1*41ec0267Sriastradh /* $NetBSD: drm_selftest.c,v 1.2 2021/12/18 23:45:44 riastradh Exp $ */
24e390cabSriastradh
34e390cabSriastradh /*
44e390cabSriastradh * Copyright © 2016 Intel Corporation
54e390cabSriastradh *
64e390cabSriastradh * Permission is hereby granted, free of charge, to any person obtaining a
74e390cabSriastradh * copy of this software and associated documentation files (the "Software"),
84e390cabSriastradh * to deal in the Software without restriction, including without limitation
94e390cabSriastradh * the rights to use, copy, modify, merge, publish, distribute, sublicense,
104e390cabSriastradh * and/or sell copies of the Software, and to permit persons to whom the
114e390cabSriastradh * Software is furnished to do so, subject to the following conditions:
124e390cabSriastradh *
134e390cabSriastradh * The above copyright notice and this permission notice (including the next
144e390cabSriastradh * paragraph) shall be included in all copies or substantial portions of the
154e390cabSriastradh * Software.
164e390cabSriastradh *
174e390cabSriastradh * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
184e390cabSriastradh * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
194e390cabSriastradh * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
204e390cabSriastradh * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
214e390cabSriastradh * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
224e390cabSriastradh * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
234e390cabSriastradh * IN THE SOFTWARE.
244e390cabSriastradh */
254e390cabSriastradh
264e390cabSriastradh #include <sys/cdefs.h>
27*41ec0267Sriastradh __KERNEL_RCSID(0, "$NetBSD: drm_selftest.c,v 1.2 2021/12/18 23:45:44 riastradh Exp $");
284e390cabSriastradh
294e390cabSriastradh #include <linux/compiler.h>
304e390cabSriastradh
314e390cabSriastradh #define selftest(name, func) __idx_##name,
324e390cabSriastradh enum {
334e390cabSriastradh #include TESTS
344e390cabSriastradh };
354e390cabSriastradh #undef selftest
364e390cabSriastradh
374e390cabSriastradh #define selftest(n, f) [__idx_##n] = { .name = #n, .func = f },
384e390cabSriastradh static struct drm_selftest {
394e390cabSriastradh bool enabled;
404e390cabSriastradh const char *name;
414e390cabSriastradh int (*func)(void *);
424e390cabSriastradh } selftests[] = {
434e390cabSriastradh #include TESTS
444e390cabSriastradh };
454e390cabSriastradh #undef selftest
464e390cabSriastradh
474e390cabSriastradh /* Embed the line number into the parameter name so that we can order tests */
484e390cabSriastradh #define param(n) __PASTE(igt__, __PASTE(__PASTE(__LINE__, __), n))
494e390cabSriastradh #define selftest_0(n, func, id) \
504e390cabSriastradh module_param_named(id, selftests[__idx_##n].enabled, bool, 0400);
514e390cabSriastradh #define selftest(n, func) selftest_0(n, func, param(n))
524e390cabSriastradh #include TESTS
534e390cabSriastradh #undef selftest
544e390cabSriastradh
set_default_test_all(struct drm_selftest * st,unsigned long count)554e390cabSriastradh static void set_default_test_all(struct drm_selftest *st, unsigned long count)
564e390cabSriastradh {
574e390cabSriastradh unsigned long i;
584e390cabSriastradh
594e390cabSriastradh for (i = 0; i < count; i++)
604e390cabSriastradh if (st[i].enabled)
614e390cabSriastradh return;
624e390cabSriastradh
634e390cabSriastradh for (i = 0; i < count; i++)
644e390cabSriastradh st[i].enabled = true;
654e390cabSriastradh }
664e390cabSriastradh
run_selftests(struct drm_selftest * st,unsigned long count,void * data)674e390cabSriastradh static int run_selftests(struct drm_selftest *st,
684e390cabSriastradh unsigned long count,
694e390cabSriastradh void *data)
704e390cabSriastradh {
714e390cabSriastradh int err = 0;
724e390cabSriastradh
734e390cabSriastradh set_default_test_all(st, count);
744e390cabSriastradh
754e390cabSriastradh /* Tests are listed in natural order in drm_*_selftests.h */
764e390cabSriastradh for (; count--; st++) {
774e390cabSriastradh if (!st->enabled)
784e390cabSriastradh continue;
794e390cabSriastradh
804e390cabSriastradh pr_debug("drm: Running %s\n", st->name);
814e390cabSriastradh err = st->func(data);
824e390cabSriastradh if (err)
834e390cabSriastradh break;
844e390cabSriastradh }
854e390cabSriastradh
864e390cabSriastradh if (WARN(err > 0 || err == -ENOTTY,
874e390cabSriastradh "%s returned %d, conflicting with selftest's magic values!\n",
884e390cabSriastradh st->name, err))
894e390cabSriastradh err = -1;
904e390cabSriastradh
914e390cabSriastradh rcu_barrier();
924e390cabSriastradh return err;
934e390cabSriastradh }
944e390cabSriastradh
954e390cabSriastradh static int __maybe_unused
__drm_subtests(const char * caller,const struct drm_subtest * st,int count,void * data)964e390cabSriastradh __drm_subtests(const char *caller,
974e390cabSriastradh const struct drm_subtest *st,
984e390cabSriastradh int count,
994e390cabSriastradh void *data)
1004e390cabSriastradh {
1014e390cabSriastradh int err;
1024e390cabSriastradh
1034e390cabSriastradh for (; count--; st++) {
1044e390cabSriastradh pr_debug("Running %s/%s\n", caller, st->name);
1054e390cabSriastradh err = st->func(data);
1064e390cabSriastradh if (err) {
1074e390cabSriastradh pr_err("%s: %s failed with error %d\n",
1084e390cabSriastradh caller, st->name, err);
1094e390cabSriastradh return err;
1104e390cabSriastradh }
1114e390cabSriastradh }
1124e390cabSriastradh
1134e390cabSriastradh return 0;
1144e390cabSriastradh }
115