lib: add subtest extra command line option handling
authorImre Deak <imre.deak@intel.com>
Fri, 2 Aug 2013 23:10:11 +0000 (02:10 +0300)
committerImre Deak <imre.deak@intel.com>
Mon, 19 Aug 2013 13:38:22 +0000 (16:38 +0300)
At the moment any command line option handling done by tests will
interfere with the option handling of the subtest interface. To fix this
add a new version of the subtest_init function accepting optional short
and long command line options. Merge these together with the subtest
interface's own long options and handle both together in the same
getopt_long call.

Signed-off-by: Imre Deak <imre.deak@intel.com>
lib/drmtest.c
lib/drmtest.h

index 64ebfee..109fbc1 100644 (file)
@@ -670,7 +670,22 @@ void __igt_fixture_end(void)
        longjmp(igt_subtest_jmpbuf, 1);
 }
 
-void igt_subtest_init(int argc, char **argv)
+static void print_usage(const char *command_str, const char *help_str,
+                       bool output_on_stderr)
+{
+       FILE *f = output_on_stderr ? stderr : stdout;
+
+       fprintf(f, "Usage: %s [OPTIONS]\n"
+                  "  --list-subtests\n"
+                  "  --run-subtest <pattern>\n"
+                  "%s\n", command_str, help_str);
+}
+
+int igt_subtest_init_parse_opts(int argc, char **argv,
+                               const char *extra_short_opts,
+                               struct option *extra_long_opts,
+                               const char *help_str,
+                               igt_opt_handler_t extra_opt_handler)
 {
        int c, option_index = 0;
        static struct option long_options[] = {
@@ -678,26 +693,84 @@ void igt_subtest_init(int argc, char **argv)
                {"run-subtest", 1, 0, 'r'},
                {NULL, 0, 0, 0,}
        };
+       struct option help_opt =
+               {"help", 0, 0, 'h'};
+       const char *command_str;
+       char *short_opts;
+       struct option *combined_opts;
+       int extra_opt_count;
+       int all_opt_count;
+       int ret = 0;
 
        test_with_subtests = true;
 
-       /* supress getopt errors about unknown options */
-       opterr = 0;
-       /* restrict the option parsing to long option names to avoid collisions
-        * with options the test declares */
-       while((c = getopt_long(argc, argv, "",
-                              long_options, &option_index)) != -1) {
+       command_str = argv[0];
+       if (strrchr(command_str, '/'))
+               command_str = strrchr(command_str, '/') + 1;
+
+       /* First calculate space for all passed-in extra long options */
+       all_opt_count = 0;
+       while (extra_long_opts && extra_long_opts[all_opt_count].name)
+               all_opt_count++;
+       extra_opt_count = all_opt_count;
+
+       /* Add space for a long help option for any passed-in help string */
+       if (help_str)
+               all_opt_count++;
+       /* Add space for the subtest long options (and the final NULL entry) */
+       all_opt_count += ARRAY_SIZE(long_options);
+
+       combined_opts = malloc(all_opt_count * sizeof(*combined_opts));
+       memcpy(combined_opts, extra_long_opts,
+              extra_opt_count * sizeof(*combined_opts));
+
+       if (help_str) {
+               combined_opts[extra_opt_count] = help_opt;
+               extra_opt_count++;
+       }
+       /* Copy the subtest long options (and the final NULL entry) */
+       memcpy(&combined_opts[extra_opt_count], long_options,
+               ARRAY_SIZE(long_options) * sizeof(*combined_opts));
+
+       ret = asprintf(&short_opts, "%s%s",
+                      extra_short_opts ? extra_short_opts : "",
+                      help_str ? "h" : "");
+       assert(ret >= 0);
+
+       while ((c = getopt_long(argc, argv, short_opts, combined_opts,
+                              &option_index)) != -1) {
                switch(c) {
                case 'l':
-                       list_subtests = true;
-                       goto out;
+                       if (!run_single_subtest)
+                               list_subtests = true;
+                       break;
                case 'r':
-                       run_single_subtest = strdup(optarg);
+                       if (!list_subtests)
+                               run_single_subtest = strdup(optarg);
+                       break;
+               case '?':
+               case 'h':
+                       print_usage(command_str, help_str, c == '?');
+                       ret = c == '?' ? -2 : -1;
                        goto out;
+               default:
+                       ret = extra_opt_handler(c, option_index);
+                       if (ret)
+                               goto out;
                }
        }
 
 out:
+       return ret;
+}
+
+void igt_subtest_init(int argc, char **argv)
+{
+       /* supress getopt errors about unknown options */
+       opterr = 0;
+
+       (void)igt_subtest_init_parse_opts(argc, argv, NULL, NULL, NULL, NULL);
+
        /* reset opt parsing */
        optind = 1;
 }
index 8419cac..485d30e 100644 (file)
@@ -101,6 +101,13 @@ void igt_progress(const char *header, uint64_t i, uint64_t total);
 /* subtest infrastructure */
 jmp_buf igt_subtest_jmpbuf;
 void igt_subtest_init(int argc, char **argv);
+typedef int (*igt_opt_handler_t)(int opt, int opt_index);
+struct option;
+int igt_subtest_init_parse_opts(int argc, char **argv,
+                               const char *extra_short_opts,
+                               struct option *extra_long_opts,
+                               const char *help_str,
+                               igt_opt_handler_t opt_handler);
 bool __igt_run_subtest(const char *subtest_name);
 /**
  * igt_subtest/_f - Denote a subtest code block