1 // SPDX-License-Identifier: GPL-2.0
2 /* Copyright (c) 2019 Facebook */
4 #include <test_progs.h>
6 #include <sys/utsname.h>
7 #include <linux/version.h>
8 #include "test_core_extern.skel.h"
10 static uint32_t get_kernel_version(void)
12 uint32_t major, minor, patch;
16 if (sscanf(info.release, "%u.%u.%u", &major, &minor, &patch) != 3)
18 return KERNEL_VERSION(major, minor, patch);
21 #define CFG "CONFIG_BPF_SYSCALL=n\n"
23 static struct test_case {
27 struct test_core_extern__data data;
29 { .name = "default search path", .data = { .bpf_syscall = true } },
31 .name = "custom values",
32 .cfg = "CONFIG_BPF_SYSCALL=n\n"
36 "CONFIG_USHORT=30000\n"
38 "CONFIG_ULONG=0xDEADBEEFC0DE\n"
39 "CONFIG_STR=\"abracad\"\n"
44 .tristate_val = TRI_MODULE,
49 .ulong_val = 0xDEADBEEFC0DE,
54 { .name = "tristate (y)", .cfg = CFG"CONFIG_TRISTATE=y\n",
55 .data = { .tristate_val = TRI_YES } },
56 { .name = "tristate (n)", .cfg = CFG"CONFIG_TRISTATE=n\n",
57 .data = { .tristate_val = TRI_NO } },
58 { .name = "tristate (m)", .cfg = CFG"CONFIG_TRISTATE=m\n",
59 .data = { .tristate_val = TRI_MODULE } },
60 { .name = "tristate (int)", .fails = 1, .cfg = CFG"CONFIG_TRISTATE=1" },
61 { .name = "tristate (bad)", .fails = 1, .cfg = CFG"CONFIG_TRISTATE=M" },
63 { .name = "bool (y)", .cfg = CFG"CONFIG_BOOL=y\n",
64 .data = { .bool_val = true } },
65 { .name = "bool (n)", .cfg = CFG"CONFIG_BOOL=n\n",
66 .data = { .bool_val = false } },
67 { .name = "bool (tristate)", .fails = 1, .cfg = CFG"CONFIG_BOOL=m" },
68 { .name = "bool (int)", .fails = 1, .cfg = CFG"CONFIG_BOOL=1" },
70 { .name = "char (tristate)", .cfg = CFG"CONFIG_CHAR=m\n",
71 .data = { .char_val = 'm' } },
72 { .name = "char (bad)", .fails = 1, .cfg = CFG"CONFIG_CHAR=q\n" },
73 { .name = "char (empty)", .fails = 1, .cfg = CFG"CONFIG_CHAR=\n" },
74 { .name = "char (str)", .fails = 1, .cfg = CFG"CONFIG_CHAR=\"y\"\n" },
76 { .name = "str (empty)", .cfg = CFG"CONFIG_STR=\"\"\n",
77 .data = { .str_val = "\0\0\0\0\0\0\0" } },
78 { .name = "str (padded)", .cfg = CFG"CONFIG_STR=\"abra\"\n",
79 .data = { .str_val = "abra\0\0\0" } },
80 { .name = "str (too long)", .cfg = CFG"CONFIG_STR=\"abracada\"\n",
81 .data = { .str_val = "abracad" } },
82 { .name = "str (no value)", .fails = 1, .cfg = CFG"CONFIG_STR=\n" },
83 { .name = "str (bad value)", .fails = 1, .cfg = CFG"CONFIG_STR=bla\n" },
86 .name = "integer forms",
89 "CONFIG_USHORT=0462\n"
91 "CONFIG_ULONG=+1000000000000",
96 .ulong_val = 1000000000000,
99 { .name = "int (bad)", .fails = 1, .cfg = CFG"CONFIG_INT=abc" },
100 { .name = "int (str)", .fails = 1, .cfg = CFG"CONFIG_INT=\"abc\"" },
101 { .name = "int (empty)", .fails = 1, .cfg = CFG"CONFIG_INT=" },
102 { .name = "int (mixed)", .fails = 1, .cfg = CFG"CONFIG_INT=123abc" },
103 { .name = "int (max)", .cfg = CFG"CONFIG_INT=2147483647",
104 .data = { .int_val = 2147483647 } },
105 { .name = "int (min)", .cfg = CFG"CONFIG_INT=-2147483648",
106 .data = { .int_val = -2147483648 } },
107 { .name = "int (max+1)", .fails = 1, .cfg = CFG"CONFIG_INT=2147483648" },
108 { .name = "int (min-1)", .fails = 1, .cfg = CFG"CONFIG_INT=-2147483649" },
109 { .name = "ushort (max)", .cfg = CFG"CONFIG_USHORT=65535",
110 .data = { .ushort_val = 65535 } },
111 { .name = "ushort (min)", .cfg = CFG"CONFIG_USHORT=0",
112 .data = { .ushort_val = 0 } },
113 { .name = "ushort (max+1)", .fails = 1, .cfg = CFG"CONFIG_USHORT=65536" },
114 { .name = "ushort (min-1)", .fails = 1, .cfg = CFG"CONFIG_USHORT=-1" },
115 { .name = "u64 (max)", .cfg = CFG"CONFIG_ULONG=0xffffffffffffffff",
116 .data = { .ulong_val = 0xffffffffffffffff } },
117 { .name = "u64 (min)", .cfg = CFG"CONFIG_ULONG=0",
118 .data = { .ulong_val = 0 } },
119 { .name = "u64 (max+1)", .fails = 1, .cfg = CFG"CONFIG_ULONG=0x10000000000000000" },
122 void test_core_extern(void)
124 const uint32_t kern_ver = get_kernel_version();
126 struct test_core_extern *skel = NULL;
128 int n = sizeof(*skel->data) / sizeof(uint64_t);
130 for (i = 0; i < ARRAY_SIZE(test_cases); i++) {
131 struct test_case *t = &test_cases[i];
132 DECLARE_LIBBPF_OPTS(bpf_object_open_opts, opts,
136 if (!test__start_subtest(t->name))
139 skel = test_core_extern__open_opts(&opts);
140 if (!ASSERT_OK_PTR(skel, "skel_open"))
142 err = test_core_extern__load(skel);
144 ASSERT_ERR(err, "skel_load_should_fail");
146 } else if (!ASSERT_OK(err, "skel_load")) {
149 err = test_core_extern__attach(skel);
150 if (!ASSERT_OK(err, "attach_raw_tp"))
155 t->data.kern_ver = kern_ver;
156 t->data.missing_val = 0xDEADC0DE;
157 got = (uint64_t *)skel->data;
158 exp = (uint64_t *)&t->data;
159 for (j = 0; j < n; j++) {
160 ASSERT_EQ(got[j], exp[j], "result");
163 test_core_extern__destroy(skel);