xref: /freebsd/contrib/llvm-project/llvm/lib/Target/PowerPC/PPCExpandAtomicPseudoInsts.cpp (revision a90b9d0159070121c221b966469c3e36d912bf82)
1 //===-- PPCExpandAtomicPseudoInsts.cpp - Expand atomic pseudo instrs. -----===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 //
9 // This file contains a pass that expands atomic pseudo instructions into
10 // target instructions post RA. With such method, LL/SC loop is considered as
11 // a whole blob and make spilling unlikely happens in the LL/SC loop.
12 //
13 //===----------------------------------------------------------------------===//
14 
15 #include "MCTargetDesc/PPCPredicates.h"
16 #include "PPC.h"
17 #include "PPCInstrInfo.h"
18 #include "PPCTargetMachine.h"
19 
20 #include "llvm/CodeGen/LivePhysRegs.h"
21 #include "llvm/CodeGen/MachineFunctionPass.h"
22 #include "llvm/CodeGen/MachineInstrBuilder.h"
23 
24 using namespace llvm;
25 
26 #define DEBUG_TYPE "ppc-atomic-expand"
27 
28 namespace {
29 
30 class PPCExpandAtomicPseudo : public MachineFunctionPass {
31 public:
32   const PPCInstrInfo *TII;
33   const PPCRegisterInfo *TRI;
34   static char ID;
35 
36   PPCExpandAtomicPseudo() : MachineFunctionPass(ID) {
37     initializePPCExpandAtomicPseudoPass(*PassRegistry::getPassRegistry());
38   }
39 
40   bool runOnMachineFunction(MachineFunction &MF) override;
41 
42 private:
43   bool expandMI(MachineBasicBlock &MBB, MachineInstr &MI,
44                 MachineBasicBlock::iterator &NMBBI);
45   bool expandAtomicRMW128(MachineBasicBlock &MBB, MachineInstr &MI,
46                           MachineBasicBlock::iterator &NMBBI);
47   bool expandAtomicCmpSwap128(MachineBasicBlock &MBB, MachineInstr &MI,
48                               MachineBasicBlock::iterator &NMBBI);
49 };
50 
51 static void PairedCopy(const PPCInstrInfo *TII, MachineBasicBlock &MBB,
52                        MachineBasicBlock::iterator MBBI, const DebugLoc &DL,
53                        Register Dest0, Register Dest1, Register Src0,
54                        Register Src1) {
55   const MCInstrDesc &OR = TII->get(PPC::OR8);
56   const MCInstrDesc &XOR = TII->get(PPC::XOR8);
57   if (Dest0 == Src1 && Dest1 == Src0) {
58     // The most tricky case, swapping values.
59     BuildMI(MBB, MBBI, DL, XOR, Dest0).addReg(Dest0).addReg(Dest1);
60     BuildMI(MBB, MBBI, DL, XOR, Dest1).addReg(Dest0).addReg(Dest1);
61     BuildMI(MBB, MBBI, DL, XOR, Dest0).addReg(Dest0).addReg(Dest1);
62   } else if (Dest0 != Src0 || Dest1 != Src1) {
63     if (Dest0 == Src1 || Dest1 != Src0) {
64       BuildMI(MBB, MBBI, DL, OR, Dest1).addReg(Src1).addReg(Src1);
65       BuildMI(MBB, MBBI, DL, OR, Dest0).addReg(Src0).addReg(Src0);
66     } else {
67       BuildMI(MBB, MBBI, DL, OR, Dest0).addReg(Src0).addReg(Src0);
68       BuildMI(MBB, MBBI, DL, OR, Dest1).addReg(Src1).addReg(Src1);
69     }
70   }
71 }
72 
73 bool PPCExpandAtomicPseudo::runOnMachineFunction(MachineFunction &MF) {
74   bool Changed = false;
75   TII = static_cast<const PPCInstrInfo *>(MF.getSubtarget().getInstrInfo());
76   TRI = &TII->getRegisterInfo();
77   for (MachineBasicBlock &MBB : MF) {
78     for (MachineBasicBlock::iterator MBBI = MBB.begin(), MBBE = MBB.end();
79          MBBI != MBBE;) {
80       MachineInstr &MI = *MBBI;
81       MachineBasicBlock::iterator NMBBI = std::next(MBBI);
82       Changed |= expandMI(MBB, MI, NMBBI);
83       MBBI = NMBBI;
84     }
85   }
86   if (Changed)
87     MF.RenumberBlocks();
88   return Changed;
89 }
90 
91 bool PPCExpandAtomicPseudo::expandMI(MachineBasicBlock &MBB, MachineInstr &MI,
92                                      MachineBasicBlock::iterator &NMBBI) {
93   switch (MI.getOpcode()) {
94   case PPC::ATOMIC_SWAP_I128:
95   case PPC::ATOMIC_LOAD_ADD_I128:
96   case PPC::ATOMIC_LOAD_SUB_I128:
97   case PPC::ATOMIC_LOAD_XOR_I128:
98   case PPC::ATOMIC_LOAD_NAND_I128:
99   case PPC::ATOMIC_LOAD_AND_I128:
100   case PPC::ATOMIC_LOAD_OR_I128:
101     return expandAtomicRMW128(MBB, MI, NMBBI);
102   case PPC::ATOMIC_CMP_SWAP_I128:
103     return expandAtomicCmpSwap128(MBB, MI, NMBBI);
104   case PPC::BUILD_QUADWORD: {
105     Register Dst = MI.getOperand(0).getReg();
106     Register DstHi = TRI->getSubReg(Dst, PPC::sub_gp8_x0);
107     Register DstLo = TRI->getSubReg(Dst, PPC::sub_gp8_x1);
108     Register Lo = MI.getOperand(1).getReg();
109     Register Hi = MI.getOperand(2).getReg();
110     PairedCopy(TII, MBB, MI, MI.getDebugLoc(), DstHi, DstLo, Hi, Lo);
111     MI.eraseFromParent();
112     return true;
113   }
114   default:
115     return false;
116   }
117 }
118 
119 bool PPCExpandAtomicPseudo::expandAtomicRMW128(
120     MachineBasicBlock &MBB, MachineInstr &MI,
121     MachineBasicBlock::iterator &NMBBI) {
122   const MCInstrDesc &LL = TII->get(PPC::LQARX);
123   const MCInstrDesc &SC = TII->get(PPC::STQCX);
124   DebugLoc DL = MI.getDebugLoc();
125   MachineFunction *MF = MBB.getParent();
126   const BasicBlock *BB = MBB.getBasicBlock();
127   // Create layout of control flow.
128   MachineFunction::iterator MFI = ++MBB.getIterator();
129   MachineBasicBlock *LoopMBB = MF->CreateMachineBasicBlock(BB);
130   MachineBasicBlock *ExitMBB = MF->CreateMachineBasicBlock(BB);
131   MF->insert(MFI, LoopMBB);
132   MF->insert(MFI, ExitMBB);
133   ExitMBB->splice(ExitMBB->begin(), &MBB, std::next(MI.getIterator()),
134                   MBB.end());
135   ExitMBB->transferSuccessorsAndUpdatePHIs(&MBB);
136   MBB.addSuccessor(LoopMBB);
137 
138   // For non-min/max operations, control flow is kinda like:
139   // MBB:
140   //   ...
141   // LoopMBB:
142   //   lqarx in, ptr
143   //   addc out.sub_x1, in.sub_x1, op.sub_x1
144   //   adde out.sub_x0, in.sub_x0, op.sub_x0
145   //   stqcx out, ptr
146   //   bne- LoopMBB
147   // ExitMBB:
148   //   ...
149   Register Old = MI.getOperand(0).getReg();
150   Register OldHi = TRI->getSubReg(Old, PPC::sub_gp8_x0);
151   Register OldLo = TRI->getSubReg(Old, PPC::sub_gp8_x1);
152   Register Scratch = MI.getOperand(1).getReg();
153   Register ScratchHi = TRI->getSubReg(Scratch, PPC::sub_gp8_x0);
154   Register ScratchLo = TRI->getSubReg(Scratch, PPC::sub_gp8_x1);
155   Register RA = MI.getOperand(2).getReg();
156   Register RB = MI.getOperand(3).getReg();
157   Register IncrLo = MI.getOperand(4).getReg();
158   Register IncrHi = MI.getOperand(5).getReg();
159   unsigned RMWOpcode = MI.getOpcode();
160 
161   MachineBasicBlock *CurrentMBB = LoopMBB;
162   BuildMI(CurrentMBB, DL, LL, Old).addReg(RA).addReg(RB);
163 
164   switch (RMWOpcode) {
165   case PPC::ATOMIC_SWAP_I128:
166     PairedCopy(TII, *CurrentMBB, CurrentMBB->end(), DL, ScratchHi, ScratchLo,
167                IncrHi, IncrLo);
168     break;
169   case PPC::ATOMIC_LOAD_ADD_I128:
170     BuildMI(CurrentMBB, DL, TII->get(PPC::ADDC8), ScratchLo)
171         .addReg(IncrLo)
172         .addReg(OldLo);
173     BuildMI(CurrentMBB, DL, TII->get(PPC::ADDE8), ScratchHi)
174         .addReg(IncrHi)
175         .addReg(OldHi);
176     break;
177   case PPC::ATOMIC_LOAD_SUB_I128:
178     BuildMI(CurrentMBB, DL, TII->get(PPC::SUBFC8), ScratchLo)
179         .addReg(IncrLo)
180         .addReg(OldLo);
181     BuildMI(CurrentMBB, DL, TII->get(PPC::SUBFE8), ScratchHi)
182         .addReg(IncrHi)
183         .addReg(OldHi);
184     break;
185 
186 #define TRIVIAL_ATOMICRMW(Opcode, Instr)                                       \
187   case Opcode:                                                                 \
188     BuildMI(CurrentMBB, DL, TII->get((Instr)), ScratchLo)                      \
189         .addReg(IncrLo)                                                        \
190         .addReg(OldLo);                                                        \
191     BuildMI(CurrentMBB, DL, TII->get((Instr)), ScratchHi)                      \
192         .addReg(IncrHi)                                                        \
193         .addReg(OldHi);                                                        \
194     break
195 
196     TRIVIAL_ATOMICRMW(PPC::ATOMIC_LOAD_OR_I128, PPC::OR8);
197     TRIVIAL_ATOMICRMW(PPC::ATOMIC_LOAD_XOR_I128, PPC::XOR8);
198     TRIVIAL_ATOMICRMW(PPC::ATOMIC_LOAD_AND_I128, PPC::AND8);
199     TRIVIAL_ATOMICRMW(PPC::ATOMIC_LOAD_NAND_I128, PPC::NAND8);
200 #undef TRIVIAL_ATOMICRMW
201   default:
202     llvm_unreachable("Unhandled atomic RMW operation");
203   }
204   BuildMI(CurrentMBB, DL, SC).addReg(Scratch).addReg(RA).addReg(RB);
205   BuildMI(CurrentMBB, DL, TII->get(PPC::BCC))
206       .addImm(PPC::PRED_NE)
207       .addReg(PPC::CR0)
208       .addMBB(LoopMBB);
209   CurrentMBB->addSuccessor(LoopMBB);
210   CurrentMBB->addSuccessor(ExitMBB);
211   bool anyChange = false;
212   do {
213     anyChange = recomputeLiveIns(*ExitMBB) || recomputeLiveIns(*LoopMBB);
214   } while (anyChange);
215   NMBBI = MBB.end();
216   MI.eraseFromParent();
217   return true;
218 }
219 
220 bool PPCExpandAtomicPseudo::expandAtomicCmpSwap128(
221     MachineBasicBlock &MBB, MachineInstr &MI,
222     MachineBasicBlock::iterator &NMBBI) {
223   const MCInstrDesc &LL = TII->get(PPC::LQARX);
224   const MCInstrDesc &SC = TII->get(PPC::STQCX);
225   DebugLoc DL = MI.getDebugLoc();
226   MachineFunction *MF = MBB.getParent();
227   const BasicBlock *BB = MBB.getBasicBlock();
228   Register Old = MI.getOperand(0).getReg();
229   Register OldHi = TRI->getSubReg(Old, PPC::sub_gp8_x0);
230   Register OldLo = TRI->getSubReg(Old, PPC::sub_gp8_x1);
231   Register Scratch = MI.getOperand(1).getReg();
232   Register ScratchHi = TRI->getSubReg(Scratch, PPC::sub_gp8_x0);
233   Register ScratchLo = TRI->getSubReg(Scratch, PPC::sub_gp8_x1);
234   Register RA = MI.getOperand(2).getReg();
235   Register RB = MI.getOperand(3).getReg();
236   Register CmpLo = MI.getOperand(4).getReg();
237   Register CmpHi = MI.getOperand(5).getReg();
238   Register NewLo = MI.getOperand(6).getReg();
239   Register NewHi = MI.getOperand(7).getReg();
240   // Create layout of control flow.
241   // loop:
242   //   old = lqarx ptr
243   //   <compare old, cmp>
244   //   bne 0, exit
245   // succ:
246   //   stqcx new ptr
247   //   bne 0, loop
248   // exit:
249   //   ....
250   MachineFunction::iterator MFI = ++MBB.getIterator();
251   MachineBasicBlock *LoopCmpMBB = MF->CreateMachineBasicBlock(BB);
252   MachineBasicBlock *CmpSuccMBB = MF->CreateMachineBasicBlock(BB);
253   MachineBasicBlock *ExitMBB = MF->CreateMachineBasicBlock(BB);
254   MF->insert(MFI, LoopCmpMBB);
255   MF->insert(MFI, CmpSuccMBB);
256   MF->insert(MFI, ExitMBB);
257   ExitMBB->splice(ExitMBB->begin(), &MBB, std::next(MI.getIterator()),
258                   MBB.end());
259   ExitMBB->transferSuccessorsAndUpdatePHIs(&MBB);
260   MBB.addSuccessor(LoopCmpMBB);
261   // Build loop.
262   MachineBasicBlock *CurrentMBB = LoopCmpMBB;
263   BuildMI(CurrentMBB, DL, LL, Old).addReg(RA).addReg(RB);
264   BuildMI(CurrentMBB, DL, TII->get(PPC::XOR8), ScratchLo)
265       .addReg(OldLo)
266       .addReg(CmpLo);
267   BuildMI(CurrentMBB, DL, TII->get(PPC::XOR8), ScratchHi)
268       .addReg(OldHi)
269       .addReg(CmpHi);
270   BuildMI(CurrentMBB, DL, TII->get(PPC::OR8_rec), ScratchLo)
271       .addReg(ScratchLo)
272       .addReg(ScratchHi);
273   BuildMI(CurrentMBB, DL, TII->get(PPC::BCC))
274       .addImm(PPC::PRED_NE)
275       .addReg(PPC::CR0)
276       .addMBB(ExitMBB);
277   CurrentMBB->addSuccessor(CmpSuccMBB);
278   CurrentMBB->addSuccessor(ExitMBB);
279   // Build succ.
280   CurrentMBB = CmpSuccMBB;
281   PairedCopy(TII, *CurrentMBB, CurrentMBB->end(), DL, ScratchHi, ScratchLo,
282              NewHi, NewLo);
283   BuildMI(CurrentMBB, DL, SC).addReg(Scratch).addReg(RA).addReg(RB);
284   BuildMI(CurrentMBB, DL, TII->get(PPC::BCC))
285       .addImm(PPC::PRED_NE)
286       .addReg(PPC::CR0)
287       .addMBB(LoopCmpMBB);
288   CurrentMBB->addSuccessor(LoopCmpMBB);
289   CurrentMBB->addSuccessor(ExitMBB);
290 
291   bool anyChange = false;
292   do {
293     anyChange = recomputeLiveIns(*ExitMBB) || recomputeLiveIns(*CmpSuccMBB) ||
294                 recomputeLiveIns(*LoopCmpMBB);
295   } while (anyChange);
296   NMBBI = MBB.end();
297   MI.eraseFromParent();
298   return true;
299 }
300 
301 } // namespace
302 
303 INITIALIZE_PASS(PPCExpandAtomicPseudo, DEBUG_TYPE, "PowerPC Expand Atomic",
304                 false, false)
305 
306 char PPCExpandAtomicPseudo::ID = 0;
307 FunctionPass *llvm::createPPCExpandAtomicPseudoPass() {
308   return new PPCExpandAtomicPseudo();
309 }
310