Skip to content

Commit 7ca0627

Browse files
author
George Rimar
committedApr 6, 2016
[ELF] - Do not handle ELF and program header as dummy sections.
ELF and program header are not part of OutputSections list anymore. That helps to avoid having and working with functions like dummySectionsNum(). Still keeping them as sections helps to simplify the code. Differential revision: http://reviews.llvm.org/D18743 llvm-svn: 265522
1 parent 12fd504 commit 7ca0627

File tree

1 file changed

+28
-27
lines changed

1 file changed

+28
-27
lines changed
 

‎lld/ELF/Writer.cpp

+28-27
Original file line numberDiff line numberDiff line change
@@ -28,10 +28,6 @@ using namespace llvm::object;
2828
using namespace lld;
2929
using namespace lld::elf;
3030

31-
// Usually there are 2 dummies sections: ELF header and program header.
32-
// Relocatable output does not require program headers to be created.
33-
static unsigned dummySectionsNum() { return Config->Relocatable ? 1 : 2; }
34-
3531
namespace {
3632
// The writer writes a SymbolTable result to a file.
3733
template <class ELFT> class Writer {
@@ -75,6 +71,7 @@ template <class ELFT> class Writer {
7571
void assignAddresses();
7672
void assignFileOffsets();
7773
void setPhdrs();
74+
void fixHeaders();
7875
void fixSectionAlignments();
7976
void fixAbsoluteSymbols();
8077
void openFile();
@@ -103,14 +100,6 @@ template <class ELFT> class Writer {
103100
std::vector<OutputSectionBase<ELFT> *> OutputSections;
104101
std::vector<std::unique_ptr<OutputSectionBase<ELFT>>> OwningSections;
105102

106-
// We create a section for the ELF header and one for the program headers.
107-
ArrayRef<OutputSectionBase<ELFT> *> getSections() const {
108-
return makeArrayRef(OutputSections).slice(dummySectionsNum());
109-
}
110-
unsigned getNumSections() const {
111-
return OutputSections.size() + 1 - dummySectionsNum();
112-
}
113-
114103
void addRelIpltSymbols();
115104
void addStartEndSymbols();
116105
void addStartStopSymbols(OutputSectionBase<ELFT> *Sec);
@@ -223,6 +212,7 @@ template <class ELFT> void Writer<ELFT>::run() {
223212
assignFileOffsets();
224213
} else {
225214
createPhdrs();
215+
fixHeaders();
226216
fixSectionAlignments();
227217
assignAddresses();
228218
assignFileOffsets();
@@ -983,10 +973,6 @@ template <class ELFT> static void sortCtorsDtors(OutputSectionBase<ELFT> *S) {
983973

984974
// Create output section objects and add them to OutputSections.
985975
template <class ELFT> void Writer<ELFT>::createSections() {
986-
OutputSections.push_back(Out<ELFT>::ElfHeader);
987-
if (!Config->Relocatable)
988-
OutputSections.push_back(Out<ELFT>::ProgramHeaders);
989-
990976
// Add .interp first because some loaders want to see that section
991977
// on the first page of the executable file when loaded into memory.
992978
if (needsInterpSection())
@@ -1075,7 +1061,7 @@ template <class ELFT> void Writer<ELFT>::createSections() {
10751061
}
10761062
}
10771063

1078-
for (OutputSectionBase<ELFT> *Sec : getSections())
1064+
for (OutputSectionBase<ELFT> *Sec : OutputSections)
10791065
Sec->assignOffsets();
10801066

10811067
// Now that we have defined all possible symbols including linker-
@@ -1112,11 +1098,11 @@ template <class ELFT> void Writer<ELFT>::createSections() {
11121098
std::stable_sort(OutputSections.begin(), OutputSections.end(),
11131099
compareSections<ELFT>);
11141100

1115-
for (unsigned I = dummySectionsNum(), N = OutputSections.size(); I < N; ++I)
1116-
OutputSections[I]->SectionIndex = I + 1 - dummySectionsNum();
1117-
1118-
for (OutputSectionBase<ELFT> *Sec : getSections())
1101+
unsigned I = 1;
1102+
for (OutputSectionBase<ELFT> *Sec : OutputSections) {
1103+
Sec->SectionIndex = I++;
11191104
Sec->setSHName(Out<ELFT>::ShStrTab->addString(Sec->getName()));
1105+
}
11201106

11211107
// Finalizers fix each section's size.
11221108
// .dynsym is finalized early since that may fill up .gnu.hash.
@@ -1284,6 +1270,7 @@ template <class ELFT> void Writer<ELFT>::createPhdrs() {
12841270
uintX_t Flags = PF_R;
12851271
Phdr *Load = AddHdr(PT_LOAD, Flags);
12861272
AddSec(*Load, Out<ELFT>::ElfHeader);
1273+
AddSec(*Load, Out<ELFT>::ProgramHeaders);
12871274

12881275
Phdr TlsHdr(PT_TLS, PF_R);
12891276
Phdr RelRo(PT_GNU_RELRO, PF_R);
@@ -1371,11 +1358,23 @@ template <class ELFT> void Writer<ELFT>::fixSectionAlignments() {
13711358
}
13721359
}
13731360

1361+
// We should set file offsets and VAs for elf header and program headers
1362+
// sections. These are special, we do not include them into output sections
1363+
// list, but have them to simplify the code.
1364+
template <class ELFT> void Writer<ELFT>::fixHeaders() {
1365+
Out<ELFT>::ElfHeader->setVA(Target->getVAStart());
1366+
Out<ELFT>::ElfHeader->setFileOffset(0);
1367+
uintX_t Off = Out<ELFT>::ElfHeader->getSize();
1368+
Out<ELFT>::ProgramHeaders->setVA(Off + Target->getVAStart());
1369+
Out<ELFT>::ProgramHeaders->setFileOffset(Off);
1370+
}
1371+
13741372
// Assign VAs (addresses at run-time) to output sections.
13751373
template <class ELFT> void Writer<ELFT>::assignAddresses() {
1376-
uintX_t ThreadBssOffset = 0;
1377-
uintX_t VA = Target->getVAStart();
1374+
uintX_t VA = Target->getVAStart() + Out<ELFT>::ElfHeader->getSize() +
1375+
Out<ELFT>::ProgramHeaders->getSize();
13781376

1377+
uintX_t ThreadBssOffset = 0;
13791378
for (OutputSectionBase<ELFT> *Sec : OutputSections) {
13801379
uintX_t Align = Sec->getAlign();
13811380
if (Sec->PageAlign)
@@ -1397,7 +1396,9 @@ template <class ELFT> void Writer<ELFT>::assignAddresses() {
13971396

13981397
// Assign file offsets to output sections.
13991398
template <class ELFT> void Writer<ELFT>::assignFileOffsets() {
1400-
uintX_t Off = 0;
1399+
uintX_t Off =
1400+
Out<ELFT>::ElfHeader->getSize() + Out<ELFT>::ProgramHeaders->getSize();
1401+
14011402
for (OutputSectionBase<ELFT> *Sec : OutputSections) {
14021403
if (Sec->getType() == SHT_NOBITS) {
14031404
Sec->setFileOffset(Off);
@@ -1411,7 +1412,7 @@ template <class ELFT> void Writer<ELFT>::assignFileOffsets() {
14111412
Off += Sec->getSize();
14121413
}
14131414
SectionHeaderOff = alignTo(Off, sizeof(uintX_t));
1414-
FileSize = SectionHeaderOff + getNumSections() * sizeof(Elf_Shdr);
1415+
FileSize = SectionHeaderOff + (OutputSections.size() + 1) * sizeof(Elf_Shdr);
14151416
}
14161417

14171418
// Finalize the program headers. We call this function after we assign
@@ -1541,7 +1542,7 @@ template <class ELFT> void Writer<ELFT>::writeHeader() {
15411542
EHdr->e_ehsize = sizeof(Elf_Ehdr);
15421543
EHdr->e_phnum = Phdrs.size();
15431544
EHdr->e_shentsize = sizeof(Elf_Shdr);
1544-
EHdr->e_shnum = getNumSections();
1545+
EHdr->e_shnum = OutputSections.size() + 1;
15451546
EHdr->e_shstrndx = Out<ELFT>::ShStrTab->SectionIndex;
15461547

15471548
if (Config->EMachine == EM_MIPS)
@@ -1559,7 +1560,7 @@ template <class ELFT> void Writer<ELFT>::writeHeader() {
15591560

15601561
// Write the section header table. Note that the first table entry is null.
15611562
auto *SHdrs = reinterpret_cast<Elf_Shdr *>(Buf + EHdr->e_shoff);
1562-
for (OutputSectionBase<ELFT> *Sec : getSections())
1563+
for (OutputSectionBase<ELFT> *Sec : OutputSections)
15631564
Sec->writeHeaderTo(++SHdrs);
15641565
}
15651566

0 commit comments

Comments
 (0)
Please sign in to comment.