[PATCH i-g-t v2 4/6] lib/kunit: Support writable filter* parameters of kunit module

Lucas De Marchi lucas.demarchi at intel.com
Thu Feb 1 01:01:17 UTC 2024


On Wed, Jan 31, 2024 at 07:03:51PM +0100, Janusz Krzysztofik wrote:
>Instead of wasting resources on reloading the base Kunit module each time
>a different set of filter parameters is needed, try to write the required
>values to sysfs representation of those parameters.  If that fails (e.g.
>on older LTS kernels with read-only filter parameters), fall back to
>legacy "no list of test cases available" mode as we do on kernels with
>those module parameters not supported at all.
>
>This modification will also serve as a workaround for the issue of
>impossibility to unload the base Kunit module on Xe platforms, available
>to IGT as soon as the module supports writable filter parameters.
>
>v2: Work around ineffective writes of empty strings to sysfs module
>    parameter files (Lucas) by using human readable non-empty strings that
>    give the same results as default NULLs,
>  - drop fallback to reload of base Kunit module method if assigning new
>    values to module parameters via sysfs fails (Lucas), instead use the
>    existing fallback to blind execution like if getting a list of test
>    cases was not supported at all,
>  - split move of open_parameters() helper up in the source file as well
>    as cleanup of base KUnit module unloading to separate patches (Kamil),
>  - address the issue of the second paragraph of commit description
>    suggesting two separate changes combined in one patch (Kamil) by
>    rewording the description.
>
>Signed-off-by: Janusz Krzysztofik <janusz.krzysztofik at linux.intel.com>
>Cc: Lucas De Marchi <lucas.demarchi at intel.com>
>Cc: Kamil Konieczny <kamil.konieczny at linux.intel.com>
>---
> lib/igt_kmod.c | 114 ++++++++++++++++++++++++++++++++++++-------------
> 1 file changed, 84 insertions(+), 30 deletions(-)
>
>diff --git a/lib/igt_kmod.c b/lib/igt_kmod.c
>index d2c28d0a64..1d57ea708d 100644
>--- a/lib/igt_kmod.c
>+++ b/lib/igt_kmod.c
>@@ -811,6 +811,56 @@ static int open_parameters(const char *module_name)
> 	return open(path, O_RDONLY);
> }
>
>+static const char *kunit_params[3] = {
>+	"filter_glob", "filter", "filter_action",
>+};
>+
>+static bool kunit_set_params(const char **values)
>+{
>+	/*
>+	 * Reapplying default NULLs to module parameters via sysfs seems not
>+	 * possible, and applying empty strings doesn't work as expected,
>+	 * leaving current values untouched.  As a workaround, use
>+	 * human-readable non-empty strings that exhibit the same behaviour
>+	 * as if default NULLs were in place.  In case of filter_action
>+	 * parameter there is no such obvious non-empty string, however,
>+	 * that's not a problem for us since even if it was previously set
>+	 * to "skip" and we leave it as is, our "module!=none" default filter
>+	 * guarantees that no test cases are filtered out to be skipped.
>+	 */
>+	const char *defaults[ARRAY_SIZE(kunit_params)] = {
>+	     /* filter_glob, filter,	     filter_action */
>+		"*",	     "module!=none", "",
>+	};
>+	int i, params;
>+	bool ret;
>+
>+	params = open_parameters("kunit");
>+	ret = !igt_debug_on(params < 0);
>+
>+	for (i = 0; ret && i < ARRAY_SIZE(kunit_params); i++) {
>+		char *param = igt_sysfs_get(params, kunit_params[i]);

why do we read-modify-write? I'd rather just write the values. An
optimization by checking previous value belongs in the kernel, when it
makes an actual difference. Something like this (but with error
handling):

	int dirfd = open("/sys/modules/kunit/parameters", O_RDONLY);
	igt_sysfs_set(dirfd, "filter_glob", ...);
	igt_sysfs_set(dirfd, "filter", ...);
	igt_sysfs_set(dirfd, "filter_action", ...);
	close(dirfd);

an abstraction could better be something like:

	igt_sysfs_set_all("/sys/modules/kunit/parameters",
			  attr1, val1,
			  attr2, val2,
			  attr3, val2,
			  NULL);
>+
>+		if (igt_debug_on_f(!param, "param: %s\n", kunit_params[i])) {
>+			ret = false;
>+			break;
>+		}
>+
>+		ret = !strcmp(param, values[i] ?: defaults[i]);
>+		free(param);
>+		if (!ret)
>+			ret = !igt_debug_on(!igt_sysfs_set(params,
>+							   kunit_params[i],
>+							   values[i] ?:
>+							   defaults[i]));


filter_action is "". How does this fix the issue of writing a 0-length
string? If we try to go back and forth between "skip" and "", this will
fail.  A workaround would be to make defaults above to be something else
like "run", but we should really fix this in igt.  I'm submitting a
patch for that.

Lucas de Marchi

>+	}
>+
>+	if (params >= 0)
>+		close(params);
>+
>+	return ret;
>+}
>+
> struct modprobe_data {
> 	struct kmod_module *kmod;
> 	const char *opts;
>@@ -1100,6 +1150,18 @@ static void kunit_get_tests(struct igt_list_head *tests,
> 			    const char *filter,
> 			    const char *opts)
> {
>+	/*
>+	 * To get a list of test cases provided by a kunit test module, ask the
>+	 * generic kunit module to respond with SKIP result for each test found.
>+	 * We could also try to use action=list kunit parameter to get the
>+	 * listing, however, parsing a KTAP report -- something that we already
>+	 * can do perfectly -- seems to be more safe than extracting a test case
>+	 * list of unknown length from /dev/kmsg.
>+	 */
>+	const char *values[ARRAY_SIZE(kunit_params)] = {
>+	     /* filter_glob, filter,	    filter_action */
>+		NULL,	     "module=none", "skip",
>+	};
> 	char *suite_name = NULL, *case_name = NULL;
> 	struct igt_ktap_result *r, *rn;
> 	struct igt_ktap_results *ktap;
>@@ -1113,27 +1175,21 @@ static void kunit_get_tests(struct igt_list_head *tests,
>
> 	igt_skip_on(lseek(tst->kmsg, 0, SEEK_END) < 0);
>
>-	/*
>-	 * To get a list of test cases provided by a kunit test module, ask the
>-	 * generic kunit module to respond with SKIP result for each test found.
>-	 * We could also use action=list kunit parameter to get the listing,
>-	 * however, parsing a KTAP report -- something that we already can do
>-	 * perfectly -- seems to be more safe than extracting a test case list
>-	 * of unknown length from /dev/kmsg.
>-	 *
>-	 * TODO: drop the following workaround, which is required by LTS kernel
>-	 *       v6.1 not capable of listing test cases when built as a module.
>-	 * If loading the kunit module with required parameters fails then
>-	 * assume that we are running on a kernel with missing test case listing
>-	 * capabilities.  Dont's skip but just return with empty list of test
>-	 * cases, that should tell the caller to use a legacy method of
>-	 * iterating over KTAP results collected from blind execution of all
>-	 * Kunit test cases provided by a Kunit test module.
>-	 */
>-	igt_ignore_warn(igt_kmod_unload("kunit", KMOD_REMOVE_FORCE));
>-	if (igt_debug_on(igt_kmod_load("kunit",
>-				       "filter=module=none filter_action=skip")))
>+	if (igt_debug_on(!kunit_set_params(values))) {
>+		/*
>+		 * TODO: drop the following workaround, required by LTS kernel
>+		 *       v6.1 not capable of listing test cases when built as as
>+		 *       module, when no longer needed.
>+		 * If updating writable filter parameters of the kunit base
>+		 * module with required values then assume that we are running
>+		 * on a kernel with missing test case listing capabilities.
>+		 * Don't skip but just return with empty list of test cases,
>+		 * which should tell the caller to use a legacy method of
>+		 * iterating over KTAP results collected from blind execution
>+		 * of all Kunit test cases provided by a Kunit test module.
>+		 */
> 		return;
>+	}
>
> 	igt_skip_on(modprobe(tst->kmod, opts));
>
>@@ -1200,16 +1256,11 @@ static void __igt_kunit(struct igt_ktest *tst,
> 			      t->case_name) {
>
> 			if (!modprobe.thread) {
>-				/*
>-				 * Since we have successfully loaded the kunit
>-				 * base module with non-default parameters in
>-				 * order to get a list of test cases, now we
>-				 * have to unload it so it is then automatically
>-				 * reloaded with default parameter values when
>-				 * we load the test module again for execution.
>-				 */
>-				igt_skip_on(igt_kmod_unload("kunit",
>-							    KMOD_REMOVE_FORCE));
>+				const char *values[ARRAY_SIZE(kunit_params)] = {
>+					NULL, NULL, NULL,
>+				};
>+
>+				igt_require(kunit_set_params(values));
>
> 				igt_assert_eq(pthread_mutexattr_init(&attr), 0);
> 				igt_assert_eq(pthread_mutexattr_setrobust(&attr,
>@@ -1378,6 +1429,9 @@ void igt_kunit(const char *module_name, const char *name, const char *opts)
> 		igt_assert(igt_list_empty(&tests));
> 	}
>
>+	/* We need the base KUnit module loaded if not built-in */
>+	igt_ignore_warn(igt_kmod_load("kunit", NULL));
>+
> 	/*
> 	 * We need to use igt_subtest here, as otherwise it may crash with:
> 	 *  skipping is allowed only in fixtures, subtests or igt_simple_main
>-- 
>2.43.0
>


More information about the igt-dev mailing list