mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/next/linux-next.git
synced 2025-01-01 10:42:11 +00:00
daa22f5a78
Summary of the changes worth highlighting from most interesting to boring below: * Christoph Hellwig's symbol_get() fix to Nvidia's efforts to circumvent the protection he put in place in year 2020 to prevent proprietary modules from using GPL only symbols, and also ensuring proprietary modules which export symbols grandfather their taint. That was done through year 2020 commit262e6ae708
("modules: inherit TAINT_PROPRIETARY_MODULE"). Christoph's new fix is done by clarifing __symbol_get() was only ever intended to prevent module reference loops by Linux kernel modules and so making it only find symbols exported via EXPORT_SYMBOL_GPL(). The circumvention tactic used by Nvidia was to use symbol_get() to purposely swift through proprietary module symbols and completley bypass our traditional EXPORT_SYMBOL*() annotations and community agreed upon restrictions. A small set of preamble patches fix up a few symbols which just needed adjusting for this on two modules, the rtc ds1685 and the networking enetc module. Two other modules just needed some build fixing and removal of use of __symbol_get() as they can't ever be modular, as was done by Arnd on the ARM pxa module and Christoph did on the mmc au1xmmc driver. This is a good reminder to us that symbol_get() is just a hack to address things which should be fixed through Kconfig at build time as was done in the later patches, and so ultimately it should just go. * Extremely late minor fix for old module layout055f23b74b
("module: check for exit sections in layout_sections() instead of module_init_section()") by James Morse for arm64. Note that this layout thing is old, it is *not* Song Liu's commitac3b432839
("module: replace module_layout with module_memory"). The issue however is very odd to run into and so there was no hurry to get this in fast. * Although the fix did not go through the modules tree I'd like to highlight the fix by Peter Zijlstra in commit5409730962
("x86/static_call: Fix __static_call_fixup()") now merged in your tree which came out of what was originally suspected to be a fallout of the the newer module layout changes by Song Liu commitac3b432839
("module: replace module_layout with module_memory") instead of module_init_section()"). Thanks to the report by Christian Bricart and the debugging by Song Liu & Peter that turned to be noted as a kernel regression in place since v5.19 through commitee88d363d1
("x86,static_call: Use alternative RET encoding"). I highlight this to reflect and clarify that we haven't seen more fallout fromac3b432839
("module: replace module_layout with module_memory"). * RISC-V toolchain got mapping symbol support which prefix symbols with "$" to help with alignment considerations for disassembly. This is used to differentiate between incompatible instruction encodings when disassembling. RISC-V just matches what ARM/AARCH64 did for alignment considerations and Palmer Dabbelt extended is_mapping_symbol() to accept these symbols for RISC-V. We already had support for this for all architectures but it also checked for the second character, the RISC-V check Dabbelt added was just for the "$". After a bit of testing and fallout on linux-next and based on feedback from Masahiro Yamada it was decided to simplify the check and treat the first char "$" as unique for all architectures, and so we no make is_mapping_symbol() for all archs if the symbol starts with "$". The most relevant commit for this for RISC-V on binutils was: https://sourceware.org/pipermail/binutils/2021-July/117350.html * A late fix by Andrea Righi (today) to make module zstd decompression use vmalloc() instead of kmalloc() to account for large compressed modules. I suspect we'll see similar things for other decompression algorithms soon. * samples/hw_breakpoint minor fixes by Rong Tao, Arnd Bergmann and Chen Jiahao -----BEGIN PGP SIGNATURE----- iQJGBAABCgAwFiEENnNq2KuOejlQLZofziMdCjCSiKcFAmTuShISHG1jZ3JvZkBr ZXJuZWwub3JnAAoJEM4jHQowkoin7rEQAIt9cGmkHyA6Po/Ex8DejWvSTTOQzIXk NvtGurODghWnCejZ7Yofo1T48mvgHOenDQB9qNSkVtKDyhmWCbss6wQU/5M8Mc3A G+9svkQ8H1BRzTwX3WJKF9KNMhI0HA0CXz3ED/I4iX/Q4Ffv3bgbAiitY6r48lJV PSKPzwH9QMIti6k3j+bFf2SwWCV3X2jz+btdxwY34dVFyggdYgaBNKEdrumCx4nL g0tQQxI8QgltOnwlfOPLEhdSU1yWyIWZtqtki6xksLziwTreRaw1HotgXQDpnt/S iJY9xiKN1ChtVSprQlbTb9yhFbCEGvOYGEaKl/ZsGENQjKzRWsQ+dtT8Ww6n2Y1H aJXwniv6SqCW7dCwdKo4sE7JFYDP56yFYKBLOPSPbMm6DJwTMbzLUf7TGNh6NKyl 3pqjGagJ+LTj3l9w5ur4zTrDGAmLzMpNR03+6niTM7C3TPOI1+wh5zGbvtoA/WdA ytQeOTiUsi0uyVgk50f67IC6virrxwupeyZQlYFGNuEGBClgXzzzgw/MKwg0VMvc aWhFPUOLx8/8juJ3A5qiOT+znQJ2DTqWlT+QkQ8R5qFVXEW1g9IOnhaHqDX+KB0A OPlZ9xwss2U0Zd1XhourtqhUhvcODWNzTj3oPzjdrGiBjdENz8hPKP+7HV1CG6xy RdxpSwu72kFu =IQy2 -----END PGP SIGNATURE----- Merge tag 'modules-6.6-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/mcgrof/linux Pull modules updates from Luis Chamberlain: "Summary of the changes worth highlighting from most interesting to boring below: - Christoph Hellwig's symbol_get() fix to Nvidia's efforts to circumvent the protection he put in place in year 2020 to prevent proprietary modules from using GPL only symbols, and also ensuring proprietary modules which export symbols grandfather their taint. That was done through year 2020 commit262e6ae708
("modules: inherit TAINT_PROPRIETARY_MODULE"). Christoph's new fix is done by clarifing __symbol_get() was only ever intended to prevent module reference loops by Linux kernel modules and so making it only find symbols exported via EXPORT_SYMBOL_GPL(). The circumvention tactic used by Nvidia was to use symbol_get() to purposely swift through proprietary module symbols and completely bypass our traditional EXPORT_SYMBOL*() annotations and community agreed upon restrictions. A small set of preamble patches fix up a few symbols which just needed adjusting for this on two modules, the rtc ds1685 and the networking enetc module. Two other modules just needed some build fixing and removal of use of __symbol_get() as they can't ever be modular, as was done by Arnd on the ARM pxa module and Christoph did on the mmc au1xmmc driver. This is a good reminder to us that symbol_get() is just a hack to address things which should be fixed through Kconfig at build time as was done in the later patches, and so ultimately it should just go. - Extremely late minor fix for old module layout055f23b74b
("module: check for exit sections in layout_sections() instead of module_init_section()") by James Morse for arm64. Note that this layout thing is old, it is *not* Song Liu's commitac3b432839
("module: replace module_layout with module_memory"). The issue however is very odd to run into and so there was no hurry to get this in fast. - Although the fix did not go through the modules tree I'd like to highlight the fix by Peter Zijlstra in commit5409730962
("x86/static_call: Fix __static_call_fixup()") now merged in your tree which came out of what was originally suspected to be a fallout of the the newer module layout changes by Song Liu commitac3b432839
("module: replace module_layout with module_memory") instead of module_init_section()"). Thanks to the report by Christian Bricart and the debugging by Song Liu & Peter that turned to be noted as a kernel regression in place since v5.19 through commitee88d363d1
("x86,static_call: Use alternative RET encoding"). I highlight this to reflect and clarify that we haven't seen more fallout fromac3b432839
("module: replace module_layout with module_memory"). - RISC-V toolchain got mapping symbol support which prefix symbols with "$" to help with alignment considerations for disassembly. This is used to differentiate between incompatible instruction encodings when disassembling. RISC-V just matches what ARM/AARCH64 did for alignment considerations and Palmer Dabbelt extended is_mapping_symbol() to accept these symbols for RISC-V. We already had support for this for all architectures but it also checked for the second character, the RISC-V check Dabbelt added was just for the "$". After a bit of testing and fallout on linux-next and based on feedback from Masahiro Yamada it was decided to simplify the check and treat the first char "$" as unique for all architectures, and so we no make is_mapping_symbol() for all archs if the symbol starts with "$". The most relevant commit for this for RISC-V on binutils was: https://sourceware.org/pipermail/binutils/2021-July/117350.html - A late fix by Andrea Righi (today) to make module zstd decompression use vmalloc() instead of kmalloc() to account for large compressed modules. I suspect we'll see similar things for other decompression algorithms soon. - samples/hw_breakpoint minor fixes by Rong Tao, Arnd Bergmann and Chen Jiahao" * tag 'modules-6.6-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/mcgrof/linux: module/decompress: use vmalloc() for zstd decompression workspace kallsyms: Add more debug output for selftest ARM: module: Use module_init_layout_section() to spot init sections arm64: module: Use module_init_layout_section() to spot init sections module: Expose module_init_layout_section() modules: only allow symbol_get of EXPORT_SYMBOL_GPL modules rtc: ds1685: use EXPORT_SYMBOL_GPL for ds1685_rtc_poweroff net: enetc: use EXPORT_SYMBOL_GPL for enetc_phc_index mmc: au1xmmc: force non-modular build and remove symbol_get usage ARM: pxa: remove use of symbol_get() samples/hw_breakpoint: mark sample_hbp as static samples/hw_breakpoint: fix building without module unloading samples/hw_breakpoint: Fix kernel BUG 'invalid opcode: 0000' modpost, kallsyms: Treat add '$'-prefixed symbols as mapping symbols kernel: params: Remove unnecessary ‘0’ values from err module: Ignore RISC-V mapping symbols too
470 lines
10 KiB
C
470 lines
10 KiB
C
// 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),
|
|
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),
|
|
ITEM_DATA(vmap_area_list),
|
|
#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 bool match_cleanup_name(const char *s, const char *name)
|
|
{
|
|
char *p;
|
|
int len;
|
|
|
|
if (!IS_ENABLED(CONFIG_LTO_CLANG))
|
|
return false;
|
|
|
|
p = strstr(s, ".llvm.");
|
|
if (!p)
|
|
return false;
|
|
|
|
len = strlen(name);
|
|
if (p - s != len)
|
|
return false;
|
|
|
|
return !strncmp(s, name, len);
|
|
}
|
|
|
|
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) == 0 ||
|
|
(!stat->perf && match_cleanup_name(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);
|