xref: /freebsd-src/contrib/llvm-project/llvm/lib/Target/AArch64/AArch64SLSHardening.cpp (revision 5ffd83dbcc34f10e07f6d3e968ae6365869615f4)
1*5ffd83dbSDimitry Andric //===- AArch64SLSHardening.cpp - Harden Straight Line Missspeculation -----===//
2*5ffd83dbSDimitry Andric //
3*5ffd83dbSDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4*5ffd83dbSDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
5*5ffd83dbSDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6*5ffd83dbSDimitry Andric //
7*5ffd83dbSDimitry Andric //===----------------------------------------------------------------------===//
8*5ffd83dbSDimitry Andric //
9*5ffd83dbSDimitry Andric // This file contains a pass to insert code to mitigate against side channel
10*5ffd83dbSDimitry Andric // vulnerabilities that may happen under straight line miss-speculation.
11*5ffd83dbSDimitry Andric //
12*5ffd83dbSDimitry Andric //===----------------------------------------------------------------------===//
13*5ffd83dbSDimitry Andric 
14*5ffd83dbSDimitry Andric #include "AArch64InstrInfo.h"
15*5ffd83dbSDimitry Andric #include "AArch64Subtarget.h"
16*5ffd83dbSDimitry Andric #include "Utils/AArch64BaseInfo.h"
17*5ffd83dbSDimitry Andric #include "llvm/ADT/BitVector.h"
18*5ffd83dbSDimitry Andric #include "llvm/ADT/SmallVector.h"
19*5ffd83dbSDimitry Andric #include "llvm/CodeGen/IndirectThunks.h"
20*5ffd83dbSDimitry Andric #include "llvm/CodeGen/MachineBasicBlock.h"
21*5ffd83dbSDimitry Andric #include "llvm/CodeGen/MachineFunction.h"
22*5ffd83dbSDimitry Andric #include "llvm/CodeGen/MachineFunctionPass.h"
23*5ffd83dbSDimitry Andric #include "llvm/CodeGen/MachineInstr.h"
24*5ffd83dbSDimitry Andric #include "llvm/CodeGen/MachineInstrBuilder.h"
25*5ffd83dbSDimitry Andric #include "llvm/CodeGen/MachineOperand.h"
26*5ffd83dbSDimitry Andric #include "llvm/CodeGen/MachineRegisterInfo.h"
27*5ffd83dbSDimitry Andric #include "llvm/CodeGen/RegisterScavenging.h"
28*5ffd83dbSDimitry Andric #include "llvm/IR/DebugLoc.h"
29*5ffd83dbSDimitry Andric #include "llvm/Pass.h"
30*5ffd83dbSDimitry Andric #include "llvm/Support/CodeGen.h"
31*5ffd83dbSDimitry Andric #include "llvm/Support/Debug.h"
32*5ffd83dbSDimitry Andric #include "llvm/Target/TargetMachine.h"
33*5ffd83dbSDimitry Andric #include <cassert>
34*5ffd83dbSDimitry Andric 
35*5ffd83dbSDimitry Andric using namespace llvm;
36*5ffd83dbSDimitry Andric 
37*5ffd83dbSDimitry Andric #define DEBUG_TYPE "aarch64-sls-hardening"
38*5ffd83dbSDimitry Andric 
39*5ffd83dbSDimitry Andric #define AARCH64_SLS_HARDENING_NAME "AArch64 sls hardening pass"
40*5ffd83dbSDimitry Andric 
41*5ffd83dbSDimitry Andric namespace {
42*5ffd83dbSDimitry Andric 
43*5ffd83dbSDimitry Andric class AArch64SLSHardening : public MachineFunctionPass {
44*5ffd83dbSDimitry Andric public:
45*5ffd83dbSDimitry Andric   const TargetInstrInfo *TII;
46*5ffd83dbSDimitry Andric   const TargetRegisterInfo *TRI;
47*5ffd83dbSDimitry Andric   const AArch64Subtarget *ST;
48*5ffd83dbSDimitry Andric 
49*5ffd83dbSDimitry Andric   static char ID;
50*5ffd83dbSDimitry Andric 
51*5ffd83dbSDimitry Andric   AArch64SLSHardening() : MachineFunctionPass(ID) {
52*5ffd83dbSDimitry Andric     initializeAArch64SLSHardeningPass(*PassRegistry::getPassRegistry());
53*5ffd83dbSDimitry Andric   }
54*5ffd83dbSDimitry Andric 
55*5ffd83dbSDimitry Andric   bool runOnMachineFunction(MachineFunction &Fn) override;
56*5ffd83dbSDimitry Andric 
57*5ffd83dbSDimitry Andric   StringRef getPassName() const override { return AARCH64_SLS_HARDENING_NAME; }
58*5ffd83dbSDimitry Andric 
59*5ffd83dbSDimitry Andric private:
60*5ffd83dbSDimitry Andric   bool hardenReturnsAndBRs(MachineBasicBlock &MBB) const;
61*5ffd83dbSDimitry Andric   bool hardenBLRs(MachineBasicBlock &MBB) const;
62*5ffd83dbSDimitry Andric   MachineBasicBlock &ConvertBLRToBL(MachineBasicBlock &MBB,
63*5ffd83dbSDimitry Andric                                     MachineBasicBlock::iterator) const;
64*5ffd83dbSDimitry Andric };
65*5ffd83dbSDimitry Andric 
66*5ffd83dbSDimitry Andric } // end anonymous namespace
67*5ffd83dbSDimitry Andric 
68*5ffd83dbSDimitry Andric char AArch64SLSHardening::ID = 0;
69*5ffd83dbSDimitry Andric 
70*5ffd83dbSDimitry Andric INITIALIZE_PASS(AArch64SLSHardening, "aarch64-sls-hardening",
71*5ffd83dbSDimitry Andric                 AARCH64_SLS_HARDENING_NAME, false, false)
72*5ffd83dbSDimitry Andric 
73*5ffd83dbSDimitry Andric static void insertSpeculationBarrier(const AArch64Subtarget *ST,
74*5ffd83dbSDimitry Andric                                      MachineBasicBlock &MBB,
75*5ffd83dbSDimitry Andric                                      MachineBasicBlock::iterator MBBI,
76*5ffd83dbSDimitry Andric                                      DebugLoc DL,
77*5ffd83dbSDimitry Andric                                      bool AlwaysUseISBDSB = false) {
78*5ffd83dbSDimitry Andric   assert(MBBI != MBB.begin() &&
79*5ffd83dbSDimitry Andric          "Must not insert SpeculationBarrierEndBB as only instruction in MBB.");
80*5ffd83dbSDimitry Andric   assert(std::prev(MBBI)->isBarrier() &&
81*5ffd83dbSDimitry Andric          "SpeculationBarrierEndBB must only follow unconditional control flow "
82*5ffd83dbSDimitry Andric          "instructions.");
83*5ffd83dbSDimitry Andric   assert(std::prev(MBBI)->isTerminator() &&
84*5ffd83dbSDimitry Andric          "SpeculationBarrierEndBB must only follow terminators.");
85*5ffd83dbSDimitry Andric   const TargetInstrInfo *TII = ST->getInstrInfo();
86*5ffd83dbSDimitry Andric   unsigned BarrierOpc = ST->hasSB() && !AlwaysUseISBDSB
87*5ffd83dbSDimitry Andric                             ? AArch64::SpeculationBarrierSBEndBB
88*5ffd83dbSDimitry Andric                             : AArch64::SpeculationBarrierISBDSBEndBB;
89*5ffd83dbSDimitry Andric   if (MBBI == MBB.end() ||
90*5ffd83dbSDimitry Andric       (MBBI->getOpcode() != AArch64::SpeculationBarrierSBEndBB &&
91*5ffd83dbSDimitry Andric        MBBI->getOpcode() != AArch64::SpeculationBarrierISBDSBEndBB))
92*5ffd83dbSDimitry Andric     BuildMI(MBB, MBBI, DL, TII->get(BarrierOpc));
93*5ffd83dbSDimitry Andric }
94*5ffd83dbSDimitry Andric 
95*5ffd83dbSDimitry Andric bool AArch64SLSHardening::runOnMachineFunction(MachineFunction &MF) {
96*5ffd83dbSDimitry Andric   ST = &MF.getSubtarget<AArch64Subtarget>();
97*5ffd83dbSDimitry Andric   TII = MF.getSubtarget().getInstrInfo();
98*5ffd83dbSDimitry Andric   TRI = MF.getSubtarget().getRegisterInfo();
99*5ffd83dbSDimitry Andric 
100*5ffd83dbSDimitry Andric   bool Modified = false;
101*5ffd83dbSDimitry Andric   for (auto &MBB : MF) {
102*5ffd83dbSDimitry Andric     Modified |= hardenReturnsAndBRs(MBB);
103*5ffd83dbSDimitry Andric     Modified |= hardenBLRs(MBB);
104*5ffd83dbSDimitry Andric   }
105*5ffd83dbSDimitry Andric 
106*5ffd83dbSDimitry Andric   return Modified;
107*5ffd83dbSDimitry Andric }
108*5ffd83dbSDimitry Andric 
109*5ffd83dbSDimitry Andric static bool isBLR(const MachineInstr &MI) {
110*5ffd83dbSDimitry Andric   switch (MI.getOpcode()) {
111*5ffd83dbSDimitry Andric   case AArch64::BLR:
112*5ffd83dbSDimitry Andric   case AArch64::BLRNoIP:
113*5ffd83dbSDimitry Andric     return true;
114*5ffd83dbSDimitry Andric   case AArch64::BLRAA:
115*5ffd83dbSDimitry Andric   case AArch64::BLRAB:
116*5ffd83dbSDimitry Andric   case AArch64::BLRAAZ:
117*5ffd83dbSDimitry Andric   case AArch64::BLRABZ:
118*5ffd83dbSDimitry Andric     llvm_unreachable("Currently, LLVM's code generator does not support "
119*5ffd83dbSDimitry Andric                      "producing BLRA* instructions. Therefore, there's no "
120*5ffd83dbSDimitry Andric                      "support in this pass for those instructions.");
121*5ffd83dbSDimitry Andric   }
122*5ffd83dbSDimitry Andric   return false;
123*5ffd83dbSDimitry Andric }
124*5ffd83dbSDimitry Andric 
125*5ffd83dbSDimitry Andric bool AArch64SLSHardening::hardenReturnsAndBRs(MachineBasicBlock &MBB) const {
126*5ffd83dbSDimitry Andric   if (!ST->hardenSlsRetBr())
127*5ffd83dbSDimitry Andric     return false;
128*5ffd83dbSDimitry Andric   bool Modified = false;
129*5ffd83dbSDimitry Andric   MachineBasicBlock::iterator MBBI = MBB.getFirstTerminator(), E = MBB.end();
130*5ffd83dbSDimitry Andric   MachineBasicBlock::iterator NextMBBI;
131*5ffd83dbSDimitry Andric   for (; MBBI != E; MBBI = NextMBBI) {
132*5ffd83dbSDimitry Andric     MachineInstr &MI = *MBBI;
133*5ffd83dbSDimitry Andric     NextMBBI = std::next(MBBI);
134*5ffd83dbSDimitry Andric     if (MI.isReturn() || isIndirectBranchOpcode(MI.getOpcode())) {
135*5ffd83dbSDimitry Andric       assert(MI.isTerminator());
136*5ffd83dbSDimitry Andric       insertSpeculationBarrier(ST, MBB, std::next(MBBI), MI.getDebugLoc());
137*5ffd83dbSDimitry Andric       Modified = true;
138*5ffd83dbSDimitry Andric     }
139*5ffd83dbSDimitry Andric   }
140*5ffd83dbSDimitry Andric   return Modified;
141*5ffd83dbSDimitry Andric }
142*5ffd83dbSDimitry Andric 
143*5ffd83dbSDimitry Andric static const char SLSBLRNamePrefix[] = "__llvm_slsblr_thunk_";
144*5ffd83dbSDimitry Andric 
145*5ffd83dbSDimitry Andric static const struct ThunkNameAndReg {
146*5ffd83dbSDimitry Andric   const char* Name;
147*5ffd83dbSDimitry Andric   Register Reg;
148*5ffd83dbSDimitry Andric } SLSBLRThunks[] = {
149*5ffd83dbSDimitry Andric   { "__llvm_slsblr_thunk_x0",  AArch64::X0},
150*5ffd83dbSDimitry Andric   { "__llvm_slsblr_thunk_x1",  AArch64::X1},
151*5ffd83dbSDimitry Andric   { "__llvm_slsblr_thunk_x2",  AArch64::X2},
152*5ffd83dbSDimitry Andric   { "__llvm_slsblr_thunk_x3",  AArch64::X3},
153*5ffd83dbSDimitry Andric   { "__llvm_slsblr_thunk_x4",  AArch64::X4},
154*5ffd83dbSDimitry Andric   { "__llvm_slsblr_thunk_x5",  AArch64::X5},
155*5ffd83dbSDimitry Andric   { "__llvm_slsblr_thunk_x6",  AArch64::X6},
156*5ffd83dbSDimitry Andric   { "__llvm_slsblr_thunk_x7",  AArch64::X7},
157*5ffd83dbSDimitry Andric   { "__llvm_slsblr_thunk_x8",  AArch64::X8},
158*5ffd83dbSDimitry Andric   { "__llvm_slsblr_thunk_x9",  AArch64::X9},
159*5ffd83dbSDimitry Andric   { "__llvm_slsblr_thunk_x10",  AArch64::X10},
160*5ffd83dbSDimitry Andric   { "__llvm_slsblr_thunk_x11",  AArch64::X11},
161*5ffd83dbSDimitry Andric   { "__llvm_slsblr_thunk_x12",  AArch64::X12},
162*5ffd83dbSDimitry Andric   { "__llvm_slsblr_thunk_x13",  AArch64::X13},
163*5ffd83dbSDimitry Andric   { "__llvm_slsblr_thunk_x14",  AArch64::X14},
164*5ffd83dbSDimitry Andric   { "__llvm_slsblr_thunk_x15",  AArch64::X15},
165*5ffd83dbSDimitry Andric   // X16 and X17 are deliberately missing, as the mitigation requires those
166*5ffd83dbSDimitry Andric   // register to not be used in BLR. See comment in ConvertBLRToBL for more
167*5ffd83dbSDimitry Andric   // details.
168*5ffd83dbSDimitry Andric   { "__llvm_slsblr_thunk_x18",  AArch64::X18},
169*5ffd83dbSDimitry Andric   { "__llvm_slsblr_thunk_x19",  AArch64::X19},
170*5ffd83dbSDimitry Andric   { "__llvm_slsblr_thunk_x20",  AArch64::X20},
171*5ffd83dbSDimitry Andric   { "__llvm_slsblr_thunk_x21",  AArch64::X21},
172*5ffd83dbSDimitry Andric   { "__llvm_slsblr_thunk_x22",  AArch64::X22},
173*5ffd83dbSDimitry Andric   { "__llvm_slsblr_thunk_x23",  AArch64::X23},
174*5ffd83dbSDimitry Andric   { "__llvm_slsblr_thunk_x24",  AArch64::X24},
175*5ffd83dbSDimitry Andric   { "__llvm_slsblr_thunk_x25",  AArch64::X25},
176*5ffd83dbSDimitry Andric   { "__llvm_slsblr_thunk_x26",  AArch64::X26},
177*5ffd83dbSDimitry Andric   { "__llvm_slsblr_thunk_x27",  AArch64::X27},
178*5ffd83dbSDimitry Andric   { "__llvm_slsblr_thunk_x28",  AArch64::X28},
179*5ffd83dbSDimitry Andric   { "__llvm_slsblr_thunk_x29",  AArch64::FP},
180*5ffd83dbSDimitry Andric   // X30 is deliberately missing, for similar reasons as X16 and X17 are
181*5ffd83dbSDimitry Andric   // missing.
182*5ffd83dbSDimitry Andric   { "__llvm_slsblr_thunk_x31",  AArch64::XZR},
183*5ffd83dbSDimitry Andric };
184*5ffd83dbSDimitry Andric 
185*5ffd83dbSDimitry Andric namespace {
186*5ffd83dbSDimitry Andric struct SLSBLRThunkInserter : ThunkInserter<SLSBLRThunkInserter> {
187*5ffd83dbSDimitry Andric   const char *getThunkPrefix() { return SLSBLRNamePrefix; }
188*5ffd83dbSDimitry Andric   bool mayUseThunk(const MachineFunction &MF) {
189*5ffd83dbSDimitry Andric     // FIXME: This could also check if there are any BLRs in the function
190*5ffd83dbSDimitry Andric     // to more accurately reflect if a thunk will be needed.
191*5ffd83dbSDimitry Andric     return MF.getSubtarget<AArch64Subtarget>().hardenSlsBlr();
192*5ffd83dbSDimitry Andric   }
193*5ffd83dbSDimitry Andric   void insertThunks(MachineModuleInfo &MMI);
194*5ffd83dbSDimitry Andric   void populateThunk(MachineFunction &MF);
195*5ffd83dbSDimitry Andric };
196*5ffd83dbSDimitry Andric } // namespace
197*5ffd83dbSDimitry Andric 
198*5ffd83dbSDimitry Andric void SLSBLRThunkInserter::insertThunks(MachineModuleInfo &MMI) {
199*5ffd83dbSDimitry Andric   // FIXME: It probably would be possible to filter which thunks to produce
200*5ffd83dbSDimitry Andric   // based on which registers are actually used in BLR instructions in this
201*5ffd83dbSDimitry Andric   // function. But would that be a worthwhile optimization?
202*5ffd83dbSDimitry Andric   for (auto T : SLSBLRThunks)
203*5ffd83dbSDimitry Andric     createThunkFunction(MMI, T.Name);
204*5ffd83dbSDimitry Andric }
205*5ffd83dbSDimitry Andric 
206*5ffd83dbSDimitry Andric void SLSBLRThunkInserter::populateThunk(MachineFunction &MF) {
207*5ffd83dbSDimitry Andric   // FIXME: How to better communicate Register number, rather than through
208*5ffd83dbSDimitry Andric   // name and lookup table?
209*5ffd83dbSDimitry Andric   assert(MF.getName().startswith(getThunkPrefix()));
210*5ffd83dbSDimitry Andric   auto ThunkIt = llvm::find_if(
211*5ffd83dbSDimitry Andric       SLSBLRThunks, [&MF](auto T) { return T.Name == MF.getName(); });
212*5ffd83dbSDimitry Andric   assert(ThunkIt != std::end(SLSBLRThunks));
213*5ffd83dbSDimitry Andric   Register ThunkReg = ThunkIt->Reg;
214*5ffd83dbSDimitry Andric 
215*5ffd83dbSDimitry Andric   const TargetInstrInfo *TII =
216*5ffd83dbSDimitry Andric       MF.getSubtarget<AArch64Subtarget>().getInstrInfo();
217*5ffd83dbSDimitry Andric   assert (MF.size() == 1);
218*5ffd83dbSDimitry Andric   MachineBasicBlock *Entry = &MF.front();
219*5ffd83dbSDimitry Andric   Entry->clear();
220*5ffd83dbSDimitry Andric 
221*5ffd83dbSDimitry Andric   //  These thunks need to consist of the following instructions:
222*5ffd83dbSDimitry Andric   //  __llvm_slsblr_thunk_xN:
223*5ffd83dbSDimitry Andric   //      BR xN
224*5ffd83dbSDimitry Andric   //      barrierInsts
225*5ffd83dbSDimitry Andric   Entry->addLiveIn(ThunkReg);
226*5ffd83dbSDimitry Andric   // MOV X16, ThunkReg == ORR X16, XZR, ThunkReg, LSL #0
227*5ffd83dbSDimitry Andric   BuildMI(Entry, DebugLoc(), TII->get(AArch64::ORRXrs), AArch64::X16)
228*5ffd83dbSDimitry Andric       .addReg(AArch64::XZR)
229*5ffd83dbSDimitry Andric       .addReg(ThunkReg)
230*5ffd83dbSDimitry Andric       .addImm(0);
231*5ffd83dbSDimitry Andric   BuildMI(Entry, DebugLoc(), TII->get(AArch64::BR)).addReg(AArch64::X16);
232*5ffd83dbSDimitry Andric   // Make sure the thunks do not make use of the SB extension in case there is
233*5ffd83dbSDimitry Andric   // a function somewhere that will call to it that for some reason disabled
234*5ffd83dbSDimitry Andric   // the SB extension locally on that function, even though it's enabled for
235*5ffd83dbSDimitry Andric   // the module otherwise. Therefore set AlwaysUseISBSDB to true.
236*5ffd83dbSDimitry Andric   insertSpeculationBarrier(&MF.getSubtarget<AArch64Subtarget>(), *Entry,
237*5ffd83dbSDimitry Andric                            Entry->end(), DebugLoc(), true /*AlwaysUseISBDSB*/);
238*5ffd83dbSDimitry Andric }
239*5ffd83dbSDimitry Andric 
240*5ffd83dbSDimitry Andric MachineBasicBlock &
241*5ffd83dbSDimitry Andric AArch64SLSHardening::ConvertBLRToBL(MachineBasicBlock &MBB,
242*5ffd83dbSDimitry Andric                                     MachineBasicBlock::iterator MBBI) const {
243*5ffd83dbSDimitry Andric   // Transform a BLR to a BL as follows:
244*5ffd83dbSDimitry Andric   // Before:
245*5ffd83dbSDimitry Andric   //   |-----------------------------|
246*5ffd83dbSDimitry Andric   //   |      ...                    |
247*5ffd83dbSDimitry Andric   //   |  instI                      |
248*5ffd83dbSDimitry Andric   //   |  BLR xN                     |
249*5ffd83dbSDimitry Andric   //   |  instJ                      |
250*5ffd83dbSDimitry Andric   //   |      ...                    |
251*5ffd83dbSDimitry Andric   //   |-----------------------------|
252*5ffd83dbSDimitry Andric   //
253*5ffd83dbSDimitry Andric   // After:
254*5ffd83dbSDimitry Andric   //   |-----------------------------|
255*5ffd83dbSDimitry Andric   //   |      ...                    |
256*5ffd83dbSDimitry Andric   //   |  instI                      |
257*5ffd83dbSDimitry Andric   //   |  BL __llvm_slsblr_thunk_xN  |
258*5ffd83dbSDimitry Andric   //   |  instJ                      |
259*5ffd83dbSDimitry Andric   //   |      ...                    |
260*5ffd83dbSDimitry Andric   //   |-----------------------------|
261*5ffd83dbSDimitry Andric   //
262*5ffd83dbSDimitry Andric   //   __llvm_slsblr_thunk_xN:
263*5ffd83dbSDimitry Andric   //   |-----------------------------|
264*5ffd83dbSDimitry Andric   //   |  BR xN                      |
265*5ffd83dbSDimitry Andric   //   |  barrierInsts               |
266*5ffd83dbSDimitry Andric   //   |-----------------------------|
267*5ffd83dbSDimitry Andric   //
268*5ffd83dbSDimitry Andric   // The __llvm_slsblr_thunk_xN thunks are created by the SLSBLRThunkInserter.
269*5ffd83dbSDimitry Andric   // This function merely needs to transform BLR xN into BL
270*5ffd83dbSDimitry Andric   // __llvm_slsblr_thunk_xN.
271*5ffd83dbSDimitry Andric   //
272*5ffd83dbSDimitry Andric   // Since linkers are allowed to clobber X16 and X17 on function calls, the
273*5ffd83dbSDimitry Andric   // above mitigation only works if the original BLR instruction was not
274*5ffd83dbSDimitry Andric   // BLR X16 nor BLR X17. Code generation before must make sure that no BLR
275*5ffd83dbSDimitry Andric   // X16|X17 was produced if the mitigation is enabled.
276*5ffd83dbSDimitry Andric 
277*5ffd83dbSDimitry Andric   MachineInstr &BLR = *MBBI;
278*5ffd83dbSDimitry Andric   assert(isBLR(BLR));
279*5ffd83dbSDimitry Andric   unsigned BLOpcode;
280*5ffd83dbSDimitry Andric   Register Reg;
281*5ffd83dbSDimitry Andric   bool RegIsKilled;
282*5ffd83dbSDimitry Andric   switch (BLR.getOpcode()) {
283*5ffd83dbSDimitry Andric   case AArch64::BLR:
284*5ffd83dbSDimitry Andric   case AArch64::BLRNoIP:
285*5ffd83dbSDimitry Andric     BLOpcode = AArch64::BL;
286*5ffd83dbSDimitry Andric     Reg = BLR.getOperand(0).getReg();
287*5ffd83dbSDimitry Andric     assert(Reg != AArch64::X16 && Reg != AArch64::X17 && Reg != AArch64::LR);
288*5ffd83dbSDimitry Andric     RegIsKilled = BLR.getOperand(0).isKill();
289*5ffd83dbSDimitry Andric     break;
290*5ffd83dbSDimitry Andric   case AArch64::BLRAA:
291*5ffd83dbSDimitry Andric   case AArch64::BLRAB:
292*5ffd83dbSDimitry Andric   case AArch64::BLRAAZ:
293*5ffd83dbSDimitry Andric   case AArch64::BLRABZ:
294*5ffd83dbSDimitry Andric     llvm_unreachable("BLRA instructions cannot yet be produced by LLVM, "
295*5ffd83dbSDimitry Andric                      "therefore there is no need to support them for now.");
296*5ffd83dbSDimitry Andric   default:
297*5ffd83dbSDimitry Andric     llvm_unreachable("unhandled BLR");
298*5ffd83dbSDimitry Andric   }
299*5ffd83dbSDimitry Andric   DebugLoc DL = BLR.getDebugLoc();
300*5ffd83dbSDimitry Andric 
301*5ffd83dbSDimitry Andric   // If we'd like to support also BLRAA and BLRAB instructions, we'd need
302*5ffd83dbSDimitry Andric   // a lot more different kind of thunks.
303*5ffd83dbSDimitry Andric   // For example, a
304*5ffd83dbSDimitry Andric   //
305*5ffd83dbSDimitry Andric   // BLRAA xN, xM
306*5ffd83dbSDimitry Andric   //
307*5ffd83dbSDimitry Andric   // instruction probably would need to be transformed to something like:
308*5ffd83dbSDimitry Andric   //
309*5ffd83dbSDimitry Andric   // BL __llvm_slsblraa_thunk_x<N>_x<M>
310*5ffd83dbSDimitry Andric   //
311*5ffd83dbSDimitry Andric   // __llvm_slsblraa_thunk_x<N>_x<M>:
312*5ffd83dbSDimitry Andric   //   BRAA x<N>, x<M>
313*5ffd83dbSDimitry Andric   //   barrierInsts
314*5ffd83dbSDimitry Andric   //
315*5ffd83dbSDimitry Andric   // Given that about 30 different values of N are possible and about 30
316*5ffd83dbSDimitry Andric   // different values of M are possible in the above, with the current way
317*5ffd83dbSDimitry Andric   // of producing indirect thunks, we'd be producing about 30 times 30, i.e.
318*5ffd83dbSDimitry Andric   // about 900 thunks (where most might not be actually called). This would
319*5ffd83dbSDimitry Andric   // multiply further by two to support both BLRAA and BLRAB variants of those
320*5ffd83dbSDimitry Andric   // instructions.
321*5ffd83dbSDimitry Andric   // If we'd want to support this, we'd probably need to look into a different
322*5ffd83dbSDimitry Andric   // way to produce thunk functions, based on which variants are actually
323*5ffd83dbSDimitry Andric   // needed, rather than producing all possible variants.
324*5ffd83dbSDimitry Andric   // So far, LLVM does never produce BLRA* instructions, so let's leave this
325*5ffd83dbSDimitry Andric   // for the future when LLVM can start producing BLRA* instructions.
326*5ffd83dbSDimitry Andric   MachineFunction &MF = *MBBI->getMF();
327*5ffd83dbSDimitry Andric   MCContext &Context = MBB.getParent()->getContext();
328*5ffd83dbSDimitry Andric   auto ThunkIt =
329*5ffd83dbSDimitry Andric       llvm::find_if(SLSBLRThunks, [Reg](auto T) { return T.Reg == Reg; });
330*5ffd83dbSDimitry Andric   assert (ThunkIt != std::end(SLSBLRThunks));
331*5ffd83dbSDimitry Andric   MCSymbol *Sym = Context.getOrCreateSymbol(ThunkIt->Name);
332*5ffd83dbSDimitry Andric 
333*5ffd83dbSDimitry Andric   MachineInstr *BL = BuildMI(MBB, MBBI, DL, TII->get(BLOpcode)).addSym(Sym);
334*5ffd83dbSDimitry Andric 
335*5ffd83dbSDimitry Andric   // Now copy the implicit operands from BLR to BL and copy other necessary
336*5ffd83dbSDimitry Andric   // info.
337*5ffd83dbSDimitry Andric   // However, both BLR and BL instructions implictly use SP and implicitly
338*5ffd83dbSDimitry Andric   // define LR. Blindly copying implicit operands would result in SP and LR
339*5ffd83dbSDimitry Andric   // operands to be present multiple times. While this may not be too much of
340*5ffd83dbSDimitry Andric   // an issue, let's avoid that for cleanliness, by removing those implicit
341*5ffd83dbSDimitry Andric   // operands from the BL created above before we copy over all implicit
342*5ffd83dbSDimitry Andric   // operands from the BLR.
343*5ffd83dbSDimitry Andric   int ImpLROpIdx = -1;
344*5ffd83dbSDimitry Andric   int ImpSPOpIdx = -1;
345*5ffd83dbSDimitry Andric   for (unsigned OpIdx = BL->getNumExplicitOperands();
346*5ffd83dbSDimitry Andric        OpIdx < BL->getNumOperands(); OpIdx++) {
347*5ffd83dbSDimitry Andric     MachineOperand Op = BL->getOperand(OpIdx);
348*5ffd83dbSDimitry Andric     if (!Op.isReg())
349*5ffd83dbSDimitry Andric       continue;
350*5ffd83dbSDimitry Andric     if (Op.getReg() == AArch64::LR && Op.isDef())
351*5ffd83dbSDimitry Andric       ImpLROpIdx = OpIdx;
352*5ffd83dbSDimitry Andric     if (Op.getReg() == AArch64::SP && !Op.isDef())
353*5ffd83dbSDimitry Andric       ImpSPOpIdx = OpIdx;
354*5ffd83dbSDimitry Andric   }
355*5ffd83dbSDimitry Andric   assert(ImpLROpIdx != -1);
356*5ffd83dbSDimitry Andric   assert(ImpSPOpIdx != -1);
357*5ffd83dbSDimitry Andric   int FirstOpIdxToRemove = std::max(ImpLROpIdx, ImpSPOpIdx);
358*5ffd83dbSDimitry Andric   int SecondOpIdxToRemove = std::min(ImpLROpIdx, ImpSPOpIdx);
359*5ffd83dbSDimitry Andric   BL->RemoveOperand(FirstOpIdxToRemove);
360*5ffd83dbSDimitry Andric   BL->RemoveOperand(SecondOpIdxToRemove);
361*5ffd83dbSDimitry Andric   // Now copy over the implicit operands from the original BLR
362*5ffd83dbSDimitry Andric   BL->copyImplicitOps(MF, BLR);
363*5ffd83dbSDimitry Andric   MF.moveCallSiteInfo(&BLR, BL);
364*5ffd83dbSDimitry Andric   // Also add the register called in the BLR as being used in the called thunk.
365*5ffd83dbSDimitry Andric   BL->addOperand(MachineOperand::CreateReg(Reg, false /*isDef*/, true /*isImp*/,
366*5ffd83dbSDimitry Andric                                            RegIsKilled /*isKill*/));
367*5ffd83dbSDimitry Andric   // Remove BLR instruction
368*5ffd83dbSDimitry Andric   MBB.erase(MBBI);
369*5ffd83dbSDimitry Andric 
370*5ffd83dbSDimitry Andric   return MBB;
371*5ffd83dbSDimitry Andric }
372*5ffd83dbSDimitry Andric 
373*5ffd83dbSDimitry Andric bool AArch64SLSHardening::hardenBLRs(MachineBasicBlock &MBB) const {
374*5ffd83dbSDimitry Andric   if (!ST->hardenSlsBlr())
375*5ffd83dbSDimitry Andric     return false;
376*5ffd83dbSDimitry Andric   bool Modified = false;
377*5ffd83dbSDimitry Andric   MachineBasicBlock::iterator MBBI = MBB.begin(), E = MBB.end();
378*5ffd83dbSDimitry Andric   MachineBasicBlock::iterator NextMBBI;
379*5ffd83dbSDimitry Andric   for (; MBBI != E; MBBI = NextMBBI) {
380*5ffd83dbSDimitry Andric     MachineInstr &MI = *MBBI;
381*5ffd83dbSDimitry Andric     NextMBBI = std::next(MBBI);
382*5ffd83dbSDimitry Andric     if (isBLR(MI)) {
383*5ffd83dbSDimitry Andric       ConvertBLRToBL(MBB, MBBI);
384*5ffd83dbSDimitry Andric       Modified = true;
385*5ffd83dbSDimitry Andric     }
386*5ffd83dbSDimitry Andric   }
387*5ffd83dbSDimitry Andric   return Modified;
388*5ffd83dbSDimitry Andric }
389*5ffd83dbSDimitry Andric 
390*5ffd83dbSDimitry Andric FunctionPass *llvm::createAArch64SLSHardeningPass() {
391*5ffd83dbSDimitry Andric   return new AArch64SLSHardening();
392*5ffd83dbSDimitry Andric }
393*5ffd83dbSDimitry Andric 
394*5ffd83dbSDimitry Andric namespace {
395*5ffd83dbSDimitry Andric class AArch64IndirectThunks : public MachineFunctionPass {
396*5ffd83dbSDimitry Andric public:
397*5ffd83dbSDimitry Andric   static char ID;
398*5ffd83dbSDimitry Andric 
399*5ffd83dbSDimitry Andric   AArch64IndirectThunks() : MachineFunctionPass(ID) {}
400*5ffd83dbSDimitry Andric 
401*5ffd83dbSDimitry Andric   StringRef getPassName() const override { return "AArch64 Indirect Thunks"; }
402*5ffd83dbSDimitry Andric 
403*5ffd83dbSDimitry Andric   bool doInitialization(Module &M) override;
404*5ffd83dbSDimitry Andric   bool runOnMachineFunction(MachineFunction &MF) override;
405*5ffd83dbSDimitry Andric 
406*5ffd83dbSDimitry Andric private:
407*5ffd83dbSDimitry Andric   std::tuple<SLSBLRThunkInserter> TIs;
408*5ffd83dbSDimitry Andric 
409*5ffd83dbSDimitry Andric   // FIXME: When LLVM moves to C++17, these can become folds
410*5ffd83dbSDimitry Andric   template <typename... ThunkInserterT>
411*5ffd83dbSDimitry Andric   static void initTIs(Module &M,
412*5ffd83dbSDimitry Andric                       std::tuple<ThunkInserterT...> &ThunkInserters) {
413*5ffd83dbSDimitry Andric     (void)std::initializer_list<int>{
414*5ffd83dbSDimitry Andric         (std::get<ThunkInserterT>(ThunkInserters).init(M), 0)...};
415*5ffd83dbSDimitry Andric   }
416*5ffd83dbSDimitry Andric   template <typename... ThunkInserterT>
417*5ffd83dbSDimitry Andric   static bool runTIs(MachineModuleInfo &MMI, MachineFunction &MF,
418*5ffd83dbSDimitry Andric                      std::tuple<ThunkInserterT...> &ThunkInserters) {
419*5ffd83dbSDimitry Andric     bool Modified = false;
420*5ffd83dbSDimitry Andric     (void)std::initializer_list<int>{
421*5ffd83dbSDimitry Andric         Modified |= std::get<ThunkInserterT>(ThunkInserters).run(MMI, MF)...};
422*5ffd83dbSDimitry Andric     return Modified;
423*5ffd83dbSDimitry Andric   }
424*5ffd83dbSDimitry Andric };
425*5ffd83dbSDimitry Andric 
426*5ffd83dbSDimitry Andric } // end anonymous namespace
427*5ffd83dbSDimitry Andric 
428*5ffd83dbSDimitry Andric char AArch64IndirectThunks::ID = 0;
429*5ffd83dbSDimitry Andric 
430*5ffd83dbSDimitry Andric FunctionPass *llvm::createAArch64IndirectThunks() {
431*5ffd83dbSDimitry Andric   return new AArch64IndirectThunks();
432*5ffd83dbSDimitry Andric }
433*5ffd83dbSDimitry Andric 
434*5ffd83dbSDimitry Andric bool AArch64IndirectThunks::doInitialization(Module &M) {
435*5ffd83dbSDimitry Andric   initTIs(M, TIs);
436*5ffd83dbSDimitry Andric   return false;
437*5ffd83dbSDimitry Andric }
438*5ffd83dbSDimitry Andric 
439*5ffd83dbSDimitry Andric bool AArch64IndirectThunks::runOnMachineFunction(MachineFunction &MF) {
440*5ffd83dbSDimitry Andric   LLVM_DEBUG(dbgs() << getPassName() << '\n');
441*5ffd83dbSDimitry Andric   auto &MMI = getAnalysis<MachineModuleInfoWrapperPass>().getMMI();
442*5ffd83dbSDimitry Andric   return runTIs(MMI, MF, TIs);
443*5ffd83dbSDimitry Andric }
444