mirror of
https://github.com/tiagovignatti/intel-gpu-tools.git
synced 2025-06-11 01:46:14 +00:00
lib/igt_core: api documentation
At most a bit of comment of function declaration movement for more polish. One tricky bit is to #ifdef out (only for gtkdoc of course) the struct option; forward declaration - gtkdoc needlessly lists it. FIXME: The struct documentation for igt_helper_process somehow doesn't get picked up ... Same issue seems to be with the igt_log_level enum, I've shoveled the relevant documentation into igt_log in free-form for now. Signed-off-by: Daniel Vetter <daniel.vetter@ffwll.ch>
This commit is contained in:
parent
d63fe1519d
commit
55e64989e5
197
lib/igt_core.c
197
lib/igt_core.c
@ -72,6 +72,17 @@
|
||||
* result propagation. Other generally useful functionality includes optional
|
||||
* structure logging infrastructure and some support code for running reduced
|
||||
* test set on in simulated hardware environments.
|
||||
*
|
||||
* When writing tests with subtests it is extremely important that nothing
|
||||
* interferes with the subtest enumeration. In i-g-t subtests are enumerated at
|
||||
* runtime, which allows powerful testcase enumeration. But it makes subtest
|
||||
* enumeration a bit more tricky since the test code needs to be careful to
|
||||
* never run any code which might fail (like trying to do privileged operations
|
||||
* or opening device driver nodes).
|
||||
*
|
||||
* To allow this i-g-t provides #igt_fixture code blocks for setup code outside
|
||||
* of subtests and automatically skips the subtest code blocks themselves. For
|
||||
* special cases igt_only_list_subtests() is also provided.
|
||||
*/
|
||||
|
||||
static unsigned int exit_handler_count;
|
||||
@ -165,6 +176,25 @@ static void oom_adjust_for_doom(void)
|
||||
igt_assert(write(fd, always_kill, sizeof(always_kill)) == sizeof(always_kill));
|
||||
}
|
||||
|
||||
/**
|
||||
* igt_subtest_init_parse_opts:
|
||||
* @argc: argc from the test's main()
|
||||
* @argv: argv from the test's main()
|
||||
* @extra_short_opts: getopt_long() compliant list with additional short options
|
||||
* @extra_long_opts: getopt_long() compliant list with additional long options
|
||||
* @help_str: help string for the additional options
|
||||
* @extra_opt_handler: handler for the additional options
|
||||
*
|
||||
* This function handles the subtest related cmdline options and allows an
|
||||
* arbitrary set of additional options. This is useful for tests which have
|
||||
* additional knobs to tune when run manually like the number of rounds execute
|
||||
* or the size of the allocated buffer objects.
|
||||
*
|
||||
* Tests without special needs should just use igt_subtest_init() or use
|
||||
* #igt_main directly instead of their own main() function.
|
||||
*
|
||||
* Returns: Forwards any option parsing errors from getopt_long.
|
||||
*/
|
||||
int igt_subtest_init_parse_opts(int argc, char **argv,
|
||||
const char *extra_short_opts,
|
||||
struct option *extra_long_opts,
|
||||
@ -274,6 +304,19 @@ static void common_init(void)
|
||||
igt_log_level = IGT_LOG_NONE;
|
||||
}
|
||||
|
||||
/**
|
||||
* igt_subtest_init:
|
||||
* @argc: argc from the test's main()
|
||||
* @argv: argv from the test's main()
|
||||
*
|
||||
* This initializes the for tests with subtests without the need for additional
|
||||
* cmdline options. It is just a simplified version of
|
||||
* igt_subtest_init_parse_opts().
|
||||
*
|
||||
* If there's not a reason to the contrary it's less error prone to just use an
|
||||
* #igt_main block instead of stitching the tests's main() function together
|
||||
* manually.
|
||||
*/
|
||||
void igt_subtest_init(int argc, char **argv)
|
||||
{
|
||||
int ret;
|
||||
@ -292,6 +335,15 @@ void igt_subtest_init(int argc, char **argv)
|
||||
common_init();
|
||||
}
|
||||
|
||||
/**
|
||||
* igt_simple_init:
|
||||
*
|
||||
* This initializes a simple test without any support for subtests.
|
||||
*
|
||||
* If there's not a reason to the contrary it's less error prone to just use an
|
||||
* #igt_simple_main block instead of stitching the tests's main() function together
|
||||
* manually.
|
||||
*/
|
||||
void igt_simple_init(void)
|
||||
{
|
||||
print_version();
|
||||
@ -330,11 +382,23 @@ bool __igt_run_subtest(const char *subtest_name)
|
||||
return (in_subtest = subtest_name);
|
||||
}
|
||||
|
||||
/**
|
||||
* igt_subtest_name:
|
||||
*
|
||||
* Returns: The name of the currently executed subtest or NULL if called from
|
||||
* outside a subtest block.
|
||||
*/
|
||||
const char *igt_subtest_name(void)
|
||||
{
|
||||
return in_subtest;
|
||||
}
|
||||
|
||||
/**
|
||||
* igt_only_list_subtests:
|
||||
*
|
||||
* Returns: Returns true if only subtest should be listed and any setup code
|
||||
* must be skipped, false otherwise.
|
||||
*/
|
||||
bool igt_only_list_subtests(void)
|
||||
{
|
||||
return list_subtests;
|
||||
@ -353,6 +417,20 @@ static void exit_subtest(const char *result)
|
||||
longjmp(igt_subtest_jmpbuf, 1);
|
||||
}
|
||||
|
||||
/**
|
||||
* igt_skip:
|
||||
* @f: format string
|
||||
* @...: optional arguments used in the format string
|
||||
*
|
||||
* Subtest aware test skipping. The format string is printed to stderr as the
|
||||
* reason why the test skipped.
|
||||
*
|
||||
* For tests with subtests this will either bail out of the current subtest or
|
||||
* mark all subsequent subtests as SKIP (presuming some global setup code
|
||||
* failed).
|
||||
*
|
||||
* For normal tests without subtest it will directly exit.
|
||||
*/
|
||||
void igt_skip(const char *f, ...)
|
||||
{
|
||||
va_list args;
|
||||
@ -407,6 +485,14 @@ void __igt_skip_check(const char *file, const int line,
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* igt_success:
|
||||
*
|
||||
* Complete a (subtest) as successfull
|
||||
*
|
||||
* This bails out of a subtests and marks it as successful. For global tests it
|
||||
* it won't bail out of anything.
|
||||
*/
|
||||
void igt_success(void)
|
||||
{
|
||||
succeeded_one = true;
|
||||
@ -414,6 +500,21 @@ void igt_success(void)
|
||||
exit_subtest("SUCCESS");
|
||||
}
|
||||
|
||||
/**
|
||||
* igt_fail:
|
||||
* @exitcode: exitcode
|
||||
*
|
||||
* Fail a testcase. The exitcode is used as the exit code of the test process.
|
||||
* It may not be 0 (which indicates success) or 77 (which indicates a skipped
|
||||
* test).
|
||||
*
|
||||
* For tests with subtests this will either bail out of the current subtest or
|
||||
* mark all subsequent subtests as FAIL (presuming some global setup code
|
||||
* failed).
|
||||
*
|
||||
* For normal tests without subtest it will directly exit with the given
|
||||
* exitcode.
|
||||
*/
|
||||
void igt_fail(int exitcode)
|
||||
{
|
||||
assert(exitcode != 0 && exitcode != 77);
|
||||
@ -473,6 +574,21 @@ void __igt_fail_assert(int exitcode, const char *file,
|
||||
igt_fail(exitcode);
|
||||
}
|
||||
|
||||
/**
|
||||
* igt_exit:
|
||||
*
|
||||
* exit() for both types (simple and with subtests) of i-g-t tests.
|
||||
*
|
||||
* This will exit the test with the right exit code when subtests have been
|
||||
* skipped. For normal tests it exits with a successful exit code, presuming
|
||||
* everything has worked out. For subtests it also checks that at least one
|
||||
* subtest has been run (save when only listing subtests.
|
||||
*
|
||||
* It is an error to normally exit a test with subtests without calling
|
||||
* igt_exit() - without it the result reporting will be wrong. To avoid such
|
||||
* issues it is highly recommended to use #igt_main instead of a hand-rolled
|
||||
* main() function.
|
||||
*/
|
||||
void igt_exit(void)
|
||||
{
|
||||
igt_exit_called = true;
|
||||
@ -562,6 +678,13 @@ bool __igt_fork_helper(struct igt_helper_process *proc)
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* igt_stop_helper:
|
||||
* @proc: #igt_helper_process structure
|
||||
*
|
||||
* Terminates a helper process. It is an error to call this on a helper process
|
||||
* which hasn't been spawned yet.
|
||||
*/
|
||||
void igt_stop_helper(struct igt_helper_process *proc)
|
||||
{
|
||||
int status, ret;
|
||||
@ -583,6 +706,13 @@ void igt_stop_helper(struct igt_helper_process *proc)
|
||||
helper_process_count--;
|
||||
}
|
||||
|
||||
/**
|
||||
* igt_wait_helper:
|
||||
* @proc: #igt_helper_process structure
|
||||
*
|
||||
* Joins a helper process. It is an error to call this on a helper process which
|
||||
* hasn't been spawned yet.
|
||||
*/
|
||||
void igt_wait_helper(struct igt_helper_process *proc)
|
||||
{
|
||||
int status;
|
||||
@ -656,9 +786,15 @@ bool __igt_fork(void)
|
||||
/**
|
||||
* igt_waitchildren:
|
||||
*
|
||||
* Wait for all children forked with igt_fork
|
||||
* Wait for all children forked with igt_fork.
|
||||
*
|
||||
* The magic here is that exit codes from children will be correctly propagated
|
||||
* to the main thread, including the relevant exitcode if a child thread failed.
|
||||
* Of course if multiple children failed with differen exitcodes the resulting
|
||||
* exitcode will be non-deterministic.
|
||||
*
|
||||
* Note that igt_skip() will not be forwarded, feature tests need to be done
|
||||
* before spawning threads with igt_fork().
|
||||
*/
|
||||
void igt_waitchildren(void)
|
||||
{
|
||||
@ -773,7 +909,10 @@ static void fatal_sig_handler(int sig)
|
||||
syscall(SYS_tgkill, pid, tid, sig);
|
||||
}
|
||||
|
||||
/*
|
||||
/**
|
||||
* igt_install_exit_handler:
|
||||
* @fn: exit handler function
|
||||
*
|
||||
* Set a handler that will be called either when the process calls exit() or
|
||||
* returns from the main function, or one of the signals in 'handled_signals'
|
||||
* is raised. MAX_EXIT_HANDLERS handlers can be installed, each of which will
|
||||
@ -782,7 +921,10 @@ static void fatal_sig_handler(int sig)
|
||||
* original signal disposition after all handlers returned.
|
||||
*
|
||||
* The handler will be passed the signal number if called due to a signal, or
|
||||
* 0 otherwise.
|
||||
* 0 otherwise. Exit handlers can also be used from test children spawned with
|
||||
* igt_fork(), but not from within helper processes spawned with
|
||||
* igt_helper_process(). The list of exit handlers is reset when forking to
|
||||
* avoid issues with children cleanup up the parent's state too early.
|
||||
*/
|
||||
void igt_install_exit_handler(igt_exit_handler_t fn)
|
||||
{
|
||||
@ -817,6 +959,12 @@ err:
|
||||
igt_assert_f(0, "failed to install the signal handler\n");
|
||||
}
|
||||
|
||||
/**
|
||||
* igt_disable_exit_handler:
|
||||
*
|
||||
* Temporarily disable all exit handlers. Useful for library code doing tricky
|
||||
* things.
|
||||
*/
|
||||
void igt_disable_exit_handler(void)
|
||||
{
|
||||
sigset_t set;
|
||||
@ -837,6 +985,12 @@ void igt_disable_exit_handler(void)
|
||||
exit_handler_disabled = true;
|
||||
}
|
||||
|
||||
/**
|
||||
* igt_enable_exit_handler:
|
||||
*
|
||||
* Re-enable all exit handlers temporarily disabled with
|
||||
* igt_disable_exit_handler().
|
||||
*/
|
||||
void igt_enable_exit_handler(void)
|
||||
{
|
||||
if (!exit_handler_disabled)
|
||||
@ -851,6 +1005,16 @@ void igt_enable_exit_handler(void)
|
||||
}
|
||||
|
||||
/* simulation enviroment support */
|
||||
|
||||
/**
|
||||
* igt_run_in_simulation:
|
||||
*
|
||||
* This function can be used to select a reduced test set when running in
|
||||
* simulation enviroments. This i-g-t mode is selected by setting the
|
||||
* INTEL_SIMULATION enviroment variable to 1.
|
||||
*
|
||||
* Returns: True when run in simulation mode, false otherwise.
|
||||
*/
|
||||
bool igt_run_in_simulation(void)
|
||||
{
|
||||
static int simulation = -1;
|
||||
@ -864,13 +1028,8 @@ bool igt_run_in_simulation(void)
|
||||
/**
|
||||
* igt_skip_on_simulation:
|
||||
*
|
||||
* Skip tests when INTEL_SIMULATION env war is set
|
||||
*
|
||||
* Skip the test when running on simulation (and that's relevant only when
|
||||
* we're not in the mode where we list the subtests).
|
||||
*
|
||||
* This function is subtest aware (since it uses igt_skip) and so can be used to
|
||||
* skip specific subtests or all subsequent subtests.
|
||||
* Skip tests when INTEL_SIMULATION environment variable is set. It uses
|
||||
* igt_skip() internally and hence is fully subtest aware.
|
||||
*/
|
||||
void igt_skip_on_simulation(void)
|
||||
{
|
||||
@ -881,6 +1040,23 @@ void igt_skip_on_simulation(void)
|
||||
}
|
||||
|
||||
/* structured logging */
|
||||
|
||||
/**
|
||||
* igt_log:
|
||||
* @level: #igt_log_level
|
||||
* @format: format string
|
||||
* @...: optional arguments used in the format string
|
||||
*
|
||||
* This is the generic structure logging helper function. i-g-t testcase should
|
||||
* output all normal message to stdout. Warning level message should be printed
|
||||
* to stderr and the test runner should treat this as an intermediate result
|
||||
* between SUCESS and FAILURE.
|
||||
*
|
||||
* The log level can be set through the IGT_LOG_LEVEL enviroment variable with
|
||||
* values "debug", "info", "warn" and "none". By default verbose debug message
|
||||
* are disabled. "none" completely disables all output and is not recommended
|
||||
* since crucial issues only reported at the IGT_LOG_WARN level are ignored.
|
||||
*/
|
||||
void igt_log(enum igt_log_level level, const char *format, ...)
|
||||
{
|
||||
va_list args;
|
||||
@ -898,4 +1074,3 @@ void igt_log(enum igt_log_level level, const char *format, ...)
|
||||
vprintf(format, args);
|
||||
va_end(args);
|
||||
}
|
||||
|
||||
|
298
lib/igt_core.h
298
lib/igt_core.h
@ -30,8 +30,6 @@
|
||||
#ifndef IGT_CORE_H
|
||||
#define IGT_CORE_H
|
||||
|
||||
bool igt_only_list_subtests(void);
|
||||
|
||||
bool __igt_fixture(void);
|
||||
void __igt_fixture_complete(void);
|
||||
void __igt_fixture_end(void) __attribute__((noreturn));
|
||||
@ -56,25 +54,42 @@ void __igt_fixture_end(void) __attribute__((noreturn));
|
||||
jmp_buf igt_subtest_jmpbuf;
|
||||
void igt_subtest_init(int argc, char **argv);
|
||||
typedef int (*igt_opt_handler_t)(int opt, int opt_index);
|
||||
#ifndef __GTK_DOC_IGNORE__ /* gtkdoc wants to document this forward decl */
|
||||
struct option;
|
||||
#endif
|
||||
int igt_subtest_init_parse_opts(int argc, char **argv,
|
||||
const char *extra_short_opts,
|
||||
struct option *extra_long_opts,
|
||||
const char *help_str,
|
||||
igt_opt_handler_t opt_handler);
|
||||
igt_opt_handler_t extra_opt_handler);
|
||||
|
||||
bool __igt_run_subtest(const char *subtest_name);
|
||||
#define __igt_tokencat2(x, y) x ## y
|
||||
|
||||
/**
|
||||
* igt_tokencat:
|
||||
* @x: first variable
|
||||
* @y: second variable
|
||||
*
|
||||
* C preprocessor helper to concatenate two variables while properly expanding
|
||||
* them.
|
||||
*/
|
||||
#define igt_tokencat(x, y) __igt_tokencat2(x, y)
|
||||
|
||||
/**
|
||||
* igt_subtest:
|
||||
* @name: name of the subtest
|
||||
*
|
||||
* Denote a subtest code block
|
||||
* This is a magic control flow block which denotes a subtest code block. Within
|
||||
* that codeblock igt_skip|success will only bail out of the subtest. The _f
|
||||
* variant accepts a printf format string, which is useful for constructing
|
||||
* combinatorial tests.
|
||||
*
|
||||
* Magic control flow which denotes a subtest code block. Within that codeblock
|
||||
* igt_skip|success will only bail out of the subtest. The _f variant accepts a
|
||||
* printf format string, which is useful for constructing combinatorial tests.
|
||||
* This is a simpler version of igt_subtest_f()
|
||||
*/
|
||||
#define igt_tokencat2(x, y) x ## y
|
||||
#define igt_tokencat(x, y) igt_tokencat2(x, y)
|
||||
#define igt_subtest(name) for (; __igt_run_subtest((name)) && \
|
||||
(setjmp(igt_subtest_jmpbuf) == 0); \
|
||||
igt_success())
|
||||
#define __igt_subtest_f(tmp, format...) \
|
||||
for (char tmp [256]; \
|
||||
snprintf( tmp , sizeof( tmp ), \
|
||||
@ -85,18 +100,29 @@ bool __igt_run_subtest(const char *subtest_name);
|
||||
|
||||
/**
|
||||
* igt_subtest_f:
|
||||
* @...: format string
|
||||
* @...: format string and optional arguments
|
||||
*
|
||||
* Denote a subtest code block
|
||||
* This is a magic control flow block which denotes a subtest code block. Within
|
||||
* that codeblock igt_skip|success will only bail out of the subtest. The _f
|
||||
* variant accepts a printf format string, which is useful for constructing
|
||||
* combinatorial tests.
|
||||
*
|
||||
* Like #igt_subtest, but also accepts a printf format string
|
||||
* Like igt_subtest(), but also accepts a printf format string instead of a
|
||||
* static string.
|
||||
*/
|
||||
#define igt_subtest_f(f...) \
|
||||
__igt_subtest_f(igt_tokencat(__tmpchar, __LINE__), f)
|
||||
#define igt_subtest(name) for (; __igt_run_subtest((name)) && \
|
||||
(setjmp(igt_subtest_jmpbuf) == 0); \
|
||||
igt_success())
|
||||
|
||||
const char *igt_subtest_name(void);
|
||||
bool igt_only_list_subtests(void);
|
||||
|
||||
/**
|
||||
* igt_main:
|
||||
*
|
||||
* This is a magic control flow block used instead of a main() function for
|
||||
* tests with subtests. Open-coding the main() function is only recommended if
|
||||
* the test needs to parse additional cmdline arguments of its own.
|
||||
*/
|
||||
#define igt_main \
|
||||
static void igt_tokencat(__real_main, __LINE__)(void); \
|
||||
int main(int argc, char **argv) { \
|
||||
@ -112,6 +138,14 @@ const char *igt_subtest_name(void);
|
||||
* Init for simple tests without subtests
|
||||
*/
|
||||
void igt_simple_init(void);
|
||||
|
||||
/**
|
||||
* igt_simple_main:
|
||||
*
|
||||
* This is a magic control flow block used instead of a main() function for
|
||||
* simple tests. Open-coding the main() function is only recommended if
|
||||
* the test needs to parse additional cmdline arguments of its own.
|
||||
*/
|
||||
#define igt_simple_main \
|
||||
static void igt_tokencat(__real_main, __LINE__)(void); \
|
||||
int main(int argc, char **argv) { \
|
||||
@ -121,62 +155,27 @@ void igt_simple_init(void);
|
||||
} \
|
||||
static void igt_tokencat(__real_main, __LINE__)(void) \
|
||||
|
||||
/**
|
||||
* igt_skip:
|
||||
*
|
||||
* Subtest aware test skipping
|
||||
*
|
||||
* For tests with subtests this will either bail out of the current subtest or
|
||||
* mark all subsequent subtests as SKIP (in case some global setup code failed).
|
||||
*
|
||||
* For normal tests without subtest it will directly exit.
|
||||
*/
|
||||
__attribute__((format(printf, 1, 2)))
|
||||
void igt_skip(const char *f, ...) __attribute__((noreturn));
|
||||
__attribute__((format(printf, 5, 6)))
|
||||
void __igt_skip_check(const char *file, const int line,
|
||||
const char *func, const char *check,
|
||||
const char *format, ...) __attribute__((noreturn));
|
||||
/**
|
||||
* igt_success:
|
||||
*
|
||||
* Complete a (subtest) as successfull
|
||||
*
|
||||
* This bails out of a subtests and marks it as successful. For global tests it
|
||||
* it won't bail out of anything.
|
||||
*/
|
||||
void igt_success(void);
|
||||
|
||||
/**
|
||||
* igt_fail:
|
||||
*
|
||||
* Fail a testcase
|
||||
*
|
||||
* For subtest it just bails out of the subtest, when run in global context it
|
||||
* will exit. Note that it won't attempt to keep on running further tests,
|
||||
* presuming that some mandatory setup failed.
|
||||
*/
|
||||
void igt_fail(int exitcode) __attribute__((noreturn));
|
||||
__attribute__((format(printf, 6, 7)))
|
||||
void __igt_fail_assert(int exitcode, const char *file,
|
||||
const int line, const char *func, const char *assertion,
|
||||
const char *format, ...)
|
||||
__attribute__((noreturn));
|
||||
/**
|
||||
* igt_exit:
|
||||
*
|
||||
* exit() for igts
|
||||
*
|
||||
* This will exit the test with the right exit code when subtests have been
|
||||
* skipped. For normal tests it exits with a successful exit code, presuming
|
||||
* everything has worked out. For subtests it also checks that at least one
|
||||
* subtest has been run (save when only listing subtests.
|
||||
*/
|
||||
void igt_exit(void) __attribute__((noreturn));
|
||||
|
||||
/**
|
||||
* igt_assert:
|
||||
* @expr: condition to test
|
||||
*
|
||||
* Fails (sub-)test if a condition is not met
|
||||
* Fails (sub-)test if the condition is not met
|
||||
*
|
||||
* Should be used everywhere where a test checks results.
|
||||
*/
|
||||
@ -184,14 +183,36 @@ void igt_exit(void) __attribute__((noreturn));
|
||||
do { if (!(expr)) \
|
||||
__igt_fail_assert(99, __FILE__, __LINE__, __func__, #expr , NULL); \
|
||||
} while (0)
|
||||
|
||||
/**
|
||||
* igt_assert_f:
|
||||
* @expr: condition to test
|
||||
* @...: format string and optional arguments
|
||||
*
|
||||
* Fails (sub-)test if the condition is not met
|
||||
*
|
||||
* Should be used everywhere where a test checks results.
|
||||
*
|
||||
* In addition to the plain igt_assert() helper this allows to print additional
|
||||
* information to help debugging test failures.
|
||||
*/
|
||||
#define igt_assert_f(expr, f...) \
|
||||
do { if (!(expr)) \
|
||||
__igt_fail_assert(99, __FILE__, __LINE__, __func__, #expr , f); \
|
||||
} while (0)
|
||||
|
||||
/**
|
||||
* igt_assert_cmptint:
|
||||
* igt_assert_cmpint:
|
||||
* @n1: first value
|
||||
* @cmp: compare operator
|
||||
* @n2: second value
|
||||
*
|
||||
* Like #igt_assert, but displays the values being compared on failure.
|
||||
* Fails (sub-)test if the condition is not met
|
||||
*
|
||||
* Should be used everywhere where a test compares two integer values.
|
||||
*
|
||||
* Like igt_assert(), but displays the values being compared on failure instead
|
||||
* of simply printing the stringified expression.
|
||||
*/
|
||||
#define igt_assert_cmpint(n1, cmp, n2) \
|
||||
do { \
|
||||
@ -204,18 +225,61 @@ void igt_exit(void) __attribute__((noreturn));
|
||||
|
||||
/**
|
||||
* igt_require:
|
||||
* @expr: condition to test
|
||||
*
|
||||
* Skip a (sub-)test if a condition is not met
|
||||
* Skip a (sub-)test if a condition is not met.
|
||||
*
|
||||
* This is useful to streamline the skip logic since it allows for a more flat
|
||||
* code control flow.
|
||||
* Should be used everywhere where a test checks results to decide about
|
||||
* skipping. This is useful to streamline the skip logic since it allows for a more flat
|
||||
* code control flow, similar to igt_assert()
|
||||
*/
|
||||
#define igt_require(expr) igt_skip_on(!(expr))
|
||||
|
||||
/**
|
||||
* igt_skip_on:
|
||||
* @expr: condition to test
|
||||
*
|
||||
* Skip a (sub-)test if a condition is met.
|
||||
*
|
||||
* Should be used everywhere where a test checks results to decide about
|
||||
* skipping. This is useful to streamline the skip logic since it allows for a more flat
|
||||
* code control flow, similar to igt_assert()
|
||||
*/
|
||||
#define igt_skip_on(expr) \
|
||||
do { if ((expr)) \
|
||||
__igt_skip_check(__FILE__, __LINE__, __func__, #expr , NULL); \
|
||||
} while (0)
|
||||
|
||||
/**
|
||||
* igt_require_f:
|
||||
* @expr: condition to test
|
||||
* @...: format string and optional arguments
|
||||
*
|
||||
* Skip a (sub-)test if a condition is not met.
|
||||
*
|
||||
* Should be used everywhere where a test checks results to decide about
|
||||
* skipping. This is useful to streamline the skip logic since it allows for a more flat
|
||||
* code control flow, similar to igt_assert()
|
||||
*
|
||||
* In addition to the plain igt_require() helper this allows to print additional
|
||||
* information to help debugging test failures.
|
||||
*/
|
||||
#define igt_require_f(expr, f...) igt_skip_on_f(!(expr), f)
|
||||
|
||||
/**
|
||||
* igt_skip_on_f:
|
||||
* @expr: condition to test
|
||||
* @...: format string and optional arguments
|
||||
*
|
||||
* Skip a (sub-)test if a condition is met.
|
||||
*
|
||||
* Should be used everywhere where a test checks results to decide about
|
||||
* skipping. This is useful to streamline the skip logic since it allows for a more flat
|
||||
* code control flow, similar to igt_assert()
|
||||
*
|
||||
* In addition to the plain igt_skip_on() helper this allows to print additional
|
||||
* information to help debugging test failures.
|
||||
*/
|
||||
#define igt_skip_on_f(expr, f...) \
|
||||
do { if ((expr)) \
|
||||
__igt_skip_check(__FILE__, __LINE__, __func__, #expr , f); \
|
||||
@ -223,21 +287,37 @@ void igt_exit(void) __attribute__((noreturn));
|
||||
|
||||
/* fork support code */
|
||||
bool __igt_fork(void);
|
||||
|
||||
/**
|
||||
* igt_fork:
|
||||
* @child: name of the int variable with the child number
|
||||
* @num_children: number of children to fork
|
||||
*
|
||||
* Fork parallel test threads with fork()
|
||||
* This is a magic control flow block which spawns parallel test threads with
|
||||
* fork().
|
||||
*
|
||||
* Joining all test threads should be done with igt_waitchildren to ensure that
|
||||
* the exit codes of all children are properly reflected in the test status.
|
||||
* The test children execute in parallel to the main test thread. Joining all
|
||||
* test threads should be done with igt_waitchildren to ensure that the exit
|
||||
* codes of all children are properly reflected in the test status.
|
||||
*
|
||||
* Note that igt_skip() will not be forwarded, feature tests need to be done
|
||||
* before spawning threads with igt_fork().
|
||||
*/
|
||||
#define igt_fork(child, num_children) \
|
||||
for (int child = 0; child < (num_children); child++) \
|
||||
for (; __igt_fork(); exit(0))
|
||||
void igt_waitchildren(void);
|
||||
|
||||
/**
|
||||
* igt_helper_process_t:
|
||||
* @running: indicates whether the process is currently running
|
||||
* @use_SIGKILL: whether the helper should be terminated with SIGKILL or SIGQUIT
|
||||
* @pid: pid of the helper if @running is true
|
||||
* @id: internal id
|
||||
*
|
||||
* Tracking structure for helper processes. Users of the i-g-t library should
|
||||
* only set @use_SIGKILL directly.
|
||||
*/
|
||||
struct igt_helper_process {
|
||||
bool running;
|
||||
bool use_SIGKILL;
|
||||
@ -245,12 +325,40 @@ struct igt_helper_process {
|
||||
int id;
|
||||
};
|
||||
bool __igt_fork_helper(struct igt_helper_process *proc);
|
||||
void igt_stop_helper(struct igt_helper_process *proc);
|
||||
void igt_wait_helper(struct igt_helper_process *proc);
|
||||
|
||||
/**
|
||||
* igt_fork_helper:
|
||||
* @proc: #igt_helper_process structure
|
||||
*
|
||||
* This is a magic control flow block which denotes an asynchronous helper
|
||||
* process block. The difference compared to igt_fork() is that failures from
|
||||
* the child process will not be forwarded, making this construct more suitable
|
||||
* for background processes. Common use cases are regular interference of the
|
||||
* main test thread through e.g. sending signals or evicting objects through
|
||||
* debugfs. Through the explicit #igt_helper_process they can also be controlled
|
||||
* in a more fine-grained way than test children spawned through igt_fork().
|
||||
*
|
||||
* For tests with subtest helper process can be started outside of a
|
||||
* #igt_subtest block.
|
||||
*
|
||||
* Calling igt_wait_helper() joins a helper process and igt_stop_helper()
|
||||
* forcefully terminates it.
|
||||
*/
|
||||
#define igt_fork_helper(proc) \
|
||||
for (; __igt_fork_helper(proc); exit(0))
|
||||
void igt_wait_helper(struct igt_helper_process *proc);
|
||||
void igt_stop_helper(struct igt_helper_process *proc);
|
||||
|
||||
/* exit handler code */
|
||||
|
||||
/**
|
||||
* igt_exit_handler_t:
|
||||
* @sig: Signal number which caused the exit or 0.
|
||||
*
|
||||
* Exit handler type used by igt_install_exit_handler(). Note that exit handlers
|
||||
* can potentially be run from signal handling contexts, the @sig parameter can
|
||||
* be used to figure this out and act accordingly.
|
||||
*/
|
||||
typedef void (*igt_exit_handler_t)(int sig);
|
||||
|
||||
/* reliable atexit helpers, also work when killed by a signal (if possible) */
|
||||
@ -260,18 +368,16 @@ void igt_disable_exit_handler(void);
|
||||
|
||||
/* helpers to automatically reduce test runtime in simulation */
|
||||
bool igt_run_in_simulation(void);
|
||||
#define SLOW_QUICK(slow,quick) (igt_run_in_simulation() ? (quick) : (slow))
|
||||
/**
|
||||
* igt_skip_on_simulation:
|
||||
* SLOW_QUICK:
|
||||
* @slow: value in simulation mode
|
||||
* @quick: value in normal mode
|
||||
*
|
||||
* Skip tests when INTEL_SIMULATION env war is set
|
||||
*
|
||||
* Skip the test when running on simulation (and that's relevant only when
|
||||
* we're not in the mode where we list the subtests).
|
||||
*
|
||||
* This function is subtest aware (since it uses igt_skip) and so can be used to
|
||||
* skip specific subtests or all subsequent subtests.
|
||||
* Simple macro to select between two values (e.g. number of test rounds or test
|
||||
* buffer size) depending upon whether i-g-t is run in simulation mode or not.
|
||||
*/
|
||||
#define SLOW_QUICK(slow,quick) (igt_run_in_simulation() ? (quick) : (slow))
|
||||
|
||||
void igt_skip_on_simulation(void);
|
||||
|
||||
/* structured logging */
|
||||
@ -283,16 +389,64 @@ enum igt_log_level {
|
||||
};
|
||||
__attribute__((format(printf, 2, 3)))
|
||||
void igt_log(enum igt_log_level level, const char *format, ...);
|
||||
|
||||
/**
|
||||
* igt_debug:
|
||||
* @...: format string and optional arguments
|
||||
*
|
||||
* Wrapper for igt_log for message at the IGT_LOG_DEBUG level.
|
||||
*/
|
||||
#define igt_debug(f...) igt_log(IGT_LOG_DEBUG, f)
|
||||
|
||||
/**
|
||||
* igt_info:
|
||||
* @...: format string and optional arguments
|
||||
*
|
||||
* Wrapper for igt_log for message at the IGT_LOG_INFO level.
|
||||
*/
|
||||
#define igt_info(f...) igt_log(IGT_LOG_INFO, f)
|
||||
|
||||
/**
|
||||
* igt_warn:
|
||||
* @...: format string and optional arguments
|
||||
*
|
||||
* Wrapper for igt_log for message at the IGT_LOG_WARN level.
|
||||
*/
|
||||
#define igt_warn(f...) igt_log(IGT_LOG_WARN, f)
|
||||
extern enum igt_log_level igt_log_level;
|
||||
|
||||
/**
|
||||
* igt_warn_on:
|
||||
* @condition: condition to test
|
||||
*
|
||||
* Print a IGT_LOG_WARN level message if a condition is not met.
|
||||
*
|
||||
* Should be used everywhere where a test checks results to decide about
|
||||
* printing warnings. This is useful to streamline the test logic since it
|
||||
* allows for a more flat code control flow, similar to igt_assert()
|
||||
*/
|
||||
#define igt_warn_on(condition) do {\
|
||||
if (condition) \
|
||||
igt_warn("Warning on condition %s in fucntion %s, file %s:%i\n", \
|
||||
#condition, __func__, __FILE__, __LINE__); \
|
||||
} while (0)
|
||||
|
||||
/**
|
||||
* igt_warn_on_f:
|
||||
* @condition: condition to test
|
||||
* @...: format string and optional arguments
|
||||
*
|
||||
* Skip a (sub-)test if a condition is not met.
|
||||
*
|
||||
* Print a IGT_LOG_WARN level message if a condition is not met.
|
||||
*
|
||||
* Should be used everywhere where a test checks results to decide about
|
||||
* printing warnings. This is useful to streamline the test logic since it
|
||||
* allows for a more flat code control flow, similar to igt_assert()
|
||||
*
|
||||
* In addition to the plain igt_warn_on_f() helper this allows to print
|
||||
* additional information (again as warnings) to help debugging test failures.
|
||||
*/
|
||||
#define igt_warn_on_f(condition, f...) do {\
|
||||
if (condition) {\
|
||||
igt_warn("Warning on condition %s in fucntion %s, file %s:%i\n", \
|
||||
|
Loading…
x
Reference in New Issue
Block a user