@@ -28,10 +28,6 @@ using namespace llvm::object;
28
28
using namespace lld ;
29
29
using namespace lld ::elf;
30
30
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
-
35
31
namespace {
36
32
// The writer writes a SymbolTable result to a file.
37
33
template <class ELFT > class Writer {
@@ -75,6 +71,7 @@ template <class ELFT> class Writer {
75
71
void assignAddresses ();
76
72
void assignFileOffsets ();
77
73
void setPhdrs ();
74
+ void fixHeaders ();
78
75
void fixSectionAlignments ();
79
76
void fixAbsoluteSymbols ();
80
77
void openFile ();
@@ -103,14 +100,6 @@ template <class ELFT> class Writer {
103
100
std::vector<OutputSectionBase<ELFT> *> OutputSections;
104
101
std::vector<std::unique_ptr<OutputSectionBase<ELFT>>> OwningSections;
105
102
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
-
114
103
void addRelIpltSymbols ();
115
104
void addStartEndSymbols ();
116
105
void addStartStopSymbols (OutputSectionBase<ELFT> *Sec);
@@ -223,6 +212,7 @@ template <class ELFT> void Writer<ELFT>::run() {
223
212
assignFileOffsets ();
224
213
} else {
225
214
createPhdrs ();
215
+ fixHeaders ();
226
216
fixSectionAlignments ();
227
217
assignAddresses ();
228
218
assignFileOffsets ();
@@ -983,10 +973,6 @@ template <class ELFT> static void sortCtorsDtors(OutputSectionBase<ELFT> *S) {
983
973
984
974
// Create output section objects and add them to OutputSections.
985
975
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
-
990
976
// Add .interp first because some loaders want to see that section
991
977
// on the first page of the executable file when loaded into memory.
992
978
if (needsInterpSection ())
@@ -1075,7 +1061,7 @@ template <class ELFT> void Writer<ELFT>::createSections() {
1075
1061
}
1076
1062
}
1077
1063
1078
- for (OutputSectionBase<ELFT> *Sec : getSections () )
1064
+ for (OutputSectionBase<ELFT> *Sec : OutputSections )
1079
1065
Sec->assignOffsets ();
1080
1066
1081
1067
// Now that we have defined all possible symbols including linker-
@@ -1112,11 +1098,11 @@ template <class ELFT> void Writer<ELFT>::createSections() {
1112
1098
std::stable_sort (OutputSections.begin (), OutputSections.end (),
1113
1099
compareSections<ELFT>);
1114
1100
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++;
1119
1104
Sec->setSHName (Out<ELFT>::ShStrTab->addString (Sec->getName ()));
1105
+ }
1120
1106
1121
1107
// Finalizers fix each section's size.
1122
1108
// .dynsym is finalized early since that may fill up .gnu.hash.
@@ -1284,6 +1270,7 @@ template <class ELFT> void Writer<ELFT>::createPhdrs() {
1284
1270
uintX_t Flags = PF_R;
1285
1271
Phdr *Load = AddHdr (PT_LOAD, Flags);
1286
1272
AddSec (*Load, Out<ELFT>::ElfHeader);
1273
+ AddSec (*Load, Out<ELFT>::ProgramHeaders);
1287
1274
1288
1275
Phdr TlsHdr (PT_TLS, PF_R);
1289
1276
Phdr RelRo (PT_GNU_RELRO, PF_R);
@@ -1371,11 +1358,23 @@ template <class ELFT> void Writer<ELFT>::fixSectionAlignments() {
1371
1358
}
1372
1359
}
1373
1360
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
+
1374
1372
// Assign VAs (addresses at run-time) to output sections.
1375
1373
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 ();
1378
1376
1377
+ uintX_t ThreadBssOffset = 0 ;
1379
1378
for (OutputSectionBase<ELFT> *Sec : OutputSections) {
1380
1379
uintX_t Align = Sec->getAlign ();
1381
1380
if (Sec->PageAlign )
@@ -1397,7 +1396,9 @@ template <class ELFT> void Writer<ELFT>::assignAddresses() {
1397
1396
1398
1397
// Assign file offsets to output sections.
1399
1398
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
+
1401
1402
for (OutputSectionBase<ELFT> *Sec : OutputSections) {
1402
1403
if (Sec->getType () == SHT_NOBITS) {
1403
1404
Sec->setFileOffset (Off);
@@ -1411,7 +1412,7 @@ template <class ELFT> void Writer<ELFT>::assignFileOffsets() {
1411
1412
Off += Sec->getSize ();
1412
1413
}
1413
1414
SectionHeaderOff = alignTo (Off, sizeof (uintX_t));
1414
- FileSize = SectionHeaderOff + getNumSections ( ) * sizeof (Elf_Shdr);
1415
+ FileSize = SectionHeaderOff + (OutputSections. size () + 1 ) * sizeof (Elf_Shdr);
1415
1416
}
1416
1417
1417
1418
// Finalize the program headers. We call this function after we assign
@@ -1541,7 +1542,7 @@ template <class ELFT> void Writer<ELFT>::writeHeader() {
1541
1542
EHdr->e_ehsize = sizeof (Elf_Ehdr);
1542
1543
EHdr->e_phnum = Phdrs.size ();
1543
1544
EHdr->e_shentsize = sizeof (Elf_Shdr);
1544
- EHdr->e_shnum = getNumSections () ;
1545
+ EHdr->e_shnum = OutputSections. size () + 1 ;
1545
1546
EHdr->e_shstrndx = Out<ELFT>::ShStrTab->SectionIndex ;
1546
1547
1547
1548
if (Config->EMachine == EM_MIPS)
@@ -1559,7 +1560,7 @@ template <class ELFT> void Writer<ELFT>::writeHeader() {
1559
1560
1560
1561
// Write the section header table. Note that the first table entry is null.
1561
1562
auto *SHdrs = reinterpret_cast <Elf_Shdr *>(Buf + EHdr->e_shoff );
1562
- for (OutputSectionBase<ELFT> *Sec : getSections () )
1563
+ for (OutputSectionBase<ELFT> *Sec : OutputSections )
1563
1564
Sec->writeHeaderTo (++SHdrs);
1564
1565
}
1565
1566
0 commit comments