summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--meta/recipes-devtools/binutils/binutils-2.25.1.inc1
-rw-r--r--meta/recipes-devtools/binutils/binutils/0016-This-patch-adds-IFUNC-support-for-arm-gold-backend.patch1004
2 files changed, 1005 insertions, 0 deletions
diff --git a/meta/recipes-devtools/binutils/binutils-2.25.1.inc b/meta/recipes-devtools/binutils/binutils-2.25.1.inc
index bf29da16f5..dc23c4d9d9 100644
--- a/meta/recipes-devtools/binutils/binutils-2.25.1.inc
+++ b/meta/recipes-devtools/binutils/binutils-2.25.1.inc
@@ -35,6 +35,7 @@ SRC_URI = "\
file://0013-Fix-an-internal-error-in-do_print_to_mapfile-seen-wi.patch \
file://0014-gold-arm-Skip-pic-check-for-R_ARM_REL32.patch \
file://0015-Fix-dynamic-list-so-that-symbols-not-in-the-list-are.patch \
+ file://0016-This-patch-adds-IFUNC-support-for-arm-gold-backend.patch \
file://binutils-octeon3.patch \
file://add-thunderx-support-for-gas.patch \
"
diff --git a/meta/recipes-devtools/binutils/binutils/0016-This-patch-adds-IFUNC-support-for-arm-gold-backend.patch b/meta/recipes-devtools/binutils/binutils/0016-This-patch-adds-IFUNC-support-for-arm-gold-backend.patch
new file mode 100644
index 0000000000..dcac308bcc
--- /dev/null
+++ b/meta/recipes-devtools/binutils/binutils/0016-This-patch-adds-IFUNC-support-for-arm-gold-backend.patch
@@ -0,0 +1,1004 @@
+From b780c9e06cabe6d8e301aaf46f33f116f3224021 Mon Sep 17 00:00:00 2001
+From: Han Shen <shenhan@google.com>
+Date: Thu, 29 Jan 2015 10:00:46 -0800
+Subject: [PATCH] This patch adds IFUNC support for arm gold backend.
+
+This is a feature required in chromeos arm development work.
+
+Tested:
+1) Built passed all-gold on x86_64 machine
+2) Tested with basic gold aarch64 ifunc unittests -
+ a) global ifunc, statically/non-statically linked
+ b) local ifunc, statically/non-statically linked
+ c) global/local, other shared library routine mixed,
+ statically/non-statically linked
+ d) arm/thumb mode ifunc
+ e) linking chrome browser passed
+---
+Upstream-Status: Backport
+Signed-off-by: Khem Raj <raj.khem@gmail.com>
+
+ elfcpp/arm.h | 5 +-
+ gold/aarch64.cc | 2 +-
+ gold/arm.cc | 593 +++++++++++++++++++++++++++++++++++++++++++++++++++-----
+ gold/output.h | 11 ++
+ 4 files changed, 561 insertions(+), 50 deletions(-)
+
+diff --git a/elfcpp/arm.h b/elfcpp/arm.h
+index 8c6b6bf..1c13dc9 100644
+--- a/elfcpp/arm.h
++++ b/elfcpp/arm.h
+@@ -192,11 +192,12 @@ enum
+ R_ARM_PRIVATE_14 = 126,
+ R_ARM_PRIVATE_15 = 127,
+ R_ARM_ME_TOO = 128, // Obsolete
+- R_ARM_THM_TLS_DESCSEQ16 = 129,// Static Thumb16
++ R_ARM_THM_TLS_DESCSEQ16 = 129,// Static Thumb16
+ R_ARM_THM_TLS_DESCSEQ32 = 130,// Static Thumb32
+ // 131 - 139 Unallocated
+ // 140 - 159 Dynamic Reserved for future allocation
+- // 160 - 255 Unallocated
++ R_ARM_IRELATIVE = 160, // Dynamic
++ // 161 - 255 Unallocated
+ };
+
+ // e_flags values used for ARM. We only support flags defined in AAELF.
+diff --git a/gold/aarch64.cc b/gold/aarch64.cc
+index afb9024..7fbbdbd 100644
+--- a/gold/aarch64.cc
++++ b/gold/aarch64.cc
+@@ -1226,7 +1226,7 @@ class Output_data_plt_aarch64 : public Output_section_data
+ // The number of PLT entries.
+ unsigned int count_;
+
+- // Number of PLT entries with R_X86_64_IRELATIVE relocs. These
++ // Number of PLT entries with R_AARCH64_IRELATIVE relocs. These
+ // follow the regular PLT entries.
+ unsigned int irelative_count_;
+
+diff --git a/gold/arm.cc b/gold/arm.cc
+index 6c472bb..8719cc9 100644
+--- a/gold/arm.cc
++++ b/gold/arm.cc
+@@ -2119,8 +2119,8 @@ class Target_arm : public Sized_target<32, big_endian>
+
+ Target_arm(const Target::Target_info* info = &arm_info)
+ : Sized_target<32, big_endian>(info),
+- got_(NULL), plt_(NULL), got_plt_(NULL), rel_dyn_(NULL),
+- copy_relocs_(elfcpp::R_ARM_COPY),
++ got_(NULL), plt_(NULL), got_plt_(NULL), got_irelative_(NULL),
++ rel_dyn_(NULL), rel_irelative_(NULL), copy_relocs_(elfcpp::R_ARM_COPY),
+ got_mod_index_offset_(-1U), tls_base_symbol_defined_(false),
+ stub_tables_(), stub_factory_(Stub_factory::get_instance()),
+ should_force_pic_veneer_(false),
+@@ -2258,6 +2258,18 @@ class Target_arm : public Sized_target<32, big_endian>
+ uint64_t
+ do_dynsym_value(const Symbol*) const;
+
++ // Return the plt address for globals. Since we have irelative plt entries,
++ // address calculation is not as straightforward as plt_address + plt_offset.
++ uint64_t
++ do_plt_address_for_global(const Symbol* gsym) const
++ { return this->plt_section()->address_for_global(gsym); }
++
++ // Return the plt address for locals. Since we have irelative plt entries,
++ // address calculation is not as straightforward as plt_address + plt_offset.
++ uint64_t
++ do_plt_address_for_local(const Relobj* relobj, unsigned int symndx) const
++ { return this->plt_section()->address_for_local(relobj, symndx); }
++
+ // Relocate a section.
+ void
+ relocate_section(const Relocate_info<32, big_endian>*,
+@@ -2357,6 +2369,10 @@ class Target_arm : public Sized_target<32, big_endian>
+ unsigned int
+ plt_entry_size() const;
+
++ // Get the section to use for IRELATIVE relocations, create it if necessary.
++ Reloc_section*
++ rel_irelative_section(Layout*);
++
+ // Map platform-specific reloc types
+ static unsigned int
+ get_real_reloc_type(unsigned int r_type);
+@@ -2448,8 +2464,11 @@ class Target_arm : public Sized_target<32, big_endian>
+ protected:
+ // Make the PLT-generator object.
+ Output_data_plt_arm<big_endian>*
+- make_data_plt(Layout* layout, Output_data_space* got_plt)
+- { return this->do_make_data_plt(layout, got_plt); }
++ make_data_plt(Layout* layout,
++ Arm_output_data_got<big_endian>* got,
++ Output_data_space* got_plt,
++ Output_data_space* got_irelative)
++ { return this->do_make_data_plt(layout, got, got_plt, got_irelative); }
+
+ // Make an ELF object.
+ Object*
+@@ -2530,9 +2549,14 @@ class Target_arm : public Sized_target<32, big_endian>
+ do_define_standard_symbols(Symbol_table*, Layout*);
+
+ virtual Output_data_plt_arm<big_endian>*
+- do_make_data_plt(Layout* layout, Output_data_space* got_plt)
++ do_make_data_plt(Layout* layout,
++ Arm_output_data_got<big_endian>* got,
++ Output_data_space* got_plt,
++ Output_data_space* got_irelative)
+ {
+- return new Output_data_plt_arm_standard<big_endian>(layout, got_plt);
++ gold_assert(got_plt != NULL && got_irelative != NULL);
++ return new Output_data_plt_arm_standard<big_endian>(
++ layout, got, got_plt, got_irelative);
+ }
+
+ private:
+@@ -2602,6 +2626,9 @@ class Target_arm : public Sized_target<32, big_endian>
+ if (sym->is_undefined() && !parameters->options().shared())
+ return false;
+
++ if (sym->type() == elfcpp::STT_GNU_IFUNC)
++ return true;
++
+ return (!parameters->doing_static_link()
+ && (sym->type() == elfcpp::STT_FUNC
+ || sym->type() == elfcpp::STT_ARM_TFUNC)
+@@ -2613,6 +2640,11 @@ class Target_arm : public Sized_target<32, big_endian>
+ inline bool
+ possible_function_pointer_reloc(unsigned int r_type);
+
++ // Whether a plt entry is needed for ifunc.
++ bool
++ reloc_needs_plt_for_ifunc(Sized_relobj_file<32, big_endian>*,
++ unsigned int r_type);
++
+ // Whether we have issued an error about a non-PIC compilation.
+ bool issued_non_pic_error_;
+ };
+@@ -2718,10 +2750,20 @@ class Target_arm : public Sized_target<32, big_endian>
+ return this->got_plt_;
+ }
+
++ // Create the PLT section.
++ void
++ make_plt_section(Symbol_table* symtab, Layout* layout);
++
+ // Create a PLT entry for a global symbol.
+ void
+ make_plt_entry(Symbol_table*, Layout*, Symbol*);
+
++ // Create a PLT entry for a local STT_GNU_IFUNC symbol.
++ void
++ make_local_ifunc_plt_entry(Symbol_table*, Layout*,
++ Sized_relobj_file<32, big_endian>* relobj,
++ unsigned int local_sym_index);
++
+ // Define the _TLS_MODULE_BASE_ symbol in the TLS segment.
+ void
+ define_tls_base_symbol(Symbol_table*, Layout*);
+@@ -2903,8 +2945,12 @@ class Target_arm : public Sized_target<32, big_endian>
+ Output_data_plt_arm<big_endian>* plt_;
+ // The GOT PLT section.
+ Output_data_space* got_plt_;
++ // The GOT section for IRELATIVE relocations.
++ Output_data_space* got_irelative_;
+ // The dynamic reloc section.
+ Reloc_section* rel_dyn_;
++ // The section to use for IRELATIVE relocs.
++ Reloc_section* rel_irelative_;
+ // Relocs saved to avoid a COPY reloc.
+ Copy_relocs<elfcpp::SHT_REL, 32, big_endian> copy_relocs_;
+ // Offset of the GOT entry for the TLS module index.
+@@ -4244,6 +4290,15 @@ Target_arm<big_endian>::got_section(Symbol_table* symtab, Layout* layout)
+ elfcpp::STB_LOCAL,
+ elfcpp::STV_HIDDEN, 0,
+ false, false);
++
++ // If there are any IRELATIVE relocations, they get GOT entries
++ // in .got.plt after the jump slot entries.
++ this->got_irelative_ = new Output_data_space(4, "** GOT IRELATIVE PLT");
++ layout->add_output_section_data(".got", elfcpp::SHT_PROGBITS,
++ (elfcpp::SHF_ALLOC | elfcpp::SHF_WRITE),
++ this->got_irelative_,
++ got_order, is_got_relro);
++
+ }
+ return this->got_;
+ }
+@@ -4257,14 +4312,43 @@ Target_arm<big_endian>::rel_dyn_section(Layout* layout)
+ if (this->rel_dyn_ == NULL)
+ {
+ gold_assert(layout != NULL);
++ // Create both relocation sections in the same place, so as to ensure
++ // their relative order in the output section.
+ this->rel_dyn_ = new Reloc_section(parameters->options().combreloc());
++ this->rel_irelative_ = new Reloc_section(false);
+ layout->add_output_section_data(".rel.dyn", elfcpp::SHT_REL,
+ elfcpp::SHF_ALLOC, this->rel_dyn_,
+ ORDER_DYNAMIC_RELOCS, false);
++ layout->add_output_section_data(".rel.dyn", elfcpp::SHT_REL,
++ elfcpp::SHF_ALLOC, this->rel_irelative_,
++ ORDER_DYNAMIC_RELOCS, false);
+ }
+ return this->rel_dyn_;
+ }
+
++
++// Get the section to use for IRELATIVE relocs, creating it if necessary. These
++// go in .rela.dyn, but only after all other dynamic relocations. They need to
++// follow the other dynamic relocations so that they can refer to global
++// variables initialized by those relocs.
++
++template<bool big_endian>
++typename Target_arm<big_endian>::Reloc_section*
++Target_arm<big_endian>::rel_irelative_section(Layout* layout)
++{
++ if (this->rel_irelative_ == NULL)
++ {
++ // Delegate the creation to rel_dyn_section so as to ensure their order in
++ // the output section.
++ this->rel_dyn_section(layout);
++ gold_assert(this->rel_irelative_ != NULL
++ && (this->rel_dyn_->output_section()
++ == this->rel_irelative_->output_section()));
++ }
++ return this->rel_irelative_;
++}
++
++
+ // Insn_template methods.
+
+ // Return byte size of an instruction template.
+@@ -7221,24 +7305,80 @@ template<bool big_endian>
+ class Output_data_plt_arm : public Output_section_data
+ {
+ public:
++ // Unlike aarch64, which records symbol value in "addend" field of relocations
++ // and could be done at the same time an IRelative reloc is created for the
++ // symbol, arm puts the symbol value into "GOT" table, which, however, is
++ // issued later in Output_data_plt_arm::do_write(). So we have a struct here
++ // to keep necessary symbol information for later use in do_write. We usually
++ // have only a very limited number of ifuncs, so the extra data required here
++ // is also limited.
++
++ struct IRelative_data
++ {
++ IRelative_data(Sized_symbol<32>* sized_symbol)
++ : symbol_is_global_(true)
++ {
++ u_.global = sized_symbol;
++ }
++
++ IRelative_data(Sized_relobj_file<32, big_endian>* relobj,
++ unsigned int index)
++ : symbol_is_global_(false)
++ {
++ u_.local.relobj = relobj;
++ u_.local.index = index;
++ }
++
++ union
++ {
++ Sized_symbol<32>* global;
++
++ struct
++ {
++ Sized_relobj_file<32, big_endian>* relobj;
++ unsigned int index;
++ } local;
++ } u_;
++
++ bool symbol_is_global_;
++ };
++
+ typedef Output_data_reloc<elfcpp::SHT_REL, true, 32, big_endian>
+ Reloc_section;
+
+- Output_data_plt_arm(Layout*, uint64_t addralign, Output_data_space*);
++ Output_data_plt_arm(Layout* layout, uint64_t addralign,
++ Arm_output_data_got<big_endian>* got,
++ Output_data_space* got_plt,
++ Output_data_space* got_irelative);
+
+ // Add an entry to the PLT.
+ void
+- add_entry(Symbol* gsym);
++ add_entry(Symbol_table* symtab, Layout* layout, Symbol* gsym);
++
++ // Add the relocation for a plt entry.
++ void
++ add_relocation(Symbol_table* symtab, Layout* layout,
++ Symbol* gsym, unsigned int got_offset);
++
++ // Add an entry to the PLT for a local STT_GNU_IFUNC symbol.
++ unsigned int
++ add_local_ifunc_entry(Symbol_table* symtab, Layout*,
++ Sized_relobj_file<32, big_endian>* relobj,
++ unsigned int local_sym_index);
+
+ // Return the .rel.plt section data.
+ const Reloc_section*
+ rel_plt() const
+ { return this->rel_; }
+
++ // Return the PLT relocation container for IRELATIVE.
++ Reloc_section*
++ rel_irelative(Symbol_table*, Layout*);
++
+ // Return the number of PLT entries.
+ unsigned int
+ entry_count() const
+- { return this->count_; }
++ { return this->count_ + this->irelative_count_; }
+
+ // Return the offset of the first non-reserved PLT entry.
+ unsigned int
+@@ -7250,6 +7390,14 @@ class Output_data_plt_arm : public Output_section_data
+ get_plt_entry_size() const
+ { return this->do_get_plt_entry_size(); }
+
++ // Return the PLT address for globals.
++ uint32_t
++ address_for_global(const Symbol*) const;
++
++ // Return the PLT address for locals.
++ uint32_t
++ address_for_local(const Relobj*, unsigned int symndx) const;
++
+ protected:
+ // Fill in the first PLT entry.
+ void
+@@ -7298,19 +7446,37 @@ class Output_data_plt_arm : public Output_section_data
+ set_final_data_size()
+ {
+ this->set_data_size(this->first_plt_entry_offset()
+- + this->count_ * this->get_plt_entry_size());
++ + ((this->count_ + this->irelative_count_)
++ * this->get_plt_entry_size()));
+ }
+
+ // Write out the PLT data.
+ void
+ do_write(Output_file*);
+
++ // Record irelative symbol data.
++ void insert_irelative_data(const IRelative_data& idata)
++ { irelative_data_vec_.push_back(idata); }
++
+ // The reloc section.
+ Reloc_section* rel_;
++ // The IRELATIVE relocs, if necessary. These must follow the
++ // regular PLT relocations.
++ Reloc_section* irelative_rel_;
++ // The .got section.
++ Arm_output_data_got<big_endian>* got_;
+ // The .got.plt section.
+ Output_data_space* got_plt_;
++ // The part of the .got.plt section used for IRELATIVE relocs.
++ Output_data_space* got_irelative_;
+ // The number of PLT entries.
+ unsigned int count_;
++ // Number of PLT entries with R_ARM_IRELATIVE relocs. These
++ // follow the regular PLT entries.
++ unsigned int irelative_count_;
++ // Vector for irelative data.
++ typedef std::vector<IRelative_data> IRelative_data_vec;
++ IRelative_data_vec irelative_data_vec_;
+ };
+
+ // Create the PLT section. The ordinary .got section is an argument,
+@@ -7318,10 +7484,14 @@ class Output_data_plt_arm : public Output_section_data
+ // section just for PLT entries.
+
+ template<bool big_endian>
+-Output_data_plt_arm<big_endian>::Output_data_plt_arm(Layout* layout,
+- uint64_t addralign,
+- Output_data_space* got_plt)
+- : Output_section_data(addralign), got_plt_(got_plt), count_(0)
++Output_data_plt_arm<big_endian>::Output_data_plt_arm(
++ Layout* layout, uint64_t addralign,
++ Arm_output_data_got<big_endian>* got,
++ Output_data_space* got_plt,
++ Output_data_space* got_irelative)
++ : Output_section_data(addralign), irelative_rel_(NULL),
++ got_(got), got_plt_(got_plt), got_irelative_(got_irelative),
++ count_(0), irelative_count_(0)
+ {
+ this->rel_ = new Reloc_section(false);
+ layout->add_output_section_data(".rel.plt", elfcpp::SHT_REL,
+@@ -7340,40 +7510,210 @@ Output_data_plt_arm<big_endian>::do_adjust_output_section(Output_section* os)
+
+ template<bool big_endian>
+ void
+-Output_data_plt_arm<big_endian>::add_entry(Symbol* gsym)
++Output_data_plt_arm<big_endian>::add_entry(Symbol_table* symtab,
++ Layout* layout,
++ Symbol* gsym)
+ {
+ gold_assert(!gsym->has_plt_offset());
+
+- // Note that when setting the PLT offset we skip the initial
+- // reserved PLT entry.
+- gsym->set_plt_offset((this->count_) * this->get_plt_entry_size()
+- + this->first_plt_entry_offset());
++ unsigned int* entry_count;
++ Output_section_data_build* got;
++
++ // We have 2 different types of plt entry here, normal and ifunc.
++
++ // For normal plt, the offset begins with first_plt_entry_offset(20), and the
++ // 1st entry offset would be 20, the second 32, third 44 ... etc.
++
++ // For ifunc plt, the offset begins with 0. So the first offset would 0,
++ // second 12, third 24 ... etc.
++
++ // IFunc plt entries *always* come after *normal* plt entries.
++
++ // Notice, when computing the plt address of a certain symbol, "plt_address +
++ // plt_offset" is no longer correct. Use target->plt_address_for_global() or
++ // target->plt_address_for_local() instead.
++
++ int begin_offset = 0;
++ if (gsym->type() == elfcpp::STT_GNU_IFUNC
++ && gsym->can_use_relative_reloc(false))
++ {
++ entry_count = &this->irelative_count_;
++ got = this->got_irelative_;
++ // For irelative plt entries, offset is relative to the end of normal plt
++ // entries, so it starts from 0.
++ begin_offset = 0;
++ // Record symbol information.
++ this->insert_irelative_data(
++ IRelative_data(symtab->get_sized_symbol<32>(gsym)));
++ }
++ else
++ {
++ entry_count = &this->count_;
++ got = this->got_plt_;
++ // Note that for normal plt entries, when setting the PLT offset we skip
++ // the initial reserved PLT entry.
++ begin_offset = this->first_plt_entry_offset();
++ }
++
++ gsym->set_plt_offset(begin_offset
++ + (*entry_count) * this->get_plt_entry_size());
+
+- ++this->count_;
++ ++(*entry_count);
+
+- section_offset_type got_offset = this->got_plt_->current_data_size();
++ section_offset_type got_offset = got->current_data_size();
+
+ // Every PLT entry needs a GOT entry which points back to the PLT
+ // entry (this will be changed by the dynamic linker, normally
+ // lazily when the function is called).
+- this->got_plt_->set_current_data_size(got_offset + 4);
++ got->set_current_data_size(got_offset + 4);
+
+ // Every PLT entry needs a reloc.
+- gsym->set_needs_dynsym_entry();
+- this->rel_->add_global(gsym, elfcpp::R_ARM_JUMP_SLOT, this->got_plt_,
+- got_offset);
++ this->add_relocation(symtab, layout, gsym, got_offset);
+
+ // Note that we don't need to save the symbol. The contents of the
+ // PLT are independent of which symbols are used. The symbols only
+ // appear in the relocations.
+ }
+
++// Add an entry to the PLT for a local STT_GNU_IFUNC symbol. Return
++// the PLT offset.
++
++template<bool big_endian>
++unsigned int
++Output_data_plt_arm<big_endian>::add_local_ifunc_entry(
++ Symbol_table* symtab,
++ Layout* layout,
++ Sized_relobj_file<32, big_endian>* relobj,
++ unsigned int local_sym_index)
++{
++ this->insert_irelative_data(IRelative_data(relobj, local_sym_index));
++
++ // Notice, when computingthe plt entry address, "plt_address + plt_offset" is
++ // no longer correct. Use target->plt_address_for_local() instead.
++ unsigned int plt_offset = this->irelative_count_ * this->get_plt_entry_size();
++ ++this->irelative_count_;
++
++ section_offset_type got_offset = this->got_irelative_->current_data_size();
++
++ // Every PLT entry needs a GOT entry which points back to the PLT
++ // entry.
++ this->got_irelative_->set_current_data_size(got_offset + 4);
++
++
++ // Every PLT entry needs a reloc.
++ Reloc_section* rel = this->rel_irelative(symtab, layout);
++ rel->add_symbolless_local_addend(relobj, local_sym_index,
++ elfcpp::R_ARM_IRELATIVE,
++ this->got_irelative_, got_offset);
++ return plt_offset;
++}
++
++
++// Add the relocation for a PLT entry.
++
++template<bool big_endian>
++void
++Output_data_plt_arm<big_endian>::add_relocation(
++ Symbol_table* symtab, Layout* layout, Symbol* gsym, unsigned int got_offset)
++{
++ if (gsym->type() == elfcpp::STT_GNU_IFUNC
++ && gsym->can_use_relative_reloc(false))
++ {
++ Reloc_section* rel = this->rel_irelative(symtab, layout);
++ rel->add_symbolless_global_addend(gsym, elfcpp::R_ARM_IRELATIVE,
++ this->got_irelative_, got_offset);
++ }
++ else
++ {
++ gsym->set_needs_dynsym_entry();
++ this->rel_->add_global(gsym, elfcpp::R_ARM_JUMP_SLOT, this->got_plt_,
++ got_offset);
++ }
++}
++
++
++// Create the irelative relocation data.
++
++template<bool big_endian>
++typename Output_data_plt_arm<big_endian>::Reloc_section*
++Output_data_plt_arm<big_endian>::rel_irelative(Symbol_table* symtab,
++ Layout* layout)
++{
++ if (this->irelative_rel_ == NULL)
++ {
++ // Since irelative relocations goes into 'rel.dyn', we delegate the
++ // creation of irelative_rel_ to where rel_dyn section gets created.
++ Target_arm<big_endian>* arm_target =
++ Target_arm<big_endian>::default_target();
++ this->irelative_rel_ = arm_target->rel_irelative_section(layout);
++
++ // Make sure we have a place for the TLSDESC relocations, in
++ // case we see any later on.
++ // this->rel_tlsdesc(layout);
++ if (parameters->doing_static_link())
++ {
++ // A statically linked executable will only have a .rel.plt section to
++ // hold R_ARM_IRELATIVE relocs for STT_GNU_IFUNC symbols. The library
++ // will use these symbols to locate the IRELATIVE relocs at program
++ // startup time.
++ symtab->define_in_output_data("__rel_iplt_start", NULL,
++ Symbol_table::PREDEFINED,
++ this->irelative_rel_, 0, 0,
++ elfcpp::STT_NOTYPE, elfcpp::STB_GLOBAL,
++ elfcpp::STV_HIDDEN, 0, false, true);
++ symtab->define_in_output_data("__rel_iplt_end", NULL,
++ Symbol_table::PREDEFINED,
++ this->irelative_rel_, 0, 0,
++ elfcpp::STT_NOTYPE, elfcpp::STB_GLOBAL,
++ elfcpp::STV_HIDDEN, 0, true, true);
++ }
++ }
++ return this->irelative_rel_;
++}
++
++
++// Return the PLT address for a global symbol.
++
++template<bool big_endian>
++uint32_t
++Output_data_plt_arm<big_endian>::address_for_global(const Symbol* gsym) const
++{
++ uint64_t begin_offset = 0;
++ if (gsym->type() == elfcpp::STT_GNU_IFUNC
++ && gsym->can_use_relative_reloc(false))
++ {
++ begin_offset = (this->first_plt_entry_offset() +
++ this->count_ * this->get_plt_entry_size());
++ }
++ return this->address() + begin_offset + gsym->plt_offset();
++}
++
++
++// Return the PLT address for a local symbol. These are always
++// IRELATIVE relocs.
++
++template<bool big_endian>
++uint32_t
++Output_data_plt_arm<big_endian>::address_for_local(
++ const Relobj* object,
++ unsigned int r_sym) const
++{
++ return (this->address()
++ + this->first_plt_entry_offset()
++ + this->count_ * this->get_plt_entry_size()
++ + object->local_plt_offset(r_sym));
++}
++
++
+ template<bool big_endian>
+ class Output_data_plt_arm_standard : public Output_data_plt_arm<big_endian>
+ {
+ public:
+- Output_data_plt_arm_standard(Layout* layout, Output_data_space* got_plt)
+- : Output_data_plt_arm<big_endian>(layout, 4, got_plt)
++ Output_data_plt_arm_standard(Layout* layout,
++ Arm_output_data_got<big_endian>* got,
++ Output_data_space* got_plt,
++ Output_data_space* got_irelative)
++ : Output_data_plt_arm<big_endian>(layout, 4, got, got_plt, got_irelative)
+ { }
+
+ protected:
+@@ -7485,8 +7825,11 @@ Output_data_plt_arm<big_endian>::do_write(Output_file* of)
+ unsigned char* const oview = of->get_output_view(offset, oview_size);
+
+ const off_t got_file_offset = this->got_plt_->offset();
++ gold_assert(got_file_offset + this->got_plt_->data_size()
++ == this->got_irelative_->offset());
+ const section_size_type got_size =
+- convert_to_section_size_type(this->got_plt_->data_size());
++ convert_to_section_size_type(this->got_plt_->data_size()
++ + this->got_irelative_->data_size());
+ unsigned char* const got_view = of->get_output_view(got_file_offset,
+ got_size);
+ unsigned char* pov = oview;
+@@ -7505,7 +7848,8 @@ Output_data_plt_arm<big_endian>::do_write(Output_file* of)
+
+ unsigned int plt_offset = this->first_plt_entry_offset();
+ unsigned int got_offset = 12;
+- const unsigned int count = this->count_;
++ const unsigned int count = this->count_ + this->irelative_count_;
++ gold_assert(this->irelative_count_ == this->irelative_data_vec_.size());
+ for (unsigned int i = 0;
+ i < count;
+ ++i,
+@@ -7518,8 +7862,33 @@ Output_data_plt_arm<big_endian>::do_write(Output_file* of)
+ this->fill_plt_entry(pov, got_address, plt_address,
+ got_offset, plt_offset);
+
+- // Set the entry in the GOT.
+- elfcpp::Swap<32, big_endian>::writeval(got_pov, plt_address);
++ Arm_address value;
++ if (i < this->count_)
++ {
++ // For non-irelative got entries, the value is the beginning of plt.
++ value = plt_address;
++ }
++ else
++ {
++ // For irelative got entries, the value is the (global/local) symbol
++ // address.
++ const IRelative_data& idata =
++ this->irelative_data_vec_[i - this->count_];
++ if (idata.symbol_is_global_)
++ {
++ // Set the entry in the GOT for irelative symbols. The content is
++ // the address of the ifunc, not the address of plt start.
++ const Sized_symbol<32>* sized_symbol = idata.u_.global;
++ gold_assert(sized_symbol->type() == elfcpp::STT_GNU_IFUNC);
++ value = sized_symbol->value();
++ }
++ else
++ {
++ value = idata.u_.local.relobj->local_symbol_value(
++ idata.u_.local.index, 0);
++ }
++ }
++ elfcpp::Swap<32, big_endian>::writeval(got_pov, value);
+ }
+
+ gold_assert(static_cast<section_size_type>(pov - oview) == oview_size);
+@@ -7529,6 +7898,7 @@ Output_data_plt_arm<big_endian>::do_write(Output_file* of)
+ of->write_output_view(got_file_offset, got_size, got_view);
+ }
+
++
+ // Create a PLT entry for a global symbol.
+
+ template<bool big_endian>
+@@ -7540,20 +7910,58 @@ Target_arm<big_endian>::make_plt_entry(Symbol_table* symtab, Layout* layout,
+ return;
+
+ if (this->plt_ == NULL)
++ this->make_plt_section(symtab, layout);
++
++ this->plt_->add_entry(symtab, layout, gsym);
++}
++
++
++// Create the PLT section.
++template<bool big_endian>
++void
++Target_arm<big_endian>::make_plt_section(
++ Symbol_table* symtab, Layout* layout)
++{
++ if (this->plt_ == NULL)
+ {
+- // Create the GOT sections first.
++ // Create the GOT section first.
+ this->got_section(symtab, layout);
+
+- this->plt_ = this->make_data_plt(layout, this->got_plt_);
++ // GOT for irelatives is create along with got.plt.
++ gold_assert(this->got_ != NULL
++ && this->got_plt_ != NULL
++ && this->got_irelative_ != NULL);
++ this->plt_ = this->make_data_plt(layout, this->got_, this->got_plt_,
++ this->got_irelative_);
+
+ layout->add_output_section_data(".plt", elfcpp::SHT_PROGBITS,
+ (elfcpp::SHF_ALLOC
+ | elfcpp::SHF_EXECINSTR),
+ this->plt_, ORDER_PLT, false);
+ }
+- this->plt_->add_entry(gsym);
+ }
+
++
++// Make a PLT entry for a local STT_GNU_IFUNC symbol.
++
++template<bool big_endian>
++void
++Target_arm<big_endian>::make_local_ifunc_plt_entry(
++ Symbol_table* symtab, Layout* layout,
++ Sized_relobj_file<32, big_endian>* relobj,
++ unsigned int local_sym_index)
++{
++ if (relobj->local_has_plt_offset(local_sym_index))
++ return;
++ if (this->plt_ == NULL)
++ this->make_plt_section(symtab, layout);
++ unsigned int plt_offset = this->plt_->add_local_ifunc_entry(symtab, layout,
++ relobj,
++ local_sym_index);
++ relobj->set_local_plt_offset(local_sym_index, plt_offset);
++}
++
++
+ // Return the number of entries in the PLT.
+
+ template<bool big_endian>
+@@ -7823,6 +8231,7 @@ Target_arm<big_endian>::Scan::check_non_pic(Relobj* object,
+ case elfcpp::R_ARM_JUMP_SLOT:
+ case elfcpp::R_ARM_ABS32:
+ case elfcpp::R_ARM_ABS32_NOI:
++ case elfcpp::R_ARM_IRELATIVE:
+ case elfcpp::R_ARM_PC24:
+ // FIXME: The following 3 types are not supported by Android's dynamic
+ // linker.
+@@ -7853,6 +8262,27 @@ Target_arm<big_endian>::Scan::check_non_pic(Relobj* object,
+ }
+ }
+
++
++// Return whether we need to make a PLT entry for a relocation of the
++// given type against a STT_GNU_IFUNC symbol.
++
++template<bool big_endian>
++bool
++Target_arm<big_endian>::Scan::reloc_needs_plt_for_ifunc(
++ Sized_relobj_file<32, big_endian>* object,
++ unsigned int r_type)
++{
++ int flags = Scan::get_reference_flags(r_type);
++ if (flags & Symbol::TLS_REF)
++ {
++ gold_error(_("%s: unsupported TLS reloc %u for IFUNC symbol"),
++ object->name().c_str(), r_type);
++ return false;
++ }
++ return flags != 0;
++}
++
++
+ // Scan a relocation for a local symbol.
+ // FIXME: This only handles a subset of relocation types used by Android
+ // on ARM v5te devices.
+@@ -7874,6 +8304,15 @@ Target_arm<big_endian>::Scan::local(Symbol_table* symtab,
+ return;
+
+ r_type = get_real_reloc_type(r_type);
++
++ // A local STT_GNU_IFUNC symbol may require a PLT entry.
++ bool is_ifunc = lsym.get_st_type() == elfcpp::STT_GNU_IFUNC;
++ if (is_ifunc && this->reloc_needs_plt_for_ifunc(object, r_type))
++ {
++ unsigned int r_sym = elfcpp::elf_r_sym<32>(reloc.get_r_info());
++ target->make_local_ifunc_plt_entry(symtab, layout, object, r_sym);
++ }
++
+ switch (r_type)
+ {
+ case elfcpp::R_ARM_NONE:
+@@ -7898,7 +8337,7 @@ Target_arm<big_endian>::Scan::local(Symbol_table* symtab,
+ // we need to add check_non_pic(object, r_type) here.
+ rel_dyn->add_local_relative(object, r_sym, elfcpp::R_ARM_RELATIVE,
+ output_section, data_shndx,
+- reloc.get_r_offset());
++ reloc.get_r_offset(), is_ifunc);
+ }
+ break;
+
+@@ -8265,6 +8704,11 @@ Target_arm<big_endian>::Scan::global(Symbol_table* symtab,
+ && strcmp(gsym->name(), "_GLOBAL_OFFSET_TABLE_") == 0)
+ target->got_section(symtab, layout);
+
++ // A STT_GNU_IFUNC symbol may require a PLT entry.
++ if (gsym->type() == elfcpp::STT_GNU_IFUNC
++ && this->reloc_needs_plt_for_ifunc(object, r_type))
++ target->make_plt_entry(symtab, layout, gsym);
++
+ r_type = get_real_reloc_type(r_type);
+ switch (r_type)
+ {
+@@ -8309,6 +8753,24 @@ Target_arm<big_endian>::Scan::global(Symbol_table* symtab,
+ }
+ else if ((r_type == elfcpp::R_ARM_ABS32
+ || r_type == elfcpp::R_ARM_ABS32_NOI)
++ && gsym->type() == elfcpp::STT_GNU_IFUNC
++ && gsym->can_use_relative_reloc(false)
++ && !gsym->is_from_dynobj()
++ && !gsym->is_undefined()
++ && !gsym->is_preemptible())
++ {
++ // Use an IRELATIVE reloc for a locally defined STT_GNU_IFUNC
++ // symbol. This makes a function address in a PIE executable
++ // match the address in a shared library that it links against.
++ Reloc_section* rel_irelative =
++ target->rel_irelative_section(layout);
++ unsigned int r_type = elfcpp::R_ARM_IRELATIVE;
++ rel_irelative->add_symbolless_global_addend(
++ gsym, r_type, output_section, object,
++ data_shndx, reloc.get_r_offset());
++ }
++ else if ((r_type == elfcpp::R_ARM_ABS32
++ || r_type == elfcpp::R_ARM_ABS32_NOI)
+ && gsym->can_use_relative_reloc(false))
+ {
+ Reloc_section* rel_dyn = target->rel_dyn_section(layout);
+@@ -8442,7 +8904,13 @@ Target_arm<big_endian>::Scan::global(Symbol_table* symtab,
+ Arm_output_data_got<big_endian>* got =
+ target->got_section(symtab, layout);
+ if (gsym->final_value_is_known())
+- got->add_global(gsym, GOT_TYPE_STANDARD);
++ {
++ // For a STT_GNU_IFUNC symbol we want the PLT address.
++ if (gsym->type() == elfcpp::STT_GNU_IFUNC)
++ got->add_global_plt(gsym, GOT_TYPE_STANDARD);
++ else
++ got->add_global(gsym, GOT_TYPE_STANDARD);
++ }
+ else
+ {
+ // If this symbol is not fully resolved, we need to add a
+@@ -8452,12 +8920,29 @@ Target_arm<big_endian>::Scan::global(Symbol_table* symtab,
+ || gsym->is_undefined()
+ || gsym->is_preemptible()
+ || (gsym->visibility() == elfcpp::STV_PROTECTED
+- && parameters->options().shared()))
++ && parameters->options().shared())
++ || (gsym->type() == elfcpp::STT_GNU_IFUNC
++ && parameters->options().output_is_position_independent()))
+ got->add_global_with_rel(gsym, GOT_TYPE_STANDARD,
+ rel_dyn, elfcpp::R_ARM_GLOB_DAT);
+ else
+ {
+- if (got->add_global(gsym, GOT_TYPE_STANDARD))
++ // For a STT_GNU_IFUNC symbol we want to write the PLT
++ // offset into the GOT, so that function pointer
++ // comparisons work correctly.
++ bool is_new;
++ if (gsym->type() != elfcpp::STT_GNU_IFUNC)
++ is_new = got->add_global(gsym, GOT_TYPE_STANDARD);
++ else
++ {
++ is_new = got->add_global_plt(gsym, GOT_TYPE_STANDARD);
++ // Tell the dynamic linker to use the PLT address
++ // when resolving relocations.
++ if (gsym->is_from_dynobj()
++ && !parameters->options().shared())
++ gsym->set_needs_dynsym_value();
++ }
++ if (is_new)
+ rel_dyn->add_global_relative(
+ gsym, elfcpp::R_ARM_RELATIVE, got,
+ gsym->got_offset(GOT_TYPE_STANDARD));
+@@ -8919,8 +9404,7 @@ Target_arm<big_endian>::Relocate::relocate(
+ if (gsym->use_plt_offset(Scan::get_reference_flags(r_type)))
+ {
+ // This uses a PLT, change the symbol value.
+- symval.set_output_value(target->plt_section()->address()
+- + gsym->plt_offset());
++ symval.set_output_value(target->plt_address_for_global(gsym));
+ psymval = &symval;
+ }
+ else if (gsym->is_weak_undefined())
+@@ -8958,6 +9442,13 @@ Target_arm<big_endian>::Relocate::relocate(
+ elfcpp::Elf_types<32>::Elf_WXword r_info = rel.get_r_info();
+ unsigned int r_sym = elfcpp::elf_r_sym<32>(r_info);
+ thumb_bit = object->local_symbol_is_thumb_function(r_sym) ? 1 : 0;
++
++ if (psymval->is_ifunc_symbol() && object->local_has_plt_offset(r_sym))
++ {
++ symval.set_output_value(
++ target->plt_address_for_local(object, r_sym));
++ psymval = &symval;
++ }
+ }
+ }
+ else
+@@ -9936,7 +10427,7 @@ uint64_t
+ Target_arm<big_endian>::do_dynsym_value(const Symbol* gsym) const
+ {
+ gold_assert(gsym->is_from_dynobj() && gsym->has_plt_offset());
+- return this->plt_section()->address() + gsym->plt_offset();
++ return this->plt_address_for_global(gsym);
+ }
+
+ // Map platform-specific relocs to real relocs
+@@ -11083,8 +11574,7 @@ Target_arm<big_endian>::scan_reloc_for_stub(
+ if (gsym->use_plt_offset(Scan::get_reference_flags(r_type)))
+ {
+ // This uses a PLT, change the symbol value.
+- symval.set_output_value(this->plt_section()->address()
+- + gsym->plt_offset());
++ symval.set_output_value(this->plt_address_for_global(gsym));
+ psymval = &symval;
+ target_is_thumb = false;
+ }
+@@ -12187,8 +12677,13 @@ class Target_arm_nacl : public Target_arm<big_endian>
+
+ protected:
+ virtual Output_data_plt_arm<big_endian>*
+- do_make_data_plt(Layout* layout, Output_data_space* got_plt)
+- { return new Output_data_plt_arm_nacl<big_endian>(layout, got_plt); }
++ do_make_data_plt(
++ Layout* layout,
++ Arm_output_data_got<big_endian>* got,
++ Output_data_space* got_plt,
++ Output_data_space* got_irelative)
++ { return new Output_data_plt_arm_nacl<big_endian>(
++ layout, got, got_plt, got_irelative); }
+
+ private:
+ static const Target::Target_info arm_nacl_info;
+@@ -12225,8 +12720,12 @@ template<bool big_endian>
+ class Output_data_plt_arm_nacl : public Output_data_plt_arm<big_endian>
+ {
+ public:
+- Output_data_plt_arm_nacl(Layout* layout, Output_data_space* got_plt)
+- : Output_data_plt_arm<big_endian>(layout, 16, got_plt)
++ Output_data_plt_arm_nacl(
++ Layout* layout,
++ Arm_output_data_got<big_endian>* got,
++ Output_data_space* got_plt,
++ Output_data_space* got_irelative)
++ : Output_data_plt_arm<big_endian>(layout, 16, got, got_plt, got_irelative)
+ { }
+
+ protected:
+diff --git a/gold/output.h b/gold/output.h
+index ba0cdaa..599c2b7 100644
+--- a/gold/output.h
++++ b/gold/output.h
+@@ -1714,6 +1714,17 @@ class Output_data_reloc<elfcpp::SHT_REL, dynamic, size, big_endian>
+ address, true, true, false, false));
+ }
+
++ void
++ add_local_relative(Sized_relobj<size, big_endian>* relobj,
++ unsigned int local_sym_index, unsigned int type,
++ Output_data* od, unsigned int shndx, Address address,
++ bool use_plt_offset)
++ {
++ this->add(od, Output_reloc_type(relobj, local_sym_index, type, shndx,
++ address, true, true, false,
++ use_plt_offset));
++ }
++
+ // Add a local relocation which does not use a symbol for the relocation,
+ // but which gets its addend from a symbol.
+
+--
+2.7.0
+