diff options
Diffstat (limited to 'bfd/elf-eh-frame.c')
-rw-r--r-- | bfd/elf-eh-frame.c | 95 |
1 files changed, 65 insertions, 30 deletions
diff --git a/bfd/elf-eh-frame.c b/bfd/elf-eh-frame.c index f215fca..14c690b 100644 --- a/bfd/elf-eh-frame.c +++ b/bfd/elf-eh-frame.c @@ -86,6 +86,7 @@ struct eh_frame_hdr_info We build it if we successfully read all .eh_frame input sections and recognize them. */ boolean table; + boolean strip; }; static bfd_vma read_unsigned_leb128 @@ -246,7 +247,7 @@ _bfd_elf_discard_section_eh_frame (abfd, info, sec, ehdrsec, struct cie_header hdr; struct cie cie; struct eh_frame_hdr_info *hdr_info; - struct eh_frame_sec_info *sec_info; + struct eh_frame_sec_info *sec_info = NULL; unsigned int leb128_tmp; unsigned int cie_usage_count, last_cie_ndx, i, offset, make_relative; Elf_Internal_Rela *rel; @@ -267,23 +268,20 @@ _bfd_elf_discard_section_eh_frame (abfd, info, sec, ehdrsec, return false; } + BFD_ASSERT (elf_section_data (ehdrsec)->sec_info_type + == ELF_INFO_TYPE_EH_FRAME_HDR); + hdr_info = (struct eh_frame_hdr_info *) + elf_section_data (ehdrsec)->sec_info; + /* Read the frame unwind information from abfd. */ ehbuf = (bfd_byte *) bfd_malloc (sec->_raw_size); - if (ehbuf == NULL - || ! bfd_get_section_contents (abfd, sec, ehbuf, (bfd_vma) 0, - sec->_raw_size)) - { - if (elf_section_data (ehdrsec)->sec_info_type - != ELF_INFO_TYPE_EH_FRAME_HDR) - { - elf_section_data (ehdrsec)->sec_info - = bfd_zmalloc (sizeof (struct eh_frame_hdr_info)); - elf_section_data (ehdrsec)->sec_info_type - = ELF_INFO_TYPE_EH_FRAME_HDR; - } - return false; - } + if (ehbuf == NULL) + goto free_no_table; + + if (! bfd_get_section_contents (abfd, sec, ehbuf, (bfd_vma) 0, + sec->_raw_size)) + goto free_no_table; if (sec->_raw_size >= 4 && bfd_get_32 (abfd, ehbuf) == 0 @@ -294,24 +292,10 @@ _bfd_elf_discard_section_eh_frame (abfd, info, sec, ehdrsec, return false; } - if (elf_section_data (ehdrsec)->sec_info_type - != ELF_INFO_TYPE_EH_FRAME_HDR) - { - hdr_info = (struct eh_frame_hdr_info *) - bfd_zmalloc (sizeof (struct eh_frame_hdr_info)); - hdr_info->table = true; - elf_section_data (ehdrsec)->sec_info = hdr_info; - elf_section_data (ehdrsec)->sec_info_type - = ELF_INFO_TYPE_EH_FRAME_HDR; - } - else - hdr_info = (struct eh_frame_hdr_info *) - elf_section_data (ehdrsec)->sec_info; - /* If .eh_frame section size doesn't fit into int, we cannot handle it (it would need to use 64-bit .eh_frame format anyway). */ if (sec->_raw_size != (unsigned int) sec->_raw_size) - return false; + goto free_no_table; ptr_size = (elf_elfheader (abfd)->e_ident[EI_CLASS] == ELFCLASS64) ? 8 : 4; @@ -650,9 +634,12 @@ _bfd_elf_discard_section_eh_frame (abfd, info, sec, ehdrsec, if (sec->_cooked_size == 0) sec->flags |= SEC_EXCLUDE; + free (ehbuf); return new_size != sec->_raw_size; free_no_table: + if (ehbuf) + free (ehbuf); if (sec_info) free (sec_info); hdr_info->table = false; @@ -686,6 +673,8 @@ _bfd_elf_discard_section_eh_frame_hdr (abfd, info, sec) hdr_info = (struct eh_frame_hdr_info *) elf_section_data (sec)->sec_info; + if (hdr_info->strip) + return false; sec->_cooked_size = EH_FRAME_HDR_SIZE; if (hdr_info->table) sec->_cooked_size += 4 + hdr_info->fde_count * 8; @@ -696,6 +685,52 @@ _bfd_elf_discard_section_eh_frame_hdr (abfd, info, sec) return true; } +/* This function is called from size_dynamic_sections. + It needs to decide whether .eh_frame_hdr should be output or not, + because later on it is too late for calling _bfd_strip_section_from_output, + since dynamic symbol table has been sized. */ + +boolean +_bfd_elf_maybe_strip_eh_frame_hdr (info) + struct bfd_link_info *info; +{ + asection *sec, *o; + bfd *abfd; + struct eh_frame_hdr_info *hdr_info; + + sec = bfd_get_section_by_name (elf_hash_table (info)->dynobj, ".eh_frame_hdr"); + if (sec == NULL) + return true; + + hdr_info + = bfd_zmalloc (sizeof (struct eh_frame_hdr_info)); + if (hdr_info == NULL) + return false; + + elf_section_data (sec)->sec_info = hdr_info; + elf_section_data (sec)->sec_info_type = ELF_INFO_TYPE_EH_FRAME_HDR; + + abfd = NULL; + if (info->eh_frame_hdr) + for (abfd = info->input_bfds; abfd != NULL; abfd = abfd->link_next) + { + /* Count only sections which have at least a single CIE or FDE. + There cannot be any CIE or FDE <= 8 bytes. */ + o = bfd_get_section_by_name (abfd, ".eh_frame"); + if (o && o->_raw_size > 8) + break; + } + + if (abfd == NULL) + { + _bfd_strip_section_from_output (info, sec); + hdr_info->strip = true; + } + else + hdr_info->table = true; + return true; +} + /* Adjust an address in the .eh_frame section. Given OFFSET within SEC, this returns the new offset in the adjusted .eh_frame section, or -1 if the address refers to a CIE/FDE which has been removed |