test_runner.c 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483
  1. #include "runners/test_runner.h"
  2. #include "bd/lfs_testbd.h"
  3. #include <getopt.h>
  4. #include <sys/types.h>
  5. // disk geometries
  6. struct test_geometry {
  7. const char *name;
  8. const test_define_t *defines;
  9. };
  10. // Note this includes the default configuration for test pre-defines
  11. #define TEST_GEOMETRY(name, read, prog, erase, count) \
  12. {name, (const test_define_t[]){ \
  13. /* READ_SIZE */ read, \
  14. /* PROG_SIZE */ prog, \
  15. /* BLOCK_SIZE */ erase, \
  16. /* BLOCK_COUNT */ count, \
  17. /* BLOCK_CYCLES */ -1, \
  18. /* CACHE_SIZE */ (64 % (prog) == 0) ? 64 : (prog), \
  19. /* LOOKAHEAD_SIZE */ 16, \
  20. /* ERASE_VALUE */ 0xff, \
  21. /* ERASE_CYCLES */ 0, \
  22. /* BADBLOCK_BEHAVIOR */ LFS_TESTBD_BADBLOCK_PROGERROR, \
  23. }}
  24. const struct test_geometry test_geometries[] = {
  25. // Made up geometry that works well for testing
  26. TEST_GEOMETRY("small", 16, 16, 512, (1024*1024)/512),
  27. TEST_GEOMETRY("medium", 16, 16, 4096, (1024*1024)/4096),
  28. TEST_GEOMETRY("big", 16, 16, 32*1024, (1024*1024)/(32*1024)),
  29. // EEPROM/NVRAM
  30. TEST_GEOMETRY("eeprom", 1, 1, 512, (1024*1024)/512),
  31. // SD/eMMC
  32. TEST_GEOMETRY("emmc", 512, 512, 512, (1024*1024)/512),
  33. // NOR flash
  34. TEST_GEOMETRY("nor", 1, 1, 4096, (1024*1024)/4096),
  35. // NAND flash
  36. TEST_GEOMETRY("nand", 4096, 4096, 32*1024, (1024*1024)/(32*1024)),
  37. };
  38. const size_t test_geometry_count = (
  39. sizeof(test_geometries) / sizeof(test_geometries[0]));
  40. // test define lookup and management
  41. const test_define_t *test_defines[3] = {NULL};
  42. const uint8_t *test_define_maps[2] = {NULL};
  43. test_define_t test_define(size_t define) {
  44. if (test_define_maps[0] && test_define_maps[0][define] != 0xff) {
  45. return test_defines[0][test_define_maps[0][define]];
  46. } else if (test_define_maps[1] && test_define_maps[1][define] != 0xff) {
  47. return test_defines[1][test_define_maps[1][define]];
  48. } else {
  49. return test_defines[2][define];
  50. }
  51. }
  52. void test_define_geometry(const struct test_geometry *geometry) {
  53. if (geometry) {
  54. test_defines[2] = geometry->defines;
  55. } else {
  56. test_defines[2] = NULL;
  57. }
  58. }
  59. void test_define_case(const struct test_case *case_, size_t perm) {
  60. if (case_ && case_->defines) {
  61. test_defines[1] = case_->defines[perm];
  62. test_define_maps[1] = case_->define_map;
  63. } else {
  64. test_defines[1] = NULL;
  65. test_define_maps[1] = NULL;
  66. }
  67. }
  68. void test_define_overrides(
  69. const struct test_suite *suite,
  70. const char *const *override_names,
  71. const test_define_t *override_defines,
  72. size_t override_count) {
  73. if (override_names && override_defines && override_count > 0) {
  74. uint8_t *define_map = malloc(suite->define_count * sizeof(uint8_t));
  75. memset(define_map, 0xff, suite->define_count * sizeof(bool));
  76. // lookup each override in the suite defines, they may have a
  77. // different index in each suite
  78. for (size_t i = 0; i < override_count; i++) {
  79. size_t j = 0;
  80. for (; j < suite->define_count; j++) {
  81. if (strcmp(override_names[i], suite->define_names[j]) == 0) {
  82. break;
  83. }
  84. }
  85. if (j < suite->define_count) {
  86. define_map[j] = i;
  87. }
  88. }
  89. test_defines[0] = override_defines;
  90. test_define_maps[0] = define_map;
  91. } else {
  92. test_defines[0] = NULL;
  93. free((uint8_t *)test_define_maps[0]);
  94. test_define_maps[0] = NULL;
  95. }
  96. }
  97. // operations we can do
  98. void summary(
  99. const char *const *override_names,
  100. const test_define_t *override_defines,
  101. size_t override_count) {
  102. (void)override_names;
  103. (void)override_defines;
  104. (void)override_count;
  105. printf("%-36s %7s %7s %7s %7s\n",
  106. "", "geoms", "suites", "cases", "perms");
  107. size_t cases = 0;
  108. size_t perms = 0;
  109. for (size_t i = 0; i < test_suite_count; i++) {
  110. cases += test_suites[i]->case_count;
  111. for (size_t j = 0; j < test_suites[i]->case_count; j++) {
  112. perms += test_suites[i]->cases[j]->permutations;
  113. }
  114. }
  115. printf("%-36s %7zu %7zu %7zu %7zu\n",
  116. "TOTAL",
  117. test_geometry_count,
  118. test_suite_count,
  119. cases,
  120. test_geometry_count*perms);
  121. }
  122. void list_suites(
  123. const char *const *override_names,
  124. const test_define_t *override_defines,
  125. size_t override_count) {
  126. (void)override_names;
  127. (void)override_defines;
  128. (void)override_count;
  129. printf("%-36s %-12s %7s %7s %7s\n",
  130. "id", "suite", "types", "cases", "perms");
  131. for (size_t i = 0; i < test_suite_count; i++) {
  132. size_t perms = 0;
  133. for (size_t j = 0; j < test_suites[i]->case_count; j++) {
  134. perms += test_suites[i]->cases[j]->permutations;
  135. }
  136. printf("%-36s %-12s %7s %7zu %7zu\n",
  137. test_suites[i]->id,
  138. test_suites[i]->name,
  139. "n", // TODO
  140. test_suites[i]->case_count,
  141. test_geometry_count*perms);
  142. }
  143. }
  144. void list_cases(
  145. const char *const *override_names,
  146. const test_define_t *override_defines,
  147. size_t override_count) {
  148. (void)override_names;
  149. (void)override_defines;
  150. (void)override_count;
  151. printf("%-36s %-12s %-12s %7s %7s\n",
  152. "id", "suite", "case", "types", "perms");
  153. for (size_t i = 0; i < test_suite_count; i++) {
  154. for (size_t j = 0; j < test_suites[i]->case_count; j++) {
  155. printf("%-36s %-12s %-12s %7s %7zu\n",
  156. test_suites[i]->cases[j]->id,
  157. test_suites[i]->name,
  158. test_suites[i]->cases[j]->name,
  159. "n", // TODO
  160. test_geometry_count
  161. * test_suites[i]->cases[j]->permutations);
  162. }
  163. }
  164. }
  165. void list_paths(
  166. const char *const *override_names,
  167. const test_define_t *override_defines,
  168. size_t override_count) {
  169. (void)override_names;
  170. (void)override_defines;
  171. (void)override_count;
  172. printf("%-36s %-36s\n", "id", "path");
  173. for (size_t i = 0; i < test_suite_count; i++) {
  174. for (size_t j = 0; j < test_suites[i]->case_count; j++) {
  175. printf("%-36s %-36s\n",
  176. test_suites[i]->cases[j]->id,
  177. test_suites[i]->cases[j]->path);
  178. }
  179. }
  180. }
  181. void list_defines(
  182. const char *const *override_names,
  183. const test_define_t *override_defines,
  184. size_t override_count) {
  185. (void)override_names;
  186. (void)override_defines;
  187. (void)override_count;
  188. // TODO
  189. }
  190. void list_geometries(
  191. const char *const *override_names,
  192. const test_define_t *override_defines,
  193. size_t override_count) {
  194. (void)override_names;
  195. (void)override_defines;
  196. (void)override_count;
  197. printf("%-36s %7s %7s %7s %7s %7s\n",
  198. "name", "read", "prog", "erase", "count", "size");
  199. for (size_t i = 0; i < test_geometry_count; i++) {
  200. test_define_geometry(&test_geometries[i]);
  201. printf("%-36s %7ju %7ju %7ju %7ju %7ju\n",
  202. test_geometries[i].name,
  203. READ_SIZE,
  204. PROG_SIZE,
  205. BLOCK_SIZE,
  206. BLOCK_COUNT,
  207. BLOCK_SIZE*BLOCK_COUNT);
  208. }
  209. }
  210. void run(
  211. const char *const *override_names,
  212. const test_define_t *override_defines,
  213. size_t override_count) {
  214. for (size_t i = 0; i < test_suite_count; i++) {
  215. test_define_overrides(
  216. test_suites[i],
  217. override_names, override_defines, override_count);
  218. for (size_t j = 0; j < test_suites[i]->case_count; j++) {
  219. for (size_t perm = 0;
  220. perm < test_geometry_count
  221. * test_suites[i]->cases[j]->permutations;
  222. perm++) {
  223. size_t case_perm = perm / test_geometry_count;
  224. size_t geom_perm = perm % test_geometry_count;
  225. // setup defines
  226. test_define_geometry(&test_geometries[geom_perm]);
  227. test_define_case(test_suites[i]->cases[j], case_perm);
  228. // create block device and configuration
  229. lfs_testbd_t bd;
  230. struct lfs_config cfg = {
  231. .context = &bd,
  232. .read = lfs_testbd_read,
  233. .prog = lfs_testbd_prog,
  234. .erase = lfs_testbd_erase,
  235. .sync = lfs_testbd_sync,
  236. .read_size = READ_SIZE,
  237. .prog_size = PROG_SIZE,
  238. .block_size = BLOCK_SIZE,
  239. .block_count = BLOCK_COUNT,
  240. .block_cycles = BLOCK_CYCLES,
  241. .cache_size = CACHE_SIZE,
  242. .lookahead_size = LOOKAHEAD_SIZE,
  243. };
  244. struct lfs_testbd_config bdcfg = {
  245. .erase_value = ERASE_VALUE,
  246. .erase_cycles = ERASE_CYCLES,
  247. .badblock_behavior = BADBLOCK_BEHAVIOR,
  248. .power_cycles = 0,
  249. };
  250. lfs_testbd_createcfg(&cfg, NULL, &bdcfg) => 0;
  251. // filter?
  252. if (test_suites[i]->cases[j]->filter) {
  253. bool filter = test_suites[i]->cases[j]->filter(
  254. &cfg, case_perm);
  255. if (!filter) {
  256. printf("skipped %s#%zu\n",
  257. test_suites[i]->cases[j]->id,
  258. perm);
  259. continue;
  260. }
  261. }
  262. // run the test
  263. printf("running %s#%zu\n", test_suites[i]->cases[j]->id, perm);
  264. test_suites[i]->cases[j]->run(&cfg, case_perm);
  265. printf("finished %s#%zu\n", test_suites[i]->cases[j]->id, perm);
  266. // cleanup
  267. lfs_testbd_destroy(&cfg) => 0;
  268. test_define_geometry(NULL);
  269. test_define_case(NULL, 0);
  270. }
  271. }
  272. test_define_overrides(NULL, NULL, NULL, 0);
  273. }
  274. }
  275. // option handling
  276. enum opt_flags {
  277. OPT_HELP = 'h',
  278. OPT_SUMMARY = 'Y',
  279. OPT_LIST_SUITES = 1,
  280. OPT_LIST_CASES = 'l',
  281. OPT_LIST_PATHS = 2,
  282. OPT_LIST_DEFINES = 3,
  283. OPT_LIST_GEOMETRIES = 4,
  284. OPT_DEFINE = 'D',
  285. };
  286. const char *short_opts = "hYlD:";
  287. const struct option long_opts[] = {
  288. {"help", no_argument, NULL, OPT_HELP},
  289. {"summary", no_argument, NULL, OPT_SUMMARY},
  290. {"list-suites", no_argument, NULL, OPT_LIST_SUITES},
  291. {"list-cases", no_argument, NULL, OPT_LIST_CASES},
  292. {"list-paths", no_argument, NULL, OPT_LIST_PATHS},
  293. {"list-defines", no_argument, NULL, OPT_LIST_DEFINES},
  294. {"list-geometries", no_argument, NULL, OPT_LIST_GEOMETRIES},
  295. {"define", required_argument, NULL, OPT_DEFINE},
  296. {NULL, 0, NULL, 0},
  297. };
  298. const char *const help_text[] = {
  299. "Show this help message.",
  300. "Show quick summary.",
  301. "List test suites.",
  302. "List test cases.",
  303. "List the path for each test case.",
  304. "List the defines for each test permutation.",
  305. "List the disk geometries used for testing.",
  306. "Override a test define.",
  307. };
  308. int main(int argc, char **argv) {
  309. void (*op)(
  310. const char *const *override_names,
  311. const test_define_t *override_defines,
  312. size_t override_count) = run;
  313. const char **override_names = NULL;
  314. test_define_t *override_defines = NULL;
  315. size_t override_count = 0;
  316. size_t override_cap = 0;
  317. // parse options
  318. while (true) {
  319. int c = getopt_long(argc, argv, short_opts, long_opts, NULL);
  320. switch (c) {
  321. // generate help message
  322. case OPT_HELP: {
  323. printf("usage: %s [options] [test_case]\n", argv[0]);
  324. printf("\n");
  325. printf("options:\n");
  326. size_t i = 0;
  327. while (long_opts[i].name) {
  328. size_t indent;
  329. if (long_opts[i].val >= '0' && long_opts[i].val < 'z') {
  330. printf(" -%c, --%-16s",
  331. long_opts[i].val,
  332. long_opts[i].name);
  333. indent = 8+strlen(long_opts[i].name);
  334. } else {
  335. printf(" --%-20s", long_opts[i].name);
  336. indent = 4+strlen(long_opts[i].name);
  337. }
  338. // a quick, hacky, byte-level method for text wrapping
  339. size_t len = strlen(help_text[i]);
  340. size_t j = 0;
  341. if (indent < 24) {
  342. printf("%.80s\n", &help_text[i][j]);
  343. j += 80;
  344. }
  345. while (j < len) {
  346. printf("%24s%.80s\n", "", &help_text[i][j]);
  347. j += 80;
  348. }
  349. i += 1;
  350. }
  351. printf("\n");
  352. exit(0);
  353. }
  354. // summary/list flags
  355. case OPT_SUMMARY:
  356. op = summary;
  357. break;
  358. case OPT_LIST_SUITES:
  359. op = list_suites;
  360. break;
  361. case OPT_LIST_CASES:
  362. op = list_cases;
  363. break;
  364. case OPT_LIST_PATHS:
  365. op = list_paths;
  366. break;
  367. case OPT_LIST_DEFINES:
  368. op = list_defines;
  369. break;
  370. case OPT_LIST_GEOMETRIES:
  371. op = list_geometries;
  372. break;
  373. // configuration
  374. case OPT_DEFINE: {
  375. // realloc if necessary
  376. override_count += 1;
  377. if (override_count > override_cap) {
  378. override_cap = (2*override_cap > 4) ? 2*override_cap : 4;
  379. override_names = realloc(override_names, override_cap
  380. * sizeof(const char *));
  381. override_defines = realloc(override_defines, override_cap
  382. * sizeof(test_define_t));
  383. }
  384. // parse into string key/test_define_t value, cannibalizing the
  385. // arg in the process
  386. char *sep = strchr(optarg, '=');
  387. char *parsed = NULL;
  388. if (!sep) {
  389. goto invalid_define;
  390. }
  391. override_defines[override_count-1]
  392. = strtoumax(sep+1, &parsed, 0);
  393. if (parsed == sep+1) {
  394. goto invalid_define;
  395. }
  396. override_names[override_count-1] = optarg;
  397. *sep = '\0';
  398. break;
  399. invalid_define:
  400. fprintf(stderr, "error: invalid define: %s\n", optarg);
  401. exit(-1);
  402. }
  403. // done parsing
  404. case -1:
  405. goto getopt_done;
  406. // unknown arg, getopt prints a message for us
  407. default:
  408. exit(-1);
  409. }
  410. }
  411. getopt_done:
  412. for (size_t i = 0; i < override_count; i++) {
  413. printf("define: %s %ju\n", override_names[i], override_defines[i]);
  414. }
  415. // do the thing
  416. op(
  417. override_names,
  418. override_defines,
  419. override_count);
  420. // cleanup (need to be done for valgrind testing)
  421. free(override_names);
  422. free(override_defines);}