| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448 |
- // SPDX-License-Identifier: GPL-2.0-or-later
- /*
- * Test the function and performance of kallsyms
- *
- * Copyright (C) Huawei Technologies Co., Ltd., 2022
- *
- * Authors: Zhen Lei <thunder.leizhen@huawei.com> Huawei
- */
- #define pr_fmt(fmt) "kallsyms_selftest: " fmt
- #include <linux/init.h>
- #include <linux/module.h>
- #include <linux/kallsyms.h>
- #include <linux/random.h>
- #include <linux/sched/clock.h>
- #include <linux/kthread.h>
- #include <linux/vmalloc.h>
- #include "kallsyms_internal.h"
- #include "kallsyms_selftest.h"
- #define MAX_NUM_OF_RECORDS 64
- struct test_stat {
- int min;
- int max;
- int save_cnt;
- int real_cnt;
- int perf;
- u64 sum;
- char *name;
- unsigned long addr;
- unsigned long addrs[MAX_NUM_OF_RECORDS];
- };
- struct test_item {
- char *name;
- unsigned long addr;
- };
- #define ITEM_FUNC(s) \
- { \
- .name = #s, \
- .addr = (unsigned long)s, \
- }
- #define ITEM_DATA(s) \
- { \
- .name = #s, \
- .addr = (unsigned long)&s, \
- }
- static int kallsyms_test_var_bss_static;
- static int kallsyms_test_var_data_static = 1;
- int kallsyms_test_var_bss;
- int kallsyms_test_var_data = 1;
- static int kallsyms_test_func_static(void)
- {
- kallsyms_test_var_bss_static++;
- kallsyms_test_var_data_static++;
- return 0;
- }
- int kallsyms_test_func(void)
- {
- return kallsyms_test_func_static();
- }
- __weak int kallsyms_test_func_weak(void)
- {
- kallsyms_test_var_bss++;
- kallsyms_test_var_data++;
- return 0;
- }
- static struct test_item test_items[] = {
- ITEM_FUNC(kallsyms_test_func_static),
- ITEM_FUNC(kallsyms_test_func),
- ITEM_FUNC(kallsyms_test_func_weak),
- ITEM_FUNC(vmalloc_noprof),
- ITEM_FUNC(vfree),
- #ifdef CONFIG_KALLSYMS_ALL
- ITEM_DATA(kallsyms_test_var_bss_static),
- ITEM_DATA(kallsyms_test_var_data_static),
- ITEM_DATA(kallsyms_test_var_bss),
- ITEM_DATA(kallsyms_test_var_data),
- #endif
- };
- static char stub_name[KSYM_NAME_LEN];
- static int stat_symbol_len(void *data, const char *name, unsigned long addr)
- {
- *(u32 *)data += strlen(name);
- return 0;
- }
- static void test_kallsyms_compression_ratio(void)
- {
- u32 pos, off, len, num;
- u32 ratio, total_size, total_len = 0;
- kallsyms_on_each_symbol(stat_symbol_len, &total_len);
- /*
- * A symbol name cannot start with a number. This stub name helps us
- * traverse the entire symbol table without finding a match. It's used
- * for subsequent performance tests, and its length is the average
- * length of all symbol names.
- */
- memset(stub_name, '4', sizeof(stub_name));
- pos = total_len / kallsyms_num_syms;
- stub_name[pos] = 0;
- pos = 0;
- num = 0;
- off = 0;
- while (pos < kallsyms_num_syms) {
- len = kallsyms_names[off];
- num++;
- off++;
- pos++;
- if ((len & 0x80) != 0) {
- len = (len & 0x7f) | (kallsyms_names[off] << 7);
- num++;
- off++;
- }
- off += len;
- }
- /*
- * 1. The length fields is not counted
- * 2. The memory occupied by array kallsyms_token_table[] and
- * kallsyms_token_index[] needs to be counted.
- */
- total_size = off - num;
- pos = kallsyms_token_index[0xff];
- total_size += pos + strlen(&kallsyms_token_table[pos]) + 1;
- total_size += 0x100 * sizeof(u16);
- pr_info(" ---------------------------------------------------------\n");
- pr_info("| nr_symbols | compressed size | original size | ratio(%%) |\n");
- pr_info("|---------------------------------------------------------|\n");
- ratio = (u32)div_u64(10000ULL * total_size, total_len);
- pr_info("| %10d | %10d | %10d | %2d.%-2d |\n",
- kallsyms_num_syms, total_size, total_len, ratio / 100, ratio % 100);
- pr_info(" ---------------------------------------------------------\n");
- }
- static int lookup_name(void *data, const char *name, unsigned long addr)
- {
- u64 t0, t1, t;
- struct test_stat *stat = (struct test_stat *)data;
- t0 = ktime_get_ns();
- (void)kallsyms_lookup_name(name);
- t1 = ktime_get_ns();
- t = t1 - t0;
- if (t < stat->min)
- stat->min = t;
- if (t > stat->max)
- stat->max = t;
- stat->real_cnt++;
- stat->sum += t;
- return 0;
- }
- static void test_perf_kallsyms_lookup_name(void)
- {
- struct test_stat stat;
- memset(&stat, 0, sizeof(stat));
- stat.min = INT_MAX;
- kallsyms_on_each_symbol(lookup_name, &stat);
- pr_info("kallsyms_lookup_name() looked up %d symbols\n", stat.real_cnt);
- pr_info("The time spent on each symbol is (ns): min=%d, max=%d, avg=%lld\n",
- stat.min, stat.max, div_u64(stat.sum, stat.real_cnt));
- }
- static int find_symbol(void *data, const char *name, unsigned long addr)
- {
- struct test_stat *stat = (struct test_stat *)data;
- if (!strcmp(name, stat->name)) {
- stat->real_cnt++;
- stat->addr = addr;
- if (stat->save_cnt < MAX_NUM_OF_RECORDS) {
- stat->addrs[stat->save_cnt] = addr;
- stat->save_cnt++;
- }
- if (stat->real_cnt == stat->max)
- return 1;
- }
- return 0;
- }
- static void test_perf_kallsyms_on_each_symbol(void)
- {
- u64 t0, t1;
- struct test_stat stat;
- memset(&stat, 0, sizeof(stat));
- stat.max = INT_MAX;
- stat.name = stub_name;
- stat.perf = 1;
- t0 = ktime_get_ns();
- kallsyms_on_each_symbol(find_symbol, &stat);
- t1 = ktime_get_ns();
- pr_info("kallsyms_on_each_symbol() traverse all: %lld ns\n", t1 - t0);
- }
- static int match_symbol(void *data, unsigned long addr)
- {
- struct test_stat *stat = (struct test_stat *)data;
- stat->real_cnt++;
- stat->addr = addr;
- if (stat->save_cnt < MAX_NUM_OF_RECORDS) {
- stat->addrs[stat->save_cnt] = addr;
- stat->save_cnt++;
- }
- if (stat->real_cnt == stat->max)
- return 1;
- return 0;
- }
- static void test_perf_kallsyms_on_each_match_symbol(void)
- {
- u64 t0, t1;
- struct test_stat stat;
- memset(&stat, 0, sizeof(stat));
- stat.max = INT_MAX;
- stat.name = stub_name;
- t0 = ktime_get_ns();
- kallsyms_on_each_match_symbol(match_symbol, stat.name, &stat);
- t1 = ktime_get_ns();
- pr_info("kallsyms_on_each_match_symbol() traverse all: %lld ns\n", t1 - t0);
- }
- static int test_kallsyms_basic_function(void)
- {
- int i, j, ret;
- int next = 0, nr_failed = 0;
- char *prefix;
- unsigned short rand;
- unsigned long addr, lookup_addr;
- char namebuf[KSYM_NAME_LEN];
- struct test_stat *stat, *stat2;
- stat = kmalloc(sizeof(*stat) * 2, GFP_KERNEL);
- if (!stat)
- return -ENOMEM;
- stat2 = stat + 1;
- prefix = "kallsyms_lookup_name() for";
- for (i = 0; i < ARRAY_SIZE(test_items); i++) {
- addr = kallsyms_lookup_name(test_items[i].name);
- if (addr != test_items[i].addr) {
- nr_failed++;
- pr_info("%s %s failed: addr=%lx, expect %lx\n",
- prefix, test_items[i].name, addr, test_items[i].addr);
- }
- }
- prefix = "kallsyms_on_each_symbol() for";
- for (i = 0; i < ARRAY_SIZE(test_items); i++) {
- memset(stat, 0, sizeof(*stat));
- stat->max = INT_MAX;
- stat->name = test_items[i].name;
- kallsyms_on_each_symbol(find_symbol, stat);
- if (stat->addr != test_items[i].addr || stat->real_cnt != 1) {
- nr_failed++;
- pr_info("%s %s failed: count=%d, addr=%lx, expect %lx\n",
- prefix, test_items[i].name,
- stat->real_cnt, stat->addr, test_items[i].addr);
- }
- }
- prefix = "kallsyms_on_each_match_symbol() for";
- for (i = 0; i < ARRAY_SIZE(test_items); i++) {
- memset(stat, 0, sizeof(*stat));
- stat->max = INT_MAX;
- stat->name = test_items[i].name;
- kallsyms_on_each_match_symbol(match_symbol, test_items[i].name, stat);
- if (stat->addr != test_items[i].addr || stat->real_cnt != 1) {
- nr_failed++;
- pr_info("%s %s failed: count=%d, addr=%lx, expect %lx\n",
- prefix, test_items[i].name,
- stat->real_cnt, stat->addr, test_items[i].addr);
- }
- }
- if (nr_failed) {
- kfree(stat);
- return -ESRCH;
- }
- for (i = 0; i < kallsyms_num_syms; i++) {
- addr = kallsyms_sym_address(i);
- if (!is_ksym_addr(addr))
- continue;
- ret = lookup_symbol_name(addr, namebuf);
- if (unlikely(ret)) {
- namebuf[0] = 0;
- pr_info("%d: lookup_symbol_name(%lx) failed\n", i, addr);
- goto failed;
- }
- lookup_addr = kallsyms_lookup_name(namebuf);
- memset(stat, 0, sizeof(*stat));
- stat->max = INT_MAX;
- kallsyms_on_each_match_symbol(match_symbol, namebuf, stat);
- /*
- * kallsyms_on_each_symbol() is too slow, randomly select some
- * symbols for test.
- */
- if (i >= next) {
- memset(stat2, 0, sizeof(*stat2));
- stat2->max = INT_MAX;
- stat2->name = namebuf;
- kallsyms_on_each_symbol(find_symbol, stat2);
- /*
- * kallsyms_on_each_symbol() and kallsyms_on_each_match_symbol()
- * need to get the same traversal result.
- */
- if (stat->addr != stat2->addr ||
- stat->real_cnt != stat2->real_cnt ||
- memcmp(stat->addrs, stat2->addrs,
- stat->save_cnt * sizeof(stat->addrs[0]))) {
- pr_info("%s: mismatch between kallsyms_on_each_symbol() and kallsyms_on_each_match_symbol()\n",
- namebuf);
- goto failed;
- }
- /*
- * The average of random increments is 128, that is, one of
- * them is tested every 128 symbols.
- */
- get_random_bytes(&rand, sizeof(rand));
- next = i + (rand & 0xff) + 1;
- }
- /* Need to be found at least once */
- if (!stat->real_cnt) {
- pr_info("%s: Never found\n", namebuf);
- goto failed;
- }
- /*
- * kallsyms_lookup_name() returns the address of the first
- * symbol found and cannot be NULL.
- */
- if (!lookup_addr) {
- pr_info("%s: NULL lookup_addr?!\n", namebuf);
- goto failed;
- }
- if (lookup_addr != stat->addrs[0]) {
- pr_info("%s: lookup_addr != stat->addrs[0]\n", namebuf);
- goto failed;
- }
- /*
- * If the addresses of all matching symbols are recorded, the
- * target address needs to be exist.
- */
- if (stat->real_cnt <= MAX_NUM_OF_RECORDS) {
- for (j = 0; j < stat->save_cnt; j++) {
- if (stat->addrs[j] == addr)
- break;
- }
- if (j == stat->save_cnt) {
- pr_info("%s: j == save_cnt?!\n", namebuf);
- goto failed;
- }
- }
- }
- kfree(stat);
- return 0;
- failed:
- pr_info("Test for %dth symbol failed: (%s) addr=%lx", i, namebuf, addr);
- kfree(stat);
- return -ESRCH;
- }
- static int test_entry(void *p)
- {
- int ret;
- do {
- schedule_timeout(5 * HZ);
- } while (system_state != SYSTEM_RUNNING);
- pr_info("start\n");
- ret = test_kallsyms_basic_function();
- if (ret) {
- pr_info("abort\n");
- return 0;
- }
- test_kallsyms_compression_ratio();
- test_perf_kallsyms_lookup_name();
- test_perf_kallsyms_on_each_symbol();
- test_perf_kallsyms_on_each_match_symbol();
- pr_info("finish\n");
- return 0;
- }
- static int __init kallsyms_test_init(void)
- {
- struct task_struct *t;
- t = kthread_create(test_entry, NULL, "kallsyms_test");
- if (IS_ERR(t)) {
- pr_info("Create kallsyms selftest task failed\n");
- return PTR_ERR(t);
- }
- kthread_bind(t, 0);
- wake_up_process(t);
- return 0;
- }
- late_initcall(kallsyms_test_init);
|