2020-08-04 23:47:42 +03:00
// SPDX-License-Identifier: GPL-2.0
2021-05-27 00:24:04 +03:00
# include <linux/reboot.h>
2020-08-04 23:47:42 +03:00
# include <kunit/test.h>
2021-02-06 03:08:52 +03:00
# include <linux/glob.h>
# include <linux/moduleparam.h>
2020-08-04 23:47:42 +03:00
/*
* These symbols point to the . kunit_test_suites section and are defined in
* include / asm - generic / vmlinux . lds . h , and consequently must be extern .
*/
extern struct kunit_suite * const * const __kunit_suites_start [ ] ;
extern struct kunit_suite * const * const __kunit_suites_end [ ] ;
# if IS_BUILTIN(CONFIG_KUNIT)
2021-04-21 05:04:27 +03:00
static char * filter_glob_param ;
2021-10-01 01:20:45 +03:00
static char * action_param ;
2021-04-21 05:04:27 +03:00
module_param_named ( filter_glob , filter_glob_param , charp , 0 ) ;
2021-02-06 03:08:52 +03:00
MODULE_PARM_DESC ( filter_glob ,
2021-09-15 00:03:48 +03:00
" Filter which KUnit test suites/tests run at boot-time, e.g. list* or list*.*del_test " ) ;
2021-10-01 01:20:45 +03:00
module_param_named ( action , action_param , charp , 0 ) ;
MODULE_PARM_DESC ( action ,
" Changes KUnit executor behavior, valid values are: \n "
" <none>: run the tests like normal \n "
" 'list' to list test names instead of running them. \n " ) ;
2021-09-15 00:03:48 +03:00
/* glob_match() needs NULL terminated strings, so we need a copy of filter_glob_param. */
struct kunit_test_filter {
char * suite_glob ;
char * test_glob ;
} ;
/* Split "suite_glob.test_glob" into two. Assumes filter_glob is not empty. */
static void kunit_parse_filter_glob ( struct kunit_test_filter * parsed ,
const char * filter_glob )
{
const int len = strlen ( filter_glob ) ;
const char * period = strchr ( filter_glob , ' . ' ) ;
if ( ! period ) {
2021-10-02 04:36:35 +03:00
parsed - > suite_glob = kzalloc ( len + 1 , GFP_KERNEL ) ;
2021-09-15 00:03:48 +03:00
parsed - > test_glob = NULL ;
strcpy ( parsed - > suite_glob , filter_glob ) ;
return ;
}
parsed - > suite_glob = kzalloc ( period - filter_glob + 1 , GFP_KERNEL ) ;
parsed - > test_glob = kzalloc ( len - ( period - filter_glob ) + 1 , GFP_KERNEL ) ;
strncpy ( parsed - > suite_glob , filter_glob , period - filter_glob ) ;
strncpy ( parsed - > test_glob , period + 1 , len - ( period - filter_glob ) ) ;
}
/* Create a copy of suite with only tests that match test_glob. */
static struct kunit_suite *
kunit_filter_tests ( struct kunit_suite * const suite , const char * test_glob )
{
int n = 0 ;
struct kunit_case * filtered , * test_case ;
struct kunit_suite * copy ;
kunit_suite_for_each_test_case ( suite , test_case ) {
if ( ! test_glob | | glob_match ( test_glob , test_case - > name ) )
+ + n ;
}
if ( n = = 0 )
return NULL ;
/* Use memcpy to workaround copy->name being const. */
copy = kmalloc ( sizeof ( * copy ) , GFP_KERNEL ) ;
2022-05-12 00:16:26 +03:00
if ( ! copy )
return ERR_PTR ( - ENOMEM ) ;
2021-09-15 00:03:48 +03:00
memcpy ( copy , suite , sizeof ( * copy ) ) ;
filtered = kcalloc ( n + 1 , sizeof ( * filtered ) , GFP_KERNEL ) ;
2022-05-12 00:16:26 +03:00
if ( ! filtered )
return ERR_PTR ( - ENOMEM ) ;
2021-09-15 00:03:48 +03:00
n = 0 ;
kunit_suite_for_each_test_case ( suite , test_case ) {
if ( ! test_glob | | glob_match ( test_glob , test_case - > name ) )
filtered [ n + + ] = * test_case ;
}
copy - > test_cases = filtered ;
return copy ;
}
2021-02-06 03:08:52 +03:00
2021-05-27 00:24:04 +03:00
static char * kunit_shutdown ;
core_param ( kunit_shutdown , kunit_shutdown , charp , 0644 ) ;
2021-02-06 03:08:52 +03:00
static struct kunit_suite * const *
2021-04-21 05:04:27 +03:00
kunit_filter_subsuite ( struct kunit_suite * const * const subsuite ,
2021-09-15 00:03:48 +03:00
struct kunit_test_filter * filter )
2021-02-06 03:08:52 +03:00
{
int i , n = 0 ;
2021-09-15 00:03:48 +03:00
struct kunit_suite * * filtered , * filtered_suite ;
2021-02-06 03:08:52 +03:00
n = 0 ;
2021-09-15 00:03:48 +03:00
for ( i = 0 ; subsuite [ i ] ; + + i ) {
if ( glob_match ( filter - > suite_glob , subsuite [ i ] - > name ) )
2021-02-06 03:08:52 +03:00
+ + n ;
}
if ( n = = 0 )
return NULL ;
filtered = kmalloc_array ( n + 1 , sizeof ( * filtered ) , GFP_KERNEL ) ;
if ( ! filtered )
2022-05-12 00:16:26 +03:00
return ERR_PTR ( - ENOMEM ) ;
2021-02-06 03:08:52 +03:00
n = 0 ;
for ( i = 0 ; subsuite [ i ] ! = NULL ; + + i ) {
2021-09-15 00:03:48 +03:00
if ( ! glob_match ( filter - > suite_glob , subsuite [ i ] - > name ) )
continue ;
filtered_suite = kunit_filter_tests ( subsuite [ i ] , filter - > test_glob ) ;
2022-05-12 00:16:26 +03:00
if ( IS_ERR ( filtered_suite ) )
return ERR_CAST ( filtered_suite ) ;
else if ( filtered_suite )
2021-09-15 00:03:48 +03:00
filtered [ n + + ] = filtered_suite ;
2021-02-06 03:08:52 +03:00
}
filtered [ n ] = NULL ;
return filtered ;
}
struct suite_set {
struct kunit_suite * const * const * start ;
struct kunit_suite * const * const * end ;
} ;
2021-09-15 00:03:48 +03:00
static void kunit_free_subsuite ( struct kunit_suite * const * subsuite )
{
unsigned int i ;
for ( i = 0 ; subsuite [ i ] ; i + + )
kfree ( subsuite [ i ] ) ;
kfree ( subsuite ) ;
}
static void kunit_free_suite_set ( struct suite_set suite_set )
{
struct kunit_suite * const * const * suites ;
for ( suites = suite_set . start ; suites < suite_set . end ; suites + + )
kunit_free_subsuite ( * suites ) ;
kfree ( suite_set . start ) ;
}
2021-04-21 05:04:27 +03:00
static struct suite_set kunit_filter_suites ( const struct suite_set * suite_set ,
2022-05-12 00:16:26 +03:00
const char * filter_glob ,
int * err )
2021-02-06 03:08:52 +03:00
{
int i ;
struct kunit_suite * const * * copy , * const * filtered_subsuite ;
struct suite_set filtered ;
2021-09-15 00:03:48 +03:00
struct kunit_test_filter filter ;
2021-02-06 03:08:52 +03:00
2021-04-21 05:04:27 +03:00
const size_t max = suite_set - > end - suite_set - > start ;
2021-02-06 03:08:52 +03:00
copy = kmalloc_array ( max , sizeof ( * filtered . start ) , GFP_KERNEL ) ;
filtered . start = copy ;
if ( ! copy ) { /* won't be able to run anything, return an empty set */
filtered . end = copy ;
return filtered ;
}
2021-09-15 00:03:48 +03:00
kunit_parse_filter_glob ( & filter , filter_glob ) ;
2021-02-06 03:08:52 +03:00
for ( i = 0 ; i < max ; + + i ) {
2021-09-15 00:03:48 +03:00
filtered_subsuite = kunit_filter_subsuite ( suite_set - > start [ i ] , & filter ) ;
2022-05-12 00:16:26 +03:00
if ( IS_ERR ( filtered_subsuite ) ) {
* err = PTR_ERR ( filtered_subsuite ) ;
return filtered ;
}
2021-02-06 03:08:52 +03:00
if ( filtered_subsuite )
* copy + + = filtered_subsuite ;
}
filtered . end = copy ;
2021-09-15 00:03:48 +03:00
kfree ( filter . suite_glob ) ;
kfree ( filter . test_glob ) ;
2021-02-06 03:08:52 +03:00
return filtered ;
}
2021-05-27 00:24:04 +03:00
static void kunit_handle_shutdown ( void )
{
if ( ! kunit_shutdown )
return ;
if ( ! strcmp ( kunit_shutdown , " poweroff " ) )
kernel_power_off ( ) ;
else if ( ! strcmp ( kunit_shutdown , " halt " ) )
kernel_halt ( ) ;
else if ( ! strcmp ( kunit_shutdown , " reboot " ) )
kernel_restart ( NULL ) ;
}
2021-02-06 03:08:52 +03:00
static void kunit_print_tap_header ( struct suite_set * suite_set )
2020-08-04 23:47:44 +03:00
{
struct kunit_suite * const * const * suites , * const * subsuite ;
int num_of_suites = 0 ;
2021-02-06 03:08:52 +03:00
for ( suites = suite_set - > start ; suites < suite_set - > end ; suites + + )
2020-08-04 23:47:44 +03:00
for ( subsuite = * suites ; * subsuite ! = NULL ; subsuite + + )
num_of_suites + + ;
pr_info ( " TAP version 14 \n " ) ;
pr_info ( " 1..%d \n " , num_of_suites ) ;
}
2021-10-01 01:20:45 +03:00
static void kunit_exec_run_tests ( struct suite_set * suite_set )
2020-08-04 23:47:42 +03:00
{
struct kunit_suite * const * const * suites ;
2021-10-01 01:20:45 +03:00
kunit_print_tap_header ( suite_set ) ;
for ( suites = suite_set - > start ; suites < suite_set - > end ; suites + + )
__kunit_test_suites_init ( * suites ) ;
}
static void kunit_exec_list_tests ( struct suite_set * suite_set )
{
unsigned int i ;
struct kunit_suite * const * const * suites ;
struct kunit_case * test_case ;
/* Hack: print a tap header so kunit.py can find the start of KUnit output. */
pr_info ( " TAP version 14 \n " ) ;
for ( suites = suite_set - > start ; suites < suite_set - > end ; suites + + )
for ( i = 0 ; ( * suites ) [ i ] ! = NULL ; i + + ) {
kunit_suite_for_each_test_case ( ( * suites ) [ i ] , test_case ) {
pr_info ( " %s.%s \n " , ( * suites ) [ i ] - > name , test_case - > name ) ;
}
}
}
int kunit_run_all_tests ( void )
{
2021-04-21 05:04:27 +03:00
struct suite_set suite_set = {
. start = __kunit_suites_start ,
. end = __kunit_suites_end ,
} ;
2022-05-13 21:37:07 +03:00
int err = 0 ;
2020-08-04 23:47:42 +03:00
2022-05-12 00:16:26 +03:00
if ( filter_glob_param ) {
suite_set = kunit_filter_suites ( & suite_set , filter_glob_param , & err ) ;
if ( err ) {
pr_err ( " kunit executor: error filtering suites: %d \n " , err ) ;
2022-05-13 21:37:07 +03:00
goto out ;
2022-05-12 00:16:26 +03:00
}
}
2021-02-06 03:08:52 +03:00
2021-10-01 01:20:45 +03:00
if ( ! action_param )
kunit_exec_run_tests ( & suite_set ) ;
else if ( strcmp ( action_param , " list " ) = = 0 )
kunit_exec_list_tests ( & suite_set ) ;
else
pr_err ( " kunit executor: unknown action '%s' \n " , action_param ) ;
2020-08-04 23:47:44 +03:00
2021-04-21 05:04:27 +03:00
if ( filter_glob_param ) { /* a copy was made of each array */
2021-09-15 00:03:48 +03:00
kunit_free_suite_set ( suite_set ) ;
2021-02-06 03:08:52 +03:00
}
2020-08-04 23:47:42 +03:00
2021-05-27 00:24:04 +03:00
2022-05-13 21:37:07 +03:00
out :
kunit_handle_shutdown ( ) ;
return err ;
2020-08-04 23:47:42 +03:00
}
2021-04-21 05:04:27 +03:00
# if IS_BUILTIN(CONFIG_KUNIT_TEST)
# include "executor_test.c"
# endif
2020-08-04 23:47:42 +03:00
# endif /* IS_BUILTIN(CONFIG_KUNIT) */