mirror of
				https://github.com/torvalds/linux.git
				synced 2025-11-04 10:40:15 +02:00 
			
		
		
		
	kunit: unify module and builtin suite definitions
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>
			
			
This commit is contained in:
		
							parent
							
								
									8370b400f5
								
							
						
					
					
						commit
						3d6e446238
					
				
					 4 changed files with 68 additions and 44 deletions
				
			
		| 
						 | 
					@ -250,42 +250,9 @@ static inline int kunit_run_all_tests(void)
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#endif /* IS_BUILTIN(CONFIG_KUNIT) */
 | 
					#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, ...)		       \
 | 
					#define __kunit_test_suites(unique_array, unique_suites, ...)		       \
 | 
				
			||||||
 | 
						MODULE_INFO(test, "Y");						       \
 | 
				
			||||||
	static struct kunit_suite *unique_array[] = { __VA_ARGS__, NULL };     \
 | 
						static struct kunit_suite *unique_array[] = { __VA_ARGS__, NULL };     \
 | 
				
			||||||
	kunit_test_suites_for_module(unique_array);			       \
 | 
					 | 
				
			||||||
	static struct kunit_suite **unique_suites			       \
 | 
						static struct kunit_suite **unique_suites			       \
 | 
				
			||||||
	__used __section(".kunit_test_suites") = unique_array
 | 
						__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.
 | 
					 * @__suites: a statically allocated list of &struct kunit_suite.
 | 
				
			||||||
 *
 | 
					 *
 | 
				
			||||||
 * Registers @suites with the test framework. See &struct kunit_suite for
 | 
					 * Registers @suites with the test framework.
 | 
				
			||||||
 * more information.
 | 
					 * This is done by placing the array of struct kunit_suite * in the
 | 
				
			||||||
 | 
					 * .kunit_test_suites ELF section.
 | 
				
			||||||
 *
 | 
					 *
 | 
				
			||||||
 * When builtin,  KUnit tests are all run via executor; this is done
 | 
					 * When builtin, KUnit tests are all run via the executor at boot, and when
 | 
				
			||||||
 * by placing the array of struct kunit_suite * in the .kunit_test_suites
 | 
					 * built as a module, they run on module load.
 | 
				
			||||||
 * 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.
 | 
					 | 
				
			||||||
 *
 | 
					 *
 | 
				
			||||||
 */
 | 
					 */
 | 
				
			||||||
#define kunit_test_suites(__suites...)						\
 | 
					#define kunit_test_suites(__suites...)						\
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -505,6 +505,11 @@ struct module {
 | 
				
			||||||
	int num_static_call_sites;
 | 
						int num_static_call_sites;
 | 
				
			||||||
	struct static_call_site *static_call_sites;
 | 
						struct static_call_site *static_call_sites;
 | 
				
			||||||
#endif
 | 
					#endif
 | 
				
			||||||
 | 
					#if IS_ENABLED(CONFIG_KUNIT)
 | 
				
			||||||
 | 
						int num_kunit_suites;
 | 
				
			||||||
 | 
						struct kunit_suite ***kunit_suites;
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#ifdef CONFIG_LIVEPATCH
 | 
					#ifdef CONFIG_LIVEPATCH
 | 
				
			||||||
	bool klp; /* Is this a livepatch module? */
 | 
						bool klp; /* Is this a livepatch module? */
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -2094,6 +2094,12 @@ static int find_module_sections(struct module *mod, struct load_info *info)
 | 
				
			||||||
					      sizeof(*mod->static_call_sites),
 | 
										      sizeof(*mod->static_call_sites),
 | 
				
			||||||
					      &mod->num_static_call_sites);
 | 
										      &mod->num_static_call_sites);
 | 
				
			||||||
#endif
 | 
					#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",
 | 
						mod->extable = section_objs(info, "__ex_table",
 | 
				
			||||||
				    sizeof(*mod->extable), &mod->num_exentries);
 | 
									    sizeof(*mod->extable), &mod->num_exentries);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -10,6 +10,7 @@
 | 
				
			||||||
#include <kunit/test.h>
 | 
					#include <kunit/test.h>
 | 
				
			||||||
#include <kunit/test-bug.h>
 | 
					#include <kunit/test-bug.h>
 | 
				
			||||||
#include <linux/kernel.h>
 | 
					#include <linux/kernel.h>
 | 
				
			||||||
 | 
					#include <linux/module.h>
 | 
				
			||||||
#include <linux/moduleparam.h>
 | 
					#include <linux/moduleparam.h>
 | 
				
			||||||
#include <linux/panic.h>
 | 
					#include <linux/panic.h>
 | 
				
			||||||
#include <linux/sched/debug.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);
 | 
					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 {
 | 
					struct kunit_kmalloc_array_params {
 | 
				
			||||||
	size_t n;
 | 
						size_t n;
 | 
				
			||||||
	size_t size;
 | 
						size_t size;
 | 
				
			||||||
| 
						 | 
					@ -707,13 +751,19 @@ EXPORT_SYMBOL_GPL(kunit_cleanup);
 | 
				
			||||||
static int __init kunit_init(void)
 | 
					static int __init kunit_init(void)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	kunit_debugfs_init();
 | 
						kunit_debugfs_init();
 | 
				
			||||||
 | 
					#ifdef CONFIG_MODULES
 | 
				
			||||||
 | 
						return register_module_notifier(&kunit_mod_nb);
 | 
				
			||||||
 | 
					#else
 | 
				
			||||||
	return 0;
 | 
						return 0;
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
late_initcall(kunit_init);
 | 
					late_initcall(kunit_init);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static void __exit kunit_exit(void)
 | 
					static void __exit kunit_exit(void)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
 | 
					#ifdef CONFIG_MODULES
 | 
				
			||||||
 | 
						unregister_module_notifier(&kunit_mod_nb);
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
	kunit_debugfs_cleanup();
 | 
						kunit_debugfs_cleanup();
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
module_exit(kunit_exit);
 | 
					module_exit(kunit_exit);
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in a new issue