Simplify ppc64 code setting toc_off.

Every function has a nominal toc pointer value, even if it isn't used,
so set toc_off for every code section to the value used in that object
file.  The thinking here was that if a code section didn't use the toc
it could use the previous object file's toc pointer value.  It can,
but doing so is only a gain if functions in that section are called
mostly from previous objects sharing the same toc.  We lose if the
functions in question are called mostly from the current object or
following objects, and it's a good bet they will probably mostly be
called from the current object.

	* elf64-ppc.c (ppc64_elf_next_input_section): Always set toc_off
	to value for object file.
This commit is contained in:
Alan Modra 2013-10-30 17:30:43 +10:30
parent eab88b547c
commit 8b974ba3e8
2 changed files with 19 additions and 31 deletions

View file

@ -1,3 +1,8 @@
2013-11-02 Alan Modra <amodra@gmail.com>
* elf64-ppc.c (ppc64_elf_next_input_section): Always set toc_off
to value for object file.
2013-11-01 Roland McGrath <mcgrathr@google.com>
* elf-nacl.c (segment_eligible_for_headers): Drop requirement that

View file

@ -11508,42 +11508,25 @@ ppc64_elf_next_input_section (struct bfd_link_info *info, asection *isec)
if (htab->multi_toc_needed)
{
/* If a code section has a function that uses the TOC then we need
to use the right TOC (obviously). Also, make sure that .opd gets
the correct TOC value for R_PPC64_TOC relocs that don't have or
can't find their function symbol (shouldn't ever happen now).
Also specially treat .fixup for the linux kernel. .fixup
contains branches, but only back to the function that hit an
exception. */
if (isec->has_toc_reloc
|| (isec->flags & SEC_CODE) == 0
|| strcmp (isec->name, ".fixup") == 0)
/* Analyse sections that aren't already flagged as needing a
valid toc pointer. Exclude .fixup for the linux kernel.
.fixup contains branches, but only back to the function that
hit an exception. */
if (!(isec->has_toc_reloc
|| (isec->flags & SEC_CODE) == 0
|| strcmp (isec->name, ".fixup") == 0
|| isec->call_check_done))
{
if (elf_gp (isec->owner) != 0)
htab->toc_curr = elf_gp (isec->owner);
}
else
{
if (!isec->call_check_done
&& toc_adjusting_stub_needed (info, isec) < 0)
if (toc_adjusting_stub_needed (info, isec) < 0)
return FALSE;
/* If we make a local call from this section, ie. a branch
without a following nop, then we have no place to put a
toc restoring insn. We must use the same toc group as
the callee.
Testing makes_toc_func_call actually tests for *any*
calls to functions that need a good toc pointer. A more
precise test would be better, as this one will set
incorrect values for pasted .init/.fini fragments.
(Fixed later in check_pasted_section.) */
if (isec->makes_toc_func_call
&& elf_gp (isec->owner) != 0)
htab->toc_curr = elf_gp (isec->owner);
}
/* Make all sections use the TOC assigned for this object file.
This will be wrong for pasted sections; We fix that in
check_pasted_section(). */
if (elf_gp (isec->owner) != 0)
htab->toc_curr = elf_gp (isec->owner);
}
/* Functions that don't use the TOC can belong in any TOC group.
Use the last TOC base. */
htab->stub_group[isec->id].toc_off = htab->toc_curr;
return TRUE;
}