diff options
Diffstat (limited to 'system/binutils/memory.patch')
-rw-r--r-- | system/binutils/memory.patch | 449 |
1 files changed, 449 insertions, 0 deletions
diff --git a/system/binutils/memory.patch b/system/binutils/memory.patch new file mode 100644 index 000000000..f2cfea130 --- /dev/null +++ b/system/binutils/memory.patch @@ -0,0 +1,449 @@ +From 9999de060bbcc7cca9dce213deeeec6593887a8e Mon Sep 17 00:00:00 2001 +From: "H.J. Lu" <hjl.tools@gmail.com> +Date: Tue, 27 Feb 2018 12:22:58 -0800 +Subject: [PATCH] Limit memory size to half of address space + +When link_info.keep_memory is TRUE, linker caches the relocation +information and symbol tables of input files in memory. On 32-bit +hosts, linker runs out of 32-bit virtual memory on input files with many +relocations. This patch limits the allocated memory size to half of +the address space for 32-bit hosts. + +bfd/ + + PR ld/18028 + * bfd-in.h (_bfd_link_keep_memory): New. + * bfd-in2.h: Regenerated. + * bfd.c (bfd): Add alloc_size. + * elf-bfd.h (_bfd_elf_link_info_read_relocs): New. + * elf32-i386.c (elf_i386_check_relocs): Use _bfd_link_keep_memory. + Update cache_size. + * elf64-x86-64.c (elf_x86_64_check_relocs): Likewise. + * elflink.c (_bfd_elf_link_read_relocs): Renamed to ... + (_bfd_elf_link_info_read_relocs): This. Update cache_size. + (_bfd_elf_link_read_relocs): New. + (_bfd_elf_link_check_relocs): Call _bfd_elf_link_info_read_relocs + instead of _bfd_elf_link_read_relocs. + (elf_link_add_object_symbols): Likewise. + (elf_link_input_bfd): Likewise. + (init_reloc_cookie_rels): Likewise. + (init_reloc_cookie): Update cache_size. Call + _bfd_elf_link_info_read_relocs instead of + _bfd_elf_link_read_relocs. + (link_info_ok): New. + (elf_gc_smash_unused_vtentry_relocs): Updated. Call + _bfd_elf_link_info_read_relocs instead of + _bfd_elf_link_read_relocs. + (bfd_elf_gc_sections): Use link_info_ok. Pass &link_info_ok + to elf_gc_smash_unused_vtentry_relocs. + * linker.c (_bfd_link_keep_memory): New. + * opncls.c (bfd_alloc): Update alloc_size. + +include/ + + PR ld/18028 + * bfdlink.h (bfd_link_info): Add cache_size and max_alloc_size. + +ld/ + + PR ld/18028 + * ldmain.c: Include "bfd_stdint.h". + (main): Set link_info.max_alloc_size to half of the address space. +--- + bfd/bfd-in.h | 2 ++ + bfd/bfd-in2.h | 5 +++ + bfd/bfd.c | 3 ++ + bfd/elf-bfd.h | 3 ++ + bfd/elf32-i386.c | 3 +- + bfd/elf64-x86-64.c | 3 +- + bfd/elflink.c | 89 ++++++++++++++++++++++++++++++++++++++++-------------- + bfd/linker.c | 33 ++++++++++++++++++++ + bfd/opncls.c | 2 ++ + include/bfdlink.h | 7 +++++ + ld/ldmain.c | 3 ++ + 11 files changed, 128 insertions(+), 25 deletions(-) + +diff --git a/bfd/bfd-in.h b/bfd/bfd-in.h +index a06cd740c0..ca4bec8aab 100644 +--- a/bfd/bfd-in.h ++++ b/bfd/bfd-in.h +@@ -600,6 +600,8 @@ struct bfd_section_already_linked; + struct bfd_elf_version_tree; + #endif + ++extern bfd_boolean _bfd_link_keep_memory (struct bfd_link_info *); ++ + extern bfd_boolean bfd_section_already_linked_table_init (void); + extern void bfd_section_already_linked_table_free (void); + extern bfd_boolean _bfd_handle_already_linked +diff --git a/bfd/bfd-in2.h b/bfd/bfd-in2.h +index 9742c1ac7f..0794bcdcdb 100644 +--- a/bfd/bfd-in2.h ++++ b/bfd/bfd-in2.h +@@ -607,6 +607,8 @@ struct bfd_section_already_linked; + struct bfd_elf_version_tree; + #endif + ++extern bfd_boolean _bfd_link_keep_memory (struct bfd_link_info *); ++ + extern bfd_boolean bfd_section_already_linked_table_init (void); + extern void bfd_section_already_linked_table_free (void); + extern bfd_boolean _bfd_handle_already_linked +@@ -6989,6 +6991,9 @@ struct bfd + be used only for archive elements. */ + int archive_pass; + ++ /* The total size of memory from bfd_alloc. */ ++ bfd_size_type alloc_size; ++ + /* Stuff only useful for object files: + The start address. */ + bfd_vma start_address; +diff --git a/bfd/bfd.c b/bfd/bfd.c +index 288b5b14fe..182b544662 100644 +--- a/bfd/bfd.c ++++ b/bfd/bfd.c +@@ -267,6 +267,9 @@ CODE_FRAGMENT + . be used only for archive elements. *} + . int archive_pass; + . ++. {* The total size of memory from bfd_alloc. *} ++. bfd_size_type alloc_size; ++. + . {* Stuff only useful for object files: + . The start address. *} + . bfd_vma start_address; +diff --git a/bfd/elf-bfd.h b/bfd/elf-bfd.h +index afd6982a92..37a7b07e51 100644 +--- a/bfd/elf-bfd.h ++++ b/bfd/elf-bfd.h +@@ -2294,6 +2294,9 @@ extern char *_bfd_elfcore_strndup + + extern Elf_Internal_Rela *_bfd_elf_link_read_relocs + (bfd *, asection *, void *, Elf_Internal_Rela *, bfd_boolean); ++extern Elf_Internal_Rela *_bfd_elf_link_info_read_relocs ++ (bfd *, struct bfd_link_info *, asection *, void *, Elf_Internal_Rela *, ++ bfd_boolean); + + extern bfd_boolean _bfd_elf_link_output_relocs + (bfd *, asection *, Elf_Internal_Shdr *, Elf_Internal_Rela *, +diff --git a/bfd/elf32-i386.c b/bfd/elf32-i386.c +index 61a14097b0..9a781dc089 100644 +--- a/bfd/elf32-i386.c ++++ b/bfd/elf32-i386.c +@@ -1909,13 +1909,14 @@ elf_i386_check_relocs (bfd *abfd, + + if (elf_section_data (sec)->this_hdr.contents != contents) + { +- if (!converted && !info->keep_memory) ++ if (!converted && !_bfd_link_keep_memory (info)) + free (contents); + else + { + /* Cache the section contents for elf_link_input_bfd if any + load is converted or --no-keep-memory isn't used. */ + elf_section_data (sec)->this_hdr.contents = contents; ++ info->cache_size += sec->size; + } + } + +diff --git a/bfd/elf64-x86-64.c b/bfd/elf64-x86-64.c +index 7016964ace..4ba15ec9fa 100644 +--- a/bfd/elf64-x86-64.c ++++ b/bfd/elf64-x86-64.c +@@ -2281,13 +2281,14 @@ elf_x86_64_check_relocs (bfd *abfd, struct bfd_link_info *info, + + if (elf_section_data (sec)->this_hdr.contents != contents) + { +- if (!converted && !info->keep_memory) ++ if (!converted && !_bfd_link_keep_memory (info)) + free (contents); + else + { + /* Cache the section contents for elf_link_input_bfd if any + load is converted or --no-keep-memory isn't used. */ + elf_section_data (sec)->this_hdr.contents = contents; ++ info->cache_size += sec->size; + } + } + +diff --git a/bfd/elflink.c b/bfd/elflink.c +index 092edaea26..831605b80a 100644 +--- a/bfd/elflink.c ++++ b/bfd/elflink.c +@@ -2475,14 +2475,16 @@ elf_link_read_relocs_from_section (bfd *abfd, + according to the KEEP_MEMORY argument. If O has two relocation + sections (both REL and RELA relocations), then the REL_HDR + relocations will appear first in INTERNAL_RELOCS, followed by the +- RELA_HDR relocations. */ ++ RELA_HDR relocations. If INFO isn't NULL and KEEP_MEMORY is TRUE, ++ update cache_size. */ + + Elf_Internal_Rela * +-_bfd_elf_link_read_relocs (bfd *abfd, +- asection *o, +- void *external_relocs, +- Elf_Internal_Rela *internal_relocs, +- bfd_boolean keep_memory) ++_bfd_elf_link_info_read_relocs (bfd *abfd, ++ struct bfd_link_info *info, ++ asection *o, ++ void *external_relocs, ++ Elf_Internal_Rela *internal_relocs, ++ bfd_boolean keep_memory) + { + void *alloc1 = NULL; + Elf_Internal_Rela *alloc2 = NULL; +@@ -2502,7 +2504,11 @@ _bfd_elf_link_read_relocs (bfd *abfd, + + size = (bfd_size_type) o->reloc_count * sizeof (Elf_Internal_Rela); + if (keep_memory) +- internal_relocs = alloc2 = (Elf_Internal_Rela *) bfd_alloc (abfd, size); ++ { ++ internal_relocs = alloc2 = (Elf_Internal_Rela *) bfd_alloc (abfd, size); ++ if (info) ++ info->cache_size += size; ++ } + else + internal_relocs = alloc2 = (Elf_Internal_Rela *) bfd_malloc (size); + if (internal_relocs == NULL) +@@ -2568,6 +2574,22 @@ _bfd_elf_link_read_relocs (bfd *abfd, + return NULL; + } + ++/* This is similar to _bfd_elf_link_info_read_relocs, except for that ++ NULL is passed to _bfd_elf_link_info_read_relocs for pointer to ++ struct bfd_link_info. */ ++ ++Elf_Internal_Rela * ++_bfd_elf_link_read_relocs (bfd *abfd, ++ asection *o, ++ void *external_relocs, ++ Elf_Internal_Rela *internal_relocs, ++ bfd_boolean keep_memory) ++{ ++ return _bfd_elf_link_info_read_relocs (abfd, NULL, o, external_relocs, ++ internal_relocs, keep_memory); ++ ++} ++ + /* Compute the size of, and allocate space for, REL_HDR which is the + section header for a section containing relocations for O. */ + +@@ -3736,8 +3758,10 @@ _bfd_elf_link_check_relocs (bfd *abfd, struct bfd_link_info *info) + || bfd_is_abs_section (o->output_section)) + continue; + +- internal_relocs = _bfd_elf_link_read_relocs (abfd, o, NULL, NULL, +- info->keep_memory); ++ internal_relocs = _bfd_elf_link_info_read_relocs (abfd, info, ++ o, NULL, ++ NULL, ++ _bfd_link_keep_memory (info)); + if (internal_relocs == NULL) + return FALSE; + +@@ -4990,9 +5014,10 @@ elf_link_add_object_symbols (bfd *abfd, struct bfd_link_info *info) + && (s->flags & SEC_DEBUGGING) != 0)) + continue; + +- internal_relocs = _bfd_elf_link_read_relocs (abfd, s, NULL, +- NULL, +- info->keep_memory); ++ internal_relocs = _bfd_elf_link_info_read_relocs (abfd, info, ++ s, NULL, ++ NULL, ++ _bfd_link_keep_memory (info)); + if (internal_relocs == NULL) + goto error_free_vers; + +@@ -10518,8 +10543,10 @@ elf_link_input_bfd (struct elf_final_link_info *flinfo, bfd *input_bfd) + + /* Get the swapped relocs. */ + internal_relocs +- = _bfd_elf_link_read_relocs (input_bfd, o, flinfo->external_relocs, +- flinfo->internal_relocs, FALSE); ++ = _bfd_elf_link_info_read_relocs (input_bfd, flinfo->info, o, ++ flinfo->external_relocs, ++ flinfo->internal_relocs, ++ FALSE); + if (internal_relocs == NULL + && o->reloc_count > 0) + return FALSE; +@@ -12684,8 +12711,12 @@ init_reloc_cookie (struct elf_reloc_cookie *cookie, + info->callbacks->einfo (_("%P%X: can not read symbols: %E\n")); + return FALSE; + } +- if (info->keep_memory) +- symtab_hdr->contents = (bfd_byte *) cookie->locsyms; ++ if (_bfd_link_keep_memory (info) ) ++ { ++ symtab_hdr->contents = (bfd_byte *) cookie->locsyms; ++ info->cache_size += (cookie->locsymcount ++ * sizeof (Elf_External_Sym_Shndx)); ++ } + } + return TRUE; + } +@@ -12718,8 +12749,9 @@ init_reloc_cookie_rels (struct elf_reloc_cookie *cookie, + } + else + { +- cookie->rels = _bfd_elf_link_read_relocs (abfd, sec, NULL, NULL, +- info->keep_memory); ++ cookie->rels = _bfd_elf_link_info_read_relocs (abfd, info, sec, ++ NULL, NULL, ++ _bfd_link_keep_memory (info)); + if (cookie->rels == NULL) + return FALSE; + cookie->rel = cookie->rels; +@@ -13233,14 +13265,21 @@ elf_gc_propagate_vtable_entries_used (struct elf_link_hash_entry *h, void *okp) + return TRUE; + } + ++struct link_info_ok ++{ ++ struct bfd_link_info *info; ++ bfd_boolean ok; ++}; ++ + static bfd_boolean +-elf_gc_smash_unused_vtentry_relocs (struct elf_link_hash_entry *h, void *okp) ++elf_gc_smash_unused_vtentry_relocs (struct elf_link_hash_entry *h, void *ptr) + { + asection *sec; + bfd_vma hstart, hend; + Elf_Internal_Rela *relstart, *relend, *rel; + const struct elf_backend_data *bed; + unsigned int log_file_align; ++ struct link_info_ok *info = (struct link_info_ok *) ptr; + + /* Take care of both those symbols that do not describe vtables as + well as those that are not loaded. */ +@@ -13256,9 +13295,10 @@ elf_gc_smash_unused_vtentry_relocs (struct elf_link_hash_entry *h, void *okp) + hstart = h->root.u.def.value; + hend = hstart + h->size; + +- relstart = _bfd_elf_link_read_relocs (sec->owner, sec, NULL, NULL, TRUE); ++ relstart = _bfd_elf_link_info_read_relocs (sec->owner, info->info, ++ sec, NULL, NULL, TRUE); + if (!relstart) +- return *(bfd_boolean *) okp = FALSE; ++ return info->ok = FALSE; + bed = get_elf_backend_data (sec->owner); + log_file_align = bed->s->log_file_align; + +@@ -13379,6 +13419,7 @@ bfd_elf_gc_sections (bfd *abfd, struct bfd_link_info *info) + elf_gc_mark_hook_fn gc_mark_hook; + const struct elf_backend_data *bed = get_elf_backend_data (abfd); + struct elf_link_hash_table *htab; ++ struct link_info_ok info_ok; + + if (!bed->can_gc_sections + || !is_elf_hash_table (info->hash)) +@@ -13420,8 +13461,10 @@ bfd_elf_gc_sections (bfd *abfd, struct bfd_link_info *info) + return FALSE; + + /* Kill the vtable relocations that were not used. */ +- elf_link_hash_traverse (htab, elf_gc_smash_unused_vtentry_relocs, &ok); +- if (!ok) ++ info_ok.info = info; ++ info_ok.ok = TRUE; ++ elf_link_hash_traverse (htab, elf_gc_smash_unused_vtentry_relocs, &info_ok); ++ if (!info_ok.ok) + return FALSE; + + /* Mark dynamically referenced symbols. */ +diff --git a/bfd/linker.c b/bfd/linker.c +index c29a6e7e10..909d768552 100644 +--- a/bfd/linker.c ++++ b/bfd/linker.c +@@ -3471,3 +3471,36 @@ _bfd_nolink_bfd_define_start_stop (struct bfd_link_info *info ATTRIBUTE_UNUSED, + { + return (struct bfd_link_hash_entry *) _bfd_ptr_bfd_null_error (sec->owner); + } ++ ++bfd_boolean ++_bfd_link_keep_memory (struct bfd_link_info * info) ++{ ++ bfd *abfd; ++ bfd_size_type size; ++ ++ if (!info->keep_memory) ++ return FALSE; ++ ++ /* Keep allocated memory size below limit only for 32-bit hosts. */ ++ if (sizeof (void *) > 4) ++ return TRUE; ++ ++ abfd = info->input_bfds; ++ size = info->cache_size; ++ do ++ { ++ if (size >= info->max_alloc_size) ++ { ++ /* Over the limit. Reduce the memory usage. */ ++ info->keep_memory = FALSE; ++ return FALSE; ++ } ++ if (!abfd) ++ break; ++ size += abfd->alloc_size; ++ abfd = abfd->link.next; ++ } ++ while (1); ++ ++ return TRUE; ++} +diff --git a/bfd/opncls.c b/bfd/opncls.c +index 16b568c8ab..86262e1231 100644 +--- a/bfd/opncls.c ++++ b/bfd/opncls.c +@@ -949,6 +949,8 @@ bfd_alloc (bfd *abfd, bfd_size_type size) + ret = objalloc_alloc ((struct objalloc *) abfd->memory, ul_size); + if (ret == NULL) + bfd_set_error (bfd_error_no_memory); ++ else ++ abfd->alloc_size += size; + return ret; + } + +diff --git a/include/bfdlink.h b/include/bfdlink.h +index 5d637acbab..f06d5171c3 100644 +--- a/include/bfdlink.h ++++ b/include/bfdlink.h +@@ -635,6 +635,13 @@ struct bfd_link_info + + /* The version information. */ + struct bfd_elf_version_tree *version_info; ++ ++ /* Size of cache. Backend can use it to keep strace cache size. */ ++ bfd_size_type cache_size; ++ ++ /* The maximum size of allocated memory. Backend can use cache_size ++ and and max_alloc_size to decide if keep_memory should be honored. */ ++ bfd_size_type max_alloc_size; + }; + + /* This structures holds a set of callback functions. These are called +diff --git a/ld/ldmain.c b/ld/ldmain.c +index b6914db5da..14c3349c97 100644 +--- a/ld/ldmain.c ++++ b/ld/ldmain.c +@@ -21,6 +21,7 @@ + + #include "sysdep.h" + #include "bfd.h" ++#include "bfd_stdint.h" + #include "safe-ctype.h" + #include "libiberty.h" + #include "progress.h" +@@ -271,6 +272,8 @@ main (int argc, char **argv) + + link_info.allow_undefined_version = TRUE; + link_info.keep_memory = TRUE; ++ /* Limit the allocated memory size to half of the address space. */ ++ link_info.max_alloc_size = ((uintptr_t) (void *) -1) / 2; + link_info.combreloc = TRUE; + link_info.strip_discarded = TRUE; + link_info.emit_hash = DEFAULT_EMIT_SYSV_HASH; |