Message ID | 20181113112745.15945-1-vincent.whitchurch@axis.com (mailing list archive) |
---|---|
State | New, archived |
Headers | show |
Series | [v3] ARM: module: Fix function kallsyms on Thumb-2 | expand |
On Tue, Nov 13, 2018 at 12:27:45PM +0100, Vincent Whitchurch wrote: > Thumb-2 functions have the lowest bit set in the symbol value in the > symtab. When kallsyms are generated for the vmlinux, the kallsyms are > generated from the output of nm, and nm clears the lowest bit. > > $ arm-linux-gnueabihf-readelf -a vmlinux | grep show_interrupts > 95947: 8015dc89 686 FUNC GLOBAL DEFAULT 2 show_interrupts > $ arm-linux-gnueabihf-nm vmlinux | grep show_interrupts > 8015dc88 T show_interrupts > $ cat /proc/kallsyms | grep show_interrupts > 8015dc88 T show_interrupts > > However, for modules, the kallsyms uses the values in the symbol table > without modification, so for functions in modules, the lowest bit is set > in kallsyms. > > $ arm-linux-gnueabihf-readelf -a drivers/net/tun.ko | grep tun_get_socket > 268: 000000e1 44 FUNC GLOBAL DEFAULT 2 tun_get_socket > $ arm-linux-gnueabihf-nm drivers/net/tun.ko | grep tun_get_socket > 000000e0 T tun_get_socket > $ cat /proc/kallsyms | grep tun_get_socket > 7fcd30e1 t tun_get_socket [tun] > > Because of this, the offset of the crashing instruction shown in oopses > is incorrect when the crash is in a module. For example, given a > tun_get_socket which starts like this, > > 000000e0 <tun_get_socket>: > e0: b500 push {lr} > e2: f7ff fffe bl 0 <__gnu_mcount_nc> > e6: 4b08 ldr r3, [pc, #32] > e8: 6942 ldr r2, [r0, #20] > ea: 429a cmp r2, r3 > ec: d002 beq.n f4 <tun_get_socket+0x14> > > a crash when tun_get_socket is called with NULL results in: > > PC is at tun_get_socket+0x7/0x2c [tun] > pc : [<7fcdb0e8>] > > which can result in the incorrect line being reported by gdb if this > symbol+offset is used there. If the crash is on the first instruction > of a function, the "PC is at" line would also report the symbol name of > the preceding function. > > To solve this, introduce a weak module_kallsyms_symbol_value() function > which arches can override to fix up these symbol values, and implement (Jumping into this patch without having reviewed previous versions in detail, so I may have misunderstood some things...) Anyway: I prefer this to the previous approach of directly hacking the symbol values in the module kallsyms table. > this for Thumb-2. We need to move elf_type() to st_other so that the > original value of st_info is preserved. Are you sure modifying st_other won't break anything? It's hard to imagine how ELF symbol visibility would be relevant in the kernel, but some arches put other stuff in st_other. See alpha, powerpc, sh for example. I haven't attempted to understand whether any of those uses matters here. Ideally, we wouldn't abuse st_info to store elf_type() in the first place, but that may be messier to solve. kallsyms could wrap the ElfXX_Sym in another struct to store extra metadata for example, but it would increase runtime memory consumption. Another option would be wedge STT_FUNC in bit 7 of st_info, say. Since elf_type() always yields an ASCII char, that bit shouldn't be used for anything today. We would of course need to wrap further access to st_info to mask that bit off as appropriate. Cheers ---Dave [...] > diff --git a/kernel/module.c b/kernel/module.c > index 49a405891587..5a588cfbb8f8 100644 > --- a/kernel/module.c > +++ b/kernel/module.c > @@ -2682,7 +2682,7 @@ static void add_kallsyms(struct module *mod, const struct load_info *info) > > /* Set types up while we still have access to sections. */ > for (i = 0; i < mod->kallsyms->num_symtab; i++) > - mod->kallsyms->symtab[i].st_info > + mod->kallsyms->symtab[i].st_other > = elf_type(&mod->kallsyms->symtab[i], info); > > /* Now populate the cut down core kallsyms for after init. */ > @@ -3916,6 +3916,11 @@ static const char *symname(struct mod_kallsyms *kallsyms, unsigned int symnum) > return kallsyms->strtab + kallsyms->symtab[symnum].st_name; > } > > +unsigned long __weak module_kallsyms_symbol_value(Elf_Sym *sym) > +{ > + return sym->st_value; > +} > + Can we make this a header inline that the arch can override? Otherwise, we add a little overhead to every arch in kallsyms core loops, just to support the arm Thumb-2 kernel case. [...] Cheers ---Dave
+++ Dave Martin [13/11/18 13:57 +0000]: >On Tue, Nov 13, 2018 at 12:27:45PM +0100, Vincent Whitchurch wrote: >> Thumb-2 functions have the lowest bit set in the symbol value in the >> symtab. When kallsyms are generated for the vmlinux, the kallsyms are >> generated from the output of nm, and nm clears the lowest bit. >> >> $ arm-linux-gnueabihf-readelf -a vmlinux | grep show_interrupts >> 95947: 8015dc89 686 FUNC GLOBAL DEFAULT 2 show_interrupts >> $ arm-linux-gnueabihf-nm vmlinux | grep show_interrupts >> 8015dc88 T show_interrupts >> $ cat /proc/kallsyms | grep show_interrupts >> 8015dc88 T show_interrupts >> >> However, for modules, the kallsyms uses the values in the symbol table >> without modification, so for functions in modules, the lowest bit is set >> in kallsyms. >> >> $ arm-linux-gnueabihf-readelf -a drivers/net/tun.ko | grep tun_get_socket >> 268: 000000e1 44 FUNC GLOBAL DEFAULT 2 tun_get_socket >> $ arm-linux-gnueabihf-nm drivers/net/tun.ko | grep tun_get_socket >> 000000e0 T tun_get_socket >> $ cat /proc/kallsyms | grep tun_get_socket >> 7fcd30e1 t tun_get_socket [tun] >> >> Because of this, the offset of the crashing instruction shown in oopses >> is incorrect when the crash is in a module. For example, given a >> tun_get_socket which starts like this, >> >> 000000e0 <tun_get_socket>: >> e0: b500 push {lr} >> e2: f7ff fffe bl 0 <__gnu_mcount_nc> >> e6: 4b08 ldr r3, [pc, #32] >> e8: 6942 ldr r2, [r0, #20] >> ea: 429a cmp r2, r3 >> ec: d002 beq.n f4 <tun_get_socket+0x14> >> >> a crash when tun_get_socket is called with NULL results in: >> >> PC is at tun_get_socket+0x7/0x2c [tun] >> pc : [<7fcdb0e8>] >> >> which can result in the incorrect line being reported by gdb if this >> symbol+offset is used there. If the crash is on the first instruction >> of a function, the "PC is at" line would also report the symbol name of >> the preceding function. >> >> To solve this, introduce a weak module_kallsyms_symbol_value() function >> which arches can override to fix up these symbol values, and implement > >(Jumping into this patch without having reviewed previous versions in >detail, so I may have misunderstood some things...) > > >Anyway: > >I prefer this to the previous approach of directly hacking the symbol >values in the module kallsyms table. > >> this for Thumb-2. We need to move elf_type() to st_other so that the >> original value of st_info is preserved. > >Are you sure modifying st_other won't break anything? > >It's hard to imagine how ELF symbol visibility would be relevant in the >kernel, but some arches put other stuff in st_other. See alpha, >powerpc, sh for example. I haven't attempted to understand whether any >of those uses matters here. Yeah, the st_other field is used to apply relocations in those arches you mention. Overwriting st_info/st_other only "works" here in the sense that that field is no longer needed after applying relocations (add_kallsyms() is called in post_relocation() in the module loader). I agree it's hacky to reuse the field in that way, though. >Ideally, we wouldn't abuse st_info to store elf_type() in the first >place, but that may be messier to solve. kallsyms could wrap the >ElfXX_Sym in another struct to store extra metadata for example, but it >would increase runtime memory consumption. Yeah, I've always thought that was ugly. I think it was done as a small optimization for kallsyms, so that we're not looking up what char to print for every symbol, so the st_info field was repurposed for this. >Another option would be wedge STT_FUNC in bit 7 of st_info, say. >Since elf_type() always yields an ASCII char, that bit shouldn't be >used for anything today. We would of course need to wrap further >access to st_info to mask that bit off as appropriate. Hm, actually on second thought, I don't think the st_size field is used *anywhere* in the module loader, not in the arch-specific relocation code nor in kallsyms. Perhaps that field could be used to store the output of elf_type instead. Then we won't run into any issues with delayed relocations with livepatch, as the st_size field isn't used at all for applying relocs anyway. Since this field is not used at runtime, I think we can use st_size instead of st_other or st_info, which are two fields that some arches do use to apply relocs. Would be great if someone could confirm/fact-check me on this. Thanks! Jessica
diff --git a/arch/arm/kernel/module.c b/arch/arm/kernel/module.c index 3ff571c2c71c..89ab84a83600 100644 --- a/arch/arm/kernel/module.c +++ b/arch/arm/kernel/module.c @@ -336,6 +336,16 @@ static const Elf_Shdr *find_mod_section(const Elf32_Ehdr *hdr, extern void fixup_pv_table(const void *, unsigned long); extern void fixup_smp(const void *, unsigned long); +#ifdef CONFIG_THUMB2_KERNEL +unsigned long module_kallsyms_symbol_value(Elf_Sym *sym) +{ + if (ELF_ST_TYPE(sym->st_info) == STT_FUNC) + return sym->st_value & ~1; + + return sym->st_value; +} +#endif + int module_finalize(const Elf32_Ehdr *hdr, const Elf_Shdr *sechdrs, struct module *mod) { diff --git a/include/linux/moduleloader.h b/include/linux/moduleloader.h index 31013c2effd3..6395409b01a4 100644 --- a/include/linux/moduleloader.h +++ b/include/linux/moduleloader.h @@ -86,6 +86,8 @@ void module_arch_cleanup(struct module *mod); /* Any cleanup before freeing mod->module_init */ void module_arch_freeing_init(struct module *mod); +unsigned long module_kallsyms_symbol_value(Elf_Sym *sym); + #ifdef CONFIG_KASAN #include <linux/kasan.h> #define MODULE_ALIGN (PAGE_SIZE << KASAN_SHADOW_SCALE_SHIFT) diff --git a/kernel/module.c b/kernel/module.c index 49a405891587..5a588cfbb8f8 100644 --- a/kernel/module.c +++ b/kernel/module.c @@ -2682,7 +2682,7 @@ static void add_kallsyms(struct module *mod, const struct load_info *info) /* Set types up while we still have access to sections. */ for (i = 0; i < mod->kallsyms->num_symtab; i++) - mod->kallsyms->symtab[i].st_info + mod->kallsyms->symtab[i].st_other = elf_type(&mod->kallsyms->symtab[i], info); /* Now populate the cut down core kallsyms for after init. */ @@ -3916,6 +3916,11 @@ static const char *symname(struct mod_kallsyms *kallsyms, unsigned int symnum) return kallsyms->strtab + kallsyms->symtab[symnum].st_name; } +unsigned long __weak module_kallsyms_symbol_value(Elf_Sym *sym) +{ + return sym->st_value; +} + static const char *get_ksymbol(struct module *mod, unsigned long addr, unsigned long *size, @@ -3934,6 +3939,9 @@ static const char *get_ksymbol(struct module *mod, /* Scan for closest preceding symbol, and next symbol. (ELF starts real symbols at 1). */ for (i = 1; i < kallsyms->num_symtab; i++) { + unsigned long thisval = module_kallsyms_symbol_value(&kallsyms->symtab[i]); + unsigned long bestval = module_kallsyms_symbol_value(&kallsyms->symtab[best]); + if (kallsyms->symtab[i].st_shndx == SHN_UNDEF) continue; @@ -3943,21 +3951,21 @@ static const char *get_ksymbol(struct module *mod, || is_arm_mapping_symbol(symname(kallsyms, i))) continue; - if (kallsyms->symtab[i].st_value <= addr - && kallsyms->symtab[i].st_value > kallsyms->symtab[best].st_value) + if (thisval <= addr + && thisval > bestval) best = i; - if (kallsyms->symtab[i].st_value > addr - && kallsyms->symtab[i].st_value < nextval) - nextval = kallsyms->symtab[i].st_value; + if (thisval > addr + && thisval < nextval) + nextval = thisval; } if (!best) return NULL; if (size) - *size = nextval - kallsyms->symtab[best].st_value; + *size = nextval - module_kallsyms_symbol_value(&kallsyms->symtab[best]); if (offset) - *offset = addr - kallsyms->symtab[best].st_value; + *offset = addr - module_kallsyms_symbol_value(&kallsyms->symtab[best]); return symname(kallsyms, best); } @@ -4060,8 +4068,8 @@ int module_get_kallsym(unsigned int symnum, unsigned long *value, char *type, continue; kallsyms = rcu_dereference_sched(mod->kallsyms); if (symnum < kallsyms->num_symtab) { - *value = kallsyms->symtab[symnum].st_value; - *type = kallsyms->symtab[symnum].st_info; + *value = module_kallsyms_symbol_value(&kallsyms->symtab[symnum]); + *type = kallsyms->symtab[symnum].st_other; strlcpy(name, symname(kallsyms, symnum), KSYM_NAME_LEN); strlcpy(module_name, mod->name, MODULE_NAME_LEN); *exported = is_exported(name, *value, mod); @@ -4082,7 +4090,7 @@ static unsigned long mod_find_symname(struct module *mod, const char *name) for (i = 0; i < kallsyms->num_symtab; i++) if (strcmp(name, symname(kallsyms, i)) == 0 && kallsyms->symtab[i].st_shndx != SHN_UNDEF) - return kallsyms->symtab[i].st_value; + return module_kallsyms_symbol_value(&kallsyms->symtab[i]); return 0; } @@ -4131,8 +4139,8 @@ int module_kallsyms_on_each_symbol(int (*fn)(void *, const char *, if (kallsyms->symtab[i].st_shndx == SHN_UNDEF) continue; - ret = fn(data, symname(kallsyms, i), - mod, kallsyms->symtab[i].st_value); + ret = fn(data, symname(kallsyms, i), mod, + module_kallsyms_symbol_value(&kallsyms->symtab[i])); if (ret != 0) return ret; }
Thumb-2 functions have the lowest bit set in the symbol value in the symtab. When kallsyms are generated for the vmlinux, the kallsyms are generated from the output of nm, and nm clears the lowest bit. $ arm-linux-gnueabihf-readelf -a vmlinux | grep show_interrupts 95947: 8015dc89 686 FUNC GLOBAL DEFAULT 2 show_interrupts $ arm-linux-gnueabihf-nm vmlinux | grep show_interrupts 8015dc88 T show_interrupts $ cat /proc/kallsyms | grep show_interrupts 8015dc88 T show_interrupts However, for modules, the kallsyms uses the values in the symbol table without modification, so for functions in modules, the lowest bit is set in kallsyms. $ arm-linux-gnueabihf-readelf -a drivers/net/tun.ko | grep tun_get_socket 268: 000000e1 44 FUNC GLOBAL DEFAULT 2 tun_get_socket $ arm-linux-gnueabihf-nm drivers/net/tun.ko | grep tun_get_socket 000000e0 T tun_get_socket $ cat /proc/kallsyms | grep tun_get_socket 7fcd30e1 t tun_get_socket [tun] Because of this, the offset of the crashing instruction shown in oopses is incorrect when the crash is in a module. For example, given a tun_get_socket which starts like this, 000000e0 <tun_get_socket>: e0: b500 push {lr} e2: f7ff fffe bl 0 <__gnu_mcount_nc> e6: 4b08 ldr r3, [pc, #32] e8: 6942 ldr r2, [r0, #20] ea: 429a cmp r2, r3 ec: d002 beq.n f4 <tun_get_socket+0x14> a crash when tun_get_socket is called with NULL results in: PC is at tun_get_socket+0x7/0x2c [tun] pc : [<7fcdb0e8>] which can result in the incorrect line being reported by gdb if this symbol+offset is used there. If the crash is on the first instruction of a function, the "PC is at" line would also report the symbol name of the preceding function. To solve this, introduce a weak module_kallsyms_symbol_value() function which arches can override to fix up these symbol values, and implement this for Thumb-2. We need to move elf_type() to st_other so that the original value of st_info is preserved. After the fix: $ cat /proc/kallsyms | grep tun_get_socket 7fcd30e0 t tun_get_socket [tun] PC is at tun_get_socket+0x8/0x2c [tun] pc : [<7fcdb0e8>] Signed-off-by: Vincent Whitchurch <vincent.whitchurch@axis.com> --- v3: Do not overwrite st_value v2: Fix build warning with !MODULES arch/arm/kernel/module.c | 10 ++++++++++ include/linux/moduleloader.h | 2 ++ kernel/module.c | 34 +++++++++++++++++++++------------- 3 files changed, 33 insertions(+), 13 deletions(-)