24#define AARCH64_POINTER_AUTH_NAME "AArch64 Pointer Authentication"
28class AArch64PointerAuthImpl {
59 return new AArch64PointerAuthLegacy();
62char AArch64PointerAuthLegacy::ID = 0;
81 auto &MF = *
MBB.getParent();
85 if (MFnI.branchProtectionPAuthLR()) {
86 CFIBuilder.buildNegateRAStateWithPC();
87 }
else if (!MF.getTarget().getTargetTriple().isOSBinFormatMachO()) {
88 CFIBuilder.buildNegateRAState();
94 auto &MFnI = *MF.
getInfo<AArch64FunctionInfo>();
95 bool UseBKey = MFnI.shouldSignWithBKey();
96 bool EmitCFI = MFnI.needsDwarfUnwindInfo(MF);
111 if (MFnI.branchProtectionPAuthLR()) {
113 MFnI.setSigningInstrLabel(PACSym);
118 if (MFnI.branchProtectionPAuthLR() && Subtarget->hasPAuthLR()) {
121 TII->get(UseBKey ? AArch64::PACIBSPPC : AArch64::PACIASPPC))
125 if (MFnI.branchProtectionPAuthLR()) {
131 TII->get(UseBKey ? AArch64::PACIBSP : AArch64::PACIASP))
134 if (!MFnI.branchProtectionPAuthLR())
138 if (!EmitCFI && NeedsWinCFI) {
144void AArch64PointerAuthImpl::authenticateLR(
146 const AArch64FunctionInfo *MFnI = MF.
getInfo<AArch64FunctionInfo>();
165 bool TerminatorIsCombinable =
166 TI !=
MBB.
end() && TI->getOpcode() == AArch64::RET;
169 if (Subtarget->hasPAuth() && TerminatorIsCombinable && !NeedsWinCFI &&
172 assert(PACSym &&
"No PAC instruction to refer to");
174 TII->get(UseBKey ? AArch64::RETABSPPCi : AArch64::RETAASPPCi))
184 BuildMI(
MBB, TI,
DL,
TII->get(UseBKey ? AArch64::RETAB : AArch64::RETAA))
192 auto &AFL = *
static_cast<const AArch64FrameLowering *
>(
194 int64_t ArgumentStackToRestore = AFL.getArgumentStackToRestore(MF,
MBB);
200 if (ArgumentStackToRestore < 0) {
212 assert(PACSym &&
"No PAC instruction to refer to");
216 unsigned AutOpc = UseBKey ? AArch64::AUTIB171615 : AArch64::AUTIA171615;
220 assert(PACSym &&
"No PAC instruction to refer to");
232 unsigned AutOpc = UseBKey ? AArch64::AUTIB1716 : AArch64::AUTIA1716;
236 unsigned AutOpc = UseBKey ? AArch64::AUTIB1716 : AArch64::AUTIA1716;
260 SmallVector<MachineInstr *, 2> SPMods;
261 if (ArgumentStackToRestore > 0) {
263 if ((
I->getOpcode() == AArch64::ADDXri ||
264 I->getOpcode() == AArch64::SUBXri) &&
265 I->getOperand(0).getReg() == AArch64::SP &&
266 I->getOperand(1).getReg() == AArch64::SP)
270 for (
auto *
MI : SPMods)
271 MI->removeFromParent();
274 assert(PACSym &&
"No PAC instruction to refer to");
277 TII->get(UseBKey ? AArch64::AUTIBSPPCi : AArch64::AUTIASPPCi))
289 TII->get(UseBKey ? AArch64::AUTIBSP : AArch64::AUTIASP))
300 for (
auto *
MI : SPMods)
327 for (
auto &
MBB : MF) {
328 for (
auto &
MI :
MBB) {
329 switch (
MI.getOpcode()) {
332 case AArch64::PAUTH_PROLOGUE:
333 case AArch64::PAUTH_EPILOGUE:
340 for (
auto It : PAuthPseudoInstrs) {
341 switch (It->getOpcode()) {
342 case AArch64::PAUTH_PROLOGUE:
345 case AArch64::PAUTH_EPILOGUE:
346 authenticateLR(MF, It);
351 It->eraseFromParent();
358bool AArch64PointerAuthLegacy::runOnMachineFunction(MachineFunction &MF) {
359 return AArch64PointerAuthImpl().run(MF);
365 const bool Changed = AArch64PointerAuthImpl().run(MF);
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
#define AARCH64_POINTER_AUTH_NAME
static void emitPACSymOffsetIntoReg(const TargetInstrInfo &TII, MachineBasicBlock &MBB, MachineBasicBlock::iterator I, DebugLoc DL, MCSymbol *PACSym, Register Reg)
static void emitPACCFI(MachineBasicBlock &MBB, MachineBasicBlock::iterator MBBI, MachineInstr::MIFlag Flags, bool EmitCFI)
MachineBasicBlock MachineBasicBlock::iterator DebugLoc DL
MachineBasicBlock MachineBasicBlock::iterator MBBI
const HexagonInstrInfo * TII
#define INITIALIZE_PASS(passName, arg, name, cfg, analysis)
AArch64FunctionInfo - This class is derived from MachineFunctionInfo and contains private AArch64-spe...
bool branchProtectionPAuthLR() const
bool needsAsyncDwarfUnwindInfo(const MachineFunction &MF) const
MCSymbol * getSigningInstrLabel() const
bool shouldSignWithBKey() const
PreservedAnalyses run(MachineFunction &MF, MachineFunctionAnalysisManager &MFAM)
const AArch64InstrInfo * getInstrInfo() const override
Represents analyses that only rely on functions' control flow.
Helper class for creating CFI instructions and inserting them into MIR.
FunctionPass class - This class is used to implement most global optimizations.
bool hasFnAttribute(Attribute::AttrKind Kind) const
Return true if the function has the attribute.
LLVM_ABI MCSymbol * createTempSymbol()
Create a temporary symbol with a unique name.
MCSymbol - Instances of this class represent a symbol name in the MC file, and MCSymbols are created ...
LLVM_ABI instr_iterator insert(instr_iterator I, MachineInstr *M)
Insert MI into the instruction list before I, possibly inside a bundle.
const MachineFunction * getParent() const
Return the MachineFunction containing this basic block.
LLVM_ABI instr_iterator erase(instr_iterator I)
Remove an instruction from the instruction list and delete it.
LLVM_ABI instr_iterator getFirstInstrTerminator()
Same getFirstTerminator but it ignores bundles and return an instr_iterator instead.
MachineInstrBundleIterator< MachineInstr > iterator
MachineFunctionPass - This class adapts the FunctionPass interface to allow convenient creation of pa...
const TargetSubtargetInfo & getSubtarget() const
getSubtarget - Return the subtarget for which this machine code is being compiled.
MCContext & getContext() const
Function & getFunction()
Return the LLVM function that this machine code represents.
Ty * getInfo()
getInfo - Keep track of various per-function pieces of information for backends that would like to do...
const TargetMachine & getTarget() const
getTarget - Return the target machine this machine code is compiled with
const MachineInstrBuilder & addReg(Register RegNo, RegState Flags={}, unsigned SubReg=0) const
Add a new virtual register operand.
const MachineInstrBuilder & setMIFlag(MachineInstr::MIFlag Flag) const
const MachineInstrBuilder & addImm(int64_t Val) const
Add a new immediate operand.
const MachineInstrBuilder & addSym(MCSymbol *Sym, unsigned char TargetFlags=0) const
const MachineInstrBuilder & copyImplicitOps(const MachineInstr &OtherMI) const
Copy all the implicit operands from OtherMI onto this one.
LLVM_ABI void setPreInstrSymbol(MachineFunction &MF, MCSymbol *Symbol)
Set a symbol that will be emitted just prior to the instruction itself.
A set of analyses that are preserved following a run of a transformation pass.
static PreservedAnalyses all()
Construct a special preserved set that preserves all passes.
PreservedAnalyses & preserveSet()
Mark an analysis set as preserved.
Wrapper class representing virtual and physical registers.
void push_back(const T &Elt)
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
int64_t getFixed() const
Returns the fixed component of the stack.
Represent a constant reference to a string, i.e.
TargetInstrInfo - Interface to description of machine instruction set.
const Triple & getTargetTriple() const
bool isOSBinFormatMachO() const
Tests whether the environment is MachO.
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
@ MO_NC
MO_NC - Indicates whether the linker is expected to check the symbol reference for overflow.
@ MO_PAGEOFF
MO_PAGEOFF - A symbol operand with this flag represents the offset of that symbol within a 4K page.
@ MO_PAGE
MO_PAGE - A symbol operand with this flag represents the pc-relative offset of the 4K page containing...
unsigned getCheckerSizeInBytes(AuthCheckMethod Method)
Returns the number of bytes added by checkAuthenticatedRegister.
AuthCheckMethod
Variants of check performed on an authenticated pointer.
@ XPACHint
Check by comparing the authenticated value with an XPAC-ed one without using PAuth instructions not e...
@ DummyLoad
Perform a load to a temporary register.
@ HighBitsNoTBI
Check by comparing bits 62 and 61 of the authenticated address.
@ None
Do not check the value at all.
@ XPAC
Similar to XPACHint but using Armv8.3-only XPAC instruction, thus not restricted to LR:
unsigned ID
LLVM IR allows to use arbitrary numbers as calling convention identifiers.
This is an optimization pass for GlobalISel generic memory operations.
MachineInstrBuilder BuildMI(MachineFunction &MF, const MIMetadata &MIMD, const MCInstrDesc &MCID)
Builder interface. Specify how to create the initial instruction itself.
FunctionPass * createAArch64PointerAuthPass()
AnalysisManager< MachineFunction > MachineFunctionAnalysisManager
LLVM_ABI PreservedAnalyses getMachineFunctionPassPreservedAnalyses()
Returns the minimum set of Analyses that all machine function passes must preserve.
void emitFrameOffset(MachineBasicBlock &MBB, MachineBasicBlock::iterator MBBI, const DebugLoc &DL, unsigned DestReg, unsigned SrcReg, StackOffset Offset, const TargetInstrInfo *TII, MachineInstr::MIFlag=MachineInstr::NoFlags, bool SetNZCV=false, bool NeedsWinCFI=false, bool *HasWinCFI=nullptr, bool EmitCFAOffset=false, StackOffset InitialOffset={}, unsigned FrameReg=AArch64::SP)
emitFrameOffset - Emit instructions as needed to set DestReg to SrcReg plus Offset.