xref: /OK3568_Linux_fs/kernel/lib/kunit/test.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun  * Base unit test (KUnit) API.
4*4882a593Smuzhiyun  *
5*4882a593Smuzhiyun  * Copyright (C) 2019, Google LLC.
6*4882a593Smuzhiyun  * Author: Brendan Higgins <brendanhiggins@google.com>
7*4882a593Smuzhiyun  */
8*4882a593Smuzhiyun 
9*4882a593Smuzhiyun #include <kunit/test.h>
10*4882a593Smuzhiyun #include <linux/kernel.h>
11*4882a593Smuzhiyun #include <linux/kref.h>
12*4882a593Smuzhiyun #include <linux/sched/debug.h>
13*4882a593Smuzhiyun #include <linux/sched.h>
14*4882a593Smuzhiyun 
15*4882a593Smuzhiyun #include "debugfs.h"
16*4882a593Smuzhiyun #include "string-stream.h"
17*4882a593Smuzhiyun #include "try-catch-impl.h"
18*4882a593Smuzhiyun 
19*4882a593Smuzhiyun /*
20*4882a593Smuzhiyun  * Append formatted message to log, size of which is limited to
21*4882a593Smuzhiyun  * KUNIT_LOG_SIZE bytes (including null terminating byte).
22*4882a593Smuzhiyun  */
kunit_log_append(char * log,const char * fmt,...)23*4882a593Smuzhiyun void kunit_log_append(char *log, const char *fmt, ...)
24*4882a593Smuzhiyun {
25*4882a593Smuzhiyun 	char line[KUNIT_LOG_SIZE];
26*4882a593Smuzhiyun 	va_list args;
27*4882a593Smuzhiyun 	int len_left;
28*4882a593Smuzhiyun 
29*4882a593Smuzhiyun 	if (!log)
30*4882a593Smuzhiyun 		return;
31*4882a593Smuzhiyun 
32*4882a593Smuzhiyun 	len_left = KUNIT_LOG_SIZE - strlen(log) - 1;
33*4882a593Smuzhiyun 	if (len_left <= 0)
34*4882a593Smuzhiyun 		return;
35*4882a593Smuzhiyun 
36*4882a593Smuzhiyun 	va_start(args, fmt);
37*4882a593Smuzhiyun 	vsnprintf(line, sizeof(line), fmt, args);
38*4882a593Smuzhiyun 	va_end(args);
39*4882a593Smuzhiyun 
40*4882a593Smuzhiyun 	strncat(log, line, len_left);
41*4882a593Smuzhiyun }
42*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(kunit_log_append);
43*4882a593Smuzhiyun 
kunit_suite_num_test_cases(struct kunit_suite * suite)44*4882a593Smuzhiyun size_t kunit_suite_num_test_cases(struct kunit_suite *suite)
45*4882a593Smuzhiyun {
46*4882a593Smuzhiyun 	struct kunit_case *test_case;
47*4882a593Smuzhiyun 	size_t len = 0;
48*4882a593Smuzhiyun 
49*4882a593Smuzhiyun 	kunit_suite_for_each_test_case(suite, test_case)
50*4882a593Smuzhiyun 		len++;
51*4882a593Smuzhiyun 
52*4882a593Smuzhiyun 	return len;
53*4882a593Smuzhiyun }
54*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(kunit_suite_num_test_cases);
55*4882a593Smuzhiyun 
kunit_print_subtest_start(struct kunit_suite * suite)56*4882a593Smuzhiyun static void kunit_print_subtest_start(struct kunit_suite *suite)
57*4882a593Smuzhiyun {
58*4882a593Smuzhiyun 	kunit_log(KERN_INFO, suite, KUNIT_SUBTEST_INDENT "# Subtest: %s",
59*4882a593Smuzhiyun 		  suite->name);
60*4882a593Smuzhiyun 	kunit_log(KERN_INFO, suite, KUNIT_SUBTEST_INDENT "1..%zd",
61*4882a593Smuzhiyun 		  kunit_suite_num_test_cases(suite));
62*4882a593Smuzhiyun }
63*4882a593Smuzhiyun 
kunit_print_ok_not_ok(void * test_or_suite,bool is_test,bool is_ok,size_t test_number,const char * description)64*4882a593Smuzhiyun static void kunit_print_ok_not_ok(void *test_or_suite,
65*4882a593Smuzhiyun 				  bool is_test,
66*4882a593Smuzhiyun 				  bool is_ok,
67*4882a593Smuzhiyun 				  size_t test_number,
68*4882a593Smuzhiyun 				  const char *description)
69*4882a593Smuzhiyun {
70*4882a593Smuzhiyun 	struct kunit_suite *suite = is_test ? NULL : test_or_suite;
71*4882a593Smuzhiyun 	struct kunit *test = is_test ? test_or_suite : NULL;
72*4882a593Smuzhiyun 
73*4882a593Smuzhiyun 	/*
74*4882a593Smuzhiyun 	 * We do not log the test suite results as doing so would
75*4882a593Smuzhiyun 	 * mean debugfs display would consist of the test suite
76*4882a593Smuzhiyun 	 * description and status prior to individual test results.
77*4882a593Smuzhiyun 	 * Hence directly printk the suite status, and we will
78*4882a593Smuzhiyun 	 * separately seq_printf() the suite status for the debugfs
79*4882a593Smuzhiyun 	 * representation.
80*4882a593Smuzhiyun 	 */
81*4882a593Smuzhiyun 	if (suite)
82*4882a593Smuzhiyun 		pr_info("%s %zd - %s\n",
83*4882a593Smuzhiyun 			kunit_status_to_string(is_ok),
84*4882a593Smuzhiyun 			test_number, description);
85*4882a593Smuzhiyun 	else
86*4882a593Smuzhiyun 		kunit_log(KERN_INFO, test, KUNIT_SUBTEST_INDENT "%s %zd - %s",
87*4882a593Smuzhiyun 			  kunit_status_to_string(is_ok),
88*4882a593Smuzhiyun 			  test_number, description);
89*4882a593Smuzhiyun }
90*4882a593Smuzhiyun 
kunit_suite_has_succeeded(struct kunit_suite * suite)91*4882a593Smuzhiyun bool kunit_suite_has_succeeded(struct kunit_suite *suite)
92*4882a593Smuzhiyun {
93*4882a593Smuzhiyun 	const struct kunit_case *test_case;
94*4882a593Smuzhiyun 
95*4882a593Smuzhiyun 	kunit_suite_for_each_test_case(suite, test_case) {
96*4882a593Smuzhiyun 		if (!test_case->success)
97*4882a593Smuzhiyun 			return false;
98*4882a593Smuzhiyun 	}
99*4882a593Smuzhiyun 
100*4882a593Smuzhiyun 	return true;
101*4882a593Smuzhiyun }
102*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(kunit_suite_has_succeeded);
103*4882a593Smuzhiyun 
kunit_print_subtest_end(struct kunit_suite * suite)104*4882a593Smuzhiyun static void kunit_print_subtest_end(struct kunit_suite *suite)
105*4882a593Smuzhiyun {
106*4882a593Smuzhiyun 	static size_t kunit_suite_counter = 1;
107*4882a593Smuzhiyun 
108*4882a593Smuzhiyun 	kunit_print_ok_not_ok((void *)suite, false,
109*4882a593Smuzhiyun 			      kunit_suite_has_succeeded(suite),
110*4882a593Smuzhiyun 			      kunit_suite_counter++,
111*4882a593Smuzhiyun 			      suite->name);
112*4882a593Smuzhiyun }
113*4882a593Smuzhiyun 
kunit_test_case_num(struct kunit_suite * suite,struct kunit_case * test_case)114*4882a593Smuzhiyun unsigned int kunit_test_case_num(struct kunit_suite *suite,
115*4882a593Smuzhiyun 				 struct kunit_case *test_case)
116*4882a593Smuzhiyun {
117*4882a593Smuzhiyun 	struct kunit_case *tc;
118*4882a593Smuzhiyun 	unsigned int i = 1;
119*4882a593Smuzhiyun 
120*4882a593Smuzhiyun 	kunit_suite_for_each_test_case(suite, tc) {
121*4882a593Smuzhiyun 		if (tc == test_case)
122*4882a593Smuzhiyun 			return i;
123*4882a593Smuzhiyun 		i++;
124*4882a593Smuzhiyun 	}
125*4882a593Smuzhiyun 
126*4882a593Smuzhiyun 	return 0;
127*4882a593Smuzhiyun }
128*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(kunit_test_case_num);
129*4882a593Smuzhiyun 
kunit_print_string_stream(struct kunit * test,struct string_stream * stream)130*4882a593Smuzhiyun static void kunit_print_string_stream(struct kunit *test,
131*4882a593Smuzhiyun 				      struct string_stream *stream)
132*4882a593Smuzhiyun {
133*4882a593Smuzhiyun 	struct string_stream_fragment *fragment;
134*4882a593Smuzhiyun 	char *buf;
135*4882a593Smuzhiyun 
136*4882a593Smuzhiyun 	if (string_stream_is_empty(stream))
137*4882a593Smuzhiyun 		return;
138*4882a593Smuzhiyun 
139*4882a593Smuzhiyun 	buf = string_stream_get_string(stream);
140*4882a593Smuzhiyun 	if (!buf) {
141*4882a593Smuzhiyun 		kunit_err(test,
142*4882a593Smuzhiyun 			  "Could not allocate buffer, dumping stream:\n");
143*4882a593Smuzhiyun 		list_for_each_entry(fragment, &stream->fragments, node) {
144*4882a593Smuzhiyun 			kunit_err(test, "%s", fragment->fragment);
145*4882a593Smuzhiyun 		}
146*4882a593Smuzhiyun 		kunit_err(test, "\n");
147*4882a593Smuzhiyun 	} else {
148*4882a593Smuzhiyun 		kunit_err(test, "%s", buf);
149*4882a593Smuzhiyun 		kunit_kfree(test, buf);
150*4882a593Smuzhiyun 	}
151*4882a593Smuzhiyun }
152*4882a593Smuzhiyun 
kunit_fail(struct kunit * test,struct kunit_assert * assert)153*4882a593Smuzhiyun static void kunit_fail(struct kunit *test, struct kunit_assert *assert)
154*4882a593Smuzhiyun {
155*4882a593Smuzhiyun 	struct string_stream *stream;
156*4882a593Smuzhiyun 
157*4882a593Smuzhiyun 	kunit_set_failure(test);
158*4882a593Smuzhiyun 
159*4882a593Smuzhiyun 	stream = alloc_string_stream(test, GFP_KERNEL);
160*4882a593Smuzhiyun 	if (!stream) {
161*4882a593Smuzhiyun 		WARN(true,
162*4882a593Smuzhiyun 		     "Could not allocate stream to print failed assertion in %s:%d\n",
163*4882a593Smuzhiyun 		     assert->file,
164*4882a593Smuzhiyun 		     assert->line);
165*4882a593Smuzhiyun 		return;
166*4882a593Smuzhiyun 	}
167*4882a593Smuzhiyun 
168*4882a593Smuzhiyun 	assert->format(assert, stream);
169*4882a593Smuzhiyun 
170*4882a593Smuzhiyun 	kunit_print_string_stream(test, stream);
171*4882a593Smuzhiyun 
172*4882a593Smuzhiyun 	WARN_ON(string_stream_destroy(stream));
173*4882a593Smuzhiyun }
174*4882a593Smuzhiyun 
kunit_abort(struct kunit * test)175*4882a593Smuzhiyun static void __noreturn kunit_abort(struct kunit *test)
176*4882a593Smuzhiyun {
177*4882a593Smuzhiyun 	kunit_try_catch_throw(&test->try_catch); /* Does not return. */
178*4882a593Smuzhiyun 
179*4882a593Smuzhiyun 	/*
180*4882a593Smuzhiyun 	 * Throw could not abort from test.
181*4882a593Smuzhiyun 	 *
182*4882a593Smuzhiyun 	 * XXX: we should never reach this line! As kunit_try_catch_throw is
183*4882a593Smuzhiyun 	 * marked __noreturn.
184*4882a593Smuzhiyun 	 */
185*4882a593Smuzhiyun 	WARN_ONCE(true, "Throw could not abort from test!\n");
186*4882a593Smuzhiyun }
187*4882a593Smuzhiyun 
kunit_do_assertion(struct kunit * test,struct kunit_assert * assert,bool pass,const char * fmt,...)188*4882a593Smuzhiyun void kunit_do_assertion(struct kunit *test,
189*4882a593Smuzhiyun 			struct kunit_assert *assert,
190*4882a593Smuzhiyun 			bool pass,
191*4882a593Smuzhiyun 			const char *fmt, ...)
192*4882a593Smuzhiyun {
193*4882a593Smuzhiyun 	va_list args;
194*4882a593Smuzhiyun 
195*4882a593Smuzhiyun 	if (pass)
196*4882a593Smuzhiyun 		return;
197*4882a593Smuzhiyun 
198*4882a593Smuzhiyun 	va_start(args, fmt);
199*4882a593Smuzhiyun 
200*4882a593Smuzhiyun 	assert->message.fmt = fmt;
201*4882a593Smuzhiyun 	assert->message.va = &args;
202*4882a593Smuzhiyun 
203*4882a593Smuzhiyun 	kunit_fail(test, assert);
204*4882a593Smuzhiyun 
205*4882a593Smuzhiyun 	va_end(args);
206*4882a593Smuzhiyun 
207*4882a593Smuzhiyun 	if (assert->type == KUNIT_ASSERTION)
208*4882a593Smuzhiyun 		kunit_abort(test);
209*4882a593Smuzhiyun }
210*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(kunit_do_assertion);
211*4882a593Smuzhiyun 
kunit_init_test(struct kunit * test,const char * name,char * log)212*4882a593Smuzhiyun void kunit_init_test(struct kunit *test, const char *name, char *log)
213*4882a593Smuzhiyun {
214*4882a593Smuzhiyun 	spin_lock_init(&test->lock);
215*4882a593Smuzhiyun 	INIT_LIST_HEAD(&test->resources);
216*4882a593Smuzhiyun 	test->name = name;
217*4882a593Smuzhiyun 	test->log = log;
218*4882a593Smuzhiyun 	if (test->log)
219*4882a593Smuzhiyun 		test->log[0] = '\0';
220*4882a593Smuzhiyun 	test->success = true;
221*4882a593Smuzhiyun }
222*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(kunit_init_test);
223*4882a593Smuzhiyun 
224*4882a593Smuzhiyun /*
225*4882a593Smuzhiyun  * Initializes and runs test case. Does not clean up or do post validations.
226*4882a593Smuzhiyun  */
kunit_run_case_internal(struct kunit * test,struct kunit_suite * suite,struct kunit_case * test_case)227*4882a593Smuzhiyun static void kunit_run_case_internal(struct kunit *test,
228*4882a593Smuzhiyun 				    struct kunit_suite *suite,
229*4882a593Smuzhiyun 				    struct kunit_case *test_case)
230*4882a593Smuzhiyun {
231*4882a593Smuzhiyun 	if (suite->init) {
232*4882a593Smuzhiyun 		int ret;
233*4882a593Smuzhiyun 
234*4882a593Smuzhiyun 		ret = suite->init(test);
235*4882a593Smuzhiyun 		if (ret) {
236*4882a593Smuzhiyun 			kunit_err(test, "failed to initialize: %d\n", ret);
237*4882a593Smuzhiyun 			kunit_set_failure(test);
238*4882a593Smuzhiyun 			return;
239*4882a593Smuzhiyun 		}
240*4882a593Smuzhiyun 	}
241*4882a593Smuzhiyun 
242*4882a593Smuzhiyun 	test_case->run_case(test);
243*4882a593Smuzhiyun }
244*4882a593Smuzhiyun 
kunit_case_internal_cleanup(struct kunit * test)245*4882a593Smuzhiyun static void kunit_case_internal_cleanup(struct kunit *test)
246*4882a593Smuzhiyun {
247*4882a593Smuzhiyun 	kunit_cleanup(test);
248*4882a593Smuzhiyun }
249*4882a593Smuzhiyun 
250*4882a593Smuzhiyun /*
251*4882a593Smuzhiyun  * Performs post validations and cleanup after a test case was run.
252*4882a593Smuzhiyun  * XXX: Should ONLY BE CALLED AFTER kunit_run_case_internal!
253*4882a593Smuzhiyun  */
kunit_run_case_cleanup(struct kunit * test,struct kunit_suite * suite)254*4882a593Smuzhiyun static void kunit_run_case_cleanup(struct kunit *test,
255*4882a593Smuzhiyun 				   struct kunit_suite *suite)
256*4882a593Smuzhiyun {
257*4882a593Smuzhiyun 	if (suite->exit)
258*4882a593Smuzhiyun 		suite->exit(test);
259*4882a593Smuzhiyun 
260*4882a593Smuzhiyun 	kunit_case_internal_cleanup(test);
261*4882a593Smuzhiyun }
262*4882a593Smuzhiyun 
263*4882a593Smuzhiyun struct kunit_try_catch_context {
264*4882a593Smuzhiyun 	struct kunit *test;
265*4882a593Smuzhiyun 	struct kunit_suite *suite;
266*4882a593Smuzhiyun 	struct kunit_case *test_case;
267*4882a593Smuzhiyun };
268*4882a593Smuzhiyun 
kunit_try_run_case(void * data)269*4882a593Smuzhiyun static void kunit_try_run_case(void *data)
270*4882a593Smuzhiyun {
271*4882a593Smuzhiyun 	struct kunit_try_catch_context *ctx = data;
272*4882a593Smuzhiyun 	struct kunit *test = ctx->test;
273*4882a593Smuzhiyun 	struct kunit_suite *suite = ctx->suite;
274*4882a593Smuzhiyun 	struct kunit_case *test_case = ctx->test_case;
275*4882a593Smuzhiyun 
276*4882a593Smuzhiyun #if (IS_ENABLED(CONFIG_KASAN) && IS_ENABLED(CONFIG_KUNIT))
277*4882a593Smuzhiyun 	current->kunit_test = test;
278*4882a593Smuzhiyun #endif /* IS_ENABLED(CONFIG_KASAN) && IS_ENABLED(CONFIG_KUNIT) */
279*4882a593Smuzhiyun 
280*4882a593Smuzhiyun 	/*
281*4882a593Smuzhiyun 	 * kunit_run_case_internal may encounter a fatal error; if it does,
282*4882a593Smuzhiyun 	 * abort will be called, this thread will exit, and finally the parent
283*4882a593Smuzhiyun 	 * thread will resume control and handle any necessary clean up.
284*4882a593Smuzhiyun 	 */
285*4882a593Smuzhiyun 	kunit_run_case_internal(test, suite, test_case);
286*4882a593Smuzhiyun 	/* This line may never be reached. */
287*4882a593Smuzhiyun 	kunit_run_case_cleanup(test, suite);
288*4882a593Smuzhiyun }
289*4882a593Smuzhiyun 
kunit_catch_run_case(void * data)290*4882a593Smuzhiyun static void kunit_catch_run_case(void *data)
291*4882a593Smuzhiyun {
292*4882a593Smuzhiyun 	struct kunit_try_catch_context *ctx = data;
293*4882a593Smuzhiyun 	struct kunit *test = ctx->test;
294*4882a593Smuzhiyun 	struct kunit_suite *suite = ctx->suite;
295*4882a593Smuzhiyun 	int try_exit_code = kunit_try_catch_get_result(&test->try_catch);
296*4882a593Smuzhiyun 
297*4882a593Smuzhiyun 	if (try_exit_code) {
298*4882a593Smuzhiyun 		kunit_set_failure(test);
299*4882a593Smuzhiyun 		/*
300*4882a593Smuzhiyun 		 * Test case could not finish, we have no idea what state it is
301*4882a593Smuzhiyun 		 * in, so don't do clean up.
302*4882a593Smuzhiyun 		 */
303*4882a593Smuzhiyun 		if (try_exit_code == -ETIMEDOUT) {
304*4882a593Smuzhiyun 			kunit_err(test, "test case timed out\n");
305*4882a593Smuzhiyun 		/*
306*4882a593Smuzhiyun 		 * Unknown internal error occurred preventing test case from
307*4882a593Smuzhiyun 		 * running, so there is nothing to clean up.
308*4882a593Smuzhiyun 		 */
309*4882a593Smuzhiyun 		} else {
310*4882a593Smuzhiyun 			kunit_err(test, "internal error occurred preventing test case from running: %d\n",
311*4882a593Smuzhiyun 				  try_exit_code);
312*4882a593Smuzhiyun 		}
313*4882a593Smuzhiyun 		return;
314*4882a593Smuzhiyun 	}
315*4882a593Smuzhiyun 
316*4882a593Smuzhiyun 	/*
317*4882a593Smuzhiyun 	 * Test case was run, but aborted. It is the test case's business as to
318*4882a593Smuzhiyun 	 * whether it failed or not, we just need to clean up.
319*4882a593Smuzhiyun 	 */
320*4882a593Smuzhiyun 	kunit_run_case_cleanup(test, suite);
321*4882a593Smuzhiyun }
322*4882a593Smuzhiyun 
323*4882a593Smuzhiyun /*
324*4882a593Smuzhiyun  * Performs all logic to run a test case. It also catches most errors that
325*4882a593Smuzhiyun  * occur in a test case and reports them as failures.
326*4882a593Smuzhiyun  */
kunit_run_case_catch_errors(struct kunit_suite * suite,struct kunit_case * test_case)327*4882a593Smuzhiyun static void kunit_run_case_catch_errors(struct kunit_suite *suite,
328*4882a593Smuzhiyun 					struct kunit_case *test_case)
329*4882a593Smuzhiyun {
330*4882a593Smuzhiyun 	struct kunit_try_catch_context context;
331*4882a593Smuzhiyun 	struct kunit_try_catch *try_catch;
332*4882a593Smuzhiyun 	struct kunit test;
333*4882a593Smuzhiyun 
334*4882a593Smuzhiyun 	kunit_init_test(&test, test_case->name, test_case->log);
335*4882a593Smuzhiyun 	try_catch = &test.try_catch;
336*4882a593Smuzhiyun 
337*4882a593Smuzhiyun 	kunit_try_catch_init(try_catch,
338*4882a593Smuzhiyun 			     &test,
339*4882a593Smuzhiyun 			     kunit_try_run_case,
340*4882a593Smuzhiyun 			     kunit_catch_run_case);
341*4882a593Smuzhiyun 	context.test = &test;
342*4882a593Smuzhiyun 	context.suite = suite;
343*4882a593Smuzhiyun 	context.test_case = test_case;
344*4882a593Smuzhiyun 	kunit_try_catch_run(try_catch, &context);
345*4882a593Smuzhiyun 
346*4882a593Smuzhiyun 	test_case->success = test.success;
347*4882a593Smuzhiyun 
348*4882a593Smuzhiyun 	kunit_print_ok_not_ok(&test, true, test_case->success,
349*4882a593Smuzhiyun 			      kunit_test_case_num(suite, test_case),
350*4882a593Smuzhiyun 			      test_case->name);
351*4882a593Smuzhiyun }
352*4882a593Smuzhiyun 
kunit_run_tests(struct kunit_suite * suite)353*4882a593Smuzhiyun int kunit_run_tests(struct kunit_suite *suite)
354*4882a593Smuzhiyun {
355*4882a593Smuzhiyun 	struct kunit_case *test_case;
356*4882a593Smuzhiyun 
357*4882a593Smuzhiyun 	kunit_print_subtest_start(suite);
358*4882a593Smuzhiyun 
359*4882a593Smuzhiyun 	kunit_suite_for_each_test_case(suite, test_case)
360*4882a593Smuzhiyun 		kunit_run_case_catch_errors(suite, test_case);
361*4882a593Smuzhiyun 
362*4882a593Smuzhiyun 	kunit_print_subtest_end(suite);
363*4882a593Smuzhiyun 
364*4882a593Smuzhiyun 	return 0;
365*4882a593Smuzhiyun }
366*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(kunit_run_tests);
367*4882a593Smuzhiyun 
kunit_init_suite(struct kunit_suite * suite)368*4882a593Smuzhiyun static void kunit_init_suite(struct kunit_suite *suite)
369*4882a593Smuzhiyun {
370*4882a593Smuzhiyun 	kunit_debugfs_create_suite(suite);
371*4882a593Smuzhiyun }
372*4882a593Smuzhiyun 
__kunit_test_suites_init(struct kunit_suite * const * const suites)373*4882a593Smuzhiyun int __kunit_test_suites_init(struct kunit_suite * const * const suites)
374*4882a593Smuzhiyun {
375*4882a593Smuzhiyun 	unsigned int i;
376*4882a593Smuzhiyun 
377*4882a593Smuzhiyun 	for (i = 0; suites[i] != NULL; i++) {
378*4882a593Smuzhiyun 		kunit_init_suite(suites[i]);
379*4882a593Smuzhiyun 		kunit_run_tests(suites[i]);
380*4882a593Smuzhiyun 	}
381*4882a593Smuzhiyun 	return 0;
382*4882a593Smuzhiyun }
383*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(__kunit_test_suites_init);
384*4882a593Smuzhiyun 
kunit_exit_suite(struct kunit_suite * suite)385*4882a593Smuzhiyun static void kunit_exit_suite(struct kunit_suite *suite)
386*4882a593Smuzhiyun {
387*4882a593Smuzhiyun 	kunit_debugfs_destroy_suite(suite);
388*4882a593Smuzhiyun }
389*4882a593Smuzhiyun 
__kunit_test_suites_exit(struct kunit_suite ** suites)390*4882a593Smuzhiyun void __kunit_test_suites_exit(struct kunit_suite **suites)
391*4882a593Smuzhiyun {
392*4882a593Smuzhiyun 	unsigned int i;
393*4882a593Smuzhiyun 
394*4882a593Smuzhiyun 	for (i = 0; suites[i] != NULL; i++)
395*4882a593Smuzhiyun 		kunit_exit_suite(suites[i]);
396*4882a593Smuzhiyun }
397*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(__kunit_test_suites_exit);
398*4882a593Smuzhiyun 
399*4882a593Smuzhiyun /*
400*4882a593Smuzhiyun  * Used for static resources and when a kunit_resource * has been created by
401*4882a593Smuzhiyun  * kunit_alloc_resource().  When an init function is supplied, @data is passed
402*4882a593Smuzhiyun  * into the init function; otherwise, we simply set the resource data field to
403*4882a593Smuzhiyun  * the data value passed in.
404*4882a593Smuzhiyun  */
kunit_add_resource(struct kunit * test,kunit_resource_init_t init,kunit_resource_free_t free,struct kunit_resource * res,void * data)405*4882a593Smuzhiyun int kunit_add_resource(struct kunit *test,
406*4882a593Smuzhiyun 		       kunit_resource_init_t init,
407*4882a593Smuzhiyun 		       kunit_resource_free_t free,
408*4882a593Smuzhiyun 		       struct kunit_resource *res,
409*4882a593Smuzhiyun 		       void *data)
410*4882a593Smuzhiyun {
411*4882a593Smuzhiyun 	int ret = 0;
412*4882a593Smuzhiyun 
413*4882a593Smuzhiyun 	res->free = free;
414*4882a593Smuzhiyun 	kref_init(&res->refcount);
415*4882a593Smuzhiyun 
416*4882a593Smuzhiyun 	if (init) {
417*4882a593Smuzhiyun 		ret = init(res, data);
418*4882a593Smuzhiyun 		if (ret)
419*4882a593Smuzhiyun 			return ret;
420*4882a593Smuzhiyun 	} else {
421*4882a593Smuzhiyun 		res->data = data;
422*4882a593Smuzhiyun 	}
423*4882a593Smuzhiyun 
424*4882a593Smuzhiyun 	spin_lock(&test->lock);
425*4882a593Smuzhiyun 	list_add_tail(&res->node, &test->resources);
426*4882a593Smuzhiyun 	/* refcount for list is established by kref_init() */
427*4882a593Smuzhiyun 	spin_unlock(&test->lock);
428*4882a593Smuzhiyun 
429*4882a593Smuzhiyun 	return ret;
430*4882a593Smuzhiyun }
431*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(kunit_add_resource);
432*4882a593Smuzhiyun 
kunit_add_named_resource(struct kunit * test,kunit_resource_init_t init,kunit_resource_free_t free,struct kunit_resource * res,const char * name,void * data)433*4882a593Smuzhiyun int kunit_add_named_resource(struct kunit *test,
434*4882a593Smuzhiyun 			     kunit_resource_init_t init,
435*4882a593Smuzhiyun 			     kunit_resource_free_t free,
436*4882a593Smuzhiyun 			     struct kunit_resource *res,
437*4882a593Smuzhiyun 			     const char *name,
438*4882a593Smuzhiyun 			     void *data)
439*4882a593Smuzhiyun {
440*4882a593Smuzhiyun 	struct kunit_resource *existing;
441*4882a593Smuzhiyun 
442*4882a593Smuzhiyun 	if (!name)
443*4882a593Smuzhiyun 		return -EINVAL;
444*4882a593Smuzhiyun 
445*4882a593Smuzhiyun 	existing = kunit_find_named_resource(test, name);
446*4882a593Smuzhiyun 	if (existing) {
447*4882a593Smuzhiyun 		kunit_put_resource(existing);
448*4882a593Smuzhiyun 		return -EEXIST;
449*4882a593Smuzhiyun 	}
450*4882a593Smuzhiyun 
451*4882a593Smuzhiyun 	res->name = name;
452*4882a593Smuzhiyun 
453*4882a593Smuzhiyun 	return kunit_add_resource(test, init, free, res, data);
454*4882a593Smuzhiyun }
455*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(kunit_add_named_resource);
456*4882a593Smuzhiyun 
kunit_alloc_and_get_resource(struct kunit * test,kunit_resource_init_t init,kunit_resource_free_t free,gfp_t internal_gfp,void * data)457*4882a593Smuzhiyun struct kunit_resource *kunit_alloc_and_get_resource(struct kunit *test,
458*4882a593Smuzhiyun 						    kunit_resource_init_t init,
459*4882a593Smuzhiyun 						    kunit_resource_free_t free,
460*4882a593Smuzhiyun 						    gfp_t internal_gfp,
461*4882a593Smuzhiyun 						    void *data)
462*4882a593Smuzhiyun {
463*4882a593Smuzhiyun 	struct kunit_resource *res;
464*4882a593Smuzhiyun 	int ret;
465*4882a593Smuzhiyun 
466*4882a593Smuzhiyun 	res = kzalloc(sizeof(*res), internal_gfp);
467*4882a593Smuzhiyun 	if (!res)
468*4882a593Smuzhiyun 		return NULL;
469*4882a593Smuzhiyun 
470*4882a593Smuzhiyun 	ret = kunit_add_resource(test, init, free, res, data);
471*4882a593Smuzhiyun 	if (!ret) {
472*4882a593Smuzhiyun 		/*
473*4882a593Smuzhiyun 		 * bump refcount for get; kunit_resource_put() should be called
474*4882a593Smuzhiyun 		 * when done.
475*4882a593Smuzhiyun 		 */
476*4882a593Smuzhiyun 		kunit_get_resource(res);
477*4882a593Smuzhiyun 		return res;
478*4882a593Smuzhiyun 	}
479*4882a593Smuzhiyun 	return NULL;
480*4882a593Smuzhiyun }
481*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(kunit_alloc_and_get_resource);
482*4882a593Smuzhiyun 
kunit_remove_resource(struct kunit * test,struct kunit_resource * res)483*4882a593Smuzhiyun void kunit_remove_resource(struct kunit *test, struct kunit_resource *res)
484*4882a593Smuzhiyun {
485*4882a593Smuzhiyun 	spin_lock(&test->lock);
486*4882a593Smuzhiyun 	list_del(&res->node);
487*4882a593Smuzhiyun 	spin_unlock(&test->lock);
488*4882a593Smuzhiyun 	kunit_put_resource(res);
489*4882a593Smuzhiyun }
490*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(kunit_remove_resource);
491*4882a593Smuzhiyun 
kunit_destroy_resource(struct kunit * test,kunit_resource_match_t match,void * match_data)492*4882a593Smuzhiyun int kunit_destroy_resource(struct kunit *test, kunit_resource_match_t match,
493*4882a593Smuzhiyun 			   void *match_data)
494*4882a593Smuzhiyun {
495*4882a593Smuzhiyun 	struct kunit_resource *res = kunit_find_resource(test, match,
496*4882a593Smuzhiyun 							 match_data);
497*4882a593Smuzhiyun 
498*4882a593Smuzhiyun 	if (!res)
499*4882a593Smuzhiyun 		return -ENOENT;
500*4882a593Smuzhiyun 
501*4882a593Smuzhiyun 	kunit_remove_resource(test, res);
502*4882a593Smuzhiyun 
503*4882a593Smuzhiyun 	/* We have a reference also via _find(); drop it. */
504*4882a593Smuzhiyun 	kunit_put_resource(res);
505*4882a593Smuzhiyun 
506*4882a593Smuzhiyun 	return 0;
507*4882a593Smuzhiyun }
508*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(kunit_destroy_resource);
509*4882a593Smuzhiyun 
510*4882a593Smuzhiyun struct kunit_kmalloc_params {
511*4882a593Smuzhiyun 	size_t size;
512*4882a593Smuzhiyun 	gfp_t gfp;
513*4882a593Smuzhiyun };
514*4882a593Smuzhiyun 
kunit_kmalloc_init(struct kunit_resource * res,void * context)515*4882a593Smuzhiyun static int kunit_kmalloc_init(struct kunit_resource *res, void *context)
516*4882a593Smuzhiyun {
517*4882a593Smuzhiyun 	struct kunit_kmalloc_params *params = context;
518*4882a593Smuzhiyun 
519*4882a593Smuzhiyun 	res->data = kmalloc(params->size, params->gfp);
520*4882a593Smuzhiyun 	if (!res->data)
521*4882a593Smuzhiyun 		return -ENOMEM;
522*4882a593Smuzhiyun 
523*4882a593Smuzhiyun 	return 0;
524*4882a593Smuzhiyun }
525*4882a593Smuzhiyun 
kunit_kmalloc_free(struct kunit_resource * res)526*4882a593Smuzhiyun static void kunit_kmalloc_free(struct kunit_resource *res)
527*4882a593Smuzhiyun {
528*4882a593Smuzhiyun 	kfree(res->data);
529*4882a593Smuzhiyun }
530*4882a593Smuzhiyun 
kunit_kmalloc(struct kunit * test,size_t size,gfp_t gfp)531*4882a593Smuzhiyun void *kunit_kmalloc(struct kunit *test, size_t size, gfp_t gfp)
532*4882a593Smuzhiyun {
533*4882a593Smuzhiyun 	struct kunit_kmalloc_params params = {
534*4882a593Smuzhiyun 		.size = size,
535*4882a593Smuzhiyun 		.gfp = gfp
536*4882a593Smuzhiyun 	};
537*4882a593Smuzhiyun 
538*4882a593Smuzhiyun 	return kunit_alloc_resource(test,
539*4882a593Smuzhiyun 				    kunit_kmalloc_init,
540*4882a593Smuzhiyun 				    kunit_kmalloc_free,
541*4882a593Smuzhiyun 				    gfp,
542*4882a593Smuzhiyun 				    &params);
543*4882a593Smuzhiyun }
544*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(kunit_kmalloc);
545*4882a593Smuzhiyun 
kunit_kfree(struct kunit * test,const void * ptr)546*4882a593Smuzhiyun void kunit_kfree(struct kunit *test, const void *ptr)
547*4882a593Smuzhiyun {
548*4882a593Smuzhiyun 	struct kunit_resource *res;
549*4882a593Smuzhiyun 
550*4882a593Smuzhiyun 	res = kunit_find_resource(test, kunit_resource_instance_match,
551*4882a593Smuzhiyun 				  (void *)ptr);
552*4882a593Smuzhiyun 
553*4882a593Smuzhiyun 	/*
554*4882a593Smuzhiyun 	 * Removing the resource from the list of resources drops the
555*4882a593Smuzhiyun 	 * reference count to 1; the final put will trigger the free.
556*4882a593Smuzhiyun 	 */
557*4882a593Smuzhiyun 	kunit_remove_resource(test, res);
558*4882a593Smuzhiyun 
559*4882a593Smuzhiyun 	kunit_put_resource(res);
560*4882a593Smuzhiyun 
561*4882a593Smuzhiyun }
562*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(kunit_kfree);
563*4882a593Smuzhiyun 
kunit_cleanup(struct kunit * test)564*4882a593Smuzhiyun void kunit_cleanup(struct kunit *test)
565*4882a593Smuzhiyun {
566*4882a593Smuzhiyun 	struct kunit_resource *res;
567*4882a593Smuzhiyun 
568*4882a593Smuzhiyun 	/*
569*4882a593Smuzhiyun 	 * test->resources is a stack - each allocation must be freed in the
570*4882a593Smuzhiyun 	 * reverse order from which it was added since one resource may depend
571*4882a593Smuzhiyun 	 * on another for its entire lifetime.
572*4882a593Smuzhiyun 	 * Also, we cannot use the normal list_for_each constructs, even the
573*4882a593Smuzhiyun 	 * safe ones because *arbitrary* nodes may be deleted when
574*4882a593Smuzhiyun 	 * kunit_resource_free is called; the list_for_each_safe variants only
575*4882a593Smuzhiyun 	 * protect against the current node being deleted, not the next.
576*4882a593Smuzhiyun 	 */
577*4882a593Smuzhiyun 	while (true) {
578*4882a593Smuzhiyun 		spin_lock(&test->lock);
579*4882a593Smuzhiyun 		if (list_empty(&test->resources)) {
580*4882a593Smuzhiyun 			spin_unlock(&test->lock);
581*4882a593Smuzhiyun 			break;
582*4882a593Smuzhiyun 		}
583*4882a593Smuzhiyun 		res = list_last_entry(&test->resources,
584*4882a593Smuzhiyun 				      struct kunit_resource,
585*4882a593Smuzhiyun 				      node);
586*4882a593Smuzhiyun 		/*
587*4882a593Smuzhiyun 		 * Need to unlock here as a resource may remove another
588*4882a593Smuzhiyun 		 * resource, and this can't happen if the test->lock
589*4882a593Smuzhiyun 		 * is held.
590*4882a593Smuzhiyun 		 */
591*4882a593Smuzhiyun 		spin_unlock(&test->lock);
592*4882a593Smuzhiyun 		kunit_remove_resource(test, res);
593*4882a593Smuzhiyun 	}
594*4882a593Smuzhiyun #if (IS_ENABLED(CONFIG_KASAN) && IS_ENABLED(CONFIG_KUNIT))
595*4882a593Smuzhiyun 	current->kunit_test = NULL;
596*4882a593Smuzhiyun #endif /* IS_ENABLED(CONFIG_KASAN) && IS_ENABLED(CONFIG_KUNIT)*/
597*4882a593Smuzhiyun }
598*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(kunit_cleanup);
599*4882a593Smuzhiyun 
kunit_init(void)600*4882a593Smuzhiyun static int __init kunit_init(void)
601*4882a593Smuzhiyun {
602*4882a593Smuzhiyun 	kunit_debugfs_init();
603*4882a593Smuzhiyun 
604*4882a593Smuzhiyun 	return 0;
605*4882a593Smuzhiyun }
606*4882a593Smuzhiyun late_initcall(kunit_init);
607*4882a593Smuzhiyun 
kunit_exit(void)608*4882a593Smuzhiyun static void __exit kunit_exit(void)
609*4882a593Smuzhiyun {
610*4882a593Smuzhiyun 	kunit_debugfs_cleanup();
611*4882a593Smuzhiyun }
612*4882a593Smuzhiyun module_exit(kunit_exit);
613*4882a593Smuzhiyun 
614*4882a593Smuzhiyun MODULE_LICENSE("GPL v2");
615