kunit: unify module and builtin suite definitions
authorJeremy Kerr <jk@codeconstruct.com.au>
Sat, 9 Jul 2022 03:19:57 +0000 (11:19 +0800)
committerShuah Khan <skhan@linuxfoundation.org>
Mon, 11 Jul 2022 23:13:09 +0000 (17:13 -0600)
Currently, KUnit runs built-in tests and tests loaded from modules
differently. For built-in tests, the kunit_test_suite{,s}() macro adds a
list of suites in the .kunit_test_suites linker section. However, for
kernel modules, a module_init() function is used to run the test suites.

This causes problems if tests are included in a module which already
defines module_init/exit_module functions, as they'll conflict with the
kunit-provided ones.

This change removes the kunit-defined module inits, and instead parses
the kunit tests from their own section in the module. After module init,
we call __kunit_test_suites_init() on the contents of that section,
which prepares and runs the suite.

This essentially unifies the module- and non-module kunit init formats.

Tested-by: MaĆ­ra Canal <maira.canal@usp.br>
Reviewed-by: Brendan Higgins <brendanhiggins@google.com>
Signed-off-by: Jeremy Kerr <jk@codeconstruct.com.au>
Signed-off-by: Daniel Latypov <dlatypov@google.com>
Signed-off-by: David Gow <davidgow@google.com>
Signed-off-by: Shuah Khan <skhan@linuxfoundation.org>
include/kunit/test.h
include/linux/module.h
kernel/module/main.c
lib/kunit/test.c

index 7646d1bcf6857da48ee29ab14aabea7b79420962..cb155d3da2847351c0b8c556fbd294c03ef582f3 100644 (file)
@@ -250,42 +250,9 @@ static inline int kunit_run_all_tests(void)
 }
 #endif /* IS_BUILTIN(CONFIG_KUNIT) */
 
-#ifdef MODULE
-/**
- * kunit_test_suites_for_module() - used to register one or more
- *                      &struct kunit_suite with KUnit.
- *
- * @__suites: a statically allocated list of &struct kunit_suite.
- *
- * Registers @__suites with the test framework. See &struct kunit_suite for
- * more information.
- *
- * If a test suite is built-in, module_init() gets translated into
- * an initcall which we don't want as the idea is that for builtins
- * the executor will manage execution.  So ensure we do not define
- * module_{init|exit} functions for the builtin case when registering
- * suites via kunit_test_suites() below.
- */
-#define kunit_test_suites_for_module(__suites)                         \
-       static int __init kunit_test_suites_init(void)                  \
-       {                                                               \
-               return __kunit_test_suites_init(__suites);              \
-       }                                                               \
-       module_init(kunit_test_suites_init);                            \
-                                                                       \
-       static void __exit kunit_test_suites_exit(void)                 \
-       {                                                               \
-               return __kunit_test_suites_exit(__suites);              \
-       }                                                               \
-       module_exit(kunit_test_suites_exit)                             \
-       MODULE_INFO(test, "Y");
-#else
-#define kunit_test_suites_for_module(__suites)
-#endif /* MODULE */
-
 #define __kunit_test_suites(unique_array, unique_suites, ...)                 \
+       MODULE_INFO(test, "Y");                                                \
        static struct kunit_suite *unique_array[] = { __VA_ARGS__, NULL };     \
-       kunit_test_suites_for_module(unique_array);                            \
        static struct kunit_suite **unique_suites                              \
        __used __section(".kunit_test_suites") = unique_array
 
@@ -295,16 +262,12 @@ static inline int kunit_run_all_tests(void)
  *
  * @__suites: a statically allocated list of &struct kunit_suite.
  *
- * Registers @suites with the test framework. See &struct kunit_suite for
- * more information.
- *
- * When builtin,  KUnit tests are all run via executor; this is done
- * by placing the array of struct kunit_suite * in the .kunit_test_suites
- * ELF section.
+ * Registers @suites with the test framework.
+ * This is done by placing the array of struct kunit_suite * in the
+ * .kunit_test_suites ELF section.
  *
- * An alternative is to build the tests as a module.  Because modules do not
- * support multiple initcall()s, we need to initialize an array of suites for a
- * module.
+ * When builtin, KUnit tests are all run via the executor at boot, and when
+ * built as a module, they run on module load.
  *
  */
 #define kunit_test_suites(__suites...)                                         \
index abd9fa916b7d93ae8006dab6c743e6d9c9396987..2490223c975d9bbf04e474c3d60f9e06490b776b 100644 (file)
@@ -505,6 +505,11 @@ struct module {
        int num_static_call_sites;
        struct static_call_site *static_call_sites;
 #endif
+#if IS_ENABLED(CONFIG_KUNIT)
+       int num_kunit_suites;
+       struct kunit_suite ***kunit_suites;
+#endif
+
 
 #ifdef CONFIG_LIVEPATCH
        bool klp; /* Is this a livepatch module? */
index 4723f1316709596bfbf975563c980f68caaa3c67..324a770f789c5a9be5b28682e02e41c9dd1ce34d 100644 (file)
@@ -2094,6 +2094,12 @@ static int find_module_sections(struct module *mod, struct load_info *info)
                                              sizeof(*mod->static_call_sites),
                                              &mod->num_static_call_sites);
 #endif
+#ifdef CONFIG_KUNIT
+       mod->kunit_suites = section_objs(info, ".kunit_test_suites",
+                                             sizeof(*mod->kunit_suites),
+                                             &mod->num_kunit_suites);
+#endif
+
        mod->extable = section_objs(info, "__ex_table",
                                    sizeof(*mod->extable), &mod->num_exentries);
 
index 8b11552dc215eea190a708e1f6aa81bc511937c1..246645eb3cef6b7c7dc3119b1a6f2afb1b9babce 100644 (file)
@@ -10,6 +10,7 @@
 #include <kunit/test.h>
 #include <kunit/test-bug.h>
 #include <linux/kernel.h>
+#include <linux/module.h>
 #include <linux/moduleparam.h>
 #include <linux/panic.h>
 #include <linux/sched/debug.h>
@@ -613,6 +614,49 @@ void __kunit_test_suites_exit(struct kunit_suite **suites)
 }
 EXPORT_SYMBOL_GPL(__kunit_test_suites_exit);
 
+#ifdef CONFIG_MODULES
+static void kunit_module_init(struct module *mod)
+{
+       unsigned int i;
+
+       for (i = 0; i < mod->num_kunit_suites; i++)
+               __kunit_test_suites_init(mod->kunit_suites[i]);
+}
+
+static void kunit_module_exit(struct module *mod)
+{
+       unsigned int i;
+
+       for (i = 0; i < mod->num_kunit_suites; i++)
+               __kunit_test_suites_exit(mod->kunit_suites[i]);
+}
+
+static int kunit_module_notify(struct notifier_block *nb, unsigned long val,
+                              void *data)
+{
+       struct module *mod = data;
+
+       switch (val) {
+       case MODULE_STATE_LIVE:
+               kunit_module_init(mod);
+               break;
+       case MODULE_STATE_GOING:
+               kunit_module_exit(mod);
+               break;
+       case MODULE_STATE_COMING:
+       case MODULE_STATE_UNFORMED:
+               break;
+       }
+
+       return 0;
+}
+
+static struct notifier_block kunit_mod_nb = {
+       .notifier_call = kunit_module_notify,
+       .priority = 0,
+};
+#endif
+
 struct kunit_kmalloc_array_params {
        size_t n;
        size_t size;
@@ -707,13 +751,19 @@ EXPORT_SYMBOL_GPL(kunit_cleanup);
 static int __init kunit_init(void)
 {
        kunit_debugfs_init();
-
+#ifdef CONFIG_MODULES
+       return register_module_notifier(&kunit_mod_nb);
+#else
        return 0;
+#endif
 }
 late_initcall(kunit_init);
 
 static void __exit kunit_exit(void)
 {
+#ifdef CONFIG_MODULES
+       unregister_module_notifier(&kunit_mod_nb);
+#endif
        kunit_debugfs_cleanup();
 }
 module_exit(kunit_exit);