forked from OSchip/llvm-project
Diagnose another case of the location counter moving backwards.
This case should be possible to handle, but it is hard: * In order to create program headers correctly, we have to scan the sections in the order they are in the file. * To find that order, we have to "execute" the linker script. * The linker script can contain SIZEOF_HEADERS. So to support this we have to start with a guess of how many headers we need (3), run the linker script and try to create the program headers. If it turns out we need more headers, we run the script again with a larger SIZEOF_HEADERS. Also, running the linker script depends on knowing the size of the sections, so we have to finalize them. But creating the program headers can change the value stored in some sections, so we have to split size finalization and content finalization. Looks like the last part is also needed for range extension thunks, so we might support this at some point. For now just report an error instead of producing broken files. llvm-svn: 295458
This commit is contained in:
parent
28096200a8
commit
679828ff92
|
@ -91,24 +91,27 @@ static bool isUnderSysroot(StringRef Path) {
|
|||
return false;
|
||||
}
|
||||
|
||||
template <class ELFT> void LinkerScript<ELFT>::setDot(Expr E, bool InSec) {
|
||||
uintX_t Val = E(Dot);
|
||||
if (Val < Dot) {
|
||||
if (InSec)
|
||||
error("unable to move location counter backward for: " + CurOutSec->Name);
|
||||
else
|
||||
error("unable to move location counter backward");
|
||||
}
|
||||
Dot = Val;
|
||||
// Update to location counter means update to section size.
|
||||
if (InSec)
|
||||
CurOutSec->Size = Dot - CurOutSec->Addr;
|
||||
}
|
||||
|
||||
// Sets value of a symbol. Two kinds of symbols are processed: synthetic
|
||||
// symbols, whose value is an offset from beginning of section and regular
|
||||
// symbols whose value is absolute.
|
||||
template <class ELFT>
|
||||
void LinkerScript<ELFT>::assignSymbol(SymbolAssignment *Cmd, bool InSec) {
|
||||
if (Cmd->Name == ".") {
|
||||
uintX_t Val = Cmd->Expression(Dot);
|
||||
if (Val < Dot) {
|
||||
if (InSec)
|
||||
error("unable to move location counter backward for: " +
|
||||
CurOutSec->Name);
|
||||
else
|
||||
error("unable to move location counter backward");
|
||||
}
|
||||
Dot = Val;
|
||||
// Update to location counter means update to section size.
|
||||
if (InSec)
|
||||
CurOutSec->Size = Dot - CurOutSec->Addr;
|
||||
setDot(Cmd->Expression, InSec);
|
||||
return;
|
||||
}
|
||||
|
||||
|
@ -566,6 +569,9 @@ void LinkerScript<ELFT>::assignOffsets(OutputSectionCommand *Cmd) {
|
|||
if (!Sec)
|
||||
return;
|
||||
|
||||
if (Cmd->AddrExpr && Sec->Flags & SHF_ALLOC)
|
||||
setDot(Cmd->AddrExpr);
|
||||
|
||||
// Handle align (e.g. ".foo : ALIGN(16) { ... }").
|
||||
if (Cmd->AlignExpr)
|
||||
Sec->updateAlignment(Cmd->AlignExpr(0));
|
||||
|
@ -801,8 +807,6 @@ void LinkerScript<ELFT>::assignAddresses(std::vector<PhdrEntry> &Phdrs) {
|
|||
}
|
||||
|
||||
auto *Cmd = cast<OutputSectionCommand>(Base.get());
|
||||
if (Cmd->AddrExpr)
|
||||
Dot = Cmd->AddrExpr(Dot);
|
||||
assignOffsets(Cmd);
|
||||
}
|
||||
|
||||
|
|
|
@ -278,6 +278,7 @@ private:
|
|||
void assignSymbol(SymbolAssignment *Cmd, bool InSec = false);
|
||||
void addSymbol(SymbolAssignment *Cmd);
|
||||
void computeInputSections(InputSectionDescription *);
|
||||
void setDot(Expr E, bool InSec = false);
|
||||
|
||||
void discard(ArrayRef<InputSectionBase<ELFT> *> V);
|
||||
|
||||
|
|
|
@ -0,0 +1,10 @@
|
|||
# REQUIRES: x86
|
||||
# RUN: llvm-mc -filetype=obj -triple=x86_64-linux %s -o %t.o
|
||||
# RUN: echo "SECTIONS { .data 0x4000 : { *(.data) } .text 0x2000 : { *(.text) } }" > %t.script
|
||||
# RUN: not ld.lld -o %t.so --script %t.script %t.o -shared 2>&1 | FileCheck %s
|
||||
|
||||
# CHECK: error: unable to move location counter backward
|
||||
|
||||
.quad 0
|
||||
.data
|
||||
.quad 0
|
Loading…
Reference in New Issue