diff options
Diffstat (limited to 'sys-devel/binutils/files/2.14')
6 files changed, 1918 insertions, 0 deletions
diff --git a/sys-devel/binutils/files/2.14/binutils-2.14.90.0.6-eh-frame-ro-2.patch b/sys-devel/binutils/files/2.14/binutils-2.14.90.0.6-eh-frame-ro-2.patch new file mode 100644 index 000000000000..c0b1c13cdaff --- /dev/null +++ b/sys-devel/binutils/files/2.14/binutils-2.14.90.0.6-eh-frame-ro-2.patch @@ -0,0 +1,427 @@ +2002-11-22 Jakub Jelinek <jakub@redhat.com> + + * ldgram.y (sect_constraint): New. + (ONLY_IF_RO, ONLY_IF_RW): New tokens. + (section): Add sect_constraint. Pass additional argument + to lang_enter_output_section_statement. + * mri.c (mri_draw_tree): Pass additional argument to + lang_enter_output_section_statement. + * emultempl/pe.em (place_orphan): Likewise. + (output_prev_sec_find): Disregard output section statements with + constraint == -1. + * emultempl/mmo.em (output_prev_sec_find): Likewise. + (mmo_place_orphan): Pass additional argument to + lang_enter_output_section_statement. + * emultempl/elf32.em (output_prev_sec_find): Disregard output section + statements with constraint == -1. + (place_orphan): Pass additional argument to + lang_enter_output_section_statement. + * ldlang.c (lang_enter_overlay_section): Likewise. + (lang_output_section_find_1): New. + (lang_output_section_find): Use it. + (lang_output_section_statement_lookup_1): New. + (lang_output_section_statement_lookup): Use it. + (check_section_callback, check_input_sections): New. + (map_input_to_output_sections): Check if all input sections + are readonly if ONLY_IF_RO or ONLY_IF_RW was seen. + (strip_excluded_output_sections): Disregard output section statements + with constraint == -1. + (lang_record_phdrs): Likewise. + (lang_enter_output_section_statement): Add constraint argument. + Use lang_output_section_statement_lookup_1. + * ldlang.h (lang_output_section_statement_type): Add constraint + and all_input_readonly fields. + (lang_enter_output_section_statement): Adjust prototype. + * ldlex.l (ONLY_IF_RO, ONLY_IF_RW): New tokens. + * scripttempl/elf.sc (.eh_frame, .gcc_except_table): Move into text + segment if all input sections are readonly. + +--- ld/emultempl/mmo.em.jj Wed Jul 23 11:08:12 2003 ++++ ld/emultempl/mmo.em Fri Sep 19 17:30:02 2003 +@@ -47,6 +47,8 @@ output_prev_sec_find (lang_output_sectio + u = lookup->next) + { + lookup = &u->output_section_statement; ++ if (lookup->constraint == -1) ++ continue; + if (lookup == os) + break; + if (lookup->bfd_section != NULL +@@ -130,7 +132,7 @@ mmo_place_orphan (lang_input_statement_t + (bfd_vma) 0, + (etree_type *) NULL, + (etree_type *) NULL, +- (etree_type *) NULL); ++ (etree_type *) NULL, 0); + + lang_add_section (&os->children, s, os, file); + +--- ld/emultempl/pe.em.jj Thu Aug 21 11:28:48 2003 ++++ ld/emultempl/pe.em Fri Sep 19 17:30:02 2003 +@@ -1453,6 +1453,8 @@ output_prev_sec_find (lang_output_sectio + u = lookup->next) + { + lookup = &u->output_section_statement; ++ if (lookup->constraint == -1) ++ continue; + if (lookup == os) + return s; + +@@ -1617,7 +1619,7 @@ gld_${EMULATION_NAME}_place_orphan (lang + (bfd_vma) 0, + (etree_type *) NULL, + (etree_type *) NULL, +- (etree_type *) NULL); ++ (etree_type *) NULL, 0); + + lang_add_section (&add_child, s, os, file); + +--- ld/emultempl/elf32.em.jj Thu Aug 21 11:28:48 2003 ++++ ld/emultempl/elf32.em Fri Sep 19 17:30:02 2003 +@@ -1016,7 +1016,8 @@ output_rel_find (asection *sec, int isdy + for (u = lang_output_section_statement.head; u; u = lookup->next) + { + lookup = &u->output_section_statement; +- if (strncmp (".rel", lookup->name, 4) == 0) ++ if (lookup->constraint != -1 ++ && strncmp (".rel", lookup->name, 4) == 0) + { + int lookrela = lookup->name[4] == 'a'; + +@@ -1264,7 +1265,7 @@ gld${EMULATION_NAME}_place_orphan (lang_ + (bfd_vma) 0, + (etree_type *) NULL, + (etree_type *) NULL, +- load_base); ++ load_base, 0); + + lang_add_section (&os->children, s, os, file); + +--- ld/scripttempl/elf.sc.jj Wed Jul 23 11:08:12 2003 ++++ ld/scripttempl/elf.sc Fri Sep 19 17:30:02 2003 +@@ -283,6 +283,8 @@ cat <<EOF + ${CREATE_SHLIB-${SBSS2}} + ${OTHER_READONLY_SECTIONS} + .eh_frame_hdr : { *(.eh_frame_hdr) } ++ .eh_frame ${RELOCATING-0} : ONLY_IF_RO { KEEP (*(.eh_frame)) } ++ .gcc_except_table ${RELOCATING-0} : ONLY_IF_RO { *(.gcc_except_table) } + + /* Adjust the address for the data segment. We want to adjust up to + the same address within the page on the next page up. */ +@@ -316,8 +318,8 @@ cat <<EOF + .data1 ${RELOCATING-0} : { *(.data1) } + .tdata ${RELOCATING-0} : { *(.tdata${RELOCATING+ .tdata.* .gnu.linkonce.td.*}) } + .tbss ${RELOCATING-0} : { *(.tbss${RELOCATING+ .tbss.* .gnu.linkonce.tb.*})${RELOCATING+ *(.tcommon)} } +- .eh_frame ${RELOCATING-0} : { KEEP (*(.eh_frame)) } +- .gcc_except_table ${RELOCATING-0} : { *(.gcc_except_table) } ++ .eh_frame ${RELOCATING-0} : ONLY_IF_RW { KEEP (*(.eh_frame)) } ++ .gcc_except_table ${RELOCATING-0} : ONLY_IF_RW { *(.gcc_except_table) } + ${WRITABLE_RODATA+${RODATA}} + ${OTHER_READWRITE_SECTIONS} + ${TEXT_DYNAMIC-${DYNAMIC}} +--- ld/ldgram.y.jj Thu Aug 21 11:28:48 2003 ++++ ld/ldgram.y Fri Sep 19 17:38:10 2003 +@@ -143,14 +143,14 @@ static int error_index; + %token ORIGIN FILL + %token LENGTH CREATE_OBJECT_SYMBOLS INPUT GROUP OUTPUT CONSTRUCTORS + %token ALIGNMOD AT SUBALIGN PROVIDE +-%type <token> assign_op atype attributes_opt ++%type <token> assign_op atype attributes_opt sect_constraint + %type <name> filename + %token CHIP LIST SECT ABSOLUTE LOAD NEWLINE ENDWORD ORDER NAMEWORD ASSERT_K + %token FORMAT PUBLIC DEFSYMEND BASE ALIAS TRUNCATE REL + %token INPUT_SCRIPT INPUT_MRI_SCRIPT INPUT_DEFSYM CASE EXTERN START + %token <name> VERS_TAG VERS_IDENTIFIER + %token GLOBAL LOCAL VERSIONK INPUT_VERSION_SCRIPT +-%token KEEP ++%token KEEP ONLY_IF_RO ONLY_IF_RW + %token EXCLUDE_FILE + %type <versyms> vers_defns + %type <versnode> vers_tag +@@ -833,22 +833,29 @@ opt_subalign: + | { $$ = 0; } + ; + ++sect_constraint: ++ ONLY_IF_RO { $$ = ONLY_IF_RO; } ++ | ONLY_IF_RW { $$ = ONLY_IF_RW; } ++ | { $$ = 0; } ++ ; ++ + section: NAME { ldlex_expression(); } + opt_exp_with_type + opt_at + opt_subalign { ldlex_popstate (); ldlex_script (); } ++ sect_constraint + '{' + { + lang_enter_output_section_statement($1, $3, + sectype, +- 0, 0, $5, $4); ++ 0, 0, $5, $4, $7); + } + statement_list_opt + '}' { ldlex_popstate (); ldlex_expression (); } + memspec_opt memspec_at_opt phdr_opt fill_opt + { + ldlex_popstate (); +- lang_leave_output_section_statement ($15, $12, $14, $13); ++ lang_leave_output_section_statement ($16, $13, $15, $14); + } + opt_comma + {} +--- ld/mri.c.jj Wed Jul 23 11:08:12 2003 ++++ ld/mri.c Fri Sep 19 17:30:02 2003 +@@ -220,7 +220,7 @@ mri_draw_tree (void) + + lang_enter_output_section_statement (p->name, base, + p->ok_to_load ? 0 : noload_section, +- 1, align, subalign, NULL); ++ 1, align, subalign, NULL, 0); + base = 0; + tmp = xmalloc (sizeof *tmp); + tmp->next = NULL; +--- ld/ldlang.h.jj Thu Aug 21 11:28:48 2003 ++++ ld/ldlang.h Fri Sep 19 17:30:02 2003 +@@ -132,6 +132,8 @@ typedef struct lang_output_section_state + + int subsection_alignment; /* alignment of components */ + int section_alignment; /* alignment of start of section */ ++ int constraint; ++ bfd_boolean all_input_readonly; + + union etree_union *load_base; + +@@ -394,7 +396,7 @@ extern lang_output_section_statement_typ + bfd_vma block_value, + etree_type *align, + etree_type *subalign, +- etree_type *); ++ etree_type *, int); + extern void lang_final + (void); + extern void lang_process +--- ld/ldlang.c.jj Thu Aug 21 11:28:48 2003 ++++ ld/ldlang.c Fri Sep 19 17:40:21 2003 +@@ -562,7 +562,7 @@ lang_memory_default (asection *section) + } + + lang_output_section_statement_type * +-lang_output_section_find (const char *const name) ++lang_output_section_find_1 (const char *const name, int constraint) + { + lang_statement_union_type *u; + lang_output_section_statement_type *lookup; +@@ -570,18 +570,26 @@ lang_output_section_find (const char *co + for (u = lang_output_section_statement.head; u != NULL; u = lookup->next) + { + lookup = &u->output_section_statement; +- if (strcmp (name, lookup->name) == 0) ++ if (strcmp (name, lookup->name) == 0 ++ && lookup->constraint != -1 ++ && (constraint == 0 || constraint == lookup->constraint)) + return lookup; + } + return NULL; + } + + lang_output_section_statement_type * +-lang_output_section_statement_lookup (const char *const name) ++lang_output_section_find (const char *const name) ++{ ++ return lang_output_section_find_1 (name, 0); ++} ++ ++lang_output_section_statement_type * ++lang_output_section_statement_lookup_1 (const char *const name, int constraint) + { + lang_output_section_statement_type *lookup; + +- lookup = lang_output_section_find (name); ++ lookup = lang_output_section_find_1 (name, constraint); + if (lookup == NULL) + { + lookup = new_stat (lang_output_section_statement, stat_ptr); +@@ -594,6 +602,7 @@ lang_output_section_statement_lookup (co + lookup->next = NULL; + lookup->bfd_section = NULL; + lookup->processed = FALSE; ++ lookup->constraint = constraint; + lookup->sectype = normal_section; + lookup->addr_tree = NULL; + lang_list_init (&lookup->children); +@@ -613,6 +622,12 @@ lang_output_section_statement_lookup (co + return lookup; + } + ++lang_output_section_statement_type * ++lang_output_section_statement_lookup (const char *const name) ++{ ++ return lang_output_section_statement_lookup_1 (name, 0); ++} ++ + static void + lang_map_flags (flagword flag) + { +@@ -1278,6 +1293,30 @@ output_section_callback (lang_wild_state + } + } + ++/* Check if all sections in a wild statement for a particular FILE ++ are readonly. */ ++ ++static void ++check_section_callback (lang_wild_statement_type *ptr ATTRIBUTE_UNUSED, ++ struct wildcard_list *sec ATTRIBUTE_UNUSED, ++ asection *section, ++ lang_input_statement_type *file ATTRIBUTE_UNUSED, ++ void *output) ++{ ++ /* Exclude sections that match UNIQUE_SECTION_LIST. */ ++ if (unique_section_p (bfd_get_section_name (file->the_bfd, section))) ++ return; ++ ++ if (section->output_section == NULL) ++ { ++ flagword flags = bfd_get_section_flags (section->owner, section); ++ ++ if ((flags & SEC_READONLY) == 0) ++ ((lang_output_section_statement_type *) output)->all_input_readonly ++ = FALSE; ++ } ++} ++ + /* This is passed a file name which must have been seen already and + added to the statement tree. We will see if it has been opened + already and had its symbols read. If not then we'll read it. */ +@@ -1923,6 +1962,40 @@ lang_place_undefineds (void) + insert_undefined (ptr->name); + } + ++/* Check for all readonly or some readwrite sections. */ ++ ++static void ++check_input_sections (lang_statement_union_type *s, ++ lang_output_section_statement_type *output_section_statement) ++{ ++ for (; s != (lang_statement_union_type *) NULL; s = s->header.next) ++ { ++ switch (s->header.type) ++ { ++ case lang_wild_statement_enum: ++ walk_wild (&s->wild_statement, check_section_callback, ++ output_section_statement); ++ if (! output_section_statement->all_input_readonly) ++ return; ++ break; ++ case lang_constructors_statement_enum: ++ check_input_sections (constructor_list.head, ++ output_section_statement); ++ if (! output_section_statement->all_input_readonly) ++ return; ++ break; ++ case lang_group_statement_enum: ++ check_input_sections (s->group_statement.children.head, ++ output_section_statement); ++ if (! output_section_statement->all_input_readonly) ++ return; ++ break; ++ default: ++ break; ++ } ++ } ++} ++ + /* Open input files and attach to output sections. */ + + static void +@@ -1943,6 +2016,23 @@ map_input_to_output_sections + output_section_statement); + break; + case lang_output_section_statement_enum: ++ if (s->output_section_statement.constraint) ++ { ++ if (s->output_section_statement.constraint == -1) ++ break; ++ s->output_section_statement.all_input_readonly = TRUE; ++ check_input_sections (s->output_section_statement.children.head, ++ &s->output_section_statement); ++ if ((s->output_section_statement.all_input_readonly ++ && s->output_section_statement.constraint == ONLY_IF_RW) ++ || (!s->output_section_statement.all_input_readonly ++ && s->output_section_statement.constraint == ONLY_IF_RO)) ++ { ++ s->output_section_statement.constraint = -1; ++ break; ++ } ++ } ++ + map_input_to_output_sections (s->output_section_statement.children.head, + target, + &s->output_section_statement); +@@ -2013,6 +2103,8 @@ strip_excluded_output_sections (void) + asection *s; + + os = &u->output_section_statement; ++ if (os->constraint == -1) ++ continue; + s = os->bfd_section; + if (s != NULL && (s->flags & SEC_EXCLUDE) != 0) + { +@@ -3857,13 +3949,15 @@ lang_enter_output_section_statement (con + bfd_vma block_value, + etree_type *align, + etree_type *subalign, +- etree_type *ebase) ++ etree_type *ebase, ++ int constraint) + { + lang_output_section_statement_type *os; + + current_section = + os = +- lang_output_section_statement_lookup (output_section_statement_name); ++ lang_output_section_statement_lookup_1 (output_section_statement_name, ++ constraint); + + /* Add this statement to tree. */ + #if 0 +@@ -4558,6 +4652,8 @@ lang_record_phdrs (void) + struct lang_output_section_phdr_list *pl; + + os = &u->output_section_statement; ++ if (os->constraint == -1) ++ continue; + + pl = os->phdrs; + if (pl != NULL) +@@ -4617,7 +4713,8 @@ lang_record_phdrs (void) + { + struct lang_output_section_phdr_list *pl; + +- if (u->output_section_statement.bfd_section == NULL) ++ if (u->output_section_statement.constraint == -1 ++ || u->output_section_statement.bfd_section == NULL) + continue; + + for (pl = u->output_section_statement.phdrs; +@@ -4689,7 +4786,7 @@ lang_enter_overlay_section (const char * + etree_type *size; + + lang_enter_output_section_statement (name, overlay_vma, normal_section, +- 0, 0, overlay_subalign, 0); ++ 0, 0, overlay_subalign, 0, 0); + + /* If this is the first section, then base the VMA of future + sections on this one. This will work correctly even if `.' is +--- ld/ldlex.l.jj Thu Aug 21 11:28:48 2003 ++++ ld/ldlex.l Fri Sep 19 17:30:02 2003 +@@ -299,6 +299,8 @@ V_IDENTIFIER [*?.$_a-zA-Z\[\]\-\!\^]([*? + <EXPRESSION,BOTH,SCRIPT>"COPY" { RTOKEN(COPY);} + <EXPRESSION,BOTH,SCRIPT>"INFO" { RTOKEN(INFO);} + <EXPRESSION,BOTH,SCRIPT>"OVERLAY" { RTOKEN(OVERLAY);} ++<EXPRESSION,BOTH,SCRIPT>"ONLY_IF_RO" { RTOKEN(ONLY_IF_RO); } ++<EXPRESSION,BOTH,SCRIPT>"ONLY_IF_RW" { RTOKEN(ONLY_IF_RW); } + <BOTH,SCRIPT>"o" { RTOKEN(ORIGIN);} + <BOTH,SCRIPT>"org" { RTOKEN(ORIGIN);} + <BOTH,SCRIPT>"l" { RTOKEN( LENGTH);} diff --git a/sys-devel/binutils/files/2.14/binutils-2.14.90.0.6-ia64-howto.patch b/sys-devel/binutils/files/2.14/binutils-2.14.90.0.6-ia64-howto.patch new file mode 100644 index 000000000000..bee66b64da38 --- /dev/null +++ b/sys-devel/binutils/files/2.14/binutils-2.14.90.0.6-ia64-howto.patch @@ -0,0 +1,37 @@ +2003-09-30 Jakub Jelinek <jakub@redhat.com> + + * elfxx-ia64.c (ia64_howto_table): Fix size of R_IA64_TPREL64[LM]SB, + R_IA64_DTPREL{32,64}[LM]SB and R_IA64_DTPMOD64[LM]SB. + +--- bfd/elfxx-ia64.c.jj 2003-09-30 16:44:32.000000000 +0200 ++++ bfd/elfxx-ia64.c 2003-09-30 17:53:11.000000000 +0200 +@@ -446,21 +446,21 @@ static reloc_howto_type ia64_howto_table + IA64_HOWTO (R_IA64_TPREL14, "TPREL14", 0, FALSE, FALSE), + IA64_HOWTO (R_IA64_TPREL22, "TPREL22", 0, FALSE, FALSE), + IA64_HOWTO (R_IA64_TPREL64I, "TPREL64I", 0, FALSE, FALSE), +- IA64_HOWTO (R_IA64_TPREL64MSB, "TPREL64MSB", 8, FALSE, FALSE), +- IA64_HOWTO (R_IA64_TPREL64LSB, "TPREL64LSB", 8, FALSE, FALSE), ++ IA64_HOWTO (R_IA64_TPREL64MSB, "TPREL64MSB", 4, FALSE, FALSE), ++ IA64_HOWTO (R_IA64_TPREL64LSB, "TPREL64LSB", 4, FALSE, FALSE), + IA64_HOWTO (R_IA64_LTOFF_TPREL22, "LTOFF_TPREL22", 0, FALSE, FALSE), + +- IA64_HOWTO (R_IA64_DTPMOD64MSB, "TPREL64MSB", 8, FALSE, FALSE), +- IA64_HOWTO (R_IA64_DTPMOD64LSB, "TPREL64LSB", 8, FALSE, FALSE), ++ IA64_HOWTO (R_IA64_DTPMOD64MSB, "TPREL64MSB", 4, FALSE, FALSE), ++ IA64_HOWTO (R_IA64_DTPMOD64LSB, "TPREL64LSB", 4, FALSE, FALSE), + IA64_HOWTO (R_IA64_LTOFF_DTPMOD22, "LTOFF_DTPMOD22", 0, FALSE, FALSE), + + IA64_HOWTO (R_IA64_DTPREL14, "DTPREL14", 0, FALSE, FALSE), + IA64_HOWTO (R_IA64_DTPREL22, "DTPREL22", 0, FALSE, FALSE), + IA64_HOWTO (R_IA64_DTPREL64I, "DTPREL64I", 0, FALSE, FALSE), +- IA64_HOWTO (R_IA64_DTPREL32MSB, "DTPREL32MSB", 4, FALSE, FALSE), +- IA64_HOWTO (R_IA64_DTPREL32LSB, "DTPREL32LSB", 4, FALSE, FALSE), +- IA64_HOWTO (R_IA64_DTPREL64MSB, "DTPREL64MSB", 8, FALSE, FALSE), +- IA64_HOWTO (R_IA64_DTPREL64LSB, "DTPREL64LSB", 8, FALSE, FALSE), ++ IA64_HOWTO (R_IA64_DTPREL32MSB, "DTPREL32MSB", 2, FALSE, FALSE), ++ IA64_HOWTO (R_IA64_DTPREL32LSB, "DTPREL32LSB", 2, FALSE, FALSE), ++ IA64_HOWTO (R_IA64_DTPREL64MSB, "DTPREL64MSB", 4, FALSE, FALSE), ++ IA64_HOWTO (R_IA64_DTPREL64LSB, "DTPREL64LSB", 4, FALSE, FALSE), + IA64_HOWTO (R_IA64_LTOFF_DTPREL22, "LTOFF_DTPREL22", 0, FALSE, FALSE), + }; + diff --git a/sys-devel/binutils/files/2.14/binutils-2.14.90.0.6-ia64-sdata.patch b/sys-devel/binutils/files/2.14/binutils-2.14.90.0.6-ia64-sdata.patch new file mode 100644 index 000000000000..55a8f67fa72d --- /dev/null +++ b/sys-devel/binutils/files/2.14/binutils-2.14.90.0.6-ia64-sdata.patch @@ -0,0 +1,33 @@ +2003-09-10 H.J. Lu <hongjiu.lu@intel.com> + + * elfxx-ia64.c (elfNN_ia64_special_sections): Mark all sections + started with ".sbss" or "".sdata" as SHF_IA_64_SHORT. + +2003-08-22 H.J. Lu <hongjiu.lu@intel.com> + + * elfxx-ia64.c (elfNN_ia64_create_dynamic_sections): Align the + .got section at 8 bytes. + +--- bfd/elfxx-ia64.c 20 Aug 2003 08:37:18 -0000 1.98 ++++ bfd/elfxx-ia64.c 10 Sep 2003 20:48:32 -0000 1.100 +@@ -1781,6 +1781,8 @@ elfNN_ia64_create_dynamic_sections (abfd + { + flagword flags = bfd_get_section_flags (abfd, ia64_info->got_sec); + bfd_set_section_flags (abfd, ia64_info->got_sec, SEC_SMALL_DATA | flags); ++ /* The .got section is always aligned at 8 bytes. */ ++ bfd_set_section_alignment (abfd, ia64_info->got_sec, 3); + } + + if (!get_pltoff (abfd, info, ia64_info)) +@@ -4746,9 +4748,9 @@ elfNN_ia64_reloc_type_class (rela) + + static struct bfd_elf_special_section const elfNN_ia64_special_sections[]= + { +- { ".sbss", 0, NULL, 0, ++ { ".sbss", 5, NULL, 0, + SHT_NOBITS, SHF_ALLOC + SHF_WRITE + SHF_IA_64_SHORT }, +- { ".sdata", 0, NULL, 0, ++ { ".sdata", 6, NULL, 0, + SHT_PROGBITS, SHF_ALLOC + SHF_WRITE + SHF_IA_64_SHORT }, + { NULL, 0, NULL, 0, + 0, 0 } diff --git a/sys-devel/binutils/files/2.14/binutils-2.14.90.0.6-ia64-speedup.patch b/sys-devel/binutils/files/2.14/binutils-2.14.90.0.6-ia64-speedup.patch new file mode 100644 index 000000000000..d93f6e98c351 --- /dev/null +++ b/sys-devel/binutils/files/2.14/binutils-2.14.90.0.6-ia64-speedup.patch @@ -0,0 +1,314 @@ +2003-09-11 Jakub Jelinek <jakub@redhat.com> + + * elfxx-ia64.c: Include objalloc.h, hashtab.h. + (struct elfNN_ia64_local_hash_entry): Remove root. Add id and r_sym + fields. + (struct elfNN_ia64_local_hash_table): Remove. + (struct elfNN_ia64_link_hash_table): Change loc_hash_table's type + to htab_t. Add loc_hash_memory field. + (elfNN_ia64_local_hash_table_init, elfNN_ia64_new_loc_hash_entry): + Removed. + (elfNN_ia64_local_htab_hash, elfNN_ia64_local_htab_eq): New + functions. + (elfNN_ia64_hash_table_create): Use hashtab.h hashtable for + loc_hash_table. Initialize loc_hash_memory. + (elfNN_ia64_hash_table_free): New function. + (elfNN_ia64_local_hash_lookup): Remove. + (elfNN_ia64_local_dyn_sym_thunk): Change into htab_traverse + callback. + (elfNN_ia64_dyn_sym_traverse): Use htab_traverse. + (get_local_sym_hash): Use hashtab.h hashtable for loc_hash_table. + (bfd_elfNN_bfd_link_hash_table_free): Define. + +--- bfd/elfxx-ia64.c.jj 2003-09-18 05:30:39.000000000 -0400 ++++ bfd/elfxx-ia64.c 2003-09-18 07:37:36.000000000 -0400 +@@ -24,6 +24,8 @@ + #include "elf-bfd.h" + #include "opcode/ia64.h" + #include "elf/ia64.h" ++#include "objalloc.h" ++#include "hashtab.h" + + /* THE RULES for all the stuff the linker creates -- + +@@ -115,7 +117,8 @@ struct elfNN_ia64_dyn_sym_info + + struct elfNN_ia64_local_hash_entry + { +- struct bfd_hash_entry root; ++ int id; ++ unsigned int r_sym; + struct elfNN_ia64_dyn_sym_info *info; + + /* TRUE if this hash entry's addends was translated for +@@ -123,12 +126,6 @@ struct elfNN_ia64_local_hash_entry + unsigned sec_merge_done : 1; + }; + +-struct elfNN_ia64_local_hash_table +-{ +- struct bfd_hash_table root; +- /* No additional fields for now. */ +-}; +- + struct elfNN_ia64_link_hash_entry + { + struct elf_link_hash_entry root; +@@ -153,7 +150,8 @@ struct elfNN_ia64_link_hash_table + unsigned self_dtpmod_done : 1;/* has self DTPMOD entry been finished? */ + bfd_vma self_dtpmod_offset; /* .got offset to self DTPMOD entry */ + +- struct elfNN_ia64_local_hash_table loc_hash_table; ++ htab_t loc_hash_table; ++ void *loc_hash_memory; + }; + + struct elfNN_ia64_allocate_data +@@ -201,12 +199,6 @@ static bfd_boolean elfNN_ia64_is_local_l + PARAMS ((bfd *abfd, const char *name)); + static bfd_boolean elfNN_ia64_dynamic_symbol_p + PARAMS ((struct elf_link_hash_entry *h, struct bfd_link_info *info, int)); +-static bfd_boolean elfNN_ia64_local_hash_table_init +- PARAMS ((struct elfNN_ia64_local_hash_table *ht, bfd *abfd, +- new_hash_entry_func new)); +-static struct bfd_hash_entry *elfNN_ia64_new_loc_hash_entry +- PARAMS ((struct bfd_hash_entry *entry, struct bfd_hash_table *table, +- const char *string)); + static struct bfd_hash_entry *elfNN_ia64_new_elf_hash_entry + PARAMS ((struct bfd_hash_entry *entry, struct bfd_hash_table *table, + const char *string)); +@@ -215,15 +207,17 @@ static void elfNN_ia64_hash_copy_indirec + struct elf_link_hash_entry *)); + static void elfNN_ia64_hash_hide_symbol + PARAMS ((struct bfd_link_info *, struct elf_link_hash_entry *, bfd_boolean)); ++static hashval_t elfNN_ia64_local_htab_hash PARAMS ((const void *)); ++static int elfNN_ia64_local_htab_eq PARAMS ((const void *ptr1, ++ const void *ptr2)); + static struct bfd_link_hash_table *elfNN_ia64_hash_table_create + PARAMS ((bfd *abfd)); +-static struct elfNN_ia64_local_hash_entry *elfNN_ia64_local_hash_lookup +- PARAMS ((struct elfNN_ia64_local_hash_table *table, const char *string, +- bfd_boolean create, bfd_boolean copy)); ++static void elfNN_ia64_hash_table_free ++ PARAMS ((struct bfd_link_hash_table *hash)); + static bfd_boolean elfNN_ia64_global_dyn_sym_thunk + PARAMS ((struct bfd_hash_entry *, PTR)); +-static bfd_boolean elfNN_ia64_local_dyn_sym_thunk +- PARAMS ((struct bfd_hash_entry *, PTR)); ++static int elfNN_ia64_local_dyn_sym_thunk ++ PARAMS ((void **, PTR)); + static void elfNN_ia64_dyn_sym_traverse + PARAMS ((struct elfNN_ia64_link_hash_table *ia64_info, + bfd_boolean (*func) (struct elfNN_ia64_dyn_sym_info *, PTR), +@@ -1520,44 +1514,6 @@ elfNN_ia64_dynamic_symbol_p (h, info, r_ + return _bfd_elf_dynamic_symbol_p (h, info, ignore_protected); + } + +-static bfd_boolean +-elfNN_ia64_local_hash_table_init (ht, abfd, new) +- struct elfNN_ia64_local_hash_table *ht; +- bfd *abfd ATTRIBUTE_UNUSED; +- new_hash_entry_func new; +-{ +- memset (ht, 0, sizeof (*ht)); +- return bfd_hash_table_init (&ht->root, new); +-} +- +-static struct bfd_hash_entry* +-elfNN_ia64_new_loc_hash_entry (entry, table, string) +- struct bfd_hash_entry *entry; +- struct bfd_hash_table *table; +- const char *string; +-{ +- struct elfNN_ia64_local_hash_entry *ret; +- ret = (struct elfNN_ia64_local_hash_entry *) entry; +- +- /* Allocate the structure if it has not already been allocated by a +- subclass. */ +- if (!ret) +- ret = bfd_hash_allocate (table, sizeof (*ret)); +- +- if (!ret) +- return 0; +- +- /* Initialize our local data. All zeros, and definitely easier +- than setting a handful of bit fields. */ +- memset (ret, 0, sizeof (*ret)); +- +- /* Call the allocation method of the superclass. */ +- ret = ((struct elfNN_ia64_local_hash_entry *) +- bfd_hash_newfunc ((struct bfd_hash_entry *) ret, table, string)); +- +- return (struct bfd_hash_entry *) ret; +-} +- + static struct bfd_hash_entry* + elfNN_ia64_new_elf_hash_entry (entry, table, string) + struct bfd_hash_entry *entry; +@@ -1657,6 +1613,33 @@ elfNN_ia64_hash_hide_symbol (info, xh, f + } + } + ++/* Compute a hash of a local hash entry. */ ++ ++static hashval_t ++elfNN_ia64_local_htab_hash (ptr) ++ const void *ptr; ++{ ++ struct elfNN_ia64_local_hash_entry *entry ++ = (struct elfNN_ia64_local_hash_entry *) ptr; ++ ++ return (((entry->id & 0xff) << 24) | ((entry->id & 0xff00) << 8)) ++ ^ entry->r_sym ^ (entry->id >> 16); ++} ++ ++/* Compare local hash entries. */ ++ ++static int ++elfNN_ia64_local_htab_eq (ptr1, ptr2) ++ const void *ptr1, *ptr2; ++{ ++ struct elfNN_ia64_local_hash_entry *entry1 ++ = (struct elfNN_ia64_local_hash_entry *) ptr1; ++ struct elfNN_ia64_local_hash_entry *entry2 ++ = (struct elfNN_ia64_local_hash_entry *) ptr2; ++ ++ return entry1->id == entry2->id && entry1->r_sym == entry2->r_sym; ++} ++ + /* Create the derived linker hash table. The IA-64 ELF port uses this + derived hash table to keep information specific to the IA-64 ElF + linker (without using static variables). */ +@@ -1678,8 +1661,10 @@ elfNN_ia64_hash_table_create (abfd) + return 0; + } + +- if (!elfNN_ia64_local_hash_table_init (&ret->loc_hash_table, abfd, +- elfNN_ia64_new_loc_hash_entry)) ++ ret->loc_hash_table = htab_try_create (1024, elfNN_ia64_local_htab_hash, ++ elfNN_ia64_local_htab_eq, NULL); ++ ret->loc_hash_memory = objalloc_create (); ++ if (!ret->loc_hash_table || !ret->loc_hash_memory) + { + free (ret); + return 0; +@@ -1688,16 +1673,19 @@ elfNN_ia64_hash_table_create (abfd) + return &ret->root.root; + } + +-/* Look up an entry in a Alpha ELF linker hash table. */ ++/* Destroy IA-64 linker hash table. */ + +-static INLINE struct elfNN_ia64_local_hash_entry * +-elfNN_ia64_local_hash_lookup(table, string, create, copy) +- struct elfNN_ia64_local_hash_table *table; +- const char *string; +- bfd_boolean create, copy; ++static void ++elfNN_ia64_hash_table_free (hash) ++ struct bfd_link_hash_table *hash; + { +- return ((struct elfNN_ia64_local_hash_entry *) +- bfd_hash_lookup (&table->root, string, create, copy)); ++ struct elfNN_ia64_link_hash_table *ia64_info ++ = (struct elfNN_ia64_link_hash_table *) hash; ++ if (ia64_info->loc_hash_table) ++ htab_delete (ia64_info->loc_hash_table); ++ if (ia64_info->loc_hash_memory) ++ objalloc_free ((struct objalloc *) ia64_info->loc_hash_memory); ++ _bfd_generic_link_hash_table_free (hash); + } + + /* Traverse both local and global hash tables. */ +@@ -1729,20 +1717,20 @@ elfNN_ia64_global_dyn_sym_thunk (xentry, + } + + static bfd_boolean +-elfNN_ia64_local_dyn_sym_thunk (xentry, xdata) +- struct bfd_hash_entry *xentry; ++elfNN_ia64_local_dyn_sym_thunk (slot, xdata) ++ void **slot; + PTR xdata; + { + struct elfNN_ia64_local_hash_entry *entry +- = (struct elfNN_ia64_local_hash_entry *) xentry; ++ = (struct elfNN_ia64_local_hash_entry *) *slot; + struct elfNN_ia64_dyn_sym_traverse_data *data + = (struct elfNN_ia64_dyn_sym_traverse_data *) xdata; + struct elfNN_ia64_dyn_sym_info *dyn_i; + + for (dyn_i = entry->info; dyn_i; dyn_i = dyn_i->next) + if (! (*data->func) (dyn_i, data->data)) +- return FALSE; +- return TRUE; ++ return 0; ++ return 1; + } + + static void +@@ -1758,8 +1746,8 @@ elfNN_ia64_dyn_sym_traverse (ia64_info, + + elf_link_hash_traverse (&ia64_info->root, + elfNN_ia64_global_dyn_sym_thunk, &xdata); +- bfd_hash_traverse (&ia64_info->loc_hash_table.root, +- elfNN_ia64_local_dyn_sym_thunk, &xdata); ++ htab_traverse (ia64_info->loc_hash_table, ++ elfNN_ia64_local_dyn_sym_thunk, &xdata); + } + + static bfd_boolean +@@ -1819,22 +1807,33 @@ get_local_sym_hash (ia64_info, abfd, rel + const Elf_Internal_Rela *rel; + bfd_boolean create; + { +- struct elfNN_ia64_local_hash_entry *ret; ++ struct elfNN_ia64_local_hash_entry e, *ret; + asection *sec = abfd->sections; +- char addr_name [34]; ++ hashval_t h = (((sec->id & 0xff) << 24) | ((sec->id & 0xff00) << 8)) ++ ^ ELFNN_R_SYM (rel->r_info) ^ (sec->id >> 16); ++ void **slot; ++ ++ e.id = sec->id; ++ e.r_sym = ELFNN_R_SYM (rel->r_info); ++ slot = htab_find_slot_with_hash (ia64_info->loc_hash_table, &e, h, ++ create ? INSERT : NO_INSERT); + +- BFD_ASSERT ((sizeof (sec->id)*2 + 1 + sizeof (unsigned long)*2 + 1) <= 34); +- BFD_ASSERT (sec); +- +- /* Construct a string for use in the elfNN_ia64_local_hash_table. +- name describes what was once anonymous memory. */ ++ if (!slot) ++ return NULL; + +- sprintf (addr_name, "%x:%lx", +- sec->id, (unsigned long) ELFNN_R_SYM (rel->r_info)); ++ if (*slot) ++ return (struct elfNN_ia64_local_hash_entry *) *slot; + +- /* Collect the canonical entry data for this address. */ +- ret = elfNN_ia64_local_hash_lookup (&ia64_info->loc_hash_table, +- addr_name, create, create); ++ ret = (struct elfNN_ia64_local_hash_entry *) ++ objalloc_alloc ((struct objalloc *) ia64_info->loc_hash_memory, ++ sizeof (struct elfNN_ia64_local_hash_entry)); ++ if (ret) ++ { ++ memset (ret, 0, sizeof (*ret)); ++ ret->id = sec->id; ++ ret->r_sym = ELFNN_R_SYM (rel->r_info); ++ *slot = ret; ++ } + return ret; + } + +@@ -4823,6 +4822,8 @@ elfNN_hpux_backend_section_from_bfd_sect + /* Stuff for the BFD linker: */ + #define bfd_elfNN_bfd_link_hash_table_create \ + elfNN_ia64_hash_table_create ++#define bfd_elfNN_bfd_link_hash_table_free \ ++ elfNN_ia64_hash_table_free + #define elf_backend_create_dynamic_sections \ + elfNN_ia64_create_dynamic_sections + #define elf_backend_check_relocs \ diff --git a/sys-devel/binutils/files/2.14/binutils-2.14.90.0.6-merge-speedup.patch b/sys-devel/binutils/files/2.14/binutils-2.14.90.0.6-merge-speedup.patch new file mode 100644 index 000000000000..671538cd05b8 --- /dev/null +++ b/sys-devel/binutils/files/2.14/binutils-2.14.90.0.6-merge-speedup.patch @@ -0,0 +1,740 @@ +2003-09-10 Lars Knoll <lars@trolltech.com> + Michael Matz <matz@suse.de> + Alan Modra <amodra@bigpond.net.au> + Jakub Jelinek <jakub@redhat.com> + + * merge.c (struct elf_strtab_hash_entry): Make len signed. + Remove u.next field. + (cmplengthentry, last4_eq, last_eq): Delete. + (strrevcmp, strrevcmp_align, is_suffix): New. + (merge_strings): Use them to implement fast suffix merging. + * elf-strtab.c (cmplengthentry, last4_eq): Delete. + (strrevcmp, is_suffix): New. + (_bfd_elf_strtab_finalize): Rework to implement fast suffix merging. + + * ld-elf/merge2.s: New test. + * ld-elf/merge2.d: New. + +--- bfd/elf-strtab.c.jj 2002-12-16 15:22:51.000000000 -0500 ++++ bfd/elf-strtab.c 2003-09-18 07:24:00.000000000 -0400 +@@ -1,5 +1,5 @@ + /* ELF strtab with GC and suffix merging support. +- Copyright 2001, 2002 Free Software Foundation, Inc. ++ Copyright 2001, 2002, 2003 Free Software Foundation, Inc. + Written by Jakub Jelinek <jakub@redhat.com>. + + This file is part of BFD, the Binary File Descriptor library. +@@ -30,15 +30,14 @@ + struct elf_strtab_hash_entry + { + struct bfd_hash_entry root; +- /* Length of this entry. */ +- unsigned int len; ++ /* Length of this entry. This includes the zero terminator. */ ++ int len; + unsigned int refcount; + union { + /* Index within the merged section. */ + bfd_size_type index; +- /* Entry this is a suffix of (if len is 0). */ ++ /* Entry this is a suffix of (if len < 0). */ + struct elf_strtab_hash_entry *suffix; +- struct elf_strtab_hash_entry *next; + } u; + }; + +@@ -158,6 +157,8 @@ _bfd_elf_strtab_add (struct elf_strtab_h + if (entry->len == 0) + { + entry->len = strlen (str) + 1; ++ /* 2G strings lose. */ ++ BFD_ASSERT (entry->len > 0); + if (tab->size == tab->alloced) + { + bfd_size_type amt = sizeof (struct elf_strtab_hash_entry *); +@@ -235,14 +236,14 @@ _bfd_elf_strtab_emit (register bfd *abfd + for (i = 1; i < tab->size; ++i) + { + register const char *str; +- register size_t len; ++ register unsigned int len; + +- str = tab->array[i]->root.string; +- len = tab->array[i]->len; + BFD_ASSERT (tab->array[i]->refcount == 0); +- if (len == 0) ++ len = tab->array[i]->len; ++ if ((int) len < 0) + continue; + ++ str = tab->array[i]->root.string; + if (bfd_bwrite (str, len, abfd) != len) + return FALSE; + +@@ -256,37 +257,38 @@ _bfd_elf_strtab_emit (register bfd *abfd + /* Compare two elf_strtab_hash_entry structures. This is called via qsort. */ + + static int +-cmplengthentry (const void *a, const void *b) ++strrevcmp (const void *a, const void *b) + { + struct elf_strtab_hash_entry *A = *(struct elf_strtab_hash_entry **) a; + struct elf_strtab_hash_entry *B = *(struct elf_strtab_hash_entry **) b; ++ unsigned int lenA = A->len; ++ unsigned int lenB = B->len; ++ const unsigned char *s = A->root.string + lenA - 1; ++ const unsigned char *t = B->root.string + lenB - 1; ++ int l = lenA < lenB ? lenA : lenB; + +- if (A->len < B->len) +- return 1; +- else if (A->len > B->len) +- return -1; +- +- return memcmp (A->root.string, B->root.string, A->len); ++ while (l) ++ { ++ if (*s != *t) ++ return (int) *s - (int) *t; ++ s--; ++ t--; ++ l--; ++ } ++ return lenA - lenB; + } + + static int +-last4_eq (const void *a, const void *b) ++is_suffix (const struct elf_strtab_hash_entry *A, ++ const struct elf_strtab_hash_entry *B) + { +- const struct elf_strtab_hash_entry *A = a; +- const struct elf_strtab_hash_entry *B = b; +- +- if (memcmp (A->root.string + A->len - 5, B->root.string + B->len - 5, 4) +- != 0) +- /* This was a hashtable collision. */ +- return 0; +- + if (A->len <= B->len) + /* B cannot be a suffix of A unless A is equal to B, which is guaranteed + not to be equal by the hash table. */ + return 0; + + return memcmp (A->root.string + (A->len - B->len), +- B->root.string, B->len - 5) == 0; ++ B->root.string, B->len - 1) == 0; + } + + /* This function assigns final string table offsets for used strings, +@@ -295,10 +297,8 @@ last4_eq (const void *a, const void *b) + void + _bfd_elf_strtab_finalize (struct elf_strtab_hash *tab) + { +- struct elf_strtab_hash_entry **array, **a, **end, *e; +- htab_t last4tab = NULL; ++ struct elf_strtab_hash_entry **array, **a, *e; + bfd_size_type size, amt; +- struct elf_strtab_hash_entry *last[256], **last_ptr[256]; + + /* GCC 2.91.66 (egcs-1.1.2) on i386 miscompiles this function when i is + a 64-bit bfd_size_type: a 64-bit target or --enable-64-bit-bfd. +@@ -306,105 +306,71 @@ _bfd_elf_strtab_finalize (struct elf_str + cycles. */ + size_t i; + +- /* Now sort the strings by length, longest first. */ +- array = NULL; ++ /* Sort the strings by suffix and length. */ + amt = tab->size * sizeof (struct elf_strtab_hash_entry *); + array = bfd_malloc (amt); + if (array == NULL) + goto alloc_failure; + +- memset (last, 0, sizeof (last)); +- for (i = 0; i < 256; ++i) +- last_ptr[i] = &last[i]; + for (i = 1, a = array; i < tab->size; ++i) +- if (tab->array[i]->refcount) +- *a++ = tab->array[i]; +- else +- tab->array[i]->len = 0; ++ { ++ e = tab->array[i]; ++ if (e->refcount) ++ { ++ *a++ = e; ++ /* Adjust the length to not include the zero terminator. */ ++ e->len -= 1; ++ } ++ else ++ e->len = 0; ++ } + + size = a - array; ++ if (size != 0) ++ { ++ qsort (array, size, sizeof (struct elf_strtab_hash_entry *), strrevcmp); + +- qsort (array, size, sizeof (struct elf_strtab_hash_entry *), cmplengthentry); ++ /* Loop over the sorted array and merge suffixes. Start from the ++ end because we want eg. + +- last4tab = htab_create_alloc (size * 4, NULL, last4_eq, NULL, calloc, free); +- if (last4tab == NULL) +- goto alloc_failure; ++ s1 -> "d" ++ s2 -> "bcd" ++ s3 -> "abcd" + +- /* Now insert the strings into hash tables (strings with last 4 characters +- and strings with last character equal), look for longer strings which +- we're suffix of. */ +- for (a = array, end = array + size; a < end; a++) +- { +- register hashval_t hash; +- unsigned int c; +- unsigned int j; +- const unsigned char *s; +- void **p; ++ to end up as + +- e = *a; +- if (e->len > 4) +- { +- s = e->root.string + e->len - 1; +- hash = 0; +- for (j = 0; j < 4; j++) +- { +- c = *--s; +- hash += c + (c << 17); +- hash ^= hash >> 2; +- } +- p = htab_find_slot_with_hash (last4tab, e, hash, INSERT); +- if (p == NULL) +- goto alloc_failure; +- if (*p) +- { +- struct elf_strtab_hash_entry *ent; ++ s3 -> "abcd" ++ s2 _____^ ++ s1 _______^ + +- ent = *p; +- e->u.suffix = ent; +- e->len = 0; +- continue; +- } +- else +- *p = e; +- } +- else ++ ie. we don't want s1 pointing into the old s2. */ ++ e = *--a; ++ e->len += 1; ++ while (--a >= array) + { +- struct elf_strtab_hash_entry *tem; ++ struct elf_strtab_hash_entry *cmp = *a; + +- c = e->root.string[e->len - 2] & 0xff; +- +- for (tem = last[c]; tem; tem = tem->u.next) +- if (tem->len > e->len +- && memcmp (tem->root.string + (tem->len - e->len), +- e->root.string, e->len - 1) == 0) +- break; +- if (tem) ++ cmp->len += 1; ++ if (is_suffix (e, cmp)) + { +- e->u.suffix = tem; +- e->len = 0; +- continue; ++ cmp->u.suffix = e; ++ cmp->len = -cmp->len; + } ++ else ++ e = cmp; + } +- +- c = e->root.string[e->len - 2] & 0xff; +- /* Put longest strings first. */ +- *last_ptr[c] = e; +- last_ptr[c] = &e->u.next; +- e->u.next = NULL; + } + + alloc_failure: + if (array) + free (array); +- if (last4tab) +- htab_delete (last4tab); + +- /* Now assign positions to the strings we want to keep. */ ++ /* Assign positions to the strings we want to keep. */ + size = 1; + for (i = 1; i < tab->size; ++i) + { + e = tab->array[i]; +- if (e->refcount && e->len) ++ if (e->refcount && e->len > 0) + { + e->u.index = size; + size += e->len; +@@ -413,12 +379,11 @@ alloc_failure: + + tab->sec_size = size; + +- /* And now adjust the rest. */ ++ /* Adjust the rest. */ + for (i = 1; i < tab->size; ++i) + { + e = tab->array[i]; +- if (e->refcount && ! e->len) +- e->u.index = e->u.suffix->u.index +- + (e->u.suffix->len - strlen (e->root.string) - 1); ++ if (e->refcount && e->len < 0) ++ e->u.index = e->u.suffix->u.index + (e->u.suffix->len + e->len); + } + } +--- bfd/merge.c.jj 2002-12-16 15:22:52.000000000 -0500 ++++ bfd/merge.c 2003-09-18 07:24:32.000000000 -0400 +@@ -1,5 +1,5 @@ + /* SEC_MERGE support. +- Copyright 2001, 2002 Free Software Foundation, Inc. ++ Copyright 2001, 2002, 2003 Free Software Foundation, Inc. + Written by Jakub Jelinek <jakub@redhat.com>. + + This file is part of BFD, the Binary File Descriptor library. +@@ -34,7 +34,7 @@ struct sec_merge_sec_info; + struct sec_merge_hash_entry + { + struct bfd_hash_entry root; +- /* Length of this entry. */ ++ /* Length of this entry. This includes the zero terminator. */ + unsigned int len; + /* Start of this string needs to be aligned to + alignment octets (not 1 << align). */ +@@ -43,8 +43,6 @@ struct sec_merge_hash_entry + { + /* Index within the merged section. */ + bfd_size_type index; +- /* Entity size (if present in suffix hash tables). */ +- unsigned int entsize; + /* Entry this is a suffix of (if alignment is 0). */ + struct sec_merge_hash_entry *suffix; + } u; +@@ -108,12 +106,10 @@ static struct sec_merge_hash_entry *sec_ + struct sec_merge_sec_info *)); + static bfd_boolean sec_merge_emit + PARAMS ((bfd *, struct sec_merge_hash_entry *)); +-static int cmplengthentry +- PARAMS ((const PTR, const PTR)); +-static int last4_eq +- PARAMS ((const PTR, const PTR)); +-static int last_eq +- PARAMS ((const PTR, const PTR)); ++static int strrevcmp PARAMS ((const PTR, const PTR)); ++static int strrevcmp_align PARAMS ((const PTR, const PTR)); ++static int is_suffix PARAMS ((const struct sec_merge_hash_entry *, ++ const struct sec_merge_hash_entry *)); + static bfd_boolean record_section + PARAMS ((struct sec_merge_info *, struct sec_merge_sec_info *)); + static void merge_strings +@@ -231,9 +227,12 @@ sec_merge_hash_lookup (table, string, al + alignment, we need to insert another copy. */ + if (hashp->alignment < alignment) + { +- /* Mark the less aligned copy as deleted. */ +- hashp->len = 0; +- hashp->alignment = 0; ++ if (create) ++ { ++ /* Mark the less aligned copy as deleted. */ ++ hashp->len = 0; ++ hashp->alignment = 0; ++ } + break; + } + return hashp; +@@ -457,85 +456,6 @@ _bfd_merge_section (abfd, psinfo, sec, p + return FALSE; + } + +-/* Compare two sec_merge_hash_entry structures. This is called via qsort. */ +- +-static int +-cmplengthentry (a, b) +- const PTR a; +- const PTR b; +-{ +- struct sec_merge_hash_entry * A = *(struct sec_merge_hash_entry **) a; +- struct sec_merge_hash_entry * B = *(struct sec_merge_hash_entry **) b; +- +- if (A->len < B->len) +- return 1; +- else if (A->len > B->len) +- return -1; +- +- return memcmp (A->root.string, B->root.string, A->len); +-} +- +-static int +-last4_eq (a, b) +- const PTR a; +- const PTR b; +-{ +- struct sec_merge_hash_entry * A = (struct sec_merge_hash_entry *) a; +- struct sec_merge_hash_entry * B = (struct sec_merge_hash_entry *) b; +- +- if (memcmp (A->root.string + A->len - 5 * A->u.entsize, +- B->root.string + B->len - 5 * A->u.entsize, +- 4 * A->u.entsize) != 0) +- /* This was a hashtable collision. */ +- return 0; +- +- if (A->len <= B->len) +- /* B cannot be a suffix of A unless A is equal to B, which is guaranteed +- not to be equal by the hash table. */ +- return 0; +- +- if (A->alignment < B->alignment +- || ((A->len - B->len) & (B->alignment - 1))) +- /* The suffix is not sufficiently aligned. */ +- return 0; +- +- return memcmp (A->root.string + (A->len - B->len), +- B->root.string, B->len - 5 * A->u.entsize) == 0; +-} +- +-static int +-last_eq (a, b) +- const PTR a; +- const PTR b; +-{ +- struct sec_merge_hash_entry * A = (struct sec_merge_hash_entry *) a; +- struct sec_merge_hash_entry * B = (struct sec_merge_hash_entry *) b; +- +- if (B->len >= 5 * A->u.entsize) +- /* Longer strings are just pushed into the hash table, +- they'll be used when looking up for very short strings. */ +- return 0; +- +- if (memcmp (A->root.string + A->len - 2 * A->u.entsize, +- B->root.string + B->len - 2 * A->u.entsize, +- A->u.entsize) != 0) +- /* This was a hashtable collision. */ +- return 0; +- +- if (A->len <= B->len) +- /* B cannot be a suffix of A unless A is equal to B, which is guaranteed +- not to be equal by the hash table. */ +- return 0; +- +- if (A->alignment < B->alignment +- || ((A->len - B->len) & (B->alignment - 1))) +- /* The suffix is not sufficiently aligned. */ +- return 0; +- +- return memcmp (A->root.string + (A->len - B->len), +- B->root.string, B->len - 2 * A->u.entsize) == 0; +-} +- + /* Record one section into the hash table. */ + static bfd_boolean + record_section (sinfo, secinfo) +@@ -620,19 +540,89 @@ error_return: + return FALSE; + } + ++static int ++strrevcmp (a, b) ++ const PTR a; ++ const PTR b; ++{ ++ struct sec_merge_hash_entry *A = *(struct sec_merge_hash_entry **) a; ++ struct sec_merge_hash_entry *B = *(struct sec_merge_hash_entry **) b; ++ int lenA = A->len; ++ int lenB = B->len; ++ const unsigned char *s = A->root.string + lenA - 1; ++ const unsigned char *t = B->root.string + lenB - 1; ++ int l = lenA < lenB ? lenA : lenB; ++ ++ while (l) ++ { ++ if (*s != *t) ++ return (int) *s - (int) *t; ++ s--; ++ t--; ++ l--; ++ } ++ return lenA - lenB; ++} ++ ++/* Like strrevcmp, but for the case where all strings have the same ++ alignment > entsize. */ ++ ++static int ++strrevcmp_align (a, b) ++ const PTR a; ++ const PTR b; ++{ ++ struct sec_merge_hash_entry *A = *(struct sec_merge_hash_entry **) a; ++ struct sec_merge_hash_entry *B = *(struct sec_merge_hash_entry **) b; ++ int lenA = A->len; ++ int lenB = B->len; ++ const unsigned char *s = A->root.string + lenA - 1; ++ const unsigned char *t = B->root.string + lenB - 1; ++ int l = lenA < lenB ? lenA : lenB; ++ ++ if ((lenA & (A->alignment - 1)) < (lenB & (A->alignment - 1))) ++ return -1; ++ if ((lenA & (A->alignment - 1)) > (lenB & (A->alignment - 1))) ++ return 1; ++ ++ while (l) ++ { ++ if (*s != *t) ++ return (int) *s - (int) *t; ++ s--; ++ t--; ++ l--; ++ } ++ return lenA - lenB; ++} ++ ++static INLINE int ++is_suffix (A, B) ++ const struct sec_merge_hash_entry *A; ++ const struct sec_merge_hash_entry *B; ++{ ++ if (A->len <= B->len) ++ /* B cannot be a suffix of A unless A is equal to B, which is guaranteed ++ not to be equal by the hash table. */ ++ return 0; ++ ++ return memcmp (A->root.string + (A->len - B->len), ++ B->root.string, B->len) == 0; ++} ++ + /* This is a helper function for _bfd_merge_sections. It attempts to + merge strings matching suffixes of longer strings. */ + static void + merge_strings (sinfo) + struct sec_merge_info *sinfo; + { +- struct sec_merge_hash_entry **array, **a, **end, *e; ++ struct sec_merge_hash_entry **array, **a, *e; + struct sec_merge_sec_info *secinfo; +- htab_t lasttab = NULL, last4tab = NULL; + bfd_size_type size, amt; ++ unsigned int alignment = 0; ++ bfd_boolean equal_align = TRUE; + +- /* Now sort the strings by length, longest first. */ +- array = NULL; ++ /* Now sort the strings */ + amt = sinfo->htab->size * sizeof (struct sec_merge_hash_entry *); + array = (struct sec_merge_hash_entry **) bfd_malloc (amt); + if (array == NULL) +@@ -640,90 +630,47 @@ merge_strings (sinfo) + + for (e = sinfo->htab->first, a = array; e; e = e->next) + if (e->alignment) +- *a++ = e; ++ { ++ *a++ = e; ++ /* Adjust the length to not include the zero terminator. */ ++ e->len -= sinfo->htab->entsize; ++ if (!alignment) ++ alignment = e->alignment; ++ else if (alignment != e->alignment) ++ equal_align = FALSE; ++ } + + sinfo->htab->size = a - array; +- +- qsort (array, (size_t) sinfo->htab->size, +- sizeof (struct sec_merge_hash_entry *), cmplengthentry); +- +- last4tab = htab_create_alloc ((size_t) sinfo->htab->size * 4, +- NULL, last4_eq, NULL, calloc, free); +- lasttab = htab_create_alloc ((size_t) sinfo->htab->size * 4, +- NULL, last_eq, NULL, calloc, free); +- if (lasttab == NULL || last4tab == NULL) +- goto alloc_failure; +- +- /* Now insert the strings into hash tables (strings with last 4 characters +- and strings with last character equal), look for longer strings which +- we're suffix of. */ +- for (a = array, end = array + sinfo->htab->size; a < end; a++) +- { +- register hashval_t hash; +- unsigned int c; +- unsigned int i; +- const unsigned char *s; +- PTR *p; +- +- e = *a; +- e->u.entsize = sinfo->htab->entsize; +- if (e->len <= e->u.entsize) +- break; +- if (e->len > 4 * e->u.entsize) +- { +- s = (const unsigned char *) (e->root.string + e->len - e->u.entsize); +- hash = 0; +- for (i = 0; i < 4 * e->u.entsize; i++) +- { +- c = *--s; +- hash += c + (c << 17); +- hash ^= hash >> 2; +- } +- p = htab_find_slot_with_hash (last4tab, e, hash, INSERT); +- if (p == NULL) +- goto alloc_failure; +- if (*p) ++ if (sinfo->htab->size != 0) ++ { ++ qsort (array, (size_t) sinfo->htab->size, ++ sizeof (struct sec_merge_hash_entry *), ++ (equal_align && alignment > sinfo->htab->entsize) ++ ? strrevcmp_align : strrevcmp); ++ ++ /* Loop over the sorted array and merge suffixes */ ++ e = *--a; ++ e->len += sinfo->htab->entsize; ++ while (--a >= array) ++ { ++ struct sec_merge_hash_entry *cmp = *a; ++ ++ cmp->len += sinfo->htab->entsize; ++ if (e->alignment >= cmp->alignment ++ && !((e->len - cmp->len) & (cmp->alignment - 1)) ++ && is_suffix (e, cmp)) + { +- struct sec_merge_hash_entry *ent; +- +- ent = (struct sec_merge_hash_entry *) *p; +- e->u.suffix = ent; +- e->alignment = 0; +- continue; ++ cmp->u.suffix = e; ++ cmp->alignment = 0; + } + else +- *p = (PTR) e; +- } +- s = (const unsigned char *) (e->root.string + e->len - e->u.entsize); +- hash = 0; +- for (i = 0; i < e->u.entsize; i++) +- { +- c = *--s; +- hash += c + (c << 17); +- hash ^= hash >> 2; +- } +- p = htab_find_slot_with_hash (lasttab, e, hash, INSERT); +- if (p == NULL) +- goto alloc_failure; +- if (*p) +- { +- struct sec_merge_hash_entry *ent; +- +- ent = (struct sec_merge_hash_entry *) *p; +- e->u.suffix = ent; +- e->alignment = 0; ++ e = cmp; + } +- else +- *p = (PTR) e; + } + + alloc_failure: + if (array) + free (array); +- if (lasttab) +- htab_delete (lasttab); +- if (last4tab) +- htab_delete (last4tab); + + /* Now assign positions to the strings we want to keep. */ + size = 0; +--- ld/testsuite/ld-elf/merge2.d.jj 2003-09-18 05:44:07.000000000 -0400 ++++ ld/testsuite/ld-elf/merge2.d 2003-09-18 05:44:07.000000000 -0400 +@@ -0,0 +1,15 @@ ++#source: merge2.s ++#ld: -T merge.ld ++#objdump: -s ++ ++.*: file format .*elf.* ++ ++Contents of section .text: ++ 1000 (3010)?0000(1030)? (3210)?0000(1032)? (3110)?0000(1031)? (3410)?0000(1034)? .* ++ 1010 (4010)?0000(1040)? (3810)?0000(1038)? (4810)?0000(1048)? (3c10)?0000(103c)? .* ++ 1020 (5010)?0000(1050)? (5410)?0000(1054)? (5810)?0000(1058)? (5010)?0000(1050)? .* ++Contents of section .rodata: ++ 1030 61626300 62000000 (78563412|12345678) 99999999 .* ++ 1040 (78563412|12345678) 00000000 99999999 00000000 .* ++ 1050 (78563412|12345678) 99999999 00000000 .* ++#pass +--- ld/testsuite/ld-elf/merge2.s.jj 2003-09-18 05:44:07.000000000 -0400 ++++ ld/testsuite/ld-elf/merge2.s 2003-09-18 05:44:07.000000000 -0400 +@@ -0,0 +1,58 @@ ++ .section .rodata.str,"aMS","progbits",1 ++.LC0: ++ .asciz "abc" ++.LC1: ++ .asciz "c" ++.LC2: ++ .asciz "bc" ++.LC3: ++ .asciz "b" ++ ++ ++ .section .rodata.str2,"aMS","progbits",4 ++ .p2align 2 ++.LC4: ++ .long 0x12345678 ++ .long 0 ++.LC5: ++ .long 0x12345678 ++ .long 0x99999999 ++ .long 0x12345678 ++ .long 0 ++.LC6: ++ .long 0x99999999 ++ .long 0 ++.LC7: ++ .long 0x99999999 ++ .long 0x12345678 ++ .long 0 ++ ++ ++ .section .rodata.m,"aM","progbits",4 ++ .p2align 2 ++.LC8: ++ .long 0x12345678 ++.LC9: ++ .long 0x99999999 ++.LC10: ++ .long 0 ++.LC11: ++ .long 0x12345678 ++ ++ ++ .text ++ .global _start ++_start: ++ .long .LC0 ++.LT0: ++ .long .LC1 ++ .long .LC2 ++ .long .LC3 ++ .long .LC4 ++ .long .LC5 ++ .long .LC6 ++ .long .LC7 ++ .long .LC8 ++ .long .LC9 ++ .long .LC10 ++ .long .LC11 diff --git a/sys-devel/binutils/files/2.14/binutils-2.14.90.0.6-sparc-cfi.patch b/sys-devel/binutils/files/2.14/binutils-2.14.90.0.6-sparc-cfi.patch new file mode 100644 index 000000000000..94342cc65633 --- /dev/null +++ b/sys-devel/binutils/files/2.14/binutils-2.14.90.0.6-sparc-cfi.patch @@ -0,0 +1,367 @@ +2003-08-29 Jakub Jelinek <jakub@redhat.com> + + * dw2gencfi.c (cfi_pseudo_table): Add cfi_window_save. + (dot_cfi, output_cfi_insn): Handle DW_CFA_GNU_window_save. + (output_cie): Don't use DW_EH_PE_pcrel if neither DIFF_EXPR_OK + nor tc_cfi_emit_pcrel_expr are defined. + (output_fde): Use tc_cfi_emit_pcrel_expr if available and + DIFF_EXPR_OK is not defined. + * config/tc-sparc.h (TARGET_USE_CFIPOP): Define. + (tc_cfi_frame_initial_instructions, tc_regname_to_dw2regnum, + tc_cfi_emit_pcrel_expr): Define. + (sparc_cfi_frame_initial_instructions, sparc_regname_to_dw2regnum, + sparc_cfi_emit_pcrel_expr): New prototypes. + (sparc_cie_data_alignment): New decl. + (DWARF2_DEFAULT_RETURN_COLUMN, DWARF2_CIE_DATA_ALIGNMENT): Define. + * config/tc-sparc.c: Include dw2gencfi.h. + (sparc_cie_data_alignment): New variable. + (md_begin): Initialize it. + (sparc_cfi_frame_initial_instructions): New function. + (sparc_regname_to_dw2regnum): Likewise. + (sparc_cfi_emit_pcrel_expr): Likewise. + * doc/as.texinfo: Document .cfi_window_save. + + * config/tc-sparc.c (s_common): Cast last argument to long and + change format string to shut up warning. +testsuite/ + * gas/cfi/cfi-sparc-1.s: New test. + * gas/cfi/cfi-sparc-1.d: New test. + * gas/cfi/cfi-sparc64-1.s: New test. + * gas/cfi/cfi-sparc64-1.d: New test. + * gas/cfi/cfi.exp: Run them. + +--- gas/dw2gencfi.c.jj 2003-06-18 13:57:59.000000000 -0400 ++++ gas/dw2gencfi.c 2003-08-29 14:00:41.000000000 -0400 +@@ -374,6 +374,7 @@ const pseudo_typeS cfi_pseudo_table[] = + { "cfi_same_value", dot_cfi, DW_CFA_same_value }, + { "cfi_remember_state", dot_cfi, DW_CFA_remember_state }, + { "cfi_restore_state", dot_cfi, DW_CFA_restore_state }, ++ { "cfi_window_save", dot_cfi, DW_CFA_GNU_window_save }, + { "cfi_escape", dot_cfi_escape, 0 }, + { NULL, NULL, 0 } + }; +@@ -529,6 +530,10 @@ dot_cfi (int arg) + cfi_add_CFA_restore_state (); + break; + ++ case DW_CFA_GNU_window_save: ++ cfi_add_CFA_insn (DW_CFA_GNU_window_save); ++ break; ++ + default: + abort (); + } +@@ -798,6 +803,10 @@ output_cfi_insn (struct cfi_insn_data *i + out_one (insn->insn); + break; + ++ case DW_CFA_GNU_window_save: ++ out_one (DW_CFA_GNU_window_save); ++ break; ++ + case CFI_escape: + { + struct cfi_escape_data *e; +@@ -838,7 +847,11 @@ output_cie (struct cie_entry *cie) + out_sleb128 (DWARF2_CIE_DATA_ALIGNMENT); /* Data alignment */ + out_one (cie->return_column); /* Return column */ + out_uleb128 (1); /* Augmentation size */ ++#if defined DIFF_EXPR_OK || defined tc_cfi_emit_pcrel_expr + out_one (DW_EH_PE_pcrel | DW_EH_PE_sdata4); ++#else ++ out_one (DW_EH_PE_sdata4); ++#endif + + if (cie->first) + for (i = cie->first; i != cie->last; i = i->next) +@@ -867,10 +880,22 @@ output_fde (struct fde_entry *fde, struc + exp.X_add_symbol = after_size_address; + exp.X_op_symbol = cie->start_address; + emit_expr (&exp, 4); /* CIE offset */ +- ++ ++#ifdef DIFF_EXPR_OK + exp.X_add_symbol = fde->start_address; + exp.X_op_symbol = symbol_temp_new_now (); + emit_expr (&exp, 4); /* Code offset */ ++#else ++ exp.X_op = O_symbol; ++ exp.X_add_symbol = fde->start_address; ++ exp.X_op_symbol = NULL; ++#ifdef tc_cfi_emit_pcrel_expr ++ tc_cfi_emit_pcrel_expr (&exp, 4); /* Code offset */ ++#else ++ emit_expr (&exp, 4); /* Code offset */ ++#endif ++ exp.X_op = O_subtract; ++#endif + + exp.X_add_symbol = fde->end_address; + exp.X_op_symbol = fde->start_address; /* Code length */ +--- gas/config/tc-sparc.h.jj 2003-01-24 18:37:51.000000000 -0500 ++++ gas/config/tc-sparc.h 2003-08-29 15:38:47.000000000 -0400 +@@ -174,6 +174,21 @@ extern void cons_fix_new_sparc + } \ + while (0) + +-#define DWARF2_LINE_MIN_INSN_LENGTH 4 ++#define TARGET_USE_CFIPOP 1 ++ ++#define tc_cfi_frame_initial_instructions sparc_cfi_frame_initial_instructions ++extern void sparc_cfi_frame_initial_instructions PARAMS ((void)); ++ ++#define tc_regname_to_dw2regnum sparc_regname_to_dw2regnum ++extern int sparc_regname_to_dw2regnum PARAMS ((const char *regname)); ++ ++#define tc_cfi_emit_pcrel_expr sparc_cfi_emit_pcrel_expr ++extern void sparc_cfi_emit_pcrel_expr PARAMS ((expressionS *, unsigned int)); ++ ++extern int sparc_cie_data_alignment; ++ ++#define DWARF2_LINE_MIN_INSN_LENGTH 4 ++#define DWARF2_DEFAULT_RETURN_COLUMN 15 ++#define DWARF2_CIE_DATA_ALIGNMENT sparc_cie_data_alignment + + /* end of tc-sparc.h */ +--- gas/config/tc-sparc.c.jj 2003-07-02 11:01:56.000000000 -0400 ++++ gas/config/tc-sparc.c 2003-08-29 15:39:10.000000000 -0400 +@@ -26,6 +26,7 @@ + #include "subsegs.h" + + #include "opcode/sparc.h" ++#include "dw2gencfi.h" + + #ifdef OBJ_ELF + #include "elf/sparc.h" +@@ -116,6 +117,9 @@ static int target_little_endian_data; + /* Symbols for global registers on v9. */ + static symbolS *globals[8]; + ++/* The dwarf2 data alignment, adjusted for 32 or 64 bit. */ ++int sparc_cie_data_alignment; ++ + /* V9 and 86x have big and little endian data, but instructions are always big + endian. The sparclet has bi-endian support but both data and insns have + the same endianness. Global `target_big_endian' is used for data. +@@ -798,6 +802,7 @@ md_begin () + if (! default_init_p) + init_default_arch (); + ++ sparc_cie_data_alignment = sparc_arch_size == 64 ? -8 : -4; + op_hash = hash_new (); + + while (i < (unsigned int) sparc_num_opcodes) +@@ -3804,8 +3809,8 @@ s_common (ignore) + { + if (S_GET_VALUE (symbolP) != (valueT) size) + { +- as_warn (_("Length of .comm \"%s\" is already %ld. Not changed to %d."), +- S_GET_NAME (symbolP), (long) S_GET_VALUE (symbolP), size); ++ as_warn (_("Length of .comm \"%s\" is already %ld. Not changed to %ld."), ++ S_GET_NAME (symbolP), (long) S_GET_VALUE (symbolP), (long) size); + } + } + else +@@ -4482,4 +4487,63 @@ cons_fix_new_sparc (frag, where, nbytes, + } + + fix_new_exp (frag, where, (int) nbytes, exp, 0, r); ++ sparc_cons_special_reloc = NULL; ++} ++ ++void ++sparc_cfi_frame_initial_instructions () ++{ ++ cfi_add_CFA_def_cfa (14, sparc_arch_size == 64 ? 0x7ff : 0); ++} ++ ++int ++sparc_regname_to_dw2regnum (const char *regname) ++{ ++ char *p, *q; ++ ++ if (!regname[0]) ++ return -1; ++ ++ q = "goli"; ++ p = strchr (q, regname[0]); ++ if (p) ++ { ++ if (regname[1] < '0' || regname[1] > '8' || regname[2]) ++ return -1; ++ return (p - q) * 8 + regname[1] - '0'; ++ } ++ if (regname[0] == 's' && regname[1] == 'p' && !regname[2]) ++ return 14; ++ if (regname[0] == 'f' && regname[1] == 'p' && !regname[2]) ++ return 30; ++ if (regname[0] == 'f' || regname[0] == 'r') ++ { ++ unsigned int regnum; ++ ++ regnum = strtoul (regname + 1, &q, 10); ++ if (p == q || *q) ++ return -1; ++ if (regnum >= ((regname[0] == 'f' ++ && SPARC_OPCODE_ARCH_V9_P (max_architecture)) ++ ? 64 : 32)) ++ return -1; ++ if (regname[0] == 'f') ++ { ++ regnum += 32; ++ if (regnum >= 64 && (regnum & 1)) ++ return -1; ++ } ++ return regnum; ++ } ++ return -1; ++} ++ ++void ++sparc_cfi_emit_pcrel_expr (expressionS *exp, unsigned int nbytes) ++{ ++ sparc_cons_special_reloc = "disp"; ++ sparc_no_align_cons = 1; ++ emit_expr (exp, nbytes); ++ sparc_no_align_cons = 0; ++ sparc_cons_special_reloc = NULL; + } +--- gas/doc/as.texinfo.jj 2003-07-29 05:40:36.000000000 -0400 ++++ gas/doc/as.texinfo 2003-08-29 15:29:21.000000000 -0400 +@@ -4050,6 +4050,9 @@ using the known displacement of the CFA + This is often easier to use, because the number will match the + code it's annotating. + ++@section @code{.cfi_window_save} ++SPARC register window has been saved. ++ + @section @code{.cfi_escape} @var{expression}[, @dots{}] + Allows the user to add arbitrary bytes to the unwind info. One + might use this to add OS-specific CFI opcodes, or generic CFI +--- gas/testsuite/gas/cfi/cfi-sparc-1.s.jj 2003-08-28 05:56:33.000000000 -0400 ++++ gas/testsuite/gas/cfi/cfi-sparc-1.s 2003-08-28 06:26:37.000000000 -0400 +@@ -0,0 +1,23 @@ ++#; $ as -o test.o -32 gas-cfi-test.s && gcc -m32 -nostdlib -o test test.o ++ ++ .file "a.c" ++ .text ++ .align 4 ++ .globl foo ++ .type foo, @function ++foo: ++ .cfi_startproc ++ save %sp, -104, %sp ++ .cfi_def_cfa_register %fp ++ .cfi_window_save ++ .cfi_register %o7, %i7 ++ add %i0, 1, %o0 ++ call bar, 0 ++ add %i0, 2, %i0 ++ call bar, 0 ++ mov %i0, %o0 ++ add %o0, 3, %o0 ++ ret ++ restore %g0, %o0, %o0 ++ .cfi_endproc ++ .size foo, .-foo +--- gas/testsuite/gas/cfi/cfi-sparc64-1.s.jj 2003-08-28 05:56:33.000000000 -0400 ++++ gas/testsuite/gas/cfi/cfi-sparc64-1.s 2003-08-28 06:26:44.000000000 -0400 +@@ -0,0 +1,26 @@ ++#; $ as -o test.o -64 gas-cfi-test.s && gcc -m64 -nostdlib -o test test.o ++ ++ .file "a.c" ++ .text ++ .align 4 ++ .globl foo ++ .type foo, @function ++foo: ++ .cfi_startproc ++ save %sp, -192, %sp ++ .cfi_def_cfa_register %fp ++ .cfi_window_save ++ .cfi_register %o7, %i7 ++ add %i0, 1, %o0 ++ add %i0, 2, %i0 ++ call bar, 0 ++ sra %o0, 0, %o0 ++ sra %i0, 0, %i0 ++ call bar, 0 ++ mov %i0, %o0 ++ add %o0, 3, %o0 ++ sra %o0, 0, %o0 ++ ret ++ restore %g0, %o0, %o0 ++ .cfi_endproc ++ .size foo, .-foo +--- gas/testsuite/gas/cfi/cfi-sparc-1.d.jj 2003-08-29 15:17:54.000000000 -0400 ++++ gas/testsuite/gas/cfi/cfi-sparc-1.d 2003-08-29 15:17:44.000000000 -0400 +@@ -0,0 +1,22 @@ ++#readelf: -wf ++#name: CFI on SPARC 32-bit ++#as: -32 ++ ++The section .eh_frame contains: ++ ++00000000 00000010 00000000 CIE ++ Version: 1 ++ Augmentation: "zR" ++ Code alignment factor: 4 ++ Data alignment factor: -4 ++ Return address column: 15 ++ Augmentation data: 1b ++ ++ DW_CFA_def_cfa: r14 ofs 0 ++ ++00000014 00000014 00000018 FDE cie=00000000 pc=0000001c..00000040 ++ DW_CFA_advance_loc: 4 to 00000020 ++ DW_CFA_def_cfa_reg: r30 ++ DW_CFA_GNU_window_save ++ DW_CFA_register: r15 in r31 ++ +--- gas/testsuite/gas/cfi/cfi-sparc64-1.d.jj 2003-08-29 15:17:51.000000000 -0400 ++++ gas/testsuite/gas/cfi/cfi-sparc64-1.d 2003-08-29 15:17:35.000000000 -0400 +@@ -0,0 +1,25 @@ ++#readelf: -wf ++#name: CFI on SPARC 64-bit ++#as: -64 ++ ++The section .eh_frame contains: ++ ++00000000 00000011 00000000 CIE ++ Version: 1 ++ Augmentation: "zR" ++ Code alignment factor: 4 ++ Data alignment factor: -8 ++ Return address column: 15 ++ Augmentation data: 1b ++ ++ DW_CFA_def_cfa: r14 ofs 2047 ++ ++00000015 00000017 00000019 FDE cie=00000000 pc=0000001d..0000004d ++ DW_CFA_advance_loc: 4 to 00000021 ++ DW_CFA_def_cfa_reg: r30 ++ DW_CFA_GNU_window_save ++ DW_CFA_register: r15 in r31 ++ DW_CFA_nop ++ DW_CFA_nop ++ DW_CFA_nop ++ +--- gas/testsuite/gas/cfi/cfi.exp.jj 2003-08-10 15:56:11.000000000 -0400 ++++ gas/testsuite/gas/cfi/cfi.exp 2003-08-29 15:23:29.000000000 -0400 +@@ -39,6 +39,16 @@ if [istarget "x86_64-*"] then { + } elseif { [istarget "m68*-*"] } then { + run_dump_test "cfi-m68k" + ++} elseif { [istarget sparc*-*-*] } then { ++ global NM ++ global NMFLAGS ++ global srcdir ++ ++ catch "exec $srcdir/lib/run $NM $NMFLAGS --help" nm_help ++ run_dump_test "cfi-sparc-1" ++ if { [regexp "elf64\[_-\]sparc" $nm_help] } then { ++ run_dump_test "cfi-sparc64-1" ++ } + } else { + return + } |