/* * Copyright (c) 2014 Fujitsu Ltd. * Author: Xiaoguang Wang * * This program is free software; you can redistribute it and/or modify it * under the terms of version 2 of the GNU General Public License as * published by the Free Software Foundation. * * This program is distributed in the hope that it would be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. * * You should have received a copy of the GNU General Public License along * with this program; if not, write the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */ #include #include #include #include #include #include "test.h" #include "usctest.h" #include "safe_macros.h" #define MAX_SANE_HARD_LINKS 65535 /* * filesystems whose subdir limit is less than MAX_SANE_HARD_LINKS * XXX: we cannot filter ext4 out, because ext2/ext3/ext4 have the * same magic number */ const long subdir_limit_whitelist[] = { TST_EXT2_OLD_MAGIC, TST_EXT234_MAGIC, TST_MINIX_MAGIC, TST_MINIX_MAGIC2, TST_MINIX2_MAGIC, TST_MINIX2_MAGIC2, TST_MINIX3_MAGIC, TST_UDF_MAGIC, TST_SYSV2_MAGIC, TST_SYSV4_MAGIC, TST_UFS_MAGIC, TST_UFS2_MAGIC, TST_F2FS_MAGIC, TST_NILFS_MAGIC, TST_EXOFS_MAGIC }; int tst_fs_fill_hardlinks_(void (*cleanup) (void), const char *dir) { unsigned int i, j; char base_filename[PATH_MAX], link_filename[PATH_MAX]; struct stat s; if (stat(dir, &s) == -1 && errno == ENOENT) SAFE_MKDIR(cleanup, dir, 0744); SAFE_STAT(cleanup, dir, &s); if (!S_ISDIR(s.st_mode)) { tst_brkm(TBROK, cleanup, "%s is not directory", dir); return 0; } sprintf(base_filename, "%s/testfile0", dir); SAFE_TOUCH(cleanup, base_filename, 0644, NULL); for (i = 1; i < MAX_SANE_HARD_LINKS; i++) { sprintf(link_filename, "%s/testfile%d", dir, i); if (link(base_filename, link_filename) == 0) continue; switch (errno) { case EMLINK: SAFE_STAT(cleanup, base_filename, &s); if (s.st_nlink != i) { tst_brkm(TBROK, cleanup, "wrong number of " "hard links for %s have %i, should be" " %d", base_filename, (int)s.st_nlink, i); return 0; } else { tst_resm(TINFO, "the maximum number of hard " "links to %s is hit: %d", base_filename, (int)s.st_nlink); return s.st_nlink; } case ENOSPC: case EDQUOT: tst_resm(TINFO | TERRNO, "link(%s, %s) failed", base_filename, link_filename); goto max_hardlinks_cleanup; default: tst_brkm(TBROK, cleanup, "link(%s, %s) failed " "unexpectedly: %s", base_filename, link_filename, strerror(errno)); return 0; } } tst_resm(TINFO, "Failed reach the hardlinks limit"); max_hardlinks_cleanup: for (j = 0; j < i; j++) { sprintf(link_filename, "%s/testfile%d", dir, j); SAFE_UNLINK(cleanup, link_filename); } return 0; } int tst_fs_fill_subdirs_(void (*cleanup) (void), const char *dir) { unsigned int i, j, whitelist_size; char dirname[PATH_MAX]; struct stat s; long fs_type; if (stat(dir, &s) == -1 && errno == ENOENT) SAFE_MKDIR(cleanup, dir, 0744); SAFE_STAT(cleanup, dir, &s); if (!S_ISDIR(s.st_mode)) { tst_brkm(TBROK, cleanup, "%s is not directory", dir); return 0; } /* for current kernel, subdir limit is not availiable for all fs */ fs_type = tst_fs_type(cleanup, dir); whitelist_size = ARRAY_SIZE(subdir_limit_whitelist); for (i = 0; i < whitelist_size; i++) { if (fs_type == subdir_limit_whitelist[i]) break; } if (i == whitelist_size) { tst_resm(TINFO, "subdir limit is not availiable for " "%s filesystem", tst_fs_type_name(fs_type)); return 0; } for (i = 0; i < MAX_SANE_HARD_LINKS; i++) { sprintf(dirname, "%s/testdir%d", dir, i); if (mkdir(dirname, 0755) == 0) continue; switch (errno) { case EMLINK: SAFE_STAT(cleanup, dir, &s); /* * i+2 because there are two links to each newly * created directory (the '.' and link from parent dir) */ if (s.st_nlink != (i + 2)) { tst_brkm(TBROK, cleanup, "%s link counts have" "%d, should be %d", dir, (int)s.st_nlink, i + 2); return 0; } else { tst_resm(TINFO, "the maximum subdirectories in " "%s is hit: %d", dir, (int)s.st_nlink); return s.st_nlink; } case ENOSPC: case EDQUOT: tst_resm(TINFO | TERRNO, "mkdir(%s, 0755) failed", dirname); goto max_subdirs_cleanup; default: tst_brkm(TBROK, cleanup, "mkdir(%s, 0755) failed " "unexpectedly: %s", dirname, strerror(errno)); return 0; } } tst_resm(TINFO, "Failed reach the subdirs limit on %s filesystem", tst_fs_type_name(fs_type)); max_subdirs_cleanup: for (j = 0; j < i; j++) { sprintf(dirname, "%s/testdir%d", dir, j); SAFE_RMDIR(cleanup, dirname); } return 0; }