1 // SPDX-License-Identifier: GPL-2.0-or-later
3 * Test the function and performance of kallsyms
5 * Copyright (C) Huawei Technologies Co., Ltd., 2022
10 #define pr_fmt(fmt) "kallsyms_selftest: " fmt
12 #include <linux/init.h>
13 #include <linux/module.h>
14 #include <linux/kallsyms.h>
15 #include <linux/random.h>
16 #include <linux/sched/clock.h>
17 #include <linux/kthread.h>
18 #include <linux/vmalloc.h>
20 #include "kallsyms_internal.h"
21 #include "kallsyms_selftest.h"
24 #define MAX_NUM_OF_RECORDS 64
35 unsigned long addrs[MAX_NUM_OF_RECORDS];
43 #define ITEM_FUNC(s) \
46 .addr = (unsigned long)s, \
49 #define ITEM_DATA(s) \
52 .addr = (unsigned long)&s, \
56 static int kallsyms_test_var_bss_static;
57 static int kallsyms_test_var_data_static = 1;
58 int kallsyms_test_var_bss;
59 int kallsyms_test_var_data = 1;
61 static int kallsyms_test_func_static(void)
63 kallsyms_test_var_bss_static++;
64 kallsyms_test_var_data_static++;
69 int kallsyms_test_func(void)
71 return kallsyms_test_func_static();
74 __weak int kallsyms_test_func_weak(void)
76 kallsyms_test_var_bss++;
77 kallsyms_test_var_data++;
81 static struct test_item test_items[] = {
82 ITEM_FUNC(kallsyms_test_func_static),
83 ITEM_FUNC(kallsyms_test_func),
84 ITEM_FUNC(kallsyms_test_func_weak),
85 ITEM_FUNC(vmalloc_noprof),
87 #ifdef CONFIG_KALLSYMS_ALL
88 ITEM_DATA(kallsyms_test_var_bss_static),
89 ITEM_DATA(kallsyms_test_var_data_static),
90 ITEM_DATA(kallsyms_test_var_bss),
91 ITEM_DATA(kallsyms_test_var_data),
95 static char stub_name[KSYM_NAME_LEN];
97 static int stat_symbol_len(void *data, const char *name, unsigned long addr)
99 *(u32 *)data += strlen(name);
104 static void test_kallsyms_compression_ratio(void)
106 u32 pos, off, len, num;
107 u32 ratio, total_size, total_len = 0;
109 kallsyms_on_each_symbol(stat_symbol_len, &total_len);
112 * A symbol name cannot start with a number. This stub name helps us
113 * traverse the entire symbol table without finding a match. It's used
114 * for subsequent performance tests, and its length is the average
115 * length of all symbol names.
117 memset(stub_name, '4', sizeof(stub_name));
118 pos = total_len / kallsyms_num_syms;
124 while (pos < kallsyms_num_syms) {
125 len = kallsyms_names[off];
129 if ((len & 0x80) != 0) {
130 len = (len & 0x7f) | (kallsyms_names[off] << 7);
138 * 1. The length fields is not counted
139 * 2. The memory occupied by array kallsyms_token_table[] and
140 * kallsyms_token_index[] needs to be counted.
142 total_size = off - num;
143 pos = kallsyms_token_index[0xff];
144 total_size += pos + strlen(&kallsyms_token_table[pos]) + 1;
145 total_size += 0x100 * sizeof(u16);
147 pr_info(" ---------------------------------------------------------\n");
148 pr_info("| nr_symbols | compressed size | original size | ratio(%%) |\n");
149 pr_info("|---------------------------------------------------------|\n");
150 ratio = (u32)div_u64(10000ULL * total_size, total_len);
151 pr_info("| %10d | %10d | %10d | %2d.%-2d |\n",
152 kallsyms_num_syms, total_size, total_len, ratio / 100, ratio % 100);
153 pr_info(" ---------------------------------------------------------\n");
156 static int lookup_name(void *data, const char *name, unsigned long addr)
159 struct test_stat *stat = (struct test_stat *)data;
162 (void)kallsyms_lookup_name(name);
178 static void test_perf_kallsyms_lookup_name(void)
180 struct test_stat stat;
182 memset(&stat, 0, sizeof(stat));
184 kallsyms_on_each_symbol(lookup_name, &stat);
185 pr_info("kallsyms_lookup_name() looked up %d symbols\n", stat.real_cnt);
186 pr_info("The time spent on each symbol is (ns): min=%d, max=%d, avg=%lld\n",
187 stat.min, stat.max, div_u64(stat.sum, stat.real_cnt));
190 static int find_symbol(void *data, const char *name, unsigned long addr)
192 struct test_stat *stat = (struct test_stat *)data;
194 if (!strcmp(name, stat->name)) {
198 if (stat->save_cnt < MAX_NUM_OF_RECORDS) {
199 stat->addrs[stat->save_cnt] = addr;
203 if (stat->real_cnt == stat->max)
210 static void test_perf_kallsyms_on_each_symbol(void)
213 struct test_stat stat;
215 memset(&stat, 0, sizeof(stat));
217 stat.name = stub_name;
220 kallsyms_on_each_symbol(find_symbol, &stat);
222 pr_info("kallsyms_on_each_symbol() traverse all: %lld ns\n", t1 - t0);
225 static int match_symbol(void *data, unsigned long addr)
227 struct test_stat *stat = (struct test_stat *)data;
232 if (stat->save_cnt < MAX_NUM_OF_RECORDS) {
233 stat->addrs[stat->save_cnt] = addr;
237 if (stat->real_cnt == stat->max)
243 static void test_perf_kallsyms_on_each_match_symbol(void)
246 struct test_stat stat;
248 memset(&stat, 0, sizeof(stat));
250 stat.name = stub_name;
252 kallsyms_on_each_match_symbol(match_symbol, stat.name, &stat);
254 pr_info("kallsyms_on_each_match_symbol() traverse all: %lld ns\n", t1 - t0);
257 static int test_kallsyms_basic_function(void)
260 int next = 0, nr_failed = 0;
263 unsigned long addr, lookup_addr;
264 char namebuf[KSYM_NAME_LEN];
265 struct test_stat *stat, *stat2;
267 stat = kmalloc(sizeof(*stat) * 2, GFP_KERNEL);
272 prefix = "kallsyms_lookup_name() for";
273 for (i = 0; i < ARRAY_SIZE(test_items); i++) {
274 addr = kallsyms_lookup_name(test_items[i].name);
275 if (addr != test_items[i].addr) {
277 pr_info("%s %s failed: addr=%lx, expect %lx\n",
278 prefix, test_items[i].name, addr, test_items[i].addr);
282 prefix = "kallsyms_on_each_symbol() for";
283 for (i = 0; i < ARRAY_SIZE(test_items); i++) {
284 memset(stat, 0, sizeof(*stat));
286 stat->name = test_items[i].name;
287 kallsyms_on_each_symbol(find_symbol, stat);
288 if (stat->addr != test_items[i].addr || stat->real_cnt != 1) {
290 pr_info("%s %s failed: count=%d, addr=%lx, expect %lx\n",
291 prefix, test_items[i].name,
292 stat->real_cnt, stat->addr, test_items[i].addr);
296 prefix = "kallsyms_on_each_match_symbol() for";
297 for (i = 0; i < ARRAY_SIZE(test_items); i++) {
298 memset(stat, 0, sizeof(*stat));
300 stat->name = test_items[i].name;
301 kallsyms_on_each_match_symbol(match_symbol, test_items[i].name, stat);
302 if (stat->addr != test_items[i].addr || stat->real_cnt != 1) {
304 pr_info("%s %s failed: count=%d, addr=%lx, expect %lx\n",
305 prefix, test_items[i].name,
306 stat->real_cnt, stat->addr, test_items[i].addr);
315 for (i = 0; i < kallsyms_num_syms; i++) {
316 addr = kallsyms_sym_address(i);
317 if (!is_ksym_addr(addr))
320 ret = lookup_symbol_name(addr, namebuf);
323 pr_info("%d: lookup_symbol_name(%lx) failed\n", i, addr);
327 lookup_addr = kallsyms_lookup_name(namebuf);
329 memset(stat, 0, sizeof(*stat));
331 kallsyms_on_each_match_symbol(match_symbol, namebuf, stat);
334 * kallsyms_on_each_symbol() is too slow, randomly select some
338 memset(stat2, 0, sizeof(*stat2));
339 stat2->max = INT_MAX;
340 stat2->name = namebuf;
341 kallsyms_on_each_symbol(find_symbol, stat2);
344 * kallsyms_on_each_symbol() and kallsyms_on_each_match_symbol()
345 * need to get the same traversal result.
347 if (stat->addr != stat2->addr ||
348 stat->real_cnt != stat2->real_cnt ||
349 memcmp(stat->addrs, stat2->addrs,
350 stat->save_cnt * sizeof(stat->addrs[0]))) {
351 pr_info("%s: mismatch between kallsyms_on_each_symbol() and kallsyms_on_each_match_symbol()\n",
357 * The average of random increments is 128, that is, one of
358 * them is tested every 128 symbols.
360 get_random_bytes(&rand, sizeof(rand));
361 next = i + (rand & 0xff) + 1;
364 /* Need to be found at least once */
365 if (!stat->real_cnt) {
366 pr_info("%s: Never found\n", namebuf);
371 * kallsyms_lookup_name() returns the address of the first
372 * symbol found and cannot be NULL.
375 pr_info("%s: NULL lookup_addr?!\n", namebuf);
378 if (lookup_addr != stat->addrs[0]) {
379 pr_info("%s: lookup_addr != stat->addrs[0]\n", namebuf);
384 * If the addresses of all matching symbols are recorded, the
385 * target address needs to be exist.
387 if (stat->real_cnt <= MAX_NUM_OF_RECORDS) {
388 for (j = 0; j < stat->save_cnt; j++) {
389 if (stat->addrs[j] == addr)
393 if (j == stat->save_cnt) {
394 pr_info("%s: j == save_cnt?!\n", namebuf);
405 pr_info("Test for %dth symbol failed: (%s) addr=%lx", i, namebuf, addr);
410 static int test_entry(void *p)
415 schedule_timeout(5 * HZ);
416 } while (system_state != SYSTEM_RUNNING);
419 ret = test_kallsyms_basic_function();
425 test_kallsyms_compression_ratio();
426 test_perf_kallsyms_lookup_name();
427 test_perf_kallsyms_on_each_symbol();
428 test_perf_kallsyms_on_each_match_symbol();
434 static int __init kallsyms_test_init(void)
436 struct task_struct *t;
438 t = kthread_create(test_entry, NULL, "kallsyms_test");
440 pr_info("Create kallsyms selftest task failed\n");
448 late_initcall(kallsyms_test_init);