static int test_id = -1;
 module_param(test_id, int, 0);
 
-static int test_range[2] = { 0, ARRAY_SIZE(tests) - 1 };
+static int test_range[2] = { 0, INT_MAX };
 module_param_array(test_range, int, NULL, 0);
 
-static __init int find_test_index(const char *test_name)
-{
-       int i;
-
-       for (i = 0; i < ARRAY_SIZE(tests); i++) {
-               if (!strcmp(tests[i].descr, test_name))
-                       return i;
-       }
-       return -1;
-}
-
-static __init int prepare_bpf_tests(void)
-{
-       if (test_id >= 0) {
-               /*
-                * if a test_id was specified, use test_range to
-                * cover only that test.
-                */
-               if (test_id >= ARRAY_SIZE(tests)) {
-                       pr_err("test_bpf: invalid test_id specified.\n");
-                       return -EINVAL;
-               }
-
-               test_range[0] = test_id;
-               test_range[1] = test_id;
-       } else if (*test_name) {
-               /*
-                * if a test_name was specified, find it and setup
-                * test_range to cover only that test.
-                */
-               int idx = find_test_index(test_name);
-
-               if (idx < 0) {
-                       pr_err("test_bpf: no test named '%s' found.\n",
-                              test_name);
-                       return -EINVAL;
-               }
-               test_range[0] = idx;
-               test_range[1] = idx;
-       } else {
-               /*
-                * check that the supplied test_range is valid.
-                */
-               if (test_range[0] >= ARRAY_SIZE(tests) ||
-                   test_range[1] >= ARRAY_SIZE(tests) ||
-                   test_range[0] < 0 || test_range[1] < 0) {
-                       pr_err("test_bpf: test_range is out of bound.\n");
-                       return -EINVAL;
-               }
-
-               if (test_range[1] < test_range[0]) {
-                       pr_err("test_bpf: test_range is ending before it starts.\n");
-                       return -EINVAL;
-               }
-       }
-
-       return 0;
-}
-
-static __init void destroy_bpf_tests(void)
-{
-}
-
 static bool exclude_test(int test_id)
 {
        return test_id < test_range[0] || test_id > test_range[1];
        for (i = 0; i < ARRAY_SIZE(skb_segment_tests); i++) {
                const struct skb_segment_test *test = &skb_segment_tests[i];
 
+               cond_resched();
+               if (exclude_test(i))
+                       continue;
+
                pr_info("#%d %s ", i, test->descr);
 
                if (test_skb_segment_single(test)) {
                int ret;
 
                cond_resched();
+               if (exclude_test(i))
+                       continue;
 
                pr_info("#%d %s ", i, test->descr);
                if (!fp) {
        return err_cnt ? -EINVAL : 0;
 }
 
+static char test_suite[32];
+module_param_string(test_suite, test_suite, sizeof(test_suite), 0);
+
+static __init int find_test_index(const char *test_name)
+{
+       int i;
+
+       if (!strcmp(test_suite, "test_bpf")) {
+               for (i = 0; i < ARRAY_SIZE(tests); i++) {
+                       if (!strcmp(tests[i].descr, test_name))
+                               return i;
+               }
+       }
+
+       if (!strcmp(test_suite, "test_tail_calls")) {
+               for (i = 0; i < ARRAY_SIZE(tail_call_tests); i++) {
+                       if (!strcmp(tail_call_tests[i].descr, test_name))
+                               return i;
+               }
+       }
+
+       if (!strcmp(test_suite, "test_skb_segment")) {
+               for (i = 0; i < ARRAY_SIZE(skb_segment_tests); i++) {
+                       if (!strcmp(skb_segment_tests[i].descr, test_name))
+                               return i;
+               }
+       }
+
+       return -1;
+}
+
+static __init int prepare_test_range(void)
+{
+       int valid_range;
+
+       if (!strcmp(test_suite, "test_bpf"))
+               valid_range = ARRAY_SIZE(tests);
+       else if (!strcmp(test_suite, "test_tail_calls"))
+               valid_range = ARRAY_SIZE(tail_call_tests);
+       else if (!strcmp(test_suite, "test_skb_segment"))
+               valid_range = ARRAY_SIZE(skb_segment_tests);
+       else
+               return 0;
+
+       if (test_id >= 0) {
+               /*
+                * if a test_id was specified, use test_range to
+                * cover only that test.
+                */
+               if (test_id >= valid_range) {
+                       pr_err("test_bpf: invalid test_id specified for '%s' suite.\n",
+                              test_suite);
+                       return -EINVAL;
+               }
+
+               test_range[0] = test_id;
+               test_range[1] = test_id;
+       } else if (*test_name) {
+               /*
+                * if a test_name was specified, find it and setup
+                * test_range to cover only that test.
+                */
+               int idx = find_test_index(test_name);
+
+               if (idx < 0) {
+                       pr_err("test_bpf: no test named '%s' found for '%s' suite.\n",
+                              test_name, test_suite);
+                       return -EINVAL;
+               }
+               test_range[0] = idx;
+               test_range[1] = idx;
+       } else if (test_range[0] != 0 || test_range[1] != INT_MAX) {
+               /*
+                * check that the supplied test_range is valid.
+                */
+               if (test_range[0] < 0 || test_range[1] >= valid_range) {
+                       pr_err("test_bpf: test_range is out of bound for '%s' suite.\n",
+                              test_suite);
+                       return -EINVAL;
+               }
+
+               if (test_range[1] < test_range[0]) {
+                       pr_err("test_bpf: test_range is ending before it starts.\n");
+                       return -EINVAL;
+               }
+       }
+
+       return 0;
+}
+
 static int __init test_bpf_init(void)
 {
        struct bpf_array *progs = NULL;
        int ret;
 
-       ret = prepare_bpf_tests();
+       if (strlen(test_suite) &&
+           strcmp(test_suite, "test_bpf") &&
+           strcmp(test_suite, "test_tail_calls") &&
+           strcmp(test_suite, "test_skb_segment")) {
+               pr_err("test_bpf: invalid test_suite '%s' specified.\n", test_suite);
+               return -EINVAL;
+       }
+
+       /*
+        * if test_suite is not specified, but test_id, test_name or test_range
+        * is specified, set 'test_bpf' as the default test suite.
+        */
+       if (!strlen(test_suite) &&
+           (test_id != -1 || strlen(test_name) ||
+           (test_range[0] != 0 || test_range[1] != INT_MAX))) {
+               pr_info("test_bpf: set 'test_bpf' as the default test_suite.\n");
+               strscpy(test_suite, "test_bpf", sizeof(test_suite));
+       }
+
+       ret = prepare_test_range();
        if (ret < 0)
                return ret;
 
-       ret = test_bpf();
-       destroy_bpf_tests();
-       if (ret)
-               return ret;
+       if (!strlen(test_suite) || !strcmp(test_suite, "test_bpf")) {
+               ret = test_bpf();
+               if (ret)
+                       return ret;
+       }
 
-       ret = prepare_tail_call_tests(&progs);
-       if (ret)
-               return ret;
-       ret = test_tail_calls(progs);
-       destroy_tail_call_tests(progs);
-       if (ret)
-               return ret;
+       if (!strlen(test_suite) || !strcmp(test_suite, "test_tail_calls")) {
+               ret = prepare_tail_call_tests(&progs);
+               if (ret)
+                       return ret;
+               ret = test_tail_calls(progs);
+               destroy_tail_call_tests(progs);
+               if (ret)
+                       return ret;
+       }
 
-       return test_skb_segment();
+       if (!strlen(test_suite) || !strcmp(test_suite, "test_skb_segment"))
+               return test_skb_segment();
+
+       return 0;
 }
 
 static void __exit test_bpf_exit(void)