kbuild: fix section mismatch check for vmlinux
vmlinux does not contain relocation entries which is used by the section mismatch checks. Reported by: Atsushi Nemoto <anemo@mba.ocn.ne.jp> Use the individual objects as inputs to overcome this limitation. In modpost check the .o files and skip non-ELF files. Signed-off-by: Sam Ravnborg <sam@ravnborg.org>
This commit is contained in:
parent
dc87c3985e
commit
85bd2fddd6
1
Makefile
1
Makefile
@ -603,6 +603,7 @@ vmlinux-init := $(head-y) $(init-y)
|
|||||||
vmlinux-main := $(core-y) $(libs-y) $(drivers-y) $(net-y)
|
vmlinux-main := $(core-y) $(libs-y) $(drivers-y) $(net-y)
|
||||||
vmlinux-all := $(vmlinux-init) $(vmlinux-main)
|
vmlinux-all := $(vmlinux-init) $(vmlinux-main)
|
||||||
vmlinux-lds := arch/$(ARCH)/kernel/vmlinux.lds
|
vmlinux-lds := arch/$(ARCH)/kernel/vmlinux.lds
|
||||||
|
export KBUILD_VMLINUX_OBJS := $(vmlinux-all)
|
||||||
|
|
||||||
# Rule to link vmlinux - also used during CONFIG_KALLSYMS
|
# Rule to link vmlinux - also used during CONFIG_KALLSYMS
|
||||||
# May be overridden by arch/$(ARCH)/Makefile
|
# May be overridden by arch/$(ARCH)/Makefile
|
||||||
|
@ -63,16 +63,16 @@ quiet_cmd_modpost = MODPOST $(words $(filter-out vmlinux FORCE, $^)) modules
|
|||||||
$(if $(KBUILD_EXTMOD),-i,-o) $(kernelsymfile) \
|
$(if $(KBUILD_EXTMOD),-i,-o) $(kernelsymfile) \
|
||||||
$(if $(KBUILD_EXTMOD),-I $(modulesymfile)) \
|
$(if $(KBUILD_EXTMOD),-I $(modulesymfile)) \
|
||||||
$(if $(KBUILD_EXTMOD),-o $(modulesymfile)) \
|
$(if $(KBUILD_EXTMOD),-o $(modulesymfile)) \
|
||||||
$(if $(KBUILD_EXTMOD)$(KBUILD_MODPOST_WARN),-w) \
|
$(if $(KBUILD_EXTMOD)$(KBUILD_MODPOST_WARN),-w)
|
||||||
$(wildcard vmlinux) $(filter-out FORCE,$^)
|
|
||||||
|
|
||||||
PHONY += __modpost
|
PHONY += __modpost
|
||||||
__modpost: $(modules:.ko=.o) FORCE
|
__modpost: $(modules:.ko=.o) FORCE
|
||||||
$(call cmd,modpost)
|
$(call cmd,modpost) $(wildcard vmlinux) $(filter-out FORCE,$^)
|
||||||
|
|
||||||
quiet_cmd_kernel-mod = MODPOST $@
|
quiet_cmd_kernel-mod = MODPOST $@
|
||||||
cmd_kernel-mod = $(cmd_modpost)
|
cmd_kernel-mod = $(cmd_modpost) $(KBUILD_VMLINUX_OBJS)
|
||||||
|
|
||||||
|
PHONY += vmlinux
|
||||||
vmlinux: FORCE
|
vmlinux: FORCE
|
||||||
$(call cmd,kernel-mod)
|
$(call cmd,kernel-mod)
|
||||||
|
|
||||||
|
@ -333,10 +333,10 @@ void release_file(void *file, unsigned long size)
|
|||||||
munmap(file, size);
|
munmap(file, size);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void parse_elf(struct elf_info *info, const char *filename)
|
static int parse_elf(struct elf_info *info, const char *filename)
|
||||||
{
|
{
|
||||||
unsigned int i;
|
unsigned int i;
|
||||||
Elf_Ehdr *hdr = info->hdr;
|
Elf_Ehdr *hdr;
|
||||||
Elf_Shdr *sechdrs;
|
Elf_Shdr *sechdrs;
|
||||||
Elf_Sym *sym;
|
Elf_Sym *sym;
|
||||||
|
|
||||||
@ -346,9 +346,18 @@ static void parse_elf(struct elf_info *info, const char *filename)
|
|||||||
exit(1);
|
exit(1);
|
||||||
}
|
}
|
||||||
info->hdr = hdr;
|
info->hdr = hdr;
|
||||||
if (info->size < sizeof(*hdr))
|
if (info->size < sizeof(*hdr)) {
|
||||||
goto truncated;
|
/* file too small, assume this is an empty .o file */
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
/* Is this a valid ELF file? */
|
||||||
|
if ((hdr->e_ident[EI_MAG0] != ELFMAG0) ||
|
||||||
|
(hdr->e_ident[EI_MAG1] != ELFMAG1) ||
|
||||||
|
(hdr->e_ident[EI_MAG2] != ELFMAG2) ||
|
||||||
|
(hdr->e_ident[EI_MAG3] != ELFMAG3)) {
|
||||||
|
/* Not an ELF file - silently ignore it */
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
/* Fix endianness in ELF header */
|
/* Fix endianness in ELF header */
|
||||||
hdr->e_shoff = TO_NATIVE(hdr->e_shoff);
|
hdr->e_shoff = TO_NATIVE(hdr->e_shoff);
|
||||||
hdr->e_shstrndx = TO_NATIVE(hdr->e_shstrndx);
|
hdr->e_shstrndx = TO_NATIVE(hdr->e_shstrndx);
|
||||||
@ -371,8 +380,10 @@ static void parse_elf(struct elf_info *info, const char *filename)
|
|||||||
= (void *)hdr + sechdrs[hdr->e_shstrndx].sh_offset;
|
= (void *)hdr + sechdrs[hdr->e_shstrndx].sh_offset;
|
||||||
const char *secname;
|
const char *secname;
|
||||||
|
|
||||||
if (sechdrs[i].sh_offset > info->size)
|
if (sechdrs[i].sh_offset > info->size) {
|
||||||
goto truncated;
|
fatal("%s is truncated. sechdrs[i].sh_offset=%u > sizeof(*hrd)=%ul\n", filename, (unsigned int)sechdrs[i].sh_offset, sizeof(*hdr));
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
secname = secstrings + sechdrs[i].sh_name;
|
secname = secstrings + sechdrs[i].sh_name;
|
||||||
if (strcmp(secname, ".modinfo") == 0) {
|
if (strcmp(secname, ".modinfo") == 0) {
|
||||||
info->modinfo = (void *)hdr + sechdrs[i].sh_offset;
|
info->modinfo = (void *)hdr + sechdrs[i].sh_offset;
|
||||||
@ -407,10 +418,7 @@ static void parse_elf(struct elf_info *info, const char *filename)
|
|||||||
sym->st_value = TO_NATIVE(sym->st_value);
|
sym->st_value = TO_NATIVE(sym->st_value);
|
||||||
sym->st_size = TO_NATIVE(sym->st_size);
|
sym->st_size = TO_NATIVE(sym->st_size);
|
||||||
}
|
}
|
||||||
return;
|
return 1;
|
||||||
|
|
||||||
truncated:
|
|
||||||
fatal("%s is truncated.\n", filename);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void parse_elf_finish(struct elf_info *info)
|
static void parse_elf_finish(struct elf_info *info)
|
||||||
@ -1089,7 +1097,8 @@ static void read_symbols(char *modname)
|
|||||||
struct elf_info info = { };
|
struct elf_info info = { };
|
||||||
Elf_Sym *sym;
|
Elf_Sym *sym;
|
||||||
|
|
||||||
parse_elf(&info, modname);
|
if (!parse_elf(&info, modname))
|
||||||
|
return;
|
||||||
|
|
||||||
mod = new_module(modname);
|
mod = new_module(modname);
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user