Linker might relax adrp + ldr got address loading to adrp + add for local non-preemptible symbols (e.g. hidden/protected symbols in executable). As usually linker doesn't change relocations properly after relaxation, so we have to handle such cases by ourselves. To do that during relocations reading we change LD64 reloc to ADD if instruction mismatch found and introduce FixRelaxationPass that searches for ADRP+ADD pairs and after performing some checks we're replacing ADRP target symbol to already fixed ADDs one. Vladislav Khmelevsky, Advanced Software Technology Lab, Huawei Differential Revision: https://reviews.llvm.org/D138097
64 lines
2.1 KiB
C++
64 lines
2.1 KiB
C++
#include "bolt/Passes/FixRelaxationPass.h"
|
|
#include "bolt/Core/ParallelUtilities.h"
|
|
|
|
using namespace llvm;
|
|
|
|
namespace llvm {
|
|
namespace bolt {
|
|
|
|
// This function finds ADRP+ADD instruction sequences that originally before
|
|
// linker relaxations were ADRP+LDR. We've modified LDR/ADD relocation properly
|
|
// during relocation reading, so its targeting right symbol. As for ADRP its
|
|
// target is wrong before this pass since we won't be able to recognize and
|
|
// properly change R_AARCH64_ADR_GOT_PAGE relocation to
|
|
// R_AARCH64_ADR_PREL_PG_HI21 during relocation reading. Now we're searching for
|
|
// ADRP+ADD sequences, checking that ADRP points to the GOT-table symbol and the
|
|
// target of ADD is another symbol. When found change ADRP symbol reference to
|
|
// the ADDs one.
|
|
void FixRelaxations::runOnFunction(BinaryFunction &BF) {
|
|
BinaryContext &BC = BF.getBinaryContext();
|
|
for (BinaryBasicBlock &BB : BF) {
|
|
for (auto II = BB.begin(); II != BB.end(); ++II) {
|
|
MCInst &Adrp = *II;
|
|
if (BC.MIB->isPseudo(Adrp) || !BC.MIB->isADRP(Adrp))
|
|
continue;
|
|
|
|
const MCSymbol *AdrpSymbol = BC.MIB->getTargetSymbol(Adrp);
|
|
if (!AdrpSymbol || AdrpSymbol->getName() != "__BOLT_got_zero")
|
|
continue;
|
|
|
|
auto NextII = std::next(II);
|
|
if (NextII == BB.end())
|
|
continue;
|
|
|
|
const MCInst &Add = *NextII;
|
|
if (!BC.MIB->matchAdrpAddPair(Adrp, Add))
|
|
continue;
|
|
|
|
const MCSymbol *Symbol = BC.MIB->getTargetSymbol(Add);
|
|
if (!Symbol || AdrpSymbol == Symbol)
|
|
continue;
|
|
|
|
const int64_t Addend = BC.MIB->getTargetAddend(Add);
|
|
BC.MIB->setOperandToSymbolRef(Adrp, /*OpNum*/ 1, Symbol, Addend,
|
|
BC.Ctx.get(), ELF::R_AARCH64_NONE);
|
|
}
|
|
}
|
|
}
|
|
|
|
void FixRelaxations::runOnFunctions(BinaryContext &BC) {
|
|
if (!BC.isAArch64() || !BC.HasRelocations)
|
|
return;
|
|
|
|
ParallelUtilities::WorkFuncTy WorkFun = [&](BinaryFunction &BF) {
|
|
runOnFunction(BF);
|
|
};
|
|
|
|
ParallelUtilities::runOnEachFunction(
|
|
BC, ParallelUtilities::SchedulingPolicy::SP_INST_LINEAR, WorkFun, nullptr,
|
|
"FixRelaxations");
|
|
}
|
|
|
|
} // namespace bolt
|
|
} // namespace llvm
|