Ciara Power <ciara.po...@intel.com> writes: > The current structure for unit testing only allows for running a > test suite with nested test cases. This means all test cases for an > autotest must be in one suite, which is not ideal. > For example, in some cases we may want to run multiple lists of test > cases that each require different setup, so should be in separate suites. > > The unit test suite struct is modified to hold a pointer to a list of > sub-testsuite pointers, along with the list of testcases as before. > Both should not be used at once, if there are sub-testsuite pointers, > that takes precedence over testcases. > > Signed-off-by: Ciara Power <ciara.po...@intel.com> > > --- > v2: > - Added macro to loop sub-testsuites. > - Added sub-testsuite summary detail. > --- > app/test/test.c | 168 ++++++++++++++++++++++++++++++++++-------------- > app/test/test.h | 1 + > 2 files changed, 122 insertions(+), 47 deletions(-) > > diff --git a/app/test/test.c b/app/test/test.c > index a795cba1bb..e401de0fdf 100644 > --- a/app/test/test.c > +++ b/app/test/test.c > @@ -41,6 +41,11 @@ extern cmdline_parse_ctx_t main_ctx[]; > suite->unit_test_cases[iter].testcase; \ > iter++, case = suite->unit_test_cases[iter]) > > +#define FOR_EACH_SUITE_TESTSUITE(iter, suite, sub_ts) > \ > + for (iter = 0, sub_ts = suite->unit_test_suites[0]; \ > + suite->unit_test_suites[iter]->suite_name != NULL; \ > + iter++, sub_ts = suite->unit_test_suites[iter]) > + > const char *prgname; /* to be set to argv[0] */ > > static const char *recursive_call; /* used in linux for MP and other tests */ > @@ -214,21 +219,46 @@ main(int argc, char **argv) > > static void > unit_test_suite_count_tcs_on_setup_fail(struct unit_test_suite *suite, > - int test_success) > + int test_success, unsigned int *sub_ts_failed, > + unsigned int *sub_ts_skipped, unsigned int *sub_ts_total) > { > struct unit_test_case tc; > - > - FOR_EACH_SUITE_TESTCASE(suite->total, suite, tc) { > - if (!tc.enabled || test_success == TEST_SKIPPED) > - suite->skipped++; > - else > - suite->failed++; > + struct unit_test_suite *ts; > + int i; > + > + if (suite->unit_test_suites) { > + FOR_EACH_SUITE_TESTSUITE(i, suite, ts) { > + unit_test_suite_count_tcs_on_setup_fail( > + ts, test_success, sub_ts_failed, > + sub_ts_skipped, sub_ts_total); > + suite->total += ts->total; > + suite->failed += ts->failed; > + suite->skipped += ts->skipped; > + if (ts->failed) > + (*sub_ts_failed)++; > + else > + (*sub_ts_skipped)++; > + (*sub_ts_total)++; > + } > + } else { > + FOR_EACH_SUITE_TESTCASE(suite->total, suite, tc) { > + if (!tc.enabled || test_success == TEST_SKIPPED) > + suite->skipped++; > + else > + suite->failed++; > + } > } > } > > static void > unit_test_suite_reset_counts(struct unit_test_suite *suite) > { > + struct unit_test_suite *ts; > + int i; > + > + if (suite->unit_test_suites) > + FOR_EACH_SUITE_TESTSUITE(i, suite, ts) > + unit_test_suite_reset_counts(ts); > suite->total = 0; > suite->executed = 0; > suite->succeeded = 0; > @@ -240,9 +270,12 @@ unit_test_suite_reset_counts(struct unit_test_suite > *suite) > int > unit_test_suite_runner(struct unit_test_suite *suite) > { > - int test_success; > + int test_success, i, ret; > const char *status; > struct unit_test_case tc; > + struct unit_test_suite *ts; > + unsigned int sub_ts_succeeded = 0, sub_ts_failed = 0; > + unsigned int sub_ts_skipped = 0, sub_ts_total = 0; > > unit_test_suite_reset_counts(suite); > > @@ -259,70 +292,111 @@ unit_test_suite_runner(struct unit_test_suite *suite) > * mark them as failed/skipped > */ > unit_test_suite_count_tcs_on_setup_fail(suite, > - test_success); > + test_success, &sub_ts_failed, > + &sub_ts_skipped, &sub_ts_total); > goto suite_summary; > } > } > > printf(" + ------------------------------------------------------- > +\n"); > > - FOR_EACH_SUITE_TESTCASE(suite->total, suite, tc) { > - if (!tc.enabled) { > - suite->skipped++; > - continue; > - } else { > - suite->executed++; > + if (suite->unit_test_suites) { > + FOR_EACH_SUITE_TESTSUITE(i, suite, ts) { > + ret = unit_test_suite_runner(ts); > + if (ret == TEST_SUCCESS) > + sub_ts_succeeded++; > + else if (ret == TEST_SKIPPED) > + sub_ts_skipped++; > + else > + sub_ts_failed++; > + sub_ts_total++; > } > + } else { > + FOR_EACH_SUITE_TESTCASE(suite->total, suite, tc) { > + if (!tc.enabled) { > + suite->skipped++; > + continue; > + } else { > + suite->executed++; > + } > + > + /* run test case setup */ > + if (tc.setup) > + test_success = tc.setup(); > + else > + test_success = TEST_SUCCESS; > + > + if (test_success == TEST_SUCCESS) { > + /* run the test case */ > + test_success = tc.testcase(); > + if (test_success == TEST_SUCCESS) > + suite->succeeded++; > + else if (test_success == TEST_SKIPPED) > + suite->skipped++; > + else if (test_success == -ENOTSUP) > + suite->unsupported++; > + else > + suite->failed++; > + } else if (test_success == -ENOTSUP) { > + suite->unsupported++; > + } else { > + suite->failed++; > + } > > - /* run test case setup */ > - if (tc.setup) > - test_success = tc.setup(); > - else > - test_success = TEST_SUCCESS; > + /* run the test case teardown */ > + if (tc.teardown) > + tc.teardown(); > > - if (test_success == TEST_SUCCESS) { > - /* run the test case */ > - test_success = tc.testcase(); > if (test_success == TEST_SUCCESS) > - suite->succeeded++; > + status = "succeeded"; > else if (test_success == TEST_SKIPPED) > - suite->skipped++; > + status = "skipped"; > else if (test_success == -ENOTSUP) > - suite->unsupported++; > + status = "unsupported"; > else > - suite->failed++; > - } else if (test_success == -ENOTSUP) { > - suite->unsupported++; > - } else { > - suite->failed++; > - } > + status = "failed"; > > - /* run the test case teardown */ > - if (tc.teardown) > - tc.teardown(); > - > - if (test_success == TEST_SUCCESS) > - status = "succeeded"; > - else if (test_success == TEST_SKIPPED) > - status = "skipped"; > - else if (test_success == -ENOTSUP) > - status = "unsupported"; > - else > - status = "failed"; > - > - printf(" + TestCase [%2d] : %s %s\n", suite->total, > - tc.name, status); > + printf(" + TestCase [%2d] : %s %s\n", suite->total, > + tc.name, status); > + } > } > > /* Run test suite teardown */ > if (suite->teardown) > suite->teardown(); > > + if (suite->unit_test_suites) > + FOR_EACH_SUITE_TESTSUITE(i, suite, ts) { > + suite->total += ts->total; > + suite->succeeded += ts->succeeded; > + suite->failed += ts->failed; > + suite->skipped += ts->skipped; > + suite->unsupported += ts->unsupported; > + suite->executed += ts->executed; > + } > + > goto suite_summary; > > suite_summary: > printf(" + ------------------------------------------------------- > +\n"); > printf(" + Test Suite Summary : %s\n", suite->suite_name); > + printf(" + ------------------------------------------------------- > +\n"); > + > + if (suite->unit_test_suites) { > + FOR_EACH_SUITE_TESTSUITE(i, suite, ts) > + printf(" + %s : %d/%d passed, %d/%d skipped, " > + "%d/%d failed, %d/%d unsupported\n", > + ts->suite_name, ts->succeeded, ts->total, > + ts->skipped, ts->total, ts->failed, ts->total, > + ts->unsupported, ts->total); > + printf(" + > ------------------------------------------------------- +\n"); > + printf(" + Sub Testsuites Total : %2d\n", sub_ts_total); > + printf(" + Sub Testsuites Skipped : %2d\n", sub_ts_skipped); > + printf(" + Sub Testsuites Passed : %2d\n", sub_ts_succeeded); > + printf(" + Sub Testsuites Failed : %2d\n", sub_ts_failed); > + printf(" + > ------------------------------------------------------- +\n"); > + } > + > printf(" + Tests Total : %2d\n", suite->total); > printf(" + Tests Skipped : %2d\n", suite->skipped); > printf(" + Tests Executed : %2d\n", suite->executed); > diff --git a/app/test/test.h b/app/test/test.h > index 10c7b496e6..e9bfc365e4 100644 > --- a/app/test/test.h > +++ b/app/test/test.h > @@ -144,6 +144,7 @@ struct unit_test_suite { > unsigned int skipped; > unsigned int failed; > unsigned int unsupported; > + struct unit_test_suite **unit_test_suites;
If it's only ever possible to either have test_suites or test_cases for iterators, maybe it's best to put in an assert for that. Either way, we probably don't need to if (suite->unit_test_suites) { ... } else { ... } It might be better to just look like: FOR_EACH_SUITE_TESTSUITE(...) { ... } FOR_EACH_SUITE_TESTCASE(...) { ... } Code looks nicer, and we can also support a mix of suite/case (unless there is a reason not to do that). WDYT? As the code exists, if I were to write a new test suite with some sub-test suites, I might do: overall_suite { .unit_test_suite = { {sub_suite1, ...}, {sub_suite2, ...} }, .unit_test_cases = { {test_case_1, ...}, {test_case_2, ...} } } And then I would be surprised if they didn't run. > struct unit_test_case unit_test_cases[]; > };