Files
clang-p2996/bolt/lib/Passes/FixRelaxationPass.cpp
Vladislav Khmelevsky 17ed8f2928 [BOLT][AArch64] Handle adrp+ld64 linker relaxations
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
2022-12-23 01:20:18 +04:00

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