+2011-07-15 Ian Lance Taylor <iant@google.com>
+
+ * i386.cc (class Output_data_plt_i386): Add layout_ field.
+ (Output_data_plt_i386::Output_data_plt_i386): Initialize layout_.
+ (Output_data_plt_i386::do_write): Write address of .dynamic
+ section to first entry in .got.plt section.
+ * x86_64.cc (class Output_data_plt_x86_64): Add layout_ field.
+ (Output_data_plt_x86_64::Output_data_plt_x86_64) [both versions]:
+ Initialize layout_.
+ (Output_data_plt_x86_64::do_write): Write address of .dynamic
+ section to first entry in .got.plt section.
+ * layout.h (Layout::dynamic_section): New function.
+
2011-07-13 Sriraman Tallam <tmsriram@google.com>
* archive.cc (Archive::get_elf_object_for_member): Add extra parameter
unsigned int got_offset;
};
+ // A pointer to the Layout class, so that we can find the .dynamic
+ // section when we write out the GOT PLT section.
+ Layout* layout_;
// The reloc section.
Reloc_section* rel_;
// The TLS_DESC relocations, if necessary. These must follow the
Output_data_plt_i386::Output_data_plt_i386(Layout* layout,
Output_data_space* got_plt,
Output_data_space* got_irelative)
- : Output_section_data(16), tls_desc_rel_(NULL), irelative_rel_(NULL),
- got_plt_(got_plt), got_irelative_(got_irelative), count_(0),
- irelative_count_(0), global_ifuncs_(), local_ifuncs_()
+ : Output_section_data(16), layout_(layout), tls_desc_rel_(NULL),
+ irelative_rel_(NULL), got_plt_(got_plt), got_irelative_(got_irelative),
+ count_(0), irelative_count_(0), global_ifuncs_(), local_ifuncs_()
{
this->rel_ = new Reloc_section(false);
layout->add_output_section_data(".rel.plt", elfcpp::SHT_REL,
unsigned char* got_pov = got_view;
- memset(got_pov, 0, 12);
- got_pov += 12;
+ // The first entry in the GOT is the address of the .dynamic section
+ // aka the PT_DYNAMIC segment. The next two entries are reserved.
+ // We saved space for them when we created the section in
+ // Target_i386::got_section.
+ Output_section* dynamic = this->layout_->dynamic_section();
+ uint32_t dynamic_addr = dynamic == NULL ? 0 : dynamic->address();
+ elfcpp::Swap<32, false>::writeval(got_pov, dynamic_addr);
+ got_pov += 4;
+ memset(got_pov, 0, 8);
+ got_pov += 8;
const int rel_size = elfcpp::Elf_sizes<32>::rel_size;
Output_data_plt_x86_64(Layout* layout, Output_data_got<64, false>* got,
Output_data_space* got_plt,
Output_data_space* got_irelative)
- : Output_section_data(16), tlsdesc_rel_(NULL), irelative_rel_(NULL),
- got_(got), got_plt_(got_plt), got_irelative_(got_irelative), count_(0),
- irelative_count_(0), tlsdesc_got_offset_(-1U), free_list_()
+ : Output_section_data(16), layout_(layout), tlsdesc_rel_(NULL),
+ irelative_rel_(NULL), got_(got), got_plt_(got_plt),
+ got_irelative_(got_irelative), count_(0), irelative_count_(0),
+ tlsdesc_got_offset_(-1U), free_list_()
{ this->init(layout); }
Output_data_plt_x86_64(Layout* layout, Output_data_got<64, false>* got,
Output_data_space* got_irelative,
unsigned int plt_count)
: Output_section_data((plt_count + 1) * plt_entry_size, 16, false),
- tlsdesc_rel_(NULL), irelative_rel_(NULL), got_(got), got_plt_(got_plt),
- got_irelative_(got_irelative), count_(plt_count), irelative_count_(0),
- tlsdesc_got_offset_(-1U), free_list_()
+ layout_(layout), tlsdesc_rel_(NULL), irelative_rel_(NULL), got_(got),
+ got_plt_(got_plt), got_irelative_(got_irelative), count_(plt_count),
+ irelative_count_(0), tlsdesc_got_offset_(-1U), free_list_()
{
this->init(layout);
void
do_write(Output_file*);
+ // A pointer to the Layout class, so that we can find the .dynamic
+ // section when we write out the GOT PLT section.
+ Layout* layout_;
// The reloc section.
Reloc_section* rel_;
// The TLSDESC relocs, if necessary. These must follow the regular
unsigned char* got_pov = got_view;
- memset(got_pov, 0, 24);
- got_pov += 24;
+ // The first entry in the GOT is the address of the .dynamic section
+ // aka the PT_DYNAMIC segment. The next two entries are reserved.
+ // We saved space for them when we created the section in
+ // Target_x86_64::got_section.
+ Output_section* dynamic = this->layout_->dynamic_section();
+ uint32_t dynamic_addr = dynamic == NULL ? 0 : dynamic->address();
+ elfcpp::Swap<64, false>::writeval(got_pov, dynamic_addr);
+ got_pov += 8;
+ memset(got_pov, 0, 16);
+ got_pov += 16;
unsigned int plt_offset = plt_entry_size;
unsigned int got_offset = 24;