xref: /linux/lib/tests/longest_symbol_kunit.c (revision a10c7949adf94356e56d5c8878f6fc3f25bd0c15)
1*a10c7949SLinus Torvalds // SPDX-License-Identifier: GPL-2.0
2*a10c7949SLinus Torvalds /*
3*a10c7949SLinus Torvalds  * Test the longest symbol length. Execute with:
4*a10c7949SLinus Torvalds  *  ./tools/testing/kunit/kunit.py run longest-symbol
5*a10c7949SLinus Torvalds  *  --arch=x86_64 --kconfig_add CONFIG_KPROBES=y --kconfig_add CONFIG_MODULES=y
6*a10c7949SLinus Torvalds  *  --kconfig_add CONFIG_RETPOLINE=n --kconfig_add CONFIG_CFI_CLANG=n
7*a10c7949SLinus Torvalds  *  --kconfig_add CONFIG_MITIGATION_RETPOLINE=n
8*a10c7949SLinus Torvalds  */
9*a10c7949SLinus Torvalds 
10*a10c7949SLinus Torvalds #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
11*a10c7949SLinus Torvalds 
12*a10c7949SLinus Torvalds #include <kunit/test.h>
13*a10c7949SLinus Torvalds #include <linux/stringify.h>
14*a10c7949SLinus Torvalds #include <linux/kprobes.h>
15*a10c7949SLinus Torvalds #include <linux/kallsyms.h>
16*a10c7949SLinus Torvalds 
17*a10c7949SLinus Torvalds #define DI(name) s##name##name
18*a10c7949SLinus Torvalds #define DDI(name) DI(n##name##name)
19*a10c7949SLinus Torvalds #define DDDI(name) DDI(n##name##name)
20*a10c7949SLinus Torvalds #define DDDDI(name) DDDI(n##name##name)
21*a10c7949SLinus Torvalds #define DDDDDI(name) DDDDI(n##name##name)
22*a10c7949SLinus Torvalds 
23*a10c7949SLinus Torvalds /*Generate a symbol whose name length is 511 */
24*a10c7949SLinus Torvalds #define LONGEST_SYM_NAME  DDDDDI(g1h2i3j4k5l6m7n)
25*a10c7949SLinus Torvalds 
26*a10c7949SLinus Torvalds #define RETURN_LONGEST_SYM 0xAAAAA
27*a10c7949SLinus Torvalds 
28*a10c7949SLinus Torvalds noinline int LONGEST_SYM_NAME(void);
LONGEST_SYM_NAME(void)29*a10c7949SLinus Torvalds noinline int LONGEST_SYM_NAME(void)
30*a10c7949SLinus Torvalds {
31*a10c7949SLinus Torvalds 	return RETURN_LONGEST_SYM;
32*a10c7949SLinus Torvalds }
33*a10c7949SLinus Torvalds 
34*a10c7949SLinus Torvalds _Static_assert(sizeof(__stringify(LONGEST_SYM_NAME)) == KSYM_NAME_LEN,
35*a10c7949SLinus Torvalds "Incorrect symbol length found. Expected KSYM_NAME_LEN: "
36*a10c7949SLinus Torvalds __stringify(KSYM_NAME_LEN) ", but found: "
37*a10c7949SLinus Torvalds __stringify(sizeof(LONGEST_SYM_NAME)));
38*a10c7949SLinus Torvalds 
test_longest_symbol(struct kunit * test)39*a10c7949SLinus Torvalds static void test_longest_symbol(struct kunit *test)
40*a10c7949SLinus Torvalds {
41*a10c7949SLinus Torvalds 	KUNIT_EXPECT_EQ(test, RETURN_LONGEST_SYM, LONGEST_SYM_NAME());
42*a10c7949SLinus Torvalds };
43*a10c7949SLinus Torvalds 
test_longest_symbol_kallsyms(struct kunit * test)44*a10c7949SLinus Torvalds static void test_longest_symbol_kallsyms(struct kunit *test)
45*a10c7949SLinus Torvalds {
46*a10c7949SLinus Torvalds 	unsigned long (*kallsyms_lookup_name)(const char *name);
47*a10c7949SLinus Torvalds 	static int (*longest_sym)(void);
48*a10c7949SLinus Torvalds 
49*a10c7949SLinus Torvalds 	struct kprobe kp = {
50*a10c7949SLinus Torvalds 		.symbol_name = "kallsyms_lookup_name",
51*a10c7949SLinus Torvalds 	};
52*a10c7949SLinus Torvalds 
53*a10c7949SLinus Torvalds 	if (register_kprobe(&kp) < 0) {
54*a10c7949SLinus Torvalds 		pr_info("%s: kprobe not registered", __func__);
55*a10c7949SLinus Torvalds 		KUNIT_FAIL(test, "test_longest_symbol kallsyms: kprobe not registered\n");
56*a10c7949SLinus Torvalds 		return;
57*a10c7949SLinus Torvalds 	}
58*a10c7949SLinus Torvalds 
59*a10c7949SLinus Torvalds 	kunit_warn(test, "test_longest_symbol kallsyms: kprobe registered\n");
60*a10c7949SLinus Torvalds 	kallsyms_lookup_name = (unsigned long (*)(const char *name))kp.addr;
61*a10c7949SLinus Torvalds 	unregister_kprobe(&kp);
62*a10c7949SLinus Torvalds 
63*a10c7949SLinus Torvalds 	longest_sym =
64*a10c7949SLinus Torvalds 		(void *) kallsyms_lookup_name(__stringify(LONGEST_SYM_NAME));
65*a10c7949SLinus Torvalds 	KUNIT_EXPECT_EQ(test, RETURN_LONGEST_SYM, longest_sym());
66*a10c7949SLinus Torvalds };
67*a10c7949SLinus Torvalds 
68*a10c7949SLinus Torvalds static struct kunit_case longest_symbol_test_cases[] = {
69*a10c7949SLinus Torvalds 	KUNIT_CASE(test_longest_symbol),
70*a10c7949SLinus Torvalds 	KUNIT_CASE(test_longest_symbol_kallsyms),
71*a10c7949SLinus Torvalds 	{}
72*a10c7949SLinus Torvalds };
73*a10c7949SLinus Torvalds 
74*a10c7949SLinus Torvalds static struct kunit_suite longest_symbol_test_suite = {
75*a10c7949SLinus Torvalds 	.name = "longest-symbol",
76*a10c7949SLinus Torvalds 	.test_cases = longest_symbol_test_cases,
77*a10c7949SLinus Torvalds };
78*a10c7949SLinus Torvalds kunit_test_suite(longest_symbol_test_suite);
79*a10c7949SLinus Torvalds 
80*a10c7949SLinus Torvalds MODULE_LICENSE("GPL");
81*a10c7949SLinus Torvalds MODULE_DESCRIPTION("Test the longest symbol length");
82*a10c7949SLinus Torvalds MODULE_AUTHOR("Sergio González Collado");
83