| Bill Schmidt | 82f1c77 | 2015-02-10 19:09:05 +0000 | [diff] [blame] | 1 | //===---------- PPCTLSDynamicCall.cpp - TLS Dynamic Call Fixup ------------===// | 
|  | 2 | // | 
|  | 3 | //                     The LLVM Compiler Infrastructure | 
|  | 4 | // | 
|  | 5 | // This file is distributed under the University of Illinois Open Source | 
|  | 6 | // License. See LICENSE.TXT for details. | 
|  | 7 | // | 
|  | 8 | //===----------------------------------------------------------------------===// | 
|  | 9 | // | 
|  | 10 | // This pass expands ADDItls{ld,gd}LADDR[32] machine instructions into | 
|  | 11 | // separate ADDItls[gd]L[32] and GETtlsADDR[32] instructions, both of | 
|  | 12 | // which define GPR3.  A copy is added from GPR3 to the target virtual | 
|  | 13 | // register of the original instruction.  The GETtlsADDR[32] is really | 
|  | 14 | // a call instruction, so its target register is constrained to be GPR3. | 
|  | 15 | // This is not true of ADDItls[gd]L[32], but there is a legacy linker | 
|  | 16 | // optimization bug that requires the target register of the addi of | 
|  | 17 | // a local- or general-dynamic TLS access sequence to be GPR3. | 
|  | 18 | // | 
|  | 19 | // This is done in a late pass so that TLS variable accesses can be | 
|  | 20 | // fully commoned by MachineCSE. | 
|  | 21 | // | 
|  | 22 | //===----------------------------------------------------------------------===// | 
|  | 23 |  | 
| Bill Schmidt | 82f1c77 | 2015-02-10 19:09:05 +0000 | [diff] [blame] | 24 | #include "PPC.h" | 
|  | 25 | #include "PPCInstrBuilder.h" | 
| Chandler Carruth | 6bda14b | 2017-06-06 11:49:48 +0000 | [diff] [blame] | 26 | #include "PPCInstrInfo.h" | 
| Bill Schmidt | 82f1c77 | 2015-02-10 19:09:05 +0000 | [diff] [blame] | 27 | #include "PPCTargetMachine.h" | 
| Matthias Braun | f842297 | 2017-12-13 02:51:04 +0000 | [diff] [blame] | 28 | #include "llvm/CodeGen/LiveIntervals.h" | 
| Bill Schmidt | 82f1c77 | 2015-02-10 19:09:05 +0000 | [diff] [blame] | 29 | #include "llvm/CodeGen/MachineFunctionPass.h" | 
|  | 30 | #include "llvm/CodeGen/MachineInstrBuilder.h" | 
|  | 31 | #include "llvm/Support/Debug.h" | 
|  | 32 | #include "llvm/Support/raw_ostream.h" | 
|  | 33 |  | 
|  | 34 | using namespace llvm; | 
|  | 35 |  | 
|  | 36 | #define DEBUG_TYPE "ppc-tls-dynamic-call" | 
|  | 37 |  | 
|  | 38 | namespace llvm { | 
|  | 39 | void initializePPCTLSDynamicCallPass(PassRegistry&); | 
|  | 40 | } | 
|  | 41 |  | 
|  | 42 | namespace { | 
|  | 43 | struct PPCTLSDynamicCall : public MachineFunctionPass { | 
|  | 44 | static char ID; | 
|  | 45 | PPCTLSDynamicCall() : MachineFunctionPass(ID) { | 
|  | 46 | initializePPCTLSDynamicCallPass(*PassRegistry::getPassRegistry()); | 
|  | 47 | } | 
|  | 48 |  | 
| Bill Schmidt | 82f1c77 | 2015-02-10 19:09:05 +0000 | [diff] [blame] | 49 | const PPCInstrInfo *TII; | 
|  | 50 | LiveIntervals *LIS; | 
|  | 51 |  | 
|  | 52 | protected: | 
|  | 53 | bool processBlock(MachineBasicBlock &MBB) { | 
|  | 54 | bool Changed = false; | 
| Hiroshi Inoue | 6989caa | 2017-06-29 14:13:38 +0000 | [diff] [blame] | 55 | bool NeedFence = true; | 
| Eric Christopher | 1df0c51 | 2015-02-20 18:44:15 +0000 | [diff] [blame] | 56 | bool Is64Bit = MBB.getParent()->getSubtarget<PPCSubtarget>().isPPC64(); | 
| Bill Schmidt | 82f1c77 | 2015-02-10 19:09:05 +0000 | [diff] [blame] | 57 |  | 
|  | 58 | for (MachineBasicBlock::iterator I = MBB.begin(), IE = MBB.end(); | 
| Hal Finkel | 82e1fc5 | 2015-05-21 23:45:49 +0000 | [diff] [blame] | 59 | I != IE;) { | 
| Duncan P. N. Exon Smith | e5a22f4 | 2016-07-27 13:24:16 +0000 | [diff] [blame] | 60 | MachineInstr &MI = *I; | 
| Bill Schmidt | 82f1c77 | 2015-02-10 19:09:05 +0000 | [diff] [blame] | 61 |  | 
| Duncan P. N. Exon Smith | e5a22f4 | 2016-07-27 13:24:16 +0000 | [diff] [blame] | 62 | if (MI.getOpcode() != PPC::ADDItlsgdLADDR && | 
|  | 63 | MI.getOpcode() != PPC::ADDItlsldLADDR && | 
|  | 64 | MI.getOpcode() != PPC::ADDItlsgdLADDR32 && | 
|  | 65 | MI.getOpcode() != PPC::ADDItlsldLADDR32) { | 
| Hiroshi Inoue | 6989caa | 2017-06-29 14:13:38 +0000 | [diff] [blame] | 66 |  | 
|  | 67 | // Although we create ADJCALLSTACKDOWN and ADJCALLSTACKUP | 
|  | 68 | // as scheduling fences, we skip creating fences if we already | 
|  | 69 | // have existing ADJCALLSTACKDOWN/UP to avoid nesting, | 
|  | 70 | // which causes verification error with -verify-machineinstrs. | 
|  | 71 | if (MI.getOpcode() == PPC::ADJCALLSTACKDOWN) | 
|  | 72 | NeedFence = false; | 
|  | 73 | else if (MI.getOpcode() == PPC::ADJCALLSTACKUP) | 
|  | 74 | NeedFence = true; | 
|  | 75 |  | 
| Hal Finkel | 82e1fc5 | 2015-05-21 23:45:49 +0000 | [diff] [blame] | 76 | ++I; | 
| Bill Schmidt | 82f1c77 | 2015-02-10 19:09:05 +0000 | [diff] [blame] | 77 | continue; | 
| Hal Finkel | 82e1fc5 | 2015-05-21 23:45:49 +0000 | [diff] [blame] | 78 | } | 
| Bill Schmidt | 82f1c77 | 2015-02-10 19:09:05 +0000 | [diff] [blame] | 79 |  | 
| Duncan P. N. Exon Smith | e5a22f4 | 2016-07-27 13:24:16 +0000 | [diff] [blame] | 80 | DEBUG(dbgs() << "TLS Dynamic Call Fixup:\n    " << MI); | 
| Bill Schmidt | 82f1c77 | 2015-02-10 19:09:05 +0000 | [diff] [blame] | 81 |  | 
| Duncan P. N. Exon Smith | e5a22f4 | 2016-07-27 13:24:16 +0000 | [diff] [blame] | 82 | unsigned OutReg = MI.getOperand(0).getReg(); | 
|  | 83 | unsigned InReg = MI.getOperand(1).getReg(); | 
|  | 84 | DebugLoc DL = MI.getDebugLoc(); | 
| Bill Schmidt | 82f1c77 | 2015-02-10 19:09:05 +0000 | [diff] [blame] | 85 | unsigned GPR3 = Is64Bit ? PPC::X3 : PPC::R3; | 
|  | 86 | unsigned Opc1, Opc2; | 
| Benjamin Kramer | 3bc1edf | 2016-07-02 11:41:39 +0000 | [diff] [blame] | 87 | const unsigned OrigRegs[] = {OutReg, InReg, GPR3}; | 
| Bill Schmidt | 82f1c77 | 2015-02-10 19:09:05 +0000 | [diff] [blame] | 88 |  | 
| Duncan P. N. Exon Smith | e5a22f4 | 2016-07-27 13:24:16 +0000 | [diff] [blame] | 89 | switch (MI.getOpcode()) { | 
| Bill Schmidt | 82f1c77 | 2015-02-10 19:09:05 +0000 | [diff] [blame] | 90 | default: | 
|  | 91 | llvm_unreachable("Opcode inconsistency error"); | 
|  | 92 | case PPC::ADDItlsgdLADDR: | 
|  | 93 | Opc1 = PPC::ADDItlsgdL; | 
|  | 94 | Opc2 = PPC::GETtlsADDR; | 
|  | 95 | break; | 
|  | 96 | case PPC::ADDItlsldLADDR: | 
|  | 97 | Opc1 = PPC::ADDItlsldL; | 
|  | 98 | Opc2 = PPC::GETtlsldADDR; | 
|  | 99 | break; | 
|  | 100 | case PPC::ADDItlsgdLADDR32: | 
|  | 101 | Opc1 = PPC::ADDItlsgdL32; | 
|  | 102 | Opc2 = PPC::GETtlsADDR32; | 
|  | 103 | break; | 
|  | 104 | case PPC::ADDItlsldLADDR32: | 
|  | 105 | Opc1 = PPC::ADDItlsldL32; | 
|  | 106 | Opc2 = PPC::GETtlsldADDR32; | 
|  | 107 | break; | 
|  | 108 | } | 
|  | 109 |  | 
| Hiroshi Inoue | 6989caa | 2017-06-29 14:13:38 +0000 | [diff] [blame] | 110 | // We create ADJCALLSTACKUP and ADJCALLSTACKDOWN around _tls_get_addr | 
|  | 111 | // as schduling fence to avoid it is scheduled before | 
|  | 112 | // mflr in the prologue and the address in LR is clobbered (PR25839). | 
|  | 113 | // We don't really need to save data to the stack - the clobbered | 
| Kyle Butt | bfcff38 | 2016-01-08 02:06:19 +0000 | [diff] [blame] | 114 | // registers are already saved when the SDNode (e.g. PPCaddiTlsgdLAddr) | 
|  | 115 | // gets translated to the pseudo instruction (e.g. ADDItlsgdLADDR). | 
| Hiroshi Inoue | 6989caa | 2017-06-29 14:13:38 +0000 | [diff] [blame] | 116 | if (NeedFence) | 
|  | 117 | BuildMI(MBB, I, DL, TII->get(PPC::ADJCALLSTACKDOWN)).addImm(0) | 
|  | 118 | .addImm(0); | 
| Kyle Butt | bfcff38 | 2016-01-08 02:06:19 +0000 | [diff] [blame] | 119 |  | 
| Bill Schmidt | 82f1c77 | 2015-02-10 19:09:05 +0000 | [diff] [blame] | 120 | // Expand into two ops built prior to the existing instruction. | 
|  | 121 | MachineInstr *Addi = BuildMI(MBB, I, DL, TII->get(Opc1), GPR3) | 
|  | 122 | .addReg(InReg); | 
| Duncan P. N. Exon Smith | e5a22f4 | 2016-07-27 13:24:16 +0000 | [diff] [blame] | 123 | Addi->addOperand(MI.getOperand(2)); | 
| Bill Schmidt | 82f1c77 | 2015-02-10 19:09:05 +0000 | [diff] [blame] | 124 |  | 
|  | 125 | // The ADDItls* instruction is the first instruction in the | 
|  | 126 | // repair range. | 
|  | 127 | MachineBasicBlock::iterator First = I; | 
|  | 128 | --First; | 
|  | 129 |  | 
|  | 130 | MachineInstr *Call = (BuildMI(MBB, I, DL, TII->get(Opc2), GPR3) | 
|  | 131 | .addReg(GPR3)); | 
| Duncan P. N. Exon Smith | e5a22f4 | 2016-07-27 13:24:16 +0000 | [diff] [blame] | 132 | Call->addOperand(MI.getOperand(3)); | 
| Bill Schmidt | 82f1c77 | 2015-02-10 19:09:05 +0000 | [diff] [blame] | 133 |  | 
| Hiroshi Inoue | 6989caa | 2017-06-29 14:13:38 +0000 | [diff] [blame] | 134 | if (NeedFence) | 
|  | 135 | BuildMI(MBB, I, DL, TII->get(PPC::ADJCALLSTACKUP)).addImm(0).addImm(0); | 
| Kyle Butt | bfcff38 | 2016-01-08 02:06:19 +0000 | [diff] [blame] | 136 |  | 
| Bill Schmidt | 82f1c77 | 2015-02-10 19:09:05 +0000 | [diff] [blame] | 137 | BuildMI(MBB, I, DL, TII->get(TargetOpcode::COPY), OutReg) | 
|  | 138 | .addReg(GPR3); | 
|  | 139 |  | 
|  | 140 | // The COPY is the last instruction in the repair range. | 
|  | 141 | MachineBasicBlock::iterator Last = I; | 
|  | 142 | --Last; | 
|  | 143 |  | 
|  | 144 | // Move past the original instruction and remove it. | 
|  | 145 | ++I; | 
| Duncan P. N. Exon Smith | e5a22f4 | 2016-07-27 13:24:16 +0000 | [diff] [blame] | 146 | MI.removeFromParent(); | 
| Bill Schmidt | 82f1c77 | 2015-02-10 19:09:05 +0000 | [diff] [blame] | 147 |  | 
|  | 148 | // Repair the live intervals. | 
|  | 149 | LIS->repairIntervalsInRange(&MBB, First, Last, OrigRegs); | 
|  | 150 | Changed = true; | 
|  | 151 | } | 
|  | 152 |  | 
|  | 153 | return Changed; | 
|  | 154 | } | 
|  | 155 |  | 
|  | 156 | public: | 
|  | 157 | bool runOnMachineFunction(MachineFunction &MF) override { | 
| Eric Christopher | 1df0c51 | 2015-02-20 18:44:15 +0000 | [diff] [blame] | 158 | TII = MF.getSubtarget<PPCSubtarget>().getInstrInfo(); | 
| Bill Schmidt | 82f1c77 | 2015-02-10 19:09:05 +0000 | [diff] [blame] | 159 | LIS = &getAnalysis<LiveIntervals>(); | 
|  | 160 |  | 
|  | 161 | bool Changed = false; | 
|  | 162 |  | 
|  | 163 | for (MachineFunction::iterator I = MF.begin(); I != MF.end();) { | 
|  | 164 | MachineBasicBlock &B = *I++; | 
|  | 165 | if (processBlock(B)) | 
|  | 166 | Changed = true; | 
|  | 167 | } | 
|  | 168 |  | 
|  | 169 | return Changed; | 
|  | 170 | } | 
|  | 171 |  | 
|  | 172 | void getAnalysisUsage(AnalysisUsage &AU) const override { | 
|  | 173 | AU.addRequired<LiveIntervals>(); | 
|  | 174 | AU.addPreserved<LiveIntervals>(); | 
|  | 175 | AU.addRequired<SlotIndexes>(); | 
|  | 176 | AU.addPreserved<SlotIndexes>(); | 
|  | 177 | MachineFunctionPass::getAnalysisUsage(AU); | 
|  | 178 | } | 
|  | 179 | }; | 
| Alexander Kornienko | f00654e | 2015-06-23 09:49:53 +0000 | [diff] [blame] | 180 | } | 
| Bill Schmidt | 82f1c77 | 2015-02-10 19:09:05 +0000 | [diff] [blame] | 181 |  | 
|  | 182 | INITIALIZE_PASS_BEGIN(PPCTLSDynamicCall, DEBUG_TYPE, | 
|  | 183 | "PowerPC TLS Dynamic Call Fixup", false, false) | 
|  | 184 | INITIALIZE_PASS_DEPENDENCY(LiveIntervals) | 
|  | 185 | INITIALIZE_PASS_DEPENDENCY(SlotIndexes) | 
|  | 186 | INITIALIZE_PASS_END(PPCTLSDynamicCall, DEBUG_TYPE, | 
|  | 187 | "PowerPC TLS Dynamic Call Fixup", false, false) | 
|  | 188 |  | 
|  | 189 | char PPCTLSDynamicCall::ID = 0; | 
|  | 190 | FunctionPass* | 
|  | 191 | llvm::createPPCTLSDynamicCallPass() { return new PPCTLSDynamicCall(); } |