2018-01-10 05:22:31 +03:00
/* SPDX-License-Identifier: GPL-2.0 */
2014-10-03 19:04:23 +04:00
/*
2020-06-22 21:16:48 +03:00
* kselftest . h : low - level kselftest framework to include from
* selftest programs . When possible , please use
* kselftest_harness . h instead .
2014-10-03 19:04:23 +04:00
*
* Copyright ( c ) 2014 Shuah Khan < shuahkh @ osg . samsung . com >
* Copyright ( c ) 2014 Samsung Electronics Co . , Ltd .
*
2020-06-22 21:16:47 +03:00
* Using this API consists of first counting how many tests your code
* has to run , and then starting up the reporting :
*
* ksft_print_header ( ) ;
* ksft_set_plan ( total_number_of_tests ) ;
*
* For each test , report any progress , debugging , etc with :
*
* ksft_print_msg ( fmt , . . . ) ;
*
* and finally report the pass / fail / skip / xfail state of the test with one of :
*
* ksft_test_result ( condition , fmt , . . . ) ;
* ksft_test_result_pass ( fmt , . . . ) ;
* ksft_test_result_fail ( fmt , . . . ) ;
* ksft_test_result_skip ( fmt , . . . ) ;
* ksft_test_result_xfail ( fmt , . . . ) ;
* ksft_test_result_error ( fmt , . . . ) ;
*
* When all tests are finished , clean up and exit the program with one of :
*
2022-03-25 04:14:18 +03:00
* ksft_finished ( ) ;
2020-06-22 21:16:47 +03:00
* ksft_exit ( condition ) ;
* ksft_exit_pass ( ) ;
* ksft_exit_fail ( ) ;
*
* If the program wants to report details on why the entire program has
* failed , it can instead exit with a message ( this is usually done when
* the program is aborting before finishing all tests ) :
*
* ksft_exit_fail_msg ( fmt , . . . ) ;
*
2014-10-03 19:04:23 +04:00
*/
# ifndef __KSELFTEST_H
# define __KSELFTEST_H
2019-07-20 03:03:32 +03:00
# include <errno.h>
2014-10-03 19:04:23 +04:00
# include <stdlib.h>
# include <unistd.h>
2017-06-28 17:40:20 +03:00
# include <stdarg.h>
2019-01-19 03:12:14 +03:00
# include <stdio.h>
2014-10-03 19:04:23 +04:00
2021-12-08 20:47:42 +03:00
# ifndef ARRAY_SIZE
# define ARRAY_SIZE(arr) (sizeof(arr) / sizeof((arr)[0]))
# endif
2022-04-26 00:01:11 +03:00
/*
* gcc cpuid . h provides __cpuid_count ( ) since v4 .4 .
* Clang / LLVM cpuid . h provides __cpuid_count ( ) since v3 .4 .0 .
*
* Provide local define for tests needing __cpuid_count ( ) because
* selftests need to work in older environments that do not yet
* have __cpuid_count ( ) .
*/
# ifndef __cpuid_count
# define __cpuid_count(level, count, a, b, c, d) \
__asm__ __volatile__ ( " cpuid \n \t " \
: " =a " ( a ) , " =b " ( b ) , " =c " ( c ) , " =d " ( d ) \
: " 0 " ( level ) , " 2 " ( count ) )
# endif
2015-05-13 07:07:56 +03:00
/* define kselftest exit codes */
# define KSFT_PASS 0
# define KSFT_FAIL 1
# define KSFT_XFAIL 2
# define KSFT_XPASS 3
2018-05-02 01:03:28 +03:00
# define KSFT_SKIP 4
2015-05-13 07:07:56 +03:00
2014-10-03 19:04:23 +04:00
/* counters */
struct ksft_count {
unsigned int ksft_pass ;
unsigned int ksft_fail ;
unsigned int ksft_xfail ;
unsigned int ksft_xpass ;
unsigned int ksft_xskip ;
2017-08-05 00:07:19 +03:00
unsigned int ksft_error ;
2014-10-03 19:04:23 +04:00
} ;
static struct ksft_count ksft_cnt ;
2019-04-25 02:12:37 +03:00
static unsigned int ksft_plan ;
2014-10-03 19:04:23 +04:00
2020-06-23 03:07:37 +03:00
static inline unsigned int ksft_test_num ( void )
2017-06-12 09:56:47 +03:00
{
return ksft_cnt . ksft_pass + ksft_cnt . ksft_fail +
ksft_cnt . ksft_xfail + ksft_cnt . ksft_xpass +
2017-08-05 00:07:19 +03:00
ksft_cnt . ksft_xskip + ksft_cnt . ksft_error ;
2017-06-12 09:56:47 +03:00
}
2014-10-03 19:04:23 +04:00
static inline void ksft_inc_pass_cnt ( void ) { ksft_cnt . ksft_pass + + ; }
static inline void ksft_inc_fail_cnt ( void ) { ksft_cnt . ksft_fail + + ; }
static inline void ksft_inc_xfail_cnt ( void ) { ksft_cnt . ksft_xfail + + ; }
static inline void ksft_inc_xpass_cnt ( void ) { ksft_cnt . ksft_xpass + + ; }
static inline void ksft_inc_xskip_cnt ( void ) { ksft_cnt . ksft_xskip + + ; }
2017-08-05 00:07:19 +03:00
static inline void ksft_inc_error_cnt ( void ) { ksft_cnt . ksft_error + + ; }
2014-10-03 19:04:23 +04:00
2017-07-24 22:55:18 +03:00
static inline int ksft_get_pass_cnt ( void ) { return ksft_cnt . ksft_pass ; }
static inline int ksft_get_fail_cnt ( void ) { return ksft_cnt . ksft_fail ; }
static inline int ksft_get_xfail_cnt ( void ) { return ksft_cnt . ksft_xfail ; }
static inline int ksft_get_xpass_cnt ( void ) { return ksft_cnt . ksft_xpass ; }
static inline int ksft_get_xskip_cnt ( void ) { return ksft_cnt . ksft_xskip ; }
2017-08-05 00:07:19 +03:00
static inline int ksft_get_error_cnt ( void ) { return ksft_cnt . ksft_error ; }
2017-07-24 22:55:18 +03:00
2017-06-12 09:56:47 +03:00
static inline void ksft_print_header ( void )
{
2018-02-22 03:11:54 +03:00
if ( ! ( getenv ( " KSFT_TAP_LEVEL " ) ) )
printf ( " TAP version 13 \n " ) ;
2017-06-12 09:56:47 +03:00
}
2019-04-25 02:12:37 +03:00
static inline void ksft_set_plan ( unsigned int plan )
{
ksft_plan = plan ;
printf ( " 1..%d \n " , ksft_plan ) ;
}
2014-10-03 19:04:23 +04:00
static inline void ksft_print_cnts ( void )
{
2019-04-25 02:12:37 +03:00
if ( ksft_plan ! = ksft_test_num ( ) )
printf ( " # Planned tests != run tests (%u != %u) \n " ,
ksft_plan , ksft_test_num ( ) ) ;
2020-06-22 21:16:47 +03:00
printf ( " # Totals: pass:%d fail:%d xfail:%d xpass:%d skip:%d error:%d \n " ,
2017-07-24 22:55:18 +03:00
ksft_cnt . ksft_pass , ksft_cnt . ksft_fail ,
ksft_cnt . ksft_xfail , ksft_cnt . ksft_xpass ,
2017-08-05 00:07:19 +03:00
ksft_cnt . ksft_xskip , ksft_cnt . ksft_error ) ;
2017-06-12 09:56:47 +03:00
}
2017-06-28 17:40:21 +03:00
static inline void ksft_print_msg ( const char * msg , . . . )
{
2019-07-20 03:03:32 +03:00
int saved_errno = errno ;
2017-06-28 17:40:21 +03:00
va_list args ;
va_start ( args , msg ) ;
printf ( " # " ) ;
2019-07-20 03:03:32 +03:00
errno = saved_errno ;
2017-06-28 17:40:21 +03:00
vprintf ( msg , args ) ;
va_end ( args ) ;
}
2017-06-28 17:40:20 +03:00
static inline void ksft_test_result_pass ( const char * msg , . . . )
2017-06-12 09:56:47 +03:00
{
2019-07-20 03:03:32 +03:00
int saved_errno = errno ;
2017-06-28 17:40:20 +03:00
va_list args ;
2017-06-12 09:56:47 +03:00
ksft_cnt . ksft_pass + + ;
2017-06-28 17:40:20 +03:00
va_start ( args , msg ) ;
printf ( " ok %d " , ksft_test_num ( ) ) ;
2019-07-20 03:03:32 +03:00
errno = saved_errno ;
2017-06-28 17:40:20 +03:00
vprintf ( msg , args ) ;
va_end ( args ) ;
2017-06-12 09:56:47 +03:00
}
2017-06-28 17:40:20 +03:00
static inline void ksft_test_result_fail ( const char * msg , . . . )
2017-06-12 09:56:47 +03:00
{
2019-07-20 03:03:32 +03:00
int saved_errno = errno ;
2017-06-28 17:40:20 +03:00
va_list args ;
2017-06-12 09:56:47 +03:00
ksft_cnt . ksft_fail + + ;
2017-06-28 17:40:20 +03:00
va_start ( args , msg ) ;
printf ( " not ok %d " , ksft_test_num ( ) ) ;
2019-07-20 03:03:32 +03:00
errno = saved_errno ;
2017-06-28 17:40:20 +03:00
vprintf ( msg , args ) ;
va_end ( args ) ;
2017-06-12 09:56:47 +03:00
}
2020-06-22 21:16:47 +03:00
/**
* ksft_test_result ( ) - Report test success based on truth of condition
*
* @ condition : if true , report test success , otherwise failure .
*/
# define ksft_test_result(condition, fmt, ...) do { \
if ( ! ! ( condition ) ) \
ksft_test_result_pass ( fmt , # # __VA_ARGS__ ) ; \
else \
ksft_test_result_fail ( fmt , # # __VA_ARGS__ ) ; \
} while ( 0 )
static inline void ksft_test_result_xfail ( const char * msg , . . . )
{
int saved_errno = errno ;
va_list args ;
ksft_cnt . ksft_xfail + + ;
va_start ( args , msg ) ;
printf ( " ok %d # XFAIL " , ksft_test_num ( ) ) ;
errno = saved_errno ;
vprintf ( msg , args ) ;
va_end ( args ) ;
}
2017-06-28 17:40:20 +03:00
static inline void ksft_test_result_skip ( const char * msg , . . . )
2017-06-12 09:56:47 +03:00
{
2019-07-20 03:03:32 +03:00
int saved_errno = errno ;
2017-06-28 17:40:20 +03:00
va_list args ;
2017-06-12 09:56:47 +03:00
ksft_cnt . ksft_xskip + + ;
2017-06-28 17:40:20 +03:00
va_start ( args , msg ) ;
2020-06-23 03:15:42 +03:00
printf ( " ok %d # SKIP " , ksft_test_num ( ) ) ;
2019-07-20 03:03:32 +03:00
errno = saved_errno ;
2017-06-28 17:40:20 +03:00
vprintf ( msg , args ) ;
va_end ( args ) ;
2014-10-03 19:04:23 +04:00
}
2020-06-22 21:16:47 +03:00
/* TODO: how does "error" differ from "fail" or "skip"? */
2017-08-05 00:07:19 +03:00
static inline void ksft_test_result_error ( const char * msg , . . . )
{
2019-07-20 03:03:32 +03:00
int saved_errno = errno ;
2017-08-05 00:07:19 +03:00
va_list args ;
ksft_cnt . ksft_error + + ;
va_start ( args , msg ) ;
printf ( " not ok %d # error " , ksft_test_num ( ) ) ;
2019-07-20 03:03:32 +03:00
errno = saved_errno ;
2017-08-05 00:07:19 +03:00
vprintf ( msg , args ) ;
va_end ( args ) ;
}
2014-10-03 19:04:23 +04:00
static inline int ksft_exit_pass ( void )
{
2017-06-12 09:56:47 +03:00
ksft_print_cnts ( ) ;
2015-05-13 07:07:56 +03:00
exit ( KSFT_PASS ) ;
2014-10-03 19:04:23 +04:00
}
2017-06-12 09:56:47 +03:00
2014-10-03 19:04:23 +04:00
static inline int ksft_exit_fail ( void )
{
2017-06-12 09:56:47 +03:00
ksft_print_cnts ( ) ;
2015-05-13 07:07:56 +03:00
exit ( KSFT_FAIL ) ;
2014-10-03 19:04:23 +04:00
}
2017-06-12 09:56:47 +03:00
2020-06-22 21:16:47 +03:00
/**
* ksft_exit ( ) - Exit selftest based on truth of condition
*
* @ condition : if true , exit self test with success , otherwise fail .
*/
# define ksft_exit(condition) do { \
if ( ! ! ( condition ) ) \
ksft_exit_pass ( ) ; \
else \
ksft_exit_fail ( ) ; \
} while ( 0 )
2022-03-25 04:14:18 +03:00
/**
* ksft_finished ( ) - Exit selftest with success if all tests passed
*/
# define ksft_finished() \
ksft_exit ( ksft_plan = = \
ksft_cnt . ksft_pass + \
ksft_cnt . ksft_xfail + \
ksft_cnt . ksft_xskip )
2017-06-28 17:40:20 +03:00
static inline int ksft_exit_fail_msg ( const char * msg , . . . )
2017-06-12 09:56:47 +03:00
{
2019-07-20 03:03:32 +03:00
int saved_errno = errno ;
2017-06-28 17:40:20 +03:00
va_list args ;
va_start ( args , msg ) ;
printf ( " Bail out! " ) ;
2019-07-20 03:03:32 +03:00
errno = saved_errno ;
2017-06-28 17:40:20 +03:00
vprintf ( msg , args ) ;
va_end ( args ) ;
2017-06-12 09:56:47 +03:00
ksft_print_cnts ( ) ;
exit ( KSFT_FAIL ) ;
}
2014-10-03 19:04:23 +04:00
static inline int ksft_exit_xfail ( void )
{
2017-06-12 09:56:47 +03:00
ksft_print_cnts ( ) ;
2015-05-13 07:07:56 +03:00
exit ( KSFT_XFAIL ) ;
2014-10-03 19:04:23 +04:00
}
2017-06-12 09:56:47 +03:00
2014-10-03 19:04:23 +04:00
static inline int ksft_exit_xpass ( void )
{
2017-06-12 09:56:47 +03:00
ksft_print_cnts ( ) ;
2015-05-13 07:07:56 +03:00
exit ( KSFT_XPASS ) ;
2014-10-03 19:04:23 +04:00
}
2017-06-12 09:56:47 +03:00
2017-06-28 17:40:20 +03:00
static inline int ksft_exit_skip ( const char * msg , . . . )
2014-10-03 19:04:23 +04:00
{
2020-06-23 03:15:42 +03:00
int saved_errno = errno ;
va_list args ;
2017-06-28 17:40:20 +03:00
2020-06-23 03:15:42 +03:00
va_start ( args , msg ) ;
2017-06-28 17:40:20 +03:00
2020-06-23 03:15:42 +03:00
/*
* FIXME : several tests misuse ksft_exit_skip so produce
* something sensible if some tests have already been run
* or a plan has been printed . Those tests should use
* ksft_test_result_skip or ksft_exit_fail_msg instead .
*/
if ( ksft_plan | | ksft_test_num ( ) ) {
ksft_cnt . ksft_xskip + + ;
printf ( " ok %d # SKIP " , 1 + ksft_test_num ( ) ) ;
} else {
printf ( " 1..0 # SKIP " ) ;
}
if ( msg ) {
2019-07-20 03:03:32 +03:00
errno = saved_errno ;
2017-06-28 17:40:20 +03:00
vprintf ( msg , args ) ;
va_end ( args ) ;
}
2020-06-23 03:15:42 +03:00
if ( ksft_test_num ( ) )
ksft_print_cnts ( ) ;
2015-05-13 07:07:56 +03:00
exit ( KSFT_SKIP ) ;
2014-10-03 19:04:23 +04:00
}
# endif /* __KSELFTEST_H */