diff options
Diffstat (limited to 'gold/output.cc')
-rw-r--r-- | gold/output.cc | 60 |
1 files changed, 49 insertions, 11 deletions
diff --git a/gold/output.cc b/gold/output.cc index fcba1135a50..90923010368 100644 --- a/gold/output.cc +++ b/gold/output.cc @@ -24,6 +24,15 @@ Output_data_const::write(Output_file* output, off_t off) output->write(off, data_.data(), data_.size()); } +// Output_segment_header methods. + +void +Output_segment_headers::write(Output_file*, off_t) +{ + // FIXME: Unimplemented. + abort(); +} + // Output_section methods. // Construct an Output_section. NAME will point into a Stringpool. @@ -35,7 +44,6 @@ Output_section::Output_section(const char* name, elfcpp::Elf_Word type, addralign_(0), entsize_(0), offset_(0), - size_(0), link_(0), info_(0), type_(type), @@ -43,6 +51,10 @@ Output_section::Output_section(const char* name, elfcpp::Elf_Word type, { } +Output_section::~Output_section() +{ +} + // Add an input section to an Output_section. We don't keep track of // input sections for an Output_section. Instead, each Object keeps // track of the Output_section for each of its input sections. @@ -60,21 +72,22 @@ Output_section::add_input_section(Object* object, const char* secname, static_cast<unsigned long>(addralign), secname); gold_exit(false); } - this->size_ = (this->size_ + addralign - 1) &~ (addralign - 1); if (addralign > this->addralign_) this->addralign_ = addralign; - off_t ret = this->size_; - this->size_ += shdr.get_sh_size(); + off_t ssize = this->get_size(); + ssize = (ssize + addralign - 1) &~ (addralign - 1); - return ret; + this->set_size(ssize + shdr.get_sh_size()); + + return size; } // Output segment methods. Output_segment::Output_segment(elfcpp::Elf_Word type, elfcpp::Elf_Word flags) - : output_sections_(), + : output_data_(), vaddr_(0), paddr_(0), memsz_(0), @@ -91,6 +104,9 @@ Output_segment::Output_segment(elfcpp::Elf_Word type, elfcpp::Elf_Word flags) void Output_segment::add_output_section(Output_section* os) { + // Update the segment flags. + this->flags_ |= os->flags() & (elfcpp::PF_R | elfcpp::PF_W | elfcpp::PF_X); + // So that PT_NOTE segments will work correctly, we need to ensure // that all SHT_NOTE sections are adjacent. This will normally // happen automatically, because all the SHT_NOTE input sections @@ -99,22 +115,44 @@ Output_segment::add_output_section(Output_section* os) // flags, and thus be in different output sections, but for the // different section flags to map into the same segment flags and // thus the same output segment. + + // Note that while there may be many input sections in an output + // section, there are normally only a few output sections in an + // output segment. This loop is expected to be fast. + if (os->type() == elfcpp::SHT_NOTE) { - for (Section_list::iterator p = this->output_sections_.begin(); - p != this->output_sections_.end(); + for (Layout::Data_list::iterator p = this->output_data_.begin(); + p != this->output_data_.end(); + ++p) + { + if ((*p)->is_section_type(elfcpp::SHT_NOTE)) + { + ++p; + this->output_data_.insert(p, os); + return; + } + } + } + + // Similarly, so that PT_TLS segments will work, we need to group + // SHF_TLS sections. + if ((os->flags() & elfcpp::SHF_TLS) != 0) + { + for (Layout::Data_list::iterator p = this->output_data_.begin(); + p != this->output_data_.end(); ++p) { - if ((*p)->type() == elfcpp::SHT_NOTE) + if ((*p)->is_section_flag_set(elfcpp::SHF_TLS)) { ++p; - this->output_sections_.insert(p, os); + this->output_data_.insert(p, os); return; } } } - this->output_sections_.push_back(os); + this->output_data_.push_back(os); } // Output_file methods. |