Commit 93684d3b authored by Sam Ravnborg's avatar Sam Ravnborg

kbuild: include symbol names in section mismatch warnings

Try to look up the symbol that is referenced. Include the symbol
name in the warning message.
Signed-off-by: default avatarSam Ravnborg <sam@ravnborg.org>
parent 8ea80ca4
...@@ -451,6 +451,29 @@ static char *get_modinfo(void *modinfo, unsigned long modinfo_len, ...@@ -451,6 +451,29 @@ static char *get_modinfo(void *modinfo, unsigned long modinfo_len,
return NULL; return NULL;
} }
/**
* Find symbol based on relocation record info.
* In some cases the symbol supplied is a valid symbol so
* return refsym. If st_name != 0 we assume this is a valid symbol.
* In other cases the symbol needs to be looked up in the symbol table
* based on section and address.
* **/
static Elf_Sym *find_elf_symbol(struct elf_info *elf, Elf_Addr addr,
Elf_Sym *relsym)
{
Elf_Sym *sym;
if (relsym->st_name != 0)
return relsym;
for (sym = elf->symtab_start; sym < elf->symtab_stop; sym++) {
if (sym->st_shndx != relsym->st_shndx)
continue;
if (sym->st_value == addr)
return sym;
}
return NULL;
}
/* /*
* Find symbols before or equal addr and after addr - in the section sec * Find symbols before or equal addr and after addr - in the section sec
**/ **/
...@@ -499,8 +522,9 @@ static void find_symbols_between(struct elf_info *elf, Elf_Addr addr, ...@@ -499,8 +522,9 @@ static void find_symbols_between(struct elf_info *elf, Elf_Addr addr,
static void warn_sec_mismatch(const char *modname, const char *fromsec, static void warn_sec_mismatch(const char *modname, const char *fromsec,
struct elf_info *elf, Elf_Sym *sym, Elf_Rela r) struct elf_info *elf, Elf_Sym *sym, Elf_Rela r)
{ {
Elf_Sym *before; const char *refsymname = "";
Elf_Sym *after; Elf_Sym *before, *after;
Elf_Sym *refsym;
Elf_Ehdr *hdr = elf->hdr; Elf_Ehdr *hdr = elf->hdr;
Elf_Shdr *sechdrs = elf->sechdrs; Elf_Shdr *sechdrs = elf->sechdrs;
const char *secstrings = (void *)hdr + const char *secstrings = (void *)hdr +
...@@ -509,29 +533,34 @@ static void warn_sec_mismatch(const char *modname, const char *fromsec, ...@@ -509,29 +533,34 @@ static void warn_sec_mismatch(const char *modname, const char *fromsec,
find_symbols_between(elf, r.r_offset, fromsec, &before, &after); find_symbols_between(elf, r.r_offset, fromsec, &before, &after);
refsym = find_elf_symbol(elf, r.r_addend, sym);
if (refsym && strlen(elf->strtab + refsym->st_name))
refsymname = elf->strtab + refsym->st_name;
if (before && after) { if (before && after) {
warn("%s - Section mismatch: reference to %s from %s " warn("%s - Section mismatch: reference to %s:%s from %s "
"between '%s' (at offset 0x%lx) and '%s'\n", "between '%s' (at offset 0x%llx) and '%s'\n",
modname, secname, fromsec, modname, secname, refsymname, fromsec,
elf->strtab + before->st_name, elf->strtab + before->st_name,
(long)(r.r_offset - before->st_value), (long long)r.r_offset,
elf->strtab + after->st_name); elf->strtab + after->st_name);
} else if (before) { } else if (before) {
warn("%s - Section mismatch: reference to %s from %s " warn("%s - Section mismatch: reference to %s:%s from %s "
"after '%s' (at offset 0x%lx)\n", "after '%s' (at offset 0x%llx)\n",
modname, secname, fromsec, modname, secname, refsymname, fromsec,
elf->strtab + before->st_name, elf->strtab + before->st_name,
(long)(r.r_offset - before->st_value)); (long long)r.r_offset);
} else if (after) { } else if (after) {
warn("%s - Section mismatch: reference to %s from %s " warn("%s - Section mismatch: reference to %s:%s from %s "
"before '%s' (at offset -0x%lx)\n", "before '%s' (at offset -0x%llx)\n",
modname, secname, fromsec, modname, secname, refsymname, fromsec,
elf->strtab + before->st_name, elf->strtab + before->st_name,
(long)(before->st_value - r.r_offset)); (long long)r.r_offset);
} else { } else {
warn("%s - Section mismatch: reference to %s from %s " warn("%s - Section mismatch: reference to %s:%s from %s "
"(offset 0x%lx)\n", "(offset 0x%llx)\n",
modname, secname, fromsec, (long)r.r_offset); modname, secname, fromsec, refsymname,
(long long)r.r_offset);
} }
} }
...@@ -575,6 +604,7 @@ static void check_sec_ref(struct module *mod, const char *modname, ...@@ -575,6 +604,7 @@ static void check_sec_ref(struct module *mod, const char *modname,
const char *secname; const char *secname;
r.r_offset = TO_NATIVE(rela->r_offset); r.r_offset = TO_NATIVE(rela->r_offset);
r.r_info = TO_NATIVE(rela->r_info); r.r_info = TO_NATIVE(rela->r_info);
r.r_addend = TO_NATIVE(rela->r_addend);
sym = elf->symtab_start + ELF_R_SYM(r.r_info); sym = elf->symtab_start + ELF_R_SYM(r.r_info);
/* Skip special sections */ /* Skip special sections */
if (sym->st_shndx >= SHN_LORESERVE) if (sym->st_shndx >= SHN_LORESERVE)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment