xref: /freebsd/contrib/llvm-project/llvm/lib/Target/Hexagon/HexagonOptAddrMode.cpp (revision 6f63e88c0166ed3e5f2805a9e667c7d24d304cf1)
1 //===- HexagonOptAddrMode.cpp ---------------------------------------------===//
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 // This implements a Hexagon-specific pass to optimize addressing mode for
9 // load/store instructions.
10 //===----------------------------------------------------------------------===//
11 
12 #include "HexagonInstrInfo.h"
13 #include "HexagonSubtarget.h"
14 #include "MCTargetDesc/HexagonBaseInfo.h"
15 #include "RDFGraph.h"
16 #include "RDFLiveness.h"
17 #include "RDFRegisters.h"
18 #include "llvm/ADT/DenseMap.h"
19 #include "llvm/ADT/DenseSet.h"
20 #include "llvm/ADT/StringRef.h"
21 #include "llvm/CodeGen/MachineBasicBlock.h"
22 #include "llvm/CodeGen/MachineDominanceFrontier.h"
23 #include "llvm/CodeGen/MachineDominators.h"
24 #include "llvm/CodeGen/MachineFunction.h"
25 #include "llvm/CodeGen/MachineFunctionPass.h"
26 #include "llvm/CodeGen/MachineInstr.h"
27 #include "llvm/CodeGen/MachineInstrBuilder.h"
28 #include "llvm/CodeGen/MachineOperand.h"
29 #include "llvm/CodeGen/MachineRegisterInfo.h"
30 #include "llvm/CodeGen/TargetSubtargetInfo.h"
31 #include "llvm/InitializePasses.h"
32 #include "llvm/MC/MCInstrDesc.h"
33 #include "llvm/Pass.h"
34 #include "llvm/Support/CommandLine.h"
35 #include "llvm/Support/Debug.h"
36 #include "llvm/Support/ErrorHandling.h"
37 #include "llvm/Support/raw_ostream.h"
38 #include <cassert>
39 #include <cstdint>
40 
41 #define DEBUG_TYPE "opt-addr-mode"
42 
43 using namespace llvm;
44 using namespace rdf;
45 
46 static cl::opt<int> CodeGrowthLimit("hexagon-amode-growth-limit",
47   cl::Hidden, cl::init(0), cl::desc("Code growth limit for address mode "
48   "optimization"));
49 
50 namespace llvm {
51 
52   FunctionPass *createHexagonOptAddrMode();
53   void initializeHexagonOptAddrModePass(PassRegistry&);
54 
55 } // end namespace llvm
56 
57 namespace {
58 
59 class HexagonOptAddrMode : public MachineFunctionPass {
60 public:
61   static char ID;
62 
63   HexagonOptAddrMode() : MachineFunctionPass(ID) {}
64 
65   StringRef getPassName() const override {
66     return "Optimize addressing mode of load/store";
67   }
68 
69   void getAnalysisUsage(AnalysisUsage &AU) const override {
70     MachineFunctionPass::getAnalysisUsage(AU);
71     AU.addRequired<MachineDominatorTree>();
72     AU.addRequired<MachineDominanceFrontier>();
73     AU.setPreservesAll();
74   }
75 
76   bool runOnMachineFunction(MachineFunction &MF) override;
77 
78 private:
79   using MISetType = DenseSet<MachineInstr *>;
80   using InstrEvalMap = DenseMap<MachineInstr *, bool>;
81 
82   MachineRegisterInfo *MRI = nullptr;
83   const HexagonInstrInfo *HII = nullptr;
84   const HexagonRegisterInfo *HRI = nullptr;
85   MachineDominatorTree *MDT = nullptr;
86   DataFlowGraph *DFG = nullptr;
87   DataFlowGraph::DefStackMap DefM;
88   Liveness *LV = nullptr;
89   MISetType Deleted;
90 
91   bool processBlock(NodeAddr<BlockNode *> BA);
92   bool xformUseMI(MachineInstr *TfrMI, MachineInstr *UseMI,
93                   NodeAddr<UseNode *> UseN, unsigned UseMOnum);
94   bool processAddUses(NodeAddr<StmtNode *> AddSN, MachineInstr *AddMI,
95                       const NodeList &UNodeList);
96   bool updateAddUses(MachineInstr *AddMI, MachineInstr *UseMI);
97   bool analyzeUses(unsigned DefR, const NodeList &UNodeList,
98                    InstrEvalMap &InstrEvalResult, short &SizeInc);
99   bool hasRepForm(MachineInstr &MI, unsigned TfrDefR);
100   bool canRemoveAddasl(NodeAddr<StmtNode *> AddAslSN, MachineInstr &MI,
101                        const NodeList &UNodeList);
102   bool isSafeToExtLR(NodeAddr<StmtNode *> SN, MachineInstr *MI,
103                      unsigned LRExtReg, const NodeList &UNodeList);
104   void getAllRealUses(NodeAddr<StmtNode *> SN, NodeList &UNodeList);
105   bool allValidCandidates(NodeAddr<StmtNode *> SA, NodeList &UNodeList);
106   short getBaseWithLongOffset(const MachineInstr &MI) const;
107   bool changeStore(MachineInstr *OldMI, MachineOperand ImmOp,
108                    unsigned ImmOpNum);
109   bool changeLoad(MachineInstr *OldMI, MachineOperand ImmOp, unsigned ImmOpNum);
110   bool changeAddAsl(NodeAddr<UseNode *> AddAslUN, MachineInstr *AddAslMI,
111                     const MachineOperand &ImmOp, unsigned ImmOpNum);
112   bool isValidOffset(MachineInstr *MI, int Offset);
113 };
114 
115 } // end anonymous namespace
116 
117 char HexagonOptAddrMode::ID = 0;
118 
119 INITIALIZE_PASS_BEGIN(HexagonOptAddrMode, "amode-opt",
120                       "Optimize addressing mode", false, false)
121 INITIALIZE_PASS_DEPENDENCY(MachineDominatorTree)
122 INITIALIZE_PASS_DEPENDENCY(MachineDominanceFrontier)
123 INITIALIZE_PASS_END(HexagonOptAddrMode, "amode-opt", "Optimize addressing mode",
124                     false, false)
125 
126 bool HexagonOptAddrMode::hasRepForm(MachineInstr &MI, unsigned TfrDefR) {
127   const MCInstrDesc &MID = MI.getDesc();
128 
129   if ((!MID.mayStore() && !MID.mayLoad()) || HII->isPredicated(MI))
130     return false;
131 
132   if (MID.mayStore()) {
133     MachineOperand StOp = MI.getOperand(MI.getNumOperands() - 1);
134     if (StOp.isReg() && StOp.getReg() == TfrDefR)
135       return false;
136   }
137 
138   if (HII->getAddrMode(MI) == HexagonII::BaseRegOffset)
139     // Tranform to Absolute plus register offset.
140     return (HII->changeAddrMode_rr_ur(MI) >= 0);
141   else if (HII->getAddrMode(MI) == HexagonII::BaseImmOffset)
142     // Tranform to absolute addressing mode.
143     return (HII->changeAddrMode_io_abs(MI) >= 0);
144 
145   return false;
146 }
147 
148 // Check if addasl instruction can be removed. This is possible only
149 // if it's feeding to only load/store instructions with base + register
150 // offset as these instruction can be tranformed to use 'absolute plus
151 // shifted register offset'.
152 // ex:
153 // Rs = ##foo
154 // Rx = addasl(Rs, Rt, #2)
155 // Rd = memw(Rx + #28)
156 // Above three instructions can be replaced with Rd = memw(Rt<<#2 + ##foo+28)
157 
158 bool HexagonOptAddrMode::canRemoveAddasl(NodeAddr<StmtNode *> AddAslSN,
159                                          MachineInstr &MI,
160                                          const NodeList &UNodeList) {
161   // check offset size in addasl. if 'offset > 3' return false
162   const MachineOperand &OffsetOp = MI.getOperand(3);
163   if (!OffsetOp.isImm() || OffsetOp.getImm() > 3)
164     return false;
165 
166   Register OffsetReg = MI.getOperand(2).getReg();
167   RegisterRef OffsetRR;
168   NodeId OffsetRegRD = 0;
169   for (NodeAddr<UseNode *> UA : AddAslSN.Addr->members_if(DFG->IsUse, *DFG)) {
170     RegisterRef RR = UA.Addr->getRegRef(*DFG);
171     if (OffsetReg == RR.Reg) {
172       OffsetRR = RR;
173       OffsetRegRD = UA.Addr->getReachingDef();
174     }
175   }
176 
177   for (auto I = UNodeList.rbegin(), E = UNodeList.rend(); I != E; ++I) {
178     NodeAddr<UseNode *> UA = *I;
179     NodeAddr<InstrNode *> IA = UA.Addr->getOwner(*DFG);
180     if (UA.Addr->getFlags() & NodeAttrs::PhiRef)
181       return false;
182     NodeAddr<RefNode*> AA = LV->getNearestAliasedRef(OffsetRR, IA);
183     if ((DFG->IsDef(AA) && AA.Id != OffsetRegRD) ||
184          AA.Addr->getReachingDef() != OffsetRegRD)
185       return false;
186 
187     MachineInstr &UseMI = *NodeAddr<StmtNode *>(IA).Addr->getCode();
188     NodeAddr<DefNode *> OffsetRegDN = DFG->addr<DefNode *>(OffsetRegRD);
189     // Reaching Def to an offset register can't be a phi.
190     if ((OffsetRegDN.Addr->getFlags() & NodeAttrs::PhiRef) &&
191         MI.getParent() != UseMI.getParent())
192     return false;
193 
194     const MCInstrDesc &UseMID = UseMI.getDesc();
195     if ((!UseMID.mayLoad() && !UseMID.mayStore()) ||
196         HII->getAddrMode(UseMI) != HexagonII::BaseImmOffset ||
197         getBaseWithLongOffset(UseMI) < 0)
198       return false;
199 
200     // Addasl output can't be a store value.
201     if (UseMID.mayStore() && UseMI.getOperand(2).isReg() &&
202         UseMI.getOperand(2).getReg() == MI.getOperand(0).getReg())
203       return false;
204 
205     for (auto &Mo : UseMI.operands())
206       if (Mo.isFI())
207         return false;
208   }
209   return true;
210 }
211 
212 bool HexagonOptAddrMode::allValidCandidates(NodeAddr<StmtNode *> SA,
213                                             NodeList &UNodeList) {
214   for (auto I = UNodeList.rbegin(), E = UNodeList.rend(); I != E; ++I) {
215     NodeAddr<UseNode *> UN = *I;
216     RegisterRef UR = UN.Addr->getRegRef(*DFG);
217     NodeSet Visited, Defs;
218     const auto &P = LV->getAllReachingDefsRec(UR, UN, Visited, Defs);
219     if (!P.second) {
220       LLVM_DEBUG({
221         dbgs() << "*** Unable to collect all reaching defs for use ***\n"
222                << PrintNode<UseNode*>(UN, *DFG) << '\n'
223                << "The program's complexity may exceed the limits.\n";
224       });
225       return false;
226     }
227     const auto &ReachingDefs = P.first;
228     if (ReachingDefs.size() > 1) {
229       LLVM_DEBUG({
230         dbgs() << "*** Multiple Reaching Defs found!!! ***\n";
231         for (auto DI : ReachingDefs) {
232           NodeAddr<UseNode *> DA = DFG->addr<UseNode *>(DI);
233           NodeAddr<StmtNode *> TempIA = DA.Addr->getOwner(*DFG);
234           dbgs() << "\t\t[Reaching Def]: "
235                  << Print<NodeAddr<InstrNode *>>(TempIA, *DFG) << "\n";
236         }
237       });
238       return false;
239     }
240   }
241   return true;
242 }
243 
244 void HexagonOptAddrMode::getAllRealUses(NodeAddr<StmtNode *> SA,
245                                         NodeList &UNodeList) {
246   for (NodeAddr<DefNode *> DA : SA.Addr->members_if(DFG->IsDef, *DFG)) {
247     LLVM_DEBUG(dbgs() << "\t\t[DefNode]: "
248                       << Print<NodeAddr<DefNode *>>(DA, *DFG) << "\n");
249     RegisterRef DR = DFG->getPRI().normalize(DA.Addr->getRegRef(*DFG));
250 
251     auto UseSet = LV->getAllReachedUses(DR, DA);
252 
253     for (auto UI : UseSet) {
254       NodeAddr<UseNode *> UA = DFG->addr<UseNode *>(UI);
255       LLVM_DEBUG({
256         NodeAddr<StmtNode *> TempIA = UA.Addr->getOwner(*DFG);
257         dbgs() << "\t\t\t[Reached Use]: "
258                << Print<NodeAddr<InstrNode *>>(TempIA, *DFG) << "\n";
259       });
260 
261       if (UA.Addr->getFlags() & NodeAttrs::PhiRef) {
262         NodeAddr<PhiNode *> PA = UA.Addr->getOwner(*DFG);
263         NodeId id = PA.Id;
264         const Liveness::RefMap &phiUse = LV->getRealUses(id);
265         LLVM_DEBUG(dbgs() << "\t\t\t\tphi real Uses"
266                           << Print<Liveness::RefMap>(phiUse, *DFG) << "\n");
267         if (!phiUse.empty()) {
268           for (auto I : phiUse) {
269             if (!DFG->getPRI().alias(RegisterRef(I.first), DR))
270               continue;
271             auto phiUseSet = I.second;
272             for (auto phiUI : phiUseSet) {
273               NodeAddr<UseNode *> phiUA = DFG->addr<UseNode *>(phiUI.first);
274               UNodeList.push_back(phiUA);
275             }
276           }
277         }
278       } else
279         UNodeList.push_back(UA);
280     }
281   }
282 }
283 
284 bool HexagonOptAddrMode::isSafeToExtLR(NodeAddr<StmtNode *> SN,
285                                        MachineInstr *MI, unsigned LRExtReg,
286                                        const NodeList &UNodeList) {
287   RegisterRef LRExtRR;
288   NodeId LRExtRegRD = 0;
289   // Iterate through all the UseNodes in SN and find the reaching def
290   // for the LRExtReg.
291   for (NodeAddr<UseNode *> UA : SN.Addr->members_if(DFG->IsUse, *DFG)) {
292     RegisterRef RR = UA.Addr->getRegRef(*DFG);
293     if (LRExtReg == RR.Reg) {
294       LRExtRR = RR;
295       LRExtRegRD = UA.Addr->getReachingDef();
296     }
297   }
298 
299   for (auto I = UNodeList.rbegin(), E = UNodeList.rend(); I != E; ++I) {
300     NodeAddr<UseNode *> UA = *I;
301     NodeAddr<InstrNode *> IA = UA.Addr->getOwner(*DFG);
302     // The reaching def of LRExtRR at load/store node should be same as the
303     // one reaching at the SN.
304     if (UA.Addr->getFlags() & NodeAttrs::PhiRef)
305       return false;
306     NodeAddr<RefNode*> AA = LV->getNearestAliasedRef(LRExtRR, IA);
307     if ((DFG->IsDef(AA) && AA.Id != LRExtRegRD) ||
308         AA.Addr->getReachingDef() != LRExtRegRD) {
309       LLVM_DEBUG(
310           dbgs() << "isSafeToExtLR: Returning false; another reaching def\n");
311       return false;
312     }
313 
314     MachineInstr *UseMI = NodeAddr<StmtNode *>(IA).Addr->getCode();
315     NodeAddr<DefNode *> LRExtRegDN = DFG->addr<DefNode *>(LRExtRegRD);
316     // Reaching Def to LRExtReg can't be a phi.
317     if ((LRExtRegDN.Addr->getFlags() & NodeAttrs::PhiRef) &&
318         MI->getParent() != UseMI->getParent())
319     return false;
320   }
321   return true;
322 }
323 
324 bool HexagonOptAddrMode::isValidOffset(MachineInstr *MI, int Offset) {
325   unsigned AlignMask = 0;
326   switch (HII->getMemAccessSize(*MI)) {
327   case HexagonII::MemAccessSize::DoubleWordAccess:
328     AlignMask = 0x7;
329     break;
330   case HexagonII::MemAccessSize::WordAccess:
331     AlignMask = 0x3;
332     break;
333   case HexagonII::MemAccessSize::HalfWordAccess:
334     AlignMask = 0x1;
335     break;
336   case HexagonII::MemAccessSize::ByteAccess:
337     AlignMask = 0x0;
338     break;
339   default:
340     return false;
341   }
342 
343   if ((AlignMask & Offset) != 0)
344     return false;
345   return HII->isValidOffset(MI->getOpcode(), Offset, HRI, false);
346 }
347 
348 bool HexagonOptAddrMode::processAddUses(NodeAddr<StmtNode *> AddSN,
349                                         MachineInstr *AddMI,
350                                         const NodeList &UNodeList) {
351 
352   Register AddDefR = AddMI->getOperand(0).getReg();
353   for (auto I = UNodeList.rbegin(), E = UNodeList.rend(); I != E; ++I) {
354     NodeAddr<UseNode *> UN = *I;
355     NodeAddr<StmtNode *> SN = UN.Addr->getOwner(*DFG);
356     MachineInstr *MI = SN.Addr->getCode();
357     const MCInstrDesc &MID = MI->getDesc();
358     if ((!MID.mayLoad() && !MID.mayStore()) ||
359         HII->getAddrMode(*MI) != HexagonII::BaseImmOffset ||
360         HII->isHVXVec(*MI))
361       return false;
362 
363     MachineOperand BaseOp = MID.mayLoad() ? MI->getOperand(1)
364                                           : MI->getOperand(0);
365 
366     if (!BaseOp.isReg() || BaseOp.getReg() != AddDefR)
367       return false;
368 
369     MachineOperand OffsetOp = MID.mayLoad() ? MI->getOperand(2)
370                                             : MI->getOperand(1);
371     if (!OffsetOp.isImm())
372       return false;
373 
374     int64_t newOffset = OffsetOp.getImm() + AddMI->getOperand(2).getImm();
375     if (!isValidOffset(MI, newOffset))
376       return false;
377 
378     // Since we'll be extending the live range of Rt in the following example,
379     // make sure that is safe. another definition of Rt doesn't exist between 'add'
380     // and load/store instruction.
381     //
382     // Ex: Rx= add(Rt,#10)
383     //     memw(Rx+#0) = Rs
384     // will be replaced with =>  memw(Rt+#10) = Rs
385     Register BaseReg = AddMI->getOperand(1).getReg();
386     if (!isSafeToExtLR(AddSN, AddMI, BaseReg, UNodeList))
387       return false;
388   }
389 
390   // Update all the uses of 'add' with the appropriate base and offset
391   // values.
392   bool Changed = false;
393   for (auto I = UNodeList.rbegin(), E = UNodeList.rend(); I != E; ++I) {
394     NodeAddr<UseNode *> UseN = *I;
395     assert(!(UseN.Addr->getFlags() & NodeAttrs::PhiRef) &&
396            "Found a PhiRef node as a real reached use!!");
397 
398     NodeAddr<StmtNode *> OwnerN = UseN.Addr->getOwner(*DFG);
399     MachineInstr *UseMI = OwnerN.Addr->getCode();
400     LLVM_DEBUG(dbgs() << "\t\t[MI <BB#" << UseMI->getParent()->getNumber()
401                       << ">]: " << *UseMI << "\n");
402     Changed |= updateAddUses(AddMI, UseMI);
403   }
404 
405   if (Changed)
406     Deleted.insert(AddMI);
407 
408   return Changed;
409 }
410 
411 bool HexagonOptAddrMode::updateAddUses(MachineInstr *AddMI,
412                                         MachineInstr *UseMI) {
413   const MachineOperand ImmOp = AddMI->getOperand(2);
414   const MachineOperand AddRegOp = AddMI->getOperand(1);
415   Register newReg = AddRegOp.getReg();
416   const MCInstrDesc &MID = UseMI->getDesc();
417 
418   MachineOperand &BaseOp = MID.mayLoad() ? UseMI->getOperand(1)
419                                          : UseMI->getOperand(0);
420   MachineOperand &OffsetOp = MID.mayLoad() ? UseMI->getOperand(2)
421                                            : UseMI->getOperand(1);
422   BaseOp.setReg(newReg);
423   BaseOp.setIsUndef(AddRegOp.isUndef());
424   BaseOp.setImplicit(AddRegOp.isImplicit());
425   OffsetOp.setImm(ImmOp.getImm() + OffsetOp.getImm());
426   MRI->clearKillFlags(newReg);
427 
428   return true;
429 }
430 
431 bool HexagonOptAddrMode::analyzeUses(unsigned tfrDefR,
432                                      const NodeList &UNodeList,
433                                      InstrEvalMap &InstrEvalResult,
434                                      short &SizeInc) {
435   bool KeepTfr = false;
436   bool HasRepInstr = false;
437   InstrEvalResult.clear();
438 
439   for (auto I = UNodeList.rbegin(), E = UNodeList.rend(); I != E; ++I) {
440     bool CanBeReplaced = false;
441     NodeAddr<UseNode *> UN = *I;
442     NodeAddr<StmtNode *> SN = UN.Addr->getOwner(*DFG);
443     MachineInstr &MI = *SN.Addr->getCode();
444     const MCInstrDesc &MID = MI.getDesc();
445     if ((MID.mayLoad() || MID.mayStore())) {
446       if (!hasRepForm(MI, tfrDefR)) {
447         KeepTfr = true;
448         continue;
449       }
450       SizeInc++;
451       CanBeReplaced = true;
452     } else if (MI.getOpcode() == Hexagon::S2_addasl_rrri) {
453       NodeList AddaslUseList;
454 
455       LLVM_DEBUG(dbgs() << "\nGetting ReachedUses for === " << MI << "\n");
456       getAllRealUses(SN, AddaslUseList);
457       // Process phi nodes.
458       if (allValidCandidates(SN, AddaslUseList) &&
459           canRemoveAddasl(SN, MI, AddaslUseList)) {
460         SizeInc += AddaslUseList.size();
461         SizeInc -= 1; // Reduce size by 1 as addasl itself can be removed.
462         CanBeReplaced = true;
463       } else
464         SizeInc++;
465     } else
466       // Currently, only load/store and addasl are handled.
467       // Some other instructions to consider -
468       // A2_add -> A2_addi
469       // M4_mpyrr_addr -> M4_mpyrr_addi
470       KeepTfr = true;
471 
472     InstrEvalResult[&MI] = CanBeReplaced;
473     HasRepInstr |= CanBeReplaced;
474   }
475 
476   // Reduce total size by 2 if original tfr can be deleted.
477   if (!KeepTfr)
478     SizeInc -= 2;
479 
480   return HasRepInstr;
481 }
482 
483 bool HexagonOptAddrMode::changeLoad(MachineInstr *OldMI, MachineOperand ImmOp,
484                                     unsigned ImmOpNum) {
485   bool Changed = false;
486   MachineBasicBlock *BB = OldMI->getParent();
487   auto UsePos = MachineBasicBlock::iterator(OldMI);
488   MachineBasicBlock::instr_iterator InsertPt = UsePos.getInstrIterator();
489   ++InsertPt;
490   unsigned OpStart;
491   unsigned OpEnd = OldMI->getNumOperands();
492   MachineInstrBuilder MIB;
493 
494   if (ImmOpNum == 1) {
495     if (HII->getAddrMode(*OldMI) == HexagonII::BaseRegOffset) {
496       short NewOpCode = HII->changeAddrMode_rr_ur(*OldMI);
497       assert(NewOpCode >= 0 && "Invalid New opcode\n");
498       MIB = BuildMI(*BB, InsertPt, OldMI->getDebugLoc(), HII->get(NewOpCode));
499       MIB.add(OldMI->getOperand(0));
500       MIB.add(OldMI->getOperand(2));
501       MIB.add(OldMI->getOperand(3));
502       MIB.add(ImmOp);
503       OpStart = 4;
504       Changed = true;
505     } else if (HII->getAddrMode(*OldMI) == HexagonII::BaseImmOffset &&
506                OldMI->getOperand(2).isImm()) {
507       short NewOpCode = HII->changeAddrMode_io_abs(*OldMI);
508       assert(NewOpCode >= 0 && "Invalid New opcode\n");
509       MIB = BuildMI(*BB, InsertPt, OldMI->getDebugLoc(), HII->get(NewOpCode))
510                 .add(OldMI->getOperand(0));
511       const GlobalValue *GV = ImmOp.getGlobal();
512       int64_t Offset = ImmOp.getOffset() + OldMI->getOperand(2).getImm();
513 
514       MIB.addGlobalAddress(GV, Offset, ImmOp.getTargetFlags());
515       OpStart = 3;
516       Changed = true;
517     } else
518       Changed = false;
519 
520     LLVM_DEBUG(dbgs() << "[Changing]: " << *OldMI << "\n");
521     LLVM_DEBUG(dbgs() << "[TO]: " << *MIB << "\n");
522   } else if (ImmOpNum == 2) {
523     if (OldMI->getOperand(3).isImm() && OldMI->getOperand(3).getImm() == 0) {
524       short NewOpCode = HII->changeAddrMode_rr_io(*OldMI);
525       assert(NewOpCode >= 0 && "Invalid New opcode\n");
526       MIB = BuildMI(*BB, InsertPt, OldMI->getDebugLoc(), HII->get(NewOpCode));
527       MIB.add(OldMI->getOperand(0));
528       MIB.add(OldMI->getOperand(1));
529       MIB.add(ImmOp);
530       OpStart = 4;
531       Changed = true;
532       LLVM_DEBUG(dbgs() << "[Changing]: " << *OldMI << "\n");
533       LLVM_DEBUG(dbgs() << "[TO]: " << *MIB << "\n");
534     }
535   }
536 
537   if (Changed)
538     for (unsigned i = OpStart; i < OpEnd; ++i)
539       MIB.add(OldMI->getOperand(i));
540 
541   return Changed;
542 }
543 
544 bool HexagonOptAddrMode::changeStore(MachineInstr *OldMI, MachineOperand ImmOp,
545                                      unsigned ImmOpNum) {
546   bool Changed = false;
547   unsigned OpStart = 0;
548   unsigned OpEnd = OldMI->getNumOperands();
549   MachineBasicBlock *BB = OldMI->getParent();
550   auto UsePos = MachineBasicBlock::iterator(OldMI);
551   MachineBasicBlock::instr_iterator InsertPt = UsePos.getInstrIterator();
552   ++InsertPt;
553   MachineInstrBuilder MIB;
554   if (ImmOpNum == 0) {
555     if (HII->getAddrMode(*OldMI) == HexagonII::BaseRegOffset) {
556       short NewOpCode = HII->changeAddrMode_rr_ur(*OldMI);
557       assert(NewOpCode >= 0 && "Invalid New opcode\n");
558       MIB = BuildMI(*BB, InsertPt, OldMI->getDebugLoc(), HII->get(NewOpCode));
559       MIB.add(OldMI->getOperand(1));
560       MIB.add(OldMI->getOperand(2));
561       MIB.add(ImmOp);
562       MIB.add(OldMI->getOperand(3));
563       OpStart = 4;
564     } else if (HII->getAddrMode(*OldMI) == HexagonII::BaseImmOffset) {
565       short NewOpCode = HII->changeAddrMode_io_abs(*OldMI);
566       assert(NewOpCode >= 0 && "Invalid New opcode\n");
567       MIB = BuildMI(*BB, InsertPt, OldMI->getDebugLoc(), HII->get(NewOpCode));
568       const GlobalValue *GV = ImmOp.getGlobal();
569       int64_t Offset = ImmOp.getOffset() + OldMI->getOperand(1).getImm();
570       MIB.addGlobalAddress(GV, Offset, ImmOp.getTargetFlags());
571       MIB.add(OldMI->getOperand(2));
572       OpStart = 3;
573     }
574     Changed = true;
575     LLVM_DEBUG(dbgs() << "[Changing]: " << *OldMI << "\n");
576     LLVM_DEBUG(dbgs() << "[TO]: " << *MIB << "\n");
577   } else if (ImmOpNum == 1 && OldMI->getOperand(2).getImm() == 0) {
578     short NewOpCode = HII->changeAddrMode_rr_io(*OldMI);
579     assert(NewOpCode >= 0 && "Invalid New opcode\n");
580     MIB = BuildMI(*BB, InsertPt, OldMI->getDebugLoc(), HII->get(NewOpCode));
581     MIB.add(OldMI->getOperand(0));
582     MIB.add(ImmOp);
583     OpStart = 3;
584     Changed = true;
585     LLVM_DEBUG(dbgs() << "[Changing]: " << *OldMI << "\n");
586     LLVM_DEBUG(dbgs() << "[TO]: " << *MIB << "\n");
587   }
588   if (Changed)
589     for (unsigned i = OpStart; i < OpEnd; ++i)
590       MIB.add(OldMI->getOperand(i));
591 
592   return Changed;
593 }
594 
595 short HexagonOptAddrMode::getBaseWithLongOffset(const MachineInstr &MI) const {
596   if (HII->getAddrMode(MI) == HexagonII::BaseImmOffset) {
597     short TempOpCode = HII->changeAddrMode_io_rr(MI);
598     return HII->changeAddrMode_rr_ur(TempOpCode);
599   }
600   return HII->changeAddrMode_rr_ur(MI);
601 }
602 
603 bool HexagonOptAddrMode::changeAddAsl(NodeAddr<UseNode *> AddAslUN,
604                                       MachineInstr *AddAslMI,
605                                       const MachineOperand &ImmOp,
606                                       unsigned ImmOpNum) {
607   NodeAddr<StmtNode *> SA = AddAslUN.Addr->getOwner(*DFG);
608 
609   LLVM_DEBUG(dbgs() << "Processing addasl :" << *AddAslMI << "\n");
610 
611   NodeList UNodeList;
612   getAllRealUses(SA, UNodeList);
613 
614   for (auto I = UNodeList.rbegin(), E = UNodeList.rend(); I != E; ++I) {
615     NodeAddr<UseNode *> UseUN = *I;
616     assert(!(UseUN.Addr->getFlags() & NodeAttrs::PhiRef) &&
617            "Can't transform this 'AddAsl' instruction!");
618 
619     NodeAddr<StmtNode *> UseIA = UseUN.Addr->getOwner(*DFG);
620     LLVM_DEBUG(dbgs() << "[InstrNode]: "
621                       << Print<NodeAddr<InstrNode *>>(UseIA, *DFG) << "\n");
622     MachineInstr *UseMI = UseIA.Addr->getCode();
623     LLVM_DEBUG(dbgs() << "[MI <" << printMBBReference(*UseMI->getParent())
624                       << ">]: " << *UseMI << "\n");
625     const MCInstrDesc &UseMID = UseMI->getDesc();
626     assert(HII->getAddrMode(*UseMI) == HexagonII::BaseImmOffset);
627 
628     auto UsePos = MachineBasicBlock::iterator(UseMI);
629     MachineBasicBlock::instr_iterator InsertPt = UsePos.getInstrIterator();
630     short NewOpCode = getBaseWithLongOffset(*UseMI);
631     assert(NewOpCode >= 0 && "Invalid New opcode\n");
632 
633     unsigned OpStart;
634     unsigned OpEnd = UseMI->getNumOperands();
635 
636     MachineBasicBlock *BB = UseMI->getParent();
637     MachineInstrBuilder MIB =
638         BuildMI(*BB, InsertPt, UseMI->getDebugLoc(), HII->get(NewOpCode));
639     // change mem(Rs + # ) -> mem(Rt << # + ##)
640     if (UseMID.mayLoad()) {
641       MIB.add(UseMI->getOperand(0));
642       MIB.add(AddAslMI->getOperand(2));
643       MIB.add(AddAslMI->getOperand(3));
644       const GlobalValue *GV = ImmOp.getGlobal();
645       MIB.addGlobalAddress(GV, UseMI->getOperand(2).getImm()+ImmOp.getOffset(),
646                            ImmOp.getTargetFlags());
647       OpStart = 3;
648     } else if (UseMID.mayStore()) {
649       MIB.add(AddAslMI->getOperand(2));
650       MIB.add(AddAslMI->getOperand(3));
651       const GlobalValue *GV = ImmOp.getGlobal();
652       MIB.addGlobalAddress(GV, UseMI->getOperand(1).getImm()+ImmOp.getOffset(),
653                            ImmOp.getTargetFlags());
654       MIB.add(UseMI->getOperand(2));
655       OpStart = 3;
656     } else
657       llvm_unreachable("Unhandled instruction");
658 
659     for (unsigned i = OpStart; i < OpEnd; ++i)
660       MIB.add(UseMI->getOperand(i));
661 
662     Deleted.insert(UseMI);
663   }
664 
665   return true;
666 }
667 
668 bool HexagonOptAddrMode::xformUseMI(MachineInstr *TfrMI, MachineInstr *UseMI,
669                                     NodeAddr<UseNode *> UseN,
670                                     unsigned UseMOnum) {
671   const MachineOperand ImmOp = TfrMI->getOperand(1);
672   const MCInstrDesc &MID = UseMI->getDesc();
673   unsigned Changed = false;
674   if (MID.mayLoad())
675     Changed = changeLoad(UseMI, ImmOp, UseMOnum);
676   else if (MID.mayStore())
677     Changed = changeStore(UseMI, ImmOp, UseMOnum);
678   else if (UseMI->getOpcode() == Hexagon::S2_addasl_rrri)
679     Changed = changeAddAsl(UseN, UseMI, ImmOp, UseMOnum);
680 
681   if (Changed)
682     Deleted.insert(UseMI);
683 
684   return Changed;
685 }
686 
687 bool HexagonOptAddrMode::processBlock(NodeAddr<BlockNode *> BA) {
688   bool Changed = false;
689 
690   for (auto IA : BA.Addr->members(*DFG)) {
691     if (!DFG->IsCode<NodeAttrs::Stmt>(IA))
692       continue;
693 
694     NodeAddr<StmtNode *> SA = IA;
695     MachineInstr *MI = SA.Addr->getCode();
696     if ((MI->getOpcode() != Hexagon::A2_tfrsi ||
697          !MI->getOperand(1).isGlobal()) &&
698         (MI->getOpcode() != Hexagon::A2_addi ||
699          !MI->getOperand(2).isImm() || HII->isConstExtended(*MI)))
700     continue;
701 
702     LLVM_DEBUG(dbgs() << "[Analyzing " << HII->getName(MI->getOpcode())
703                       << "]: " << *MI << "\n\t[InstrNode]: "
704                       << Print<NodeAddr<InstrNode *>>(IA, *DFG) << '\n');
705 
706     NodeList UNodeList;
707     getAllRealUses(SA, UNodeList);
708 
709     if (!allValidCandidates(SA, UNodeList))
710       continue;
711 
712     // Analyze all uses of 'add'. If the output of 'add' is used as an address
713     // in the base+immediate addressing mode load/store instructions, see if
714     // they can be updated to use the immediate value as an offet. Thus,
715     // providing us the opportunity to eliminate 'add'.
716     // Ex: Rx= add(Rt,#12)
717     //     memw(Rx+#0) = Rs
718     // This can be replaced with memw(Rt+#12) = Rs
719     //
720     // This transformation is only performed if all uses can be updated and
721     // the offset isn't required to be constant extended.
722     if (MI->getOpcode() == Hexagon::A2_addi) {
723       Changed |= processAddUses(SA, MI, UNodeList);
724       continue;
725     }
726 
727     short SizeInc = 0;
728     Register DefR = MI->getOperand(0).getReg();
729     InstrEvalMap InstrEvalResult;
730 
731     // Analyze all uses and calculate increase in size. Perform the optimization
732     // only if there is no increase in size.
733     if (!analyzeUses(DefR, UNodeList, InstrEvalResult, SizeInc))
734       continue;
735     if (SizeInc > CodeGrowthLimit)
736       continue;
737 
738     bool KeepTfr = false;
739 
740     LLVM_DEBUG(dbgs() << "\t[Total reached uses] : " << UNodeList.size()
741                       << "\n");
742     LLVM_DEBUG(dbgs() << "\t[Processing Reached Uses] ===\n");
743     for (auto I = UNodeList.rbegin(), E = UNodeList.rend(); I != E; ++I) {
744       NodeAddr<UseNode *> UseN = *I;
745       assert(!(UseN.Addr->getFlags() & NodeAttrs::PhiRef) &&
746              "Found a PhiRef node as a real reached use!!");
747 
748       NodeAddr<StmtNode *> OwnerN = UseN.Addr->getOwner(*DFG);
749       MachineInstr *UseMI = OwnerN.Addr->getCode();
750       LLVM_DEBUG(dbgs() << "\t\t[MI <" << printMBBReference(*UseMI->getParent())
751                         << ">]: " << *UseMI << "\n");
752 
753       int UseMOnum = -1;
754       unsigned NumOperands = UseMI->getNumOperands();
755       for (unsigned j = 0; j < NumOperands - 1; ++j) {
756         const MachineOperand &op = UseMI->getOperand(j);
757         if (op.isReg() && op.isUse() && DefR == op.getReg())
758           UseMOnum = j;
759       }
760       // It is possible that the register will not be found in any operand.
761       // This could happen, for example, when DefR = R4, but the used
762       // register is D2.
763 
764       // Change UseMI if replacement is possible. If any replacement failed,
765       // or wasn't attempted, make sure to keep the TFR.
766       bool Xformed = false;
767       if (UseMOnum >= 0 && InstrEvalResult[UseMI])
768         Xformed = xformUseMI(MI, UseMI, UseN, UseMOnum);
769       Changed |=  Xformed;
770       KeepTfr |= !Xformed;
771     }
772     if (!KeepTfr)
773       Deleted.insert(MI);
774   }
775   return Changed;
776 }
777 
778 bool HexagonOptAddrMode::runOnMachineFunction(MachineFunction &MF) {
779   if (skipFunction(MF.getFunction()))
780     return false;
781 
782   bool Changed = false;
783   auto &HST = MF.getSubtarget<HexagonSubtarget>();
784   MRI = &MF.getRegInfo();
785   HII = HST.getInstrInfo();
786   HRI = HST.getRegisterInfo();
787   const auto &MDF = getAnalysis<MachineDominanceFrontier>();
788   MDT = &getAnalysis<MachineDominatorTree>();
789   const TargetOperandInfo TOI(*HII);
790 
791   DataFlowGraph G(MF, *HII, *HRI, *MDT, MDF, TOI);
792   // Need to keep dead phis because we can propagate uses of registers into
793   // nodes dominated by those would-be phis.
794   G.build(BuildOptions::KeepDeadPhis);
795   DFG = &G;
796 
797   Liveness L(*MRI, *DFG);
798   L.computePhiInfo();
799   LV = &L;
800 
801   Deleted.clear();
802   NodeAddr<FuncNode *> FA = DFG->getFunc();
803   LLVM_DEBUG(dbgs() << "==== [RefMap#]=====:\n "
804                     << Print<NodeAddr<FuncNode *>>(FA, *DFG) << "\n");
805 
806   for (NodeAddr<BlockNode *> BA : FA.Addr->members(*DFG))
807     Changed |= processBlock(BA);
808 
809   for (auto MI : Deleted)
810     MI->eraseFromParent();
811 
812   if (Changed) {
813     G.build();
814     L.computeLiveIns();
815     L.resetLiveIns();
816     L.resetKills();
817   }
818 
819   return Changed;
820 }
821 
822 //===----------------------------------------------------------------------===//
823 //                         Public Constructor Functions
824 //===----------------------------------------------------------------------===//
825 
826 FunctionPass *llvm::createHexagonOptAddrMode() {
827   return new HexagonOptAddrMode();
828 }
829