objtool: Get rid of reloc->list
Now that all relocs are allocated in an array, the linked list is no longer needed. With allyesconfig + CONFIG_DEBUG_INFO: - Before: peak heap memory consumption: 49.02G - After: peak heap memory consumption: 45.56G Link: https://lore.kernel.org/r/71e7a2c017dbc46bb497857ec97d67214f832d10.1685464332.git.jpoimboe@kernel.org Signed-off-by: Josh Poimboeuf <jpoimboe@kernel.org>
This commit is contained in:
parent
e0a9349b4c
commit
ebcef730a1
@ -293,7 +293,6 @@ static int read_sections(struct elf *elf)
|
|||||||
sec = &elf->section_data[i];
|
sec = &elf->section_data[i];
|
||||||
|
|
||||||
INIT_LIST_HEAD(&sec->symbol_list);
|
INIT_LIST_HEAD(&sec->symbol_list);
|
||||||
INIT_LIST_HEAD(&sec->reloc_list);
|
|
||||||
|
|
||||||
s = elf_getscn(elf->elf, i);
|
s = elf_getscn(elf->elf, i);
|
||||||
if (!s) {
|
if (!s) {
|
||||||
@ -333,7 +332,7 @@ static int read_sections(struct elf *elf)
|
|||||||
elf_hash_add(section_name, &sec->name_hash, str_hash(sec->name));
|
elf_hash_add(section_name, &sec->name_hash, str_hash(sec->name));
|
||||||
|
|
||||||
if (is_reloc_sec(sec))
|
if (is_reloc_sec(sec))
|
||||||
elf->num_relocs += sec->sh.sh_size / sec->sh.sh_entsize;
|
elf->num_relocs += sec_num_entries(sec);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (opts.stats) {
|
if (opts.stats) {
|
||||||
@ -407,7 +406,7 @@ static int read_symbols(struct elf *elf)
|
|||||||
if (symtab_shndx)
|
if (symtab_shndx)
|
||||||
shndx_data = symtab_shndx->data;
|
shndx_data = symtab_shndx->data;
|
||||||
|
|
||||||
symbols_nr = symtab->sh.sh_size / symtab->sh.sh_entsize;
|
symbols_nr = sec_num_entries(symtab);
|
||||||
} else {
|
} else {
|
||||||
/*
|
/*
|
||||||
* A missing symbol table is actually possible if it's an empty
|
* A missing symbol table is actually possible if it's an empty
|
||||||
@ -701,7 +700,7 @@ __elf_create_symbol(struct elf *elf, struct symbol *sym)
|
|||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
new_idx = symtab->sh.sh_size / symtab->sh.sh_entsize;
|
new_idx = sec_num_entries(symtab);
|
||||||
|
|
||||||
if (GELF_ST_BIND(sym->sym.st_info) != STB_LOCAL)
|
if (GELF_ST_BIND(sym->sym.st_info) != STB_LOCAL)
|
||||||
goto non_local;
|
goto non_local;
|
||||||
@ -816,13 +815,13 @@ static struct reloc *elf_init_reloc(struct elf *elf, struct section *rsec,
|
|||||||
{
|
{
|
||||||
struct reloc *reloc, empty = { 0 };
|
struct reloc *reloc, empty = { 0 };
|
||||||
|
|
||||||
if (reloc_idx >= rsec->sh.sh_size / elf_rela_size(elf)) {
|
if (reloc_idx >= sec_num_entries(rsec)) {
|
||||||
WARN("%s: bad reloc_idx %u for %s with size 0x%lx",
|
WARN("%s: bad reloc_idx %u for %s with %d relocs",
|
||||||
__func__, reloc_idx, rsec->name, rsec->sh.sh_size);
|
__func__, reloc_idx, rsec->name, sec_num_entries(rsec));
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
reloc = &rsec->reloc_data[reloc_idx];
|
reloc = &rsec->relocs[reloc_idx];
|
||||||
|
|
||||||
if (memcmp(reloc, &empty, sizeof(empty))) {
|
if (memcmp(reloc, &empty, sizeof(empty))) {
|
||||||
WARN("%s: %s: reloc %d already initialized!",
|
WARN("%s: %s: reloc %d already initialized!",
|
||||||
@ -841,7 +840,6 @@ static struct reloc *elf_init_reloc(struct elf *elf, struct section *rsec,
|
|||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
list_add_tail(&reloc->sym_reloc_entry, &sym->reloc_list);
|
list_add_tail(&reloc->sym_reloc_entry, &sym->reloc_list);
|
||||||
list_add_tail(&reloc->list, &rsec->reloc_list);
|
|
||||||
elf_hash_add(reloc, &reloc->hash, reloc_hash(reloc));
|
elf_hash_add(reloc, &reloc->hash, reloc_hash(reloc));
|
||||||
|
|
||||||
return reloc;
|
return reloc;
|
||||||
@ -944,14 +942,13 @@ static int read_relocs(struct elf *elf)
|
|||||||
rsec->base->rsec = rsec;
|
rsec->base->rsec = rsec;
|
||||||
|
|
||||||
nr_reloc = 0;
|
nr_reloc = 0;
|
||||||
rsec->reloc_data = calloc(rsec->sh.sh_size / rsec->sh.sh_entsize,
|
rsec->relocs = calloc(sec_num_entries(rsec), sizeof(*reloc));
|
||||||
sizeof(*reloc));
|
if (!rsec->relocs) {
|
||||||
if (!rsec->reloc_data) {
|
|
||||||
perror("calloc");
|
perror("calloc");
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
for (i = 0; i < rsec->sh.sh_size / rsec->sh.sh_entsize; i++) {
|
for (i = 0; i < sec_num_entries(rsec); i++) {
|
||||||
reloc = &rsec->reloc_data[i];
|
reloc = &rsec->relocs[i];
|
||||||
|
|
||||||
if (read_reloc(rsec, i, reloc))
|
if (read_reloc(rsec, i, reloc))
|
||||||
return -1;
|
return -1;
|
||||||
@ -967,7 +964,6 @@ static int read_relocs(struct elf *elf)
|
|||||||
}
|
}
|
||||||
|
|
||||||
list_add_tail(&reloc->sym_reloc_entry, &sym->reloc_list);
|
list_add_tail(&reloc->sym_reloc_entry, &sym->reloc_list);
|
||||||
list_add_tail(&reloc->list, &rsec->reloc_list);
|
|
||||||
elf_hash_add(reloc, &reloc->hash, reloc_hash(reloc));
|
elf_hash_add(reloc, &reloc->hash, reloc_hash(reloc));
|
||||||
|
|
||||||
nr_reloc++;
|
nr_reloc++;
|
||||||
@ -1093,7 +1089,6 @@ struct section *elf_create_section(struct elf *elf, const char *name,
|
|||||||
memset(sec, 0, sizeof(*sec));
|
memset(sec, 0, sizeof(*sec));
|
||||||
|
|
||||||
INIT_LIST_HEAD(&sec->symbol_list);
|
INIT_LIST_HEAD(&sec->symbol_list);
|
||||||
INIT_LIST_HEAD(&sec->reloc_list);
|
|
||||||
|
|
||||||
s = elf_newscn(elf->elf);
|
s = elf_newscn(elf->elf);
|
||||||
if (!s) {
|
if (!s) {
|
||||||
@ -1186,9 +1181,8 @@ static struct section *elf_create_rela_section(struct elf *elf,
|
|||||||
rsec->sh.sh_info = sec->idx;
|
rsec->sh.sh_info = sec->idx;
|
||||||
rsec->sh.sh_flags = SHF_INFO_LINK;
|
rsec->sh.sh_flags = SHF_INFO_LINK;
|
||||||
|
|
||||||
rsec->reloc_data = calloc(rsec->sh.sh_size / rsec->sh.sh_entsize,
|
rsec->relocs = calloc(sec_num_entries(rsec), sizeof(struct reloc));
|
||||||
sizeof(struct reloc));
|
if (!rsec->relocs) {
|
||||||
if (!rsec->reloc_data) {
|
|
||||||
perror("calloc");
|
perror("calloc");
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
@ -33,14 +33,13 @@ struct section {
|
|||||||
GElf_Shdr sh;
|
GElf_Shdr sh;
|
||||||
struct rb_root_cached symbol_tree;
|
struct rb_root_cached symbol_tree;
|
||||||
struct list_head symbol_list;
|
struct list_head symbol_list;
|
||||||
struct list_head reloc_list;
|
|
||||||
struct section *base, *rsec;
|
struct section *base, *rsec;
|
||||||
struct symbol *sym;
|
struct symbol *sym;
|
||||||
Elf_Data *data;
|
Elf_Data *data;
|
||||||
char *name;
|
char *name;
|
||||||
int idx;
|
int idx;
|
||||||
bool _changed, text, rodata, noinstr, init, truncate;
|
bool _changed, text, rodata, noinstr, init, truncate;
|
||||||
struct reloc *reloc_data;
|
struct reloc *relocs;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct symbol {
|
struct symbol {
|
||||||
@ -68,7 +67,6 @@ struct symbol {
|
|||||||
};
|
};
|
||||||
|
|
||||||
struct reloc {
|
struct reloc {
|
||||||
struct list_head list;
|
|
||||||
struct hlist_node hash;
|
struct hlist_node hash;
|
||||||
union {
|
union {
|
||||||
GElf_Rela rela;
|
GElf_Rela rela;
|
||||||
@ -197,6 +195,11 @@ static inline void mark_sec_changed(struct elf *elf, struct section *sec,
|
|||||||
elf->changed |= changed;
|
elf->changed |= changed;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline unsigned int sec_num_entries(struct section *sec)
|
||||||
|
{
|
||||||
|
return sec->sh.sh_size / sec->sh.sh_entsize;
|
||||||
|
}
|
||||||
|
|
||||||
#define for_each_sec(file, sec) \
|
#define for_each_sec(file, sec) \
|
||||||
list_for_each_entry(sec, &file->elf->sections, list)
|
list_for_each_entry(sec, &file->elf->sections, list)
|
||||||
|
|
||||||
@ -210,10 +213,15 @@ static inline void mark_sec_changed(struct elf *elf, struct section *sec,
|
|||||||
sec_for_each_sym(__sec, sym)
|
sec_for_each_sym(__sec, sym)
|
||||||
|
|
||||||
#define for_each_reloc(rsec, reloc) \
|
#define for_each_reloc(rsec, reloc) \
|
||||||
list_for_each_entry(reloc, &rsec->reloc_list, list)
|
for (int __i = 0, __fake = 1; __fake; __fake = 0) \
|
||||||
|
for (reloc = rsec->relocs; \
|
||||||
|
__i < sec_num_entries(rsec); \
|
||||||
|
__i++, reloc++)
|
||||||
|
|
||||||
#define for_each_reloc_from(rsec, reloc) \
|
#define for_each_reloc_from(rsec, reloc) \
|
||||||
list_for_each_entry_from(reloc, &rsec->reloc_list, list)
|
for (int __i = reloc->idx; \
|
||||||
|
__i < sec_num_entries(rsec); \
|
||||||
|
__i++, reloc++)
|
||||||
|
|
||||||
#define OFFSET_STRIDE_BITS 4
|
#define OFFSET_STRIDE_BITS 4
|
||||||
#define OFFSET_STRIDE (1UL << OFFSET_STRIDE_BITS)
|
#define OFFSET_STRIDE (1UL << OFFSET_STRIDE_BITS)
|
||||||
|
Loading…
Reference in New Issue
Block a user