tests.c 30 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281
  1. /*
  2. * Copyright (C) 2007 Nokia Corporation.
  3. *
  4. * This program is free software; you can redistribute it and/or
  5. * modify it under the terms of the GNU General Public License
  6. * version 2 as published by the Free Software Foundation.
  7. *
  8. * This program is distributed in the hope that it will be useful, but
  9. * WITHOUT ANY WARRANTY; without even the implied warranty of
  10. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  11. * General Public License for more details.
  12. *
  13. * You should have received a copy of the GNU General Public License
  14. * along with this program; if not, write to the Free Software
  15. * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
  16. * 02110-1301 USA
  17. *
  18. * Author: Adrian Hunter
  19. */
  20. #include <unistd.h>
  21. #include <stdlib.h>
  22. #include <stdio.h>
  23. #include <string.h>
  24. #include <stdint.h>
  25. #include <fcntl.h>
  26. #include <errno.h>
  27. #include <libgen.h>
  28. #include <dirent.h>
  29. #include <ctype.h>
  30. #include <limits.h>
  31. #include <mntent.h>
  32. #include <time.h>
  33. #include <sys/types.h>
  34. #include <sys/stat.h>
  35. #include <sys/vfs.h>
  36. #include <sys/mount.h>
  37. #include <sys/statvfs.h>
  38. #include <linux/fs.h>
  39. #include <linux/jffs2.h>
  40. #include "tests.h"
  41. const char *tests_file_system_mount_dir = TESTS_DEFAULT_FILE_SYSTEM_MOUNT_DIR;
  42. const char *tests_file_system_type = TESTS_DEFAULT_FILE_SYSTEM_TYPE;
  43. int tests_ok_to_sync = 0; /* Whether to use fsync */
  44. /* General purpose test parameter to specify some aspect of test size.
  45. May be used by different tests in different ways or not at all.
  46. Set by the -z or --size option. */
  47. int64_t tests_size_parameter = 0;
  48. /* General purpose test parameter to specify some aspect of test repetition.
  49. May be used by different tests in different ways or not at all.
  50. Set by the -n, --repeat options. */
  51. int64_t tests_repeat_parameter = 0;
  52. /* General purpose test parameter to specify some aspect of test sleeping.
  53. May be used by different tests in different ways or not at all.
  54. Set by the -p, --sleep options. */
  55. int64_t tests_sleep_parameter = 0;
  56. /* Program name from argv[0] */
  57. const char *program_name = "unknown";
  58. /* General purpose test parameter to specify a file should be unlinked.
  59. May be used by different tests in different ways or not at all. */
  60. int tests_unlink_flag = 0;
  61. /* General purpose test parameter to specify a file should be closed.
  62. May be used by different tests in different ways or not at all. */
  63. int tests_close_flag = 0;
  64. /* General purpose test parameter to specify a file should be deleted.
  65. May be used by different tests in different ways or not at all. */
  66. int tests_delete_flag = 0;
  67. /* General purpose test parameter to specify a file have a hole.
  68. May be used by different tests in different ways or not at all. */
  69. int tests_hole_flag = 0;
  70. /* Whether it is ok to test on the root file system */
  71. static int rootok = 0;
  72. /* Maximum file name length of test file system (from statfs) */
  73. long tests_max_fname_len = 255;
  74. /* Function invoked by the CHECK macro */
  75. void tests_test(int test,const char *msg,const char *file,unsigned line)
  76. {
  77. int eno;
  78. time_t t;
  79. if (test)
  80. return;
  81. eno = errno;
  82. time(&t);
  83. fprintf(stderr, "Test failed: %s on %s"
  84. "Test failed: %s in %s at line %u\n",
  85. program_name, ctime(&t), msg, file, line);
  86. if (eno) {
  87. fprintf(stderr,"errno = %d\n",eno);
  88. fprintf(stderr,"strerror = %s\n",strerror(eno));
  89. }
  90. exit(1);
  91. }
  92. static int is_zero(const char *p)
  93. {
  94. for (;*p;++p)
  95. if (*p != '0')
  96. return 0;
  97. return 1;
  98. }
  99. static void fold(const char *text, int width)
  100. {
  101. int pos, bpos = 0;
  102. const char *p;
  103. char line[1024];
  104. if (width > 1023) {
  105. printf("%s\n", text);
  106. return;
  107. }
  108. p = text;
  109. pos = 0;
  110. while (p[pos]) {
  111. while (!isspace(p[pos])) {
  112. line[pos] = p[pos];
  113. if (!p[pos])
  114. break;
  115. ++pos;
  116. if (pos == width) {
  117. line[pos] = '\0';
  118. printf("%s\n", line);
  119. p += pos;
  120. pos = 0;
  121. }
  122. }
  123. while (pos < width) {
  124. line[pos] = p[pos];
  125. if (!p[pos]) {
  126. bpos = pos;
  127. break;
  128. }
  129. if (isspace(p[pos]))
  130. bpos = pos;
  131. ++pos;
  132. }
  133. line[bpos] = '\0';
  134. printf("%s\n", line);
  135. p += bpos;
  136. pos = 0;
  137. while (p[pos] && isspace(p[pos]))
  138. ++p;
  139. }
  140. }
  141. /* Handle common program options */
  142. int tests_get_args(int argc,
  143. char *argv[],
  144. const char *title,
  145. const char *desc,
  146. const char *opts)
  147. {
  148. int run_test = 0;
  149. int display_help = 0;
  150. int display_title = 0;
  151. int display_description = 0;
  152. int i;
  153. char *s;
  154. program_name = argv[0];
  155. s = getenv("TEST_FILE_SYSTEM_MOUNT_DIR");
  156. if (s)
  157. tests_file_system_mount_dir = strdup(s);
  158. s = getenv("TEST_FILE_SYSTEM_TYPE");
  159. if (s)
  160. tests_file_system_type = strdup(s);
  161. run_test = 1;
  162. rootok = 1;
  163. for (i = 1; i < argc; ++i) {
  164. if (strcmp(argv[i], "--help") == 0 ||
  165. strcmp(argv[i], "-h") == 0)
  166. display_help = 1;
  167. else if (strcmp(argv[i], "--title") == 0 ||
  168. strcmp(argv[i], "-t") == 0)
  169. display_title = 1;
  170. else if (strcmp(argv[i], "--description") == 0 ||
  171. strcmp(argv[i], "-d") == 0)
  172. display_description = 1;
  173. else if (strcmp(argv[i], "--sync") == 0 ||
  174. strcmp(argv[i], "-s") == 0)
  175. tests_ok_to_sync = 1;
  176. else if (strncmp(argv[i], "--size", 6) == 0 ||
  177. strncmp(argv[i], "-z", 2) == 0) {
  178. int64_t n;
  179. char *p;
  180. if (i+1 < argc && !isdigit(argv[i][strlen(argv[i])-1]))
  181. ++i;
  182. p = argv[i];
  183. while (*p && !isdigit(*p))
  184. ++p;
  185. n = atoll(p);
  186. if (n)
  187. tests_size_parameter = n;
  188. else {
  189. int all_zero = 1;
  190. for (; all_zero && *p; ++p)
  191. if (*p != '0')
  192. all_zero = 0;
  193. if (all_zero)
  194. tests_size_parameter = 0;
  195. else
  196. display_help = 1;
  197. }
  198. } else if (strncmp(argv[i], "--repeat", 8) == 0 ||
  199. strncmp(argv[i], "-n", 2) == 0) {
  200. int64_t n;
  201. char *p;
  202. if (i+1 < argc && !isdigit(argv[i][strlen(argv[i])-1]))
  203. ++i;
  204. p = argv[i];
  205. while (*p && !isdigit(*p))
  206. ++p;
  207. n = atoll(p);
  208. if (n || is_zero(p))
  209. tests_repeat_parameter = n;
  210. else
  211. display_help = 1;
  212. } else if (strncmp(argv[i], "--sleep", 7) == 0 ||
  213. strncmp(argv[i], "-p", 2) == 0) {
  214. int64_t n;
  215. char *p;
  216. if (i+1 < argc && !isdigit(argv[i][strlen(argv[i])-1]))
  217. ++i;
  218. p = argv[i];
  219. while (*p && !isdigit(*p))
  220. ++p;
  221. n = atoll(p);
  222. if (n || is_zero(p))
  223. tests_sleep_parameter = n;
  224. else
  225. display_help = 1;
  226. } else if (strcmp(argv[i], "--unlink") == 0 ||
  227. strcmp(argv[i], "-u") == 0)
  228. tests_unlink_flag = 1;
  229. else if (strcmp(argv[i], "--hole") == 0 ||
  230. strcmp(argv[i], "-o") == 0)
  231. tests_hole_flag = 1;
  232. else if (strcmp(argv[i], "--close") == 0 ||
  233. strcmp(argv[i], "-c") == 0)
  234. tests_close_flag = 1;
  235. else if (strcmp(argv[i], "--delete") == 0 ||
  236. strcmp(argv[i], "-e") == 0)
  237. tests_delete_flag = 1;
  238. else
  239. display_help = 1;
  240. }
  241. if (display_help) {
  242. run_test = 0;
  243. display_title = 0;
  244. display_description = 0;
  245. if (!opts)
  246. opts = "";
  247. printf("File System Test Program\n\n");
  248. printf("Test Title: %s\n\n", title);
  249. printf("Usage is: %s [ options ]\n",argv[0]);
  250. printf(" Options are:\n");
  251. printf(" -h, --help ");
  252. printf("Display this help\n");
  253. printf(" -t, --title ");
  254. printf("Display the test title\n");
  255. printf(" -d, --description ");
  256. printf("Display the test description\n");
  257. if (strchr(opts, 's')) {
  258. printf(" -s, --sync ");
  259. printf("Make use of fsync\n");
  260. }
  261. if (strchr(opts, 'z')) {
  262. printf(" -z, --size ");
  263. printf("Set size parameter\n");
  264. }
  265. if (strchr(opts, 'n')) {
  266. printf(" -n, --repeat ");
  267. printf("Set repeat parameter\n");
  268. }
  269. if (strchr(opts, 'p')) {
  270. printf(" -p, --sleep ");
  271. printf("Set sleep parameter\n");
  272. }
  273. if (strchr(opts, 'u')) {
  274. printf(" -u, --unlink ");
  275. printf("Unlink file\n");
  276. }
  277. if (strchr(opts, 'o')) {
  278. printf(" -o, --hole ");
  279. printf("Create a hole in a file\n");
  280. }
  281. if (strchr(opts, 'c')) {
  282. printf(" -c, --close ");
  283. printf("Close file\n");
  284. }
  285. if (strchr(opts, 'e')) {
  286. printf(" -e, --delete ");
  287. printf("Delete file\n");
  288. }
  289. printf("\nBy default, testing is done in directory ");
  290. printf("/mnt/test_file_system. To change this\nuse ");
  291. printf("environmental variable ");
  292. printf("TEST_FILE_SYSTEM_MOUNT_DIR. By default, ");
  293. printf("the file\nsystem tested is jffs2. To change this ");
  294. printf("set TEST_FILE_SYSTEM_TYPE.\n\n");
  295. printf("Test Description:\n");
  296. fold(desc, 80);
  297. } else {
  298. if (display_title)
  299. printf("%s\n", title);
  300. if (display_description)
  301. printf("%s\n", desc);
  302. if (display_title || display_description)
  303. if (argc == 2 || (argc == 3 &&
  304. display_title &&
  305. display_description))
  306. run_test = 0;
  307. }
  308. return run_test;
  309. }
  310. /* Return the number of files (or directories) in the given directory */
  311. unsigned tests_count_files_in_dir(const char *dir_name)
  312. {
  313. DIR *dir;
  314. struct dirent *entry;
  315. unsigned count = 0;
  316. dir = opendir(dir_name);
  317. CHECK(dir != NULL);
  318. for (;;) {
  319. errno = 0;
  320. entry = readdir(dir);
  321. if (entry) {
  322. if (strcmp(".",entry->d_name) != 0 &&
  323. strcmp("..",entry->d_name) != 0)
  324. ++count;
  325. } else {
  326. CHECK(errno == 0);
  327. break;
  328. }
  329. }
  330. CHECK(closedir(dir) != -1);
  331. return count;
  332. }
  333. /* Change to the file system mount directory, check that it is empty,
  334. matches the file system type, and is not the root file system */
  335. void tests_check_test_file_system(void)
  336. {
  337. struct statfs fs_info;
  338. struct stat f_info;
  339. struct stat root_f_info;
  340. if (chdir(tests_file_system_mount_dir) == -1 ||
  341. statfs(tests_file_system_mount_dir, &fs_info) == -1) {
  342. fprintf(stderr, "Invalid test file system mount directory:"
  343. " %s\n", tests_file_system_mount_dir);
  344. fprintf(stderr, "Use environment variable "
  345. "TEST_FILE_SYSTEM_MOUNT_DIR\n");
  346. CHECK(0);
  347. }
  348. tests_max_fname_len = fs_info.f_namelen;
  349. if (strcmp(tests_file_system_type, "jffs2") == 0 &&
  350. fs_info.f_type != JFFS2_SUPER_MAGIC) {
  351. fprintf(stderr, "File system type is not jffs2\n");
  352. CHECK(0);
  353. }
  354. /* Check that the test file system is not the root file system */
  355. if (!rootok) {
  356. CHECK(stat(tests_file_system_mount_dir, &f_info) != -1);
  357. CHECK(stat("/", &root_f_info) != -1);
  358. CHECK(f_info.st_dev != root_f_info.st_dev);
  359. }
  360. }
  361. /* Get the free space for the file system of the current directory */
  362. uint64_t tests_get_free_space(void)
  363. {
  364. struct statvfs fs_info;
  365. CHECK(statvfs(tests_file_system_mount_dir, &fs_info) != -1);
  366. return (uint64_t) fs_info.f_bavail * (uint64_t) fs_info.f_frsize;
  367. }
  368. /* Get the total space for the file system of the current directory */
  369. uint64_t tests_get_total_space(void)
  370. {
  371. struct statvfs fs_info;
  372. CHECK(statvfs(tests_file_system_mount_dir, &fs_info) != -1);
  373. return (uint64_t) fs_info.f_blocks * (uint64_t) fs_info.f_frsize;
  374. }
  375. #define WRITE_BUFFER_SIZE 32768
  376. static char write_buffer[WRITE_BUFFER_SIZE];
  377. static void init_write_buffer()
  378. {
  379. static int init = 0;
  380. if (!init) {
  381. int i, d;
  382. uint64_t u;
  383. u = RAND_MAX;
  384. u += 1;
  385. u /= 256;
  386. d = (int) u;
  387. srand(1);
  388. for (i = 0; i < WRITE_BUFFER_SIZE; ++i)
  389. write_buffer[i] = rand() / d;
  390. init = 1;
  391. }
  392. }
  393. /* Write size random bytes into file descriptor fd at the current position,
  394. returning the number of bytes actually written */
  395. uint64_t tests_fill_file(int fd, uint64_t size)
  396. {
  397. ssize_t written;
  398. size_t sz;
  399. unsigned start = 0, length;
  400. uint64_t remains;
  401. uint64_t actual_size = 0;
  402. init_write_buffer();
  403. remains = size;
  404. while (remains > 0) {
  405. length = WRITE_BUFFER_SIZE - start;
  406. if (remains > length)
  407. sz = length;
  408. else
  409. sz = (size_t) remains;
  410. written = write(fd, write_buffer + start, sz);
  411. if (written <= 0) {
  412. CHECK(errno == ENOSPC); /* File system full */
  413. errno = 0;
  414. break;
  415. }
  416. remains -= written;
  417. actual_size += written;
  418. if (written == sz)
  419. start = 0;
  420. else
  421. start += written;
  422. }
  423. tests_maybe_sync(fd);
  424. return actual_size;
  425. }
  426. /* Write size random bytes into file descriptor fd at offset,
  427. returning the number of bytes actually written */
  428. uint64_t tests_write_filled_file(int fd, off_t offset, uint64_t size)
  429. {
  430. ssize_t written;
  431. size_t sz;
  432. unsigned start = 0, length;
  433. uint64_t remains;
  434. uint64_t actual_size = 0;
  435. CHECK(lseek(fd, offset, SEEK_SET) == offset);
  436. init_write_buffer();
  437. remains = size;
  438. start = offset % WRITE_BUFFER_SIZE;
  439. while (remains > 0) {
  440. length = WRITE_BUFFER_SIZE - start;
  441. if (remains > length)
  442. sz = length;
  443. else
  444. sz = (size_t) remains;
  445. written = write(fd, write_buffer + start, sz);
  446. if (written <= 0) {
  447. CHECK(errno == ENOSPC); /* File system full */
  448. errno = 0;
  449. break;
  450. }
  451. remains -= written;
  452. actual_size += written;
  453. if (written == sz)
  454. start = 0;
  455. else
  456. start += written;
  457. }
  458. tests_maybe_sync(fd);
  459. return actual_size;
  460. }
  461. /* Check that a file written using tests_fill_file() and/or
  462. tests_write_filled_file() and/or tests_create_file()
  463. contains the expected random data */
  464. void tests_check_filled_file_fd(int fd)
  465. {
  466. ssize_t sz;
  467. char buf[WRITE_BUFFER_SIZE];
  468. CHECK(lseek(fd, 0, SEEK_SET) == 0);
  469. do {
  470. sz = read(fd, buf, WRITE_BUFFER_SIZE);
  471. CHECK(sz >= 0);
  472. CHECK(memcmp(buf, write_buffer, sz) == 0);
  473. } while (sz);
  474. }
  475. /* Check that a file written using tests_fill_file() and/or
  476. tests_write_filled_file() and/or tests_create_file()
  477. contains the expected random data */
  478. void tests_check_filled_file(const char *file_name)
  479. {
  480. int fd;
  481. fd = open(file_name, O_RDONLY);
  482. CHECK(fd != -1);
  483. tests_check_filled_file_fd(fd);
  484. CHECK(close(fd) != -1);
  485. }
  486. static void tests_sync_directory(const char *file_name)
  487. {
  488. char *path;
  489. char *dir;
  490. int fd;
  491. if (!tests_ok_to_sync)
  492. return;
  493. path = strdup(file_name);
  494. dir = dirname(path);
  495. fd = open(dir,O_RDONLY | tests_maybe_sync_flag());
  496. CHECK(fd != -1);
  497. CHECK(fsync(fd) != -1);
  498. CHECK(close(fd) != -1);
  499. free(path);
  500. }
  501. /* Delete a file */
  502. void tests_delete_file(const char *file_name)
  503. {
  504. CHECK(unlink(file_name) != -1);
  505. tests_sync_directory(file_name);
  506. }
  507. /* Create a file of size file_size */
  508. uint64_t tests_create_file(const char *file_name, uint64_t file_size)
  509. {
  510. int fd;
  511. int flags;
  512. mode_t mode;
  513. uint64_t actual_size; /* Less than size if the file system is full */
  514. flags = O_CREAT | O_TRUNC | O_WRONLY | tests_maybe_sync_flag();
  515. mode = S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP | S_IROTH;
  516. fd = open(file_name, flags, mode);
  517. if (fd == -1 && errno == ENOSPC) {
  518. errno = 0;
  519. return 0; /* File system full */
  520. }
  521. CHECK(fd != -1);
  522. actual_size = tests_fill_file(fd, file_size);
  523. CHECK(close(fd) != -1);
  524. if (file_size != 0 && actual_size == 0)
  525. tests_delete_file(file_name);
  526. else
  527. tests_sync_directory(file_name);
  528. return actual_size;
  529. }
  530. /* Calculate: free_space * numerator / denominator */
  531. uint64_t tests_get_big_file_size(unsigned numerator, unsigned denominator)
  532. {
  533. if (denominator == 0)
  534. denominator = 1;
  535. if (numerator > denominator)
  536. numerator = denominator;
  537. return numerator * (tests_get_free_space() / denominator);
  538. }
  539. /* Create file "fragment_n" where n is the file_number, and unlink it */
  540. int tests_create_orphan(unsigned file_number)
  541. {
  542. int fd;
  543. int flags;
  544. mode_t mode;
  545. char file_name[256];
  546. sprintf(file_name, "fragment_%u", file_number);
  547. flags = O_CREAT | O_TRUNC | O_RDWR | tests_maybe_sync_flag();
  548. mode = S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP | S_IROTH;
  549. fd = open(file_name, flags, mode);
  550. if (fd == -1 && (errno == ENOSPC || errno == EMFILE))
  551. return fd; /* File system full or too many open files */
  552. CHECK(fd != -1);
  553. tests_sync_directory(file_name);
  554. CHECK(unlink(file_name) != -1);
  555. return fd;
  556. }
  557. /* Write size bytes at offset to the file "fragment_n" where n is the
  558. file_number and file_number also determines the random data written
  559. i.e. seed for random numbers */
  560. unsigned tests_write_fragment_file(unsigned file_number,
  561. int fd,
  562. off_t offset,
  563. unsigned size)
  564. {
  565. int i, d;
  566. uint64_t u;
  567. ssize_t written;
  568. off_t pos;
  569. char buf[WRITE_BUFFER_SIZE];
  570. if (size > WRITE_BUFFER_SIZE)
  571. size = WRITE_BUFFER_SIZE;
  572. pos = lseek(fd, 0, SEEK_END);
  573. CHECK(pos != (off_t) -1);
  574. if (offset > pos)
  575. offset = pos;
  576. pos = lseek(fd, offset, SEEK_SET);
  577. CHECK(pos != (off_t) -1);
  578. CHECK(pos == offset);
  579. srand(file_number);
  580. while (offset--)
  581. rand();
  582. u = RAND_MAX;
  583. u += 1;
  584. u /= 256;
  585. d = (int) u;
  586. for (i = 0; i < size; ++i)
  587. buf[i] = rand() / d;
  588. written = write(fd, buf, size);
  589. if (written <= 0) {
  590. CHECK(errno == ENOSPC); /* File system full */
  591. errno = 0;
  592. written = 0;
  593. }
  594. tests_maybe_sync(fd);
  595. return (unsigned) written;
  596. }
  597. /* Write size bytes to the end of file descriptor fd using file_number
  598. to determine the random data written i.e. seed for random numbers */
  599. unsigned tests_fill_fragment_file(unsigned file_number, int fd, unsigned size)
  600. {
  601. off_t offset;
  602. offset = lseek(fd, 0, SEEK_END);
  603. CHECK(offset != (off_t) -1);
  604. return tests_write_fragment_file(file_number, fd, offset, size);
  605. }
  606. /* Write size bytes to the end of file "fragment_n" where n is the file_number
  607. and file_number also determines the random data written
  608. i.e. seed for random numbers */
  609. unsigned tests_append_to_fragment_file(unsigned file_number,
  610. unsigned size,
  611. int create)
  612. {
  613. int fd;
  614. int flags;
  615. mode_t mode;
  616. unsigned actual_growth;
  617. char file_name[256];
  618. sprintf(file_name, "fragment_%u", file_number);
  619. if (create)
  620. flags = O_CREAT | O_EXCL | O_WRONLY | tests_maybe_sync_flag();
  621. else
  622. flags = O_WRONLY | tests_maybe_sync_flag();
  623. mode = S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP | S_IROTH;
  624. fd = open(file_name, flags, mode);
  625. if (fd == -1 && errno == ENOSPC) {
  626. errno = 0;
  627. return 0; /* File system full */
  628. }
  629. CHECK(fd != -1);
  630. actual_growth = tests_fill_fragment_file(file_number, fd, size);
  631. CHECK(close(fd) != -1);
  632. if (create && !actual_growth)
  633. tests_delete_fragment_file(file_number);
  634. return actual_growth;
  635. }
  636. /* Write size bytes at offset to the file "fragment_n" where n is the
  637. file_number and file_number also determines the random data written
  638. i.e. seed for random numbers */
  639. unsigned tests_overwite_fragment_file( unsigned file_number,
  640. off_t offset,
  641. unsigned size)
  642. {
  643. int fd;
  644. unsigned actual_size;
  645. char file_name[256];
  646. sprintf(file_name, "fragment_%u", file_number);
  647. fd = open(file_name, O_RDWR | tests_maybe_sync_flag());
  648. if (fd == -1 && errno == ENOSPC) {
  649. errno = 0;
  650. return 0; /* File system full */
  651. }
  652. CHECK(fd != -1);
  653. actual_size = tests_write_fragment_file(file_number,
  654. fd, offset, size);
  655. CHECK(close(fd) != -1);
  656. return actual_size;
  657. }
  658. /* Delete file "fragment_n" where n is the file_number */
  659. void tests_delete_fragment_file(unsigned file_number)
  660. {
  661. char file_name[256];
  662. sprintf(file_name, "fragment_%u", file_number);
  663. tests_delete_file(file_name);
  664. }
  665. /* Check the random data in file "fragment_n" is what is expected */
  666. void tests_check_fragment_file_fd(unsigned file_number, int fd)
  667. {
  668. ssize_t sz, i;
  669. int d;
  670. uint64_t u;
  671. char buf[8192];
  672. CHECK(lseek(fd, 0, SEEK_SET) == 0);
  673. srand(file_number);
  674. u = RAND_MAX;
  675. u += 1;
  676. u /= 256;
  677. d = (int) u;
  678. for (;;) {
  679. sz = read(fd, buf, 8192);
  680. if (sz == 0)
  681. break;
  682. CHECK(sz >= 0);
  683. for (i = 0; i < sz; ++i)
  684. CHECK(buf[i] == (char) (rand() / d));
  685. }
  686. }
  687. /* Check the random data in file "fragment_n" is what is expected */
  688. void tests_check_fragment_file(unsigned file_number)
  689. {
  690. int fd;
  691. ssize_t sz, i;
  692. int d;
  693. uint64_t u;
  694. char file_name[256];
  695. char buf[8192];
  696. sprintf(file_name, "fragment_%u", file_number);
  697. fd = open(file_name, O_RDONLY);
  698. CHECK(fd != -1);
  699. srand(file_number);
  700. u = RAND_MAX;
  701. u += 1;
  702. u /= 256;
  703. d = (int) u;
  704. for (;;) {
  705. sz = read(fd, buf, 8192);
  706. if (sz == 0)
  707. break;
  708. CHECK(sz >= 0);
  709. for (i = 0; i < sz; ++i)
  710. CHECK(buf[i] == (char) (rand() / d));
  711. }
  712. CHECK(close(fd) != -1);
  713. }
  714. /* Central point to decide whether to use fsync */
  715. void tests_maybe_sync(int fd)
  716. {
  717. if (tests_ok_to_sync)
  718. CHECK(fsync(fd) != -1);
  719. }
  720. /* Return O_SYNC if ok to sync otherwise return 0 */
  721. int tests_maybe_sync_flag(void)
  722. {
  723. if (tests_ok_to_sync)
  724. return O_SYNC;
  725. return 0;
  726. }
  727. /* Return random number from 0 to n - 1 */
  728. size_t tests_random_no(size_t n)
  729. {
  730. uint64_t a, b;
  731. if (!n)
  732. return 0;
  733. if (n - 1 <= RAND_MAX) {
  734. a = rand();
  735. b = RAND_MAX;
  736. b += 1;
  737. } else {
  738. const uint64_t u = 1 + (uint64_t) RAND_MAX;
  739. a = rand();
  740. a *= u;
  741. a += rand();
  742. b = u * u;
  743. CHECK(n <= b);
  744. }
  745. if (RAND_MAX <= UINT32_MAX && n <= UINT32_MAX)
  746. return a * n / b;
  747. else /*if (RAND_MAX <= UINT64_MAX && n <= UINT64_MAX)*/ {
  748. uint64_t x, y;
  749. if (a < n) {
  750. x = a;
  751. y = n;
  752. } else {
  753. x = n;
  754. y = a;
  755. }
  756. return (x * (y / b)) + ((x * (y % b)) / b);
  757. }
  758. }
  759. /* Make a directory empty */
  760. void tests_clear_dir(const char *dir_name)
  761. {
  762. DIR *dir;
  763. struct dirent *entry;
  764. char buf[4096];
  765. dir = opendir(dir_name);
  766. CHECK(dir != NULL);
  767. CHECK(getcwd(buf, 4096) != NULL);
  768. CHECK(chdir(dir_name) != -1);
  769. for (;;) {
  770. errno = 0;
  771. entry = readdir(dir);
  772. if (entry) {
  773. if (strcmp(".",entry->d_name) != 0 &&
  774. strcmp("..",entry->d_name) != 0) {
  775. if (entry->d_type == DT_DIR) {
  776. tests_clear_dir(entry->d_name);
  777. CHECK(rmdir(entry->d_name) != -1);
  778. } else
  779. CHECK(unlink(entry->d_name) != -1);
  780. }
  781. } else {
  782. CHECK(errno == 0);
  783. break;
  784. }
  785. }
  786. CHECK(chdir(buf) != -1);
  787. CHECK(closedir(dir) != -1);
  788. }
  789. /* Create an empty sub-directory or small file in the current directory */
  790. int64_t tests_create_entry(char *return_name)
  791. {
  792. int fd;
  793. char name[256];
  794. for (;;) {
  795. sprintf(name, "%u", (unsigned) tests_random_no(10000000));
  796. fd = open(name, O_RDONLY);
  797. if (fd == -1)
  798. break;
  799. close(fd);
  800. }
  801. if (return_name)
  802. strcpy(return_name, name);
  803. if (tests_random_no(2)) {
  804. return tests_create_file(name, tests_random_no(4096));
  805. } else {
  806. if (mkdir(name, 0777) == -1) {
  807. CHECK(errno == ENOSPC);
  808. errno = 0;
  809. return 0;
  810. }
  811. return TESTS_EMPTY_DIR_SIZE;
  812. }
  813. }
  814. /* Remove a random file of empty sub-directory from the current directory */
  815. int64_t tests_remove_entry(void)
  816. {
  817. DIR *dir;
  818. struct dirent *entry;
  819. unsigned count = 0, pos;
  820. int64_t result = 0;
  821. dir = opendir(".");
  822. CHECK(dir != NULL);
  823. for (;;) {
  824. errno = 0;
  825. entry = readdir(dir);
  826. if (entry) {
  827. if (strcmp(".",entry->d_name) != 0 &&
  828. strcmp("..",entry->d_name) != 0)
  829. ++count;
  830. } else {
  831. CHECK(errno == 0);
  832. break;
  833. }
  834. }
  835. pos = tests_random_no(count);
  836. count = 0;
  837. rewinddir(dir);
  838. for (;;) {
  839. errno = 0;
  840. entry = readdir(dir);
  841. if (!entry) {
  842. CHECK(errno == 0);
  843. break;
  844. }
  845. if (strcmp(".",entry->d_name) != 0 &&
  846. strcmp("..",entry->d_name) != 0) {
  847. if (count == pos) {
  848. if (entry->d_type == DT_DIR) {
  849. tests_clear_dir(entry->d_name);
  850. CHECK(rmdir(entry->d_name) != -1);
  851. result = TESTS_EMPTY_DIR_SIZE;
  852. } else {
  853. struct stat st;
  854. CHECK(stat(entry->d_name, &st) != -1);
  855. result = st.st_size;
  856. CHECK(unlink(entry->d_name) != -1);
  857. }
  858. }
  859. ++count;
  860. }
  861. }
  862. CHECK(closedir(dir) != -1);
  863. return result;
  864. }
  865. /* Read mount information from /proc/mounts or /etc/mtab */
  866. static int tests_get_mount_info(struct mntent *info)
  867. {
  868. FILE *f;
  869. struct mntent *entry;
  870. int found = 0;
  871. f = fopen("/proc/mounts", "rb");
  872. if (!f)
  873. f = fopen("/etc/mtab", "rb");
  874. CHECK(f != NULL);
  875. while (!found) {
  876. entry = getmntent(f);
  877. if (entry) {
  878. if (strcmp(entry->mnt_dir,
  879. tests_file_system_mount_dir) == 0) {
  880. found = 1;
  881. *info = *entry;
  882. }
  883. } else
  884. break;
  885. }
  886. CHECK(fclose(f) == 0);
  887. return found;
  888. }
  889. /*
  890. * This funcion parses file-system options string, extracts standard mount
  891. * options from there, and saves them in the @flags variable. The non-standard
  892. * (fs-specific) mount options are left in @mnt_opts string, while the standard
  893. * ones will be removed from it.
  894. *
  895. * The reason for this perverted function is that we want to preserve mount
  896. * options when unmounting the file-system and mounting it again. But we cannot
  897. * pass standard* mount optins (like sync, ro, etc) as a string to the
  898. * 'mount()' function, because it fails. It accepts standard mount options only
  899. * as flags. And only the FS-specific mount options are accepted in form of a
  900. * string.
  901. */
  902. static int process_mount_options(char **mnt_opts, unsigned long *flags)
  903. {
  904. char *tmp, *opts, *p;
  905. const char *opt;
  906. /*
  907. * We are going to use 'strtok()' which modifies the original string,
  908. * so duplicate it.
  909. */
  910. tmp = strdup(*mnt_opts);
  911. if (!tmp)
  912. goto out_mem;
  913. p = opts = calloc(1, strlen(*mnt_opts) + 1);
  914. if (!opts) {
  915. free(tmp);
  916. goto out_mem;
  917. }
  918. *flags = 0;
  919. opt = strtok(tmp, ",");
  920. while (opt) {
  921. if (!strcmp(opt, "rw"))
  922. ;
  923. else if (!strcmp(opt, "ro"))
  924. *flags |= MS_RDONLY;
  925. else if (!strcmp(opt, "dirsync"))
  926. *flags |= MS_DIRSYNC;
  927. else if (!strcmp(opt, "noatime"))
  928. *flags |= MS_NOATIME;
  929. else if (!strcmp(opt, "nodiratime"))
  930. *flags |= MS_NODIRATIME;
  931. else if (!strcmp(opt, "noexec"))
  932. *flags |= MS_NOEXEC;
  933. else if (!strcmp(opt, "nosuid"))
  934. *flags |= MS_NOSUID;
  935. else if (!strcmp(opt, "relatime"))
  936. *flags |= MS_RELATIME;
  937. else if (!strcmp(opt, "sync"))
  938. *flags |= MS_SYNCHRONOUS;
  939. else {
  940. int len = strlen(opt);
  941. if (p != opts)
  942. *p++ = ',';
  943. memcpy(p, opt, len);
  944. p += len;
  945. *p = '\0';
  946. }
  947. opt = strtok(NULL, ",");
  948. }
  949. free(tmp);
  950. *mnt_opts = opts;
  951. return 0;
  952. out_mem:
  953. fprintf(stderr, "cannot allocate memory\n");
  954. return 1;
  955. }
  956. /*
  957. * Re-mount test file system. Randomly choose how to do this: re-mount R/O then
  958. * re-mount R/W, or unmount, then mount R/W, or unmount then mount R/O then
  959. * re-mount R/W, etc. This should improve test coverage.
  960. */
  961. void tests_remount(void)
  962. {
  963. int err;
  964. struct mntent mount_info;
  965. const char *source, *target, *filesystemtype;
  966. char cwd[4096], *data;
  967. unsigned long mountflags, flags;
  968. unsigned int rorw1, um, um_ro, um_rorw, rorw2;
  969. CHECK(tests_get_mount_info(&mount_info));
  970. if (strcmp(mount_info.mnt_dir,"/") == 0)
  971. return;
  972. /* Save current working directory */
  973. CHECK(getcwd(cwd, 4096) != NULL);
  974. /* Temporarily change working directory to '/' */
  975. CHECK(chdir("/") != -1);
  976. /* Choose what to do */
  977. rorw1 = tests_random_no(2);
  978. um = tests_random_no(2);
  979. um_ro = tests_random_no(2);
  980. um_rorw = tests_random_no(2);
  981. rorw2 = tests_random_no(2);
  982. if (rorw1 + um + rorw2 == 0)
  983. um = 1;
  984. source = mount_info.mnt_fsname;
  985. target = tests_file_system_mount_dir;
  986. filesystemtype = tests_file_system_type;
  987. data = mount_info.mnt_opts;
  988. process_mount_options(&data, &mountflags);
  989. if (rorw1) {
  990. /* Re-mount R/O and then re-mount R/W */
  991. flags = mountflags | MS_RDONLY | MS_REMOUNT;
  992. err = mount(source, target, filesystemtype, flags, data);
  993. CHECK(err != -1);
  994. flags = mountflags | MS_REMOUNT;
  995. flags &= ~((unsigned long)MS_RDONLY);
  996. err = mount(source, target, filesystemtype, flags, data);
  997. CHECK(err != -1);
  998. }
  999. if (um) {
  1000. /* Unmount and mount */
  1001. if (um_ro) {
  1002. /* But re-mount R/O before unmounting */
  1003. flags = mountflags | MS_RDONLY | MS_REMOUNT;
  1004. err = mount(source, target, filesystemtype,
  1005. flags, data);
  1006. CHECK(err != -1);
  1007. }
  1008. CHECK(umount(target) != -1);
  1009. if (!um_rorw) {
  1010. /* Mount R/W straight away */
  1011. err = mount(source, target, filesystemtype,
  1012. mountflags, data);
  1013. CHECK(err != -1);
  1014. } else {
  1015. /* Mount R/O and then re-mount R/W */
  1016. err = mount(source, target, filesystemtype,
  1017. mountflags | MS_RDONLY, data);
  1018. CHECK(err != -1);
  1019. flags = mountflags | MS_REMOUNT;
  1020. flags &= ~((unsigned long)MS_RDONLY);
  1021. err = mount(source, target, filesystemtype,
  1022. flags, data);
  1023. CHECK(err != -1);
  1024. }
  1025. }
  1026. if (rorw2) {
  1027. /* Re-mount R/O and then re-mount R/W */
  1028. flags = mountflags | MS_RDONLY | MS_REMOUNT;
  1029. err = mount(source, target, filesystemtype, flags, data);
  1030. CHECK(err != -1);
  1031. flags = mountflags | MS_REMOUNT;
  1032. flags &= ~((unsigned long)MS_RDONLY);
  1033. err = mount(source, target, filesystemtype, flags, data);
  1034. CHECK(err != -1);
  1035. }
  1036. /* Restore the previous working directory */
  1037. CHECK(chdir(cwd) != -1);
  1038. }
  1039. /* Un-mount or re-mount test file system */
  1040. static void tests_mnt(int mnt)
  1041. {
  1042. static struct mntent mount_info;
  1043. const char *source;
  1044. const char *target;
  1045. const char *filesystemtype;
  1046. unsigned long mountflags;
  1047. char *data;
  1048. static char cwd[4096];
  1049. if (mnt == 0) {
  1050. CHECK(tests_get_mount_info(&mount_info));
  1051. if (strcmp(mount_info.mnt_dir,"/") == 0)
  1052. return;
  1053. CHECK(getcwd(cwd, 4096) != NULL);
  1054. CHECK(chdir("/") != -1);
  1055. CHECK(umount(tests_file_system_mount_dir) != -1);
  1056. } else {
  1057. source = mount_info.mnt_fsname;
  1058. target = tests_file_system_mount_dir;
  1059. filesystemtype = tests_file_system_type;
  1060. data = mount_info.mnt_opts;
  1061. process_mount_options(&data, &mountflags);
  1062. CHECK(mount(source, target, filesystemtype, mountflags, data)
  1063. != -1);
  1064. CHECK(chdir(cwd) != -1);
  1065. }
  1066. }
  1067. /* Unmount test file system */
  1068. void tests_unmount(void)
  1069. {
  1070. tests_mnt(0);
  1071. }
  1072. /* Mount test file system */
  1073. void tests_mount(void)
  1074. {
  1075. tests_mnt(1);
  1076. }
  1077. /* Check whether the test file system is also the root file system */
  1078. int tests_fs_is_rootfs(void)
  1079. {
  1080. struct stat f_info;
  1081. struct stat root_f_info;
  1082. CHECK(stat(tests_file_system_mount_dir, &f_info) != -1);
  1083. CHECK(stat("/", &root_f_info) != -1);
  1084. if (f_info.st_dev == root_f_info.st_dev)
  1085. return 1;
  1086. else
  1087. return 0;
  1088. }
  1089. /* Try to make a directory empty */
  1090. void tests_try_to_clear_dir(const char *dir_name)
  1091. {
  1092. DIR *dir;
  1093. struct dirent *entry;
  1094. char buf[4096];
  1095. dir = opendir(dir_name);
  1096. if (dir == NULL)
  1097. return;
  1098. if (getcwd(buf, 4096) == NULL || chdir(dir_name) == -1) {
  1099. closedir(dir);
  1100. return;
  1101. }
  1102. for (;;) {
  1103. errno = 0;
  1104. entry = readdir(dir);
  1105. if (entry) {
  1106. if (strcmp(".",entry->d_name) != 0 &&
  1107. strcmp("..",entry->d_name) != 0) {
  1108. if (entry->d_type == DT_DIR) {
  1109. tests_try_to_clear_dir(entry->d_name);
  1110. rmdir(entry->d_name);
  1111. } else
  1112. unlink(entry->d_name);
  1113. }
  1114. } else {
  1115. CHECK(errno == 0);
  1116. break;
  1117. }
  1118. }
  1119. if (chdir(buf) < 0)
  1120. perror("chdir");
  1121. closedir(dir);
  1122. }
  1123. /* Check whether the test file system is also the current file system */
  1124. int tests_fs_is_currfs(void)
  1125. {
  1126. struct stat f_info;
  1127. struct stat curr_f_info;
  1128. CHECK(stat(tests_file_system_mount_dir, &f_info) != -1);
  1129. CHECK(stat(".", &curr_f_info) != -1);
  1130. if (f_info.st_dev == curr_f_info.st_dev)
  1131. return 1;
  1132. else
  1133. return 0;
  1134. }
  1135. #define PID_BUF_SIZE 64
  1136. /* Concatenate a pid to a string in a signal safe way */
  1137. void tests_cat_pid(char *buf, const char *name, pid_t pid)
  1138. {
  1139. char *p;
  1140. unsigned x;
  1141. const char digits[] = "0123456789";
  1142. char pid_buf[PID_BUF_SIZE];
  1143. x = (unsigned) pid;
  1144. p = pid_buf + PID_BUF_SIZE;
  1145. *--p = '\0';
  1146. if (x)
  1147. while (x) {
  1148. *--p = digits[x % 10];
  1149. x /= 10;
  1150. }
  1151. else
  1152. *--p = '0';
  1153. buf[0] = '\0';
  1154. strcat(buf, name);
  1155. strcat(buf, p);
  1156. }