xref: /freebsd-src/contrib/llvm-project/llvm/lib/CodeGen/ExecutionDomainFix.cpp (revision 06c3fb2749bda94cb5201f81ffdb8fa6c3161b2e)
10b57cec5SDimitry Andric //===- ExecutionDomainFix.cpp - Fix execution domain issues ----*- C++ -*--===//
20b57cec5SDimitry Andric //
30b57cec5SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
40b57cec5SDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
50b57cec5SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
60b57cec5SDimitry Andric //
70b57cec5SDimitry Andric //===----------------------------------------------------------------------===//
80b57cec5SDimitry Andric 
90b57cec5SDimitry Andric #include "llvm/CodeGen/ExecutionDomainFix.h"
100b57cec5SDimitry Andric #include "llvm/CodeGen/MachineRegisterInfo.h"
110b57cec5SDimitry Andric #include "llvm/CodeGen/TargetInstrInfo.h"
128bcb0991SDimitry Andric #include "llvm/Support/Debug.h"
130b57cec5SDimitry Andric 
140b57cec5SDimitry Andric using namespace llvm;
150b57cec5SDimitry Andric 
160b57cec5SDimitry Andric #define DEBUG_TYPE "execution-deps-fix"
170b57cec5SDimitry Andric 
180b57cec5SDimitry Andric iterator_range<SmallVectorImpl<int>::const_iterator>
regIndices(unsigned Reg) const190b57cec5SDimitry Andric ExecutionDomainFix::regIndices(unsigned Reg) const {
200b57cec5SDimitry Andric   assert(Reg < AliasMap.size() && "Invalid register");
210b57cec5SDimitry Andric   const auto &Entry = AliasMap[Reg];
220b57cec5SDimitry Andric   return make_range(Entry.begin(), Entry.end());
230b57cec5SDimitry Andric }
240b57cec5SDimitry Andric 
alloc(int domain)250b57cec5SDimitry Andric DomainValue *ExecutionDomainFix::alloc(int domain) {
260b57cec5SDimitry Andric   DomainValue *dv = Avail.empty() ? new (Allocator.Allocate()) DomainValue
270b57cec5SDimitry Andric                                   : Avail.pop_back_val();
280b57cec5SDimitry Andric   if (domain >= 0)
290b57cec5SDimitry Andric     dv->addDomain(domain);
300b57cec5SDimitry Andric   assert(dv->Refs == 0 && "Reference count wasn't cleared");
310b57cec5SDimitry Andric   assert(!dv->Next && "Chained DomainValue shouldn't have been recycled");
320b57cec5SDimitry Andric   return dv;
330b57cec5SDimitry Andric }
340b57cec5SDimitry Andric 
release(DomainValue * DV)350b57cec5SDimitry Andric void ExecutionDomainFix::release(DomainValue *DV) {
360b57cec5SDimitry Andric   while (DV) {
370b57cec5SDimitry Andric     assert(DV->Refs && "Bad DomainValue");
380b57cec5SDimitry Andric     if (--DV->Refs)
390b57cec5SDimitry Andric       return;
400b57cec5SDimitry Andric 
410b57cec5SDimitry Andric     // There are no more DV references. Collapse any contained instructions.
420b57cec5SDimitry Andric     if (DV->AvailableDomains && !DV->isCollapsed())
430b57cec5SDimitry Andric       collapse(DV, DV->getFirstDomain());
440b57cec5SDimitry Andric 
450b57cec5SDimitry Andric     DomainValue *Next = DV->Next;
460b57cec5SDimitry Andric     DV->clear();
470b57cec5SDimitry Andric     Avail.push_back(DV);
480b57cec5SDimitry Andric     // Also release the next DomainValue in the chain.
490b57cec5SDimitry Andric     DV = Next;
500b57cec5SDimitry Andric   }
510b57cec5SDimitry Andric }
520b57cec5SDimitry Andric 
resolve(DomainValue * & DVRef)530b57cec5SDimitry Andric DomainValue *ExecutionDomainFix::resolve(DomainValue *&DVRef) {
540b57cec5SDimitry Andric   DomainValue *DV = DVRef;
550b57cec5SDimitry Andric   if (!DV || !DV->Next)
560b57cec5SDimitry Andric     return DV;
570b57cec5SDimitry Andric 
580b57cec5SDimitry Andric   // DV has a chain. Find the end.
590b57cec5SDimitry Andric   do
600b57cec5SDimitry Andric     DV = DV->Next;
610b57cec5SDimitry Andric   while (DV->Next);
620b57cec5SDimitry Andric 
630b57cec5SDimitry Andric   // Update DVRef to point to DV.
640b57cec5SDimitry Andric   retain(DV);
650b57cec5SDimitry Andric   release(DVRef);
660b57cec5SDimitry Andric   DVRef = DV;
670b57cec5SDimitry Andric   return DV;
680b57cec5SDimitry Andric }
690b57cec5SDimitry Andric 
setLiveReg(int rx,DomainValue * dv)700b57cec5SDimitry Andric void ExecutionDomainFix::setLiveReg(int rx, DomainValue *dv) {
710b57cec5SDimitry Andric   assert(unsigned(rx) < NumRegs && "Invalid index");
720b57cec5SDimitry Andric   assert(!LiveRegs.empty() && "Must enter basic block first.");
730b57cec5SDimitry Andric 
740b57cec5SDimitry Andric   if (LiveRegs[rx] == dv)
750b57cec5SDimitry Andric     return;
760b57cec5SDimitry Andric   if (LiveRegs[rx])
770b57cec5SDimitry Andric     release(LiveRegs[rx]);
780b57cec5SDimitry Andric   LiveRegs[rx] = retain(dv);
790b57cec5SDimitry Andric }
800b57cec5SDimitry Andric 
kill(int rx)810b57cec5SDimitry Andric void ExecutionDomainFix::kill(int rx) {
820b57cec5SDimitry Andric   assert(unsigned(rx) < NumRegs && "Invalid index");
830b57cec5SDimitry Andric   assert(!LiveRegs.empty() && "Must enter basic block first.");
840b57cec5SDimitry Andric   if (!LiveRegs[rx])
850b57cec5SDimitry Andric     return;
860b57cec5SDimitry Andric 
870b57cec5SDimitry Andric   release(LiveRegs[rx]);
880b57cec5SDimitry Andric   LiveRegs[rx] = nullptr;
890b57cec5SDimitry Andric }
900b57cec5SDimitry Andric 
force(int rx,unsigned domain)910b57cec5SDimitry Andric void ExecutionDomainFix::force(int rx, unsigned domain) {
920b57cec5SDimitry Andric   assert(unsigned(rx) < NumRegs && "Invalid index");
930b57cec5SDimitry Andric   assert(!LiveRegs.empty() && "Must enter basic block first.");
940b57cec5SDimitry Andric   if (DomainValue *dv = LiveRegs[rx]) {
950b57cec5SDimitry Andric     if (dv->isCollapsed())
960b57cec5SDimitry Andric       dv->addDomain(domain);
970b57cec5SDimitry Andric     else if (dv->hasDomain(domain))
980b57cec5SDimitry Andric       collapse(dv, domain);
990b57cec5SDimitry Andric     else {
1000b57cec5SDimitry Andric       // This is an incompatible open DomainValue. Collapse it to whatever and
1010b57cec5SDimitry Andric       // force the new value into domain. This costs a domain crossing.
1020b57cec5SDimitry Andric       collapse(dv, dv->getFirstDomain());
1030b57cec5SDimitry Andric       assert(LiveRegs[rx] && "Not live after collapse?");
1040b57cec5SDimitry Andric       LiveRegs[rx]->addDomain(domain);
1050b57cec5SDimitry Andric     }
1060b57cec5SDimitry Andric   } else {
1070b57cec5SDimitry Andric     // Set up basic collapsed DomainValue.
1080b57cec5SDimitry Andric     setLiveReg(rx, alloc(domain));
1090b57cec5SDimitry Andric   }
1100b57cec5SDimitry Andric }
1110b57cec5SDimitry Andric 
collapse(DomainValue * dv,unsigned domain)1120b57cec5SDimitry Andric void ExecutionDomainFix::collapse(DomainValue *dv, unsigned domain) {
1130b57cec5SDimitry Andric   assert(dv->hasDomain(domain) && "Cannot collapse");
1140b57cec5SDimitry Andric 
1150b57cec5SDimitry Andric   // Collapse all the instructions.
1160b57cec5SDimitry Andric   while (!dv->Instrs.empty())
1170b57cec5SDimitry Andric     TII->setExecutionDomain(*dv->Instrs.pop_back_val(), domain);
1180b57cec5SDimitry Andric   dv->setSingleDomain(domain);
1190b57cec5SDimitry Andric 
1200b57cec5SDimitry Andric   // If there are multiple users, give them new, unique DomainValues.
1210b57cec5SDimitry Andric   if (!LiveRegs.empty() && dv->Refs > 1)
1220b57cec5SDimitry Andric     for (unsigned rx = 0; rx != NumRegs; ++rx)
1230b57cec5SDimitry Andric       if (LiveRegs[rx] == dv)
1240b57cec5SDimitry Andric         setLiveReg(rx, alloc(domain));
1250b57cec5SDimitry Andric }
1260b57cec5SDimitry Andric 
merge(DomainValue * A,DomainValue * B)1270b57cec5SDimitry Andric bool ExecutionDomainFix::merge(DomainValue *A, DomainValue *B) {
1280b57cec5SDimitry Andric   assert(!A->isCollapsed() && "Cannot merge into collapsed");
1290b57cec5SDimitry Andric   assert(!B->isCollapsed() && "Cannot merge from collapsed");
1300b57cec5SDimitry Andric   if (A == B)
1310b57cec5SDimitry Andric     return true;
1320b57cec5SDimitry Andric   // Restrict to the domains that A and B have in common.
1330b57cec5SDimitry Andric   unsigned common = A->getCommonDomains(B->AvailableDomains);
1340b57cec5SDimitry Andric   if (!common)
1350b57cec5SDimitry Andric     return false;
1360b57cec5SDimitry Andric   A->AvailableDomains = common;
1370b57cec5SDimitry Andric   A->Instrs.append(B->Instrs.begin(), B->Instrs.end());
1380b57cec5SDimitry Andric 
1390b57cec5SDimitry Andric   // Clear the old DomainValue so we won't try to swizzle instructions twice.
1400b57cec5SDimitry Andric   B->clear();
1410b57cec5SDimitry Andric   // All uses of B are referred to A.
1420b57cec5SDimitry Andric   B->Next = retain(A);
1430b57cec5SDimitry Andric 
1440b57cec5SDimitry Andric   for (unsigned rx = 0; rx != NumRegs; ++rx) {
1450b57cec5SDimitry Andric     assert(!LiveRegs.empty() && "no space allocated for live registers");
1460b57cec5SDimitry Andric     if (LiveRegs[rx] == B)
1470b57cec5SDimitry Andric       setLiveReg(rx, A);
1480b57cec5SDimitry Andric   }
1490b57cec5SDimitry Andric   return true;
1500b57cec5SDimitry Andric }
1510b57cec5SDimitry Andric 
enterBasicBlock(const LoopTraversal::TraversedMBBInfo & TraversedMBB)1520b57cec5SDimitry Andric void ExecutionDomainFix::enterBasicBlock(
1530b57cec5SDimitry Andric     const LoopTraversal::TraversedMBBInfo &TraversedMBB) {
1540b57cec5SDimitry Andric 
1550b57cec5SDimitry Andric   MachineBasicBlock *MBB = TraversedMBB.MBB;
1560b57cec5SDimitry Andric 
1570b57cec5SDimitry Andric   // Set up LiveRegs to represent registers entering MBB.
1580b57cec5SDimitry Andric   // Set default domain values to 'no domain' (nullptr)
1590b57cec5SDimitry Andric   if (LiveRegs.empty())
1600b57cec5SDimitry Andric     LiveRegs.assign(NumRegs, nullptr);
1610b57cec5SDimitry Andric 
1620b57cec5SDimitry Andric   // This is the entry block.
1630b57cec5SDimitry Andric   if (MBB->pred_empty()) {
1640b57cec5SDimitry Andric     LLVM_DEBUG(dbgs() << printMBBReference(*MBB) << ": entry\n");
1650b57cec5SDimitry Andric     return;
1660b57cec5SDimitry Andric   }
1670b57cec5SDimitry Andric 
1680b57cec5SDimitry Andric   // Try to coalesce live-out registers from predecessors.
1690b57cec5SDimitry Andric   for (MachineBasicBlock *pred : MBB->predecessors()) {
1700b57cec5SDimitry Andric     assert(unsigned(pred->getNumber()) < MBBOutRegsInfos.size() &&
1710b57cec5SDimitry Andric            "Should have pre-allocated MBBInfos for all MBBs");
1720b57cec5SDimitry Andric     LiveRegsDVInfo &Incoming = MBBOutRegsInfos[pred->getNumber()];
1730b57cec5SDimitry Andric     // Incoming is null if this is a backedge from a BB
1740b57cec5SDimitry Andric     // we haven't processed yet
1750b57cec5SDimitry Andric     if (Incoming.empty())
1760b57cec5SDimitry Andric       continue;
1770b57cec5SDimitry Andric 
1780b57cec5SDimitry Andric     for (unsigned rx = 0; rx != NumRegs; ++rx) {
1790b57cec5SDimitry Andric       DomainValue *pdv = resolve(Incoming[rx]);
1800b57cec5SDimitry Andric       if (!pdv)
1810b57cec5SDimitry Andric         continue;
1820b57cec5SDimitry Andric       if (!LiveRegs[rx]) {
1830b57cec5SDimitry Andric         setLiveReg(rx, pdv);
1840b57cec5SDimitry Andric         continue;
1850b57cec5SDimitry Andric       }
1860b57cec5SDimitry Andric 
1870b57cec5SDimitry Andric       // We have a live DomainValue from more than one predecessor.
1880b57cec5SDimitry Andric       if (LiveRegs[rx]->isCollapsed()) {
1890b57cec5SDimitry Andric         // We are already collapsed, but predecessor is not. Force it.
1900b57cec5SDimitry Andric         unsigned Domain = LiveRegs[rx]->getFirstDomain();
1910b57cec5SDimitry Andric         if (!pdv->isCollapsed() && pdv->hasDomain(Domain))
1920b57cec5SDimitry Andric           collapse(pdv, Domain);
1930b57cec5SDimitry Andric         continue;
1940b57cec5SDimitry Andric       }
1950b57cec5SDimitry Andric 
1960b57cec5SDimitry Andric       // Currently open, merge in predecessor.
1970b57cec5SDimitry Andric       if (!pdv->isCollapsed())
1980b57cec5SDimitry Andric         merge(LiveRegs[rx], pdv);
1990b57cec5SDimitry Andric       else
2000b57cec5SDimitry Andric         force(rx, pdv->getFirstDomain());
2010b57cec5SDimitry Andric     }
2020b57cec5SDimitry Andric   }
2030b57cec5SDimitry Andric   LLVM_DEBUG(dbgs() << printMBBReference(*MBB)
2040b57cec5SDimitry Andric                     << (!TraversedMBB.IsDone ? ": incomplete\n"
2050b57cec5SDimitry Andric                                              : ": all preds known\n"));
2060b57cec5SDimitry Andric }
2070b57cec5SDimitry Andric 
leaveBasicBlock(const LoopTraversal::TraversedMBBInfo & TraversedMBB)2080b57cec5SDimitry Andric void ExecutionDomainFix::leaveBasicBlock(
2090b57cec5SDimitry Andric     const LoopTraversal::TraversedMBBInfo &TraversedMBB) {
2100b57cec5SDimitry Andric   assert(!LiveRegs.empty() && "Must enter basic block first.");
2110b57cec5SDimitry Andric   unsigned MBBNumber = TraversedMBB.MBB->getNumber();
2120b57cec5SDimitry Andric   assert(MBBNumber < MBBOutRegsInfos.size() &&
2130b57cec5SDimitry Andric          "Unexpected basic block number.");
2140b57cec5SDimitry Andric   // Save register clearances at end of MBB - used by enterBasicBlock().
2150b57cec5SDimitry Andric   for (DomainValue *OldLiveReg : MBBOutRegsInfos[MBBNumber]) {
2160b57cec5SDimitry Andric     release(OldLiveReg);
2170b57cec5SDimitry Andric   }
2180b57cec5SDimitry Andric   MBBOutRegsInfos[MBBNumber] = LiveRegs;
2190b57cec5SDimitry Andric   LiveRegs.clear();
2200b57cec5SDimitry Andric }
2210b57cec5SDimitry Andric 
visitInstr(MachineInstr * MI)2220b57cec5SDimitry Andric bool ExecutionDomainFix::visitInstr(MachineInstr *MI) {
2230b57cec5SDimitry Andric   // Update instructions with explicit execution domains.
2240b57cec5SDimitry Andric   std::pair<uint16_t, uint16_t> DomP = TII->getExecutionDomain(*MI);
2250b57cec5SDimitry Andric   if (DomP.first) {
2260b57cec5SDimitry Andric     if (DomP.second)
2270b57cec5SDimitry Andric       visitSoftInstr(MI, DomP.second);
2280b57cec5SDimitry Andric     else
2290b57cec5SDimitry Andric       visitHardInstr(MI, DomP.first);
2300b57cec5SDimitry Andric   }
2310b57cec5SDimitry Andric 
2320b57cec5SDimitry Andric   return !DomP.first;
2330b57cec5SDimitry Andric }
2340b57cec5SDimitry Andric 
processDefs(MachineInstr * MI,bool Kill)2350b57cec5SDimitry Andric void ExecutionDomainFix::processDefs(MachineInstr *MI, bool Kill) {
2360b57cec5SDimitry Andric   assert(!MI->isDebugInstr() && "Won't process debug values");
2370b57cec5SDimitry Andric   const MCInstrDesc &MCID = MI->getDesc();
2380b57cec5SDimitry Andric   for (unsigned i = 0,
2390b57cec5SDimitry Andric                 e = MI->isVariadic() ? MI->getNumOperands() : MCID.getNumDefs();
2400b57cec5SDimitry Andric        i != e; ++i) {
2410b57cec5SDimitry Andric     MachineOperand &MO = MI->getOperand(i);
2420b57cec5SDimitry Andric     if (!MO.isReg())
2430b57cec5SDimitry Andric       continue;
2440b57cec5SDimitry Andric     if (MO.isUse())
2450b57cec5SDimitry Andric       continue;
2460b57cec5SDimitry Andric     for (int rx : regIndices(MO.getReg())) {
2470b57cec5SDimitry Andric       // This instruction explicitly defines rx.
2480b57cec5SDimitry Andric       LLVM_DEBUG(dbgs() << printReg(RC->getRegister(rx), TRI) << ":\t" << *MI);
2490b57cec5SDimitry Andric 
2500b57cec5SDimitry Andric       // Kill off domains redefined by generic instructions.
2510b57cec5SDimitry Andric       if (Kill)
2520b57cec5SDimitry Andric         kill(rx);
2530b57cec5SDimitry Andric     }
2540b57cec5SDimitry Andric   }
2550b57cec5SDimitry Andric }
2560b57cec5SDimitry Andric 
visitHardInstr(MachineInstr * mi,unsigned domain)2570b57cec5SDimitry Andric void ExecutionDomainFix::visitHardInstr(MachineInstr *mi, unsigned domain) {
2580b57cec5SDimitry Andric   // Collapse all uses.
2590b57cec5SDimitry Andric   for (unsigned i = mi->getDesc().getNumDefs(),
2600b57cec5SDimitry Andric                 e = mi->getDesc().getNumOperands();
2610b57cec5SDimitry Andric        i != e; ++i) {
2620b57cec5SDimitry Andric     MachineOperand &mo = mi->getOperand(i);
2630b57cec5SDimitry Andric     if (!mo.isReg())
2640b57cec5SDimitry Andric       continue;
2650b57cec5SDimitry Andric     for (int rx : regIndices(mo.getReg())) {
2660b57cec5SDimitry Andric       force(rx, domain);
2670b57cec5SDimitry Andric     }
2680b57cec5SDimitry Andric   }
2690b57cec5SDimitry Andric 
2700b57cec5SDimitry Andric   // Kill all defs and force them.
2710b57cec5SDimitry Andric   for (unsigned i = 0, e = mi->getDesc().getNumDefs(); i != e; ++i) {
2720b57cec5SDimitry Andric     MachineOperand &mo = mi->getOperand(i);
2730b57cec5SDimitry Andric     if (!mo.isReg())
2740b57cec5SDimitry Andric       continue;
2750b57cec5SDimitry Andric     for (int rx : regIndices(mo.getReg())) {
2760b57cec5SDimitry Andric       kill(rx);
2770b57cec5SDimitry Andric       force(rx, domain);
2780b57cec5SDimitry Andric     }
2790b57cec5SDimitry Andric   }
2800b57cec5SDimitry Andric }
2810b57cec5SDimitry Andric 
visitSoftInstr(MachineInstr * mi,unsigned mask)2820b57cec5SDimitry Andric void ExecutionDomainFix::visitSoftInstr(MachineInstr *mi, unsigned mask) {
2830b57cec5SDimitry Andric   // Bitmask of available domains for this instruction after taking collapsed
2840b57cec5SDimitry Andric   // operands into account.
2850b57cec5SDimitry Andric   unsigned available = mask;
2860b57cec5SDimitry Andric 
2870b57cec5SDimitry Andric   // Scan the explicit use operands for incoming domains.
2880b57cec5SDimitry Andric   SmallVector<int, 4> used;
2890b57cec5SDimitry Andric   if (!LiveRegs.empty())
2900b57cec5SDimitry Andric     for (unsigned i = mi->getDesc().getNumDefs(),
2910b57cec5SDimitry Andric                   e = mi->getDesc().getNumOperands();
2920b57cec5SDimitry Andric          i != e; ++i) {
2930b57cec5SDimitry Andric       MachineOperand &mo = mi->getOperand(i);
2940b57cec5SDimitry Andric       if (!mo.isReg())
2950b57cec5SDimitry Andric         continue;
2960b57cec5SDimitry Andric       for (int rx : regIndices(mo.getReg())) {
2970b57cec5SDimitry Andric         DomainValue *dv = LiveRegs[rx];
2980b57cec5SDimitry Andric         if (dv == nullptr)
2990b57cec5SDimitry Andric           continue;
3000b57cec5SDimitry Andric         // Bitmask of domains that dv and available have in common.
3010b57cec5SDimitry Andric         unsigned common = dv->getCommonDomains(available);
3020b57cec5SDimitry Andric         // Is it possible to use this collapsed register for free?
3030b57cec5SDimitry Andric         if (dv->isCollapsed()) {
3040b57cec5SDimitry Andric           // Restrict available domains to the ones in common with the operand.
3050b57cec5SDimitry Andric           // If there are no common domains, we must pay the cross-domain
3060b57cec5SDimitry Andric           // penalty for this operand.
3070b57cec5SDimitry Andric           if (common)
3080b57cec5SDimitry Andric             available = common;
3090b57cec5SDimitry Andric         } else if (common)
3100b57cec5SDimitry Andric           // Open DomainValue is compatible, save it for merging.
3110b57cec5SDimitry Andric           used.push_back(rx);
3120b57cec5SDimitry Andric         else
3130b57cec5SDimitry Andric           // Open DomainValue is not compatible with instruction. It is useless
3140b57cec5SDimitry Andric           // now.
3150b57cec5SDimitry Andric           kill(rx);
3160b57cec5SDimitry Andric       }
3170b57cec5SDimitry Andric     }
3180b57cec5SDimitry Andric 
3190b57cec5SDimitry Andric   // If the collapsed operands force a single domain, propagate the collapse.
3200b57cec5SDimitry Andric   if (isPowerOf2_32(available)) {
321*06c3fb27SDimitry Andric     unsigned domain = llvm::countr_zero(available);
3220b57cec5SDimitry Andric     TII->setExecutionDomain(*mi, domain);
3230b57cec5SDimitry Andric     visitHardInstr(mi, domain);
3240b57cec5SDimitry Andric     return;
3250b57cec5SDimitry Andric   }
3260b57cec5SDimitry Andric 
3270b57cec5SDimitry Andric   // Kill off any remaining uses that don't match available, and build a list of
3280b57cec5SDimitry Andric   // incoming DomainValues that we want to merge.
3290b57cec5SDimitry Andric   SmallVector<int, 4> Regs;
3300b57cec5SDimitry Andric   for (int rx : used) {
3310b57cec5SDimitry Andric     assert(!LiveRegs.empty() && "no space allocated for live registers");
3320b57cec5SDimitry Andric     DomainValue *&LR = LiveRegs[rx];
3330b57cec5SDimitry Andric     // This useless DomainValue could have been missed above.
3340b57cec5SDimitry Andric     if (!LR->getCommonDomains(available)) {
3350b57cec5SDimitry Andric       kill(rx);
3360b57cec5SDimitry Andric       continue;
3370b57cec5SDimitry Andric     }
3380b57cec5SDimitry Andric     // Sorted insertion.
3390b57cec5SDimitry Andric     // Enables giving priority to the latest domains during merging.
3400b57cec5SDimitry Andric     const int Def = RDA->getReachingDef(mi, RC->getRegister(rx));
3410b57cec5SDimitry Andric     auto I = partition_point(Regs, [&](int I) {
3420b57cec5SDimitry Andric       return RDA->getReachingDef(mi, RC->getRegister(I)) <= Def;
3430b57cec5SDimitry Andric     });
3440b57cec5SDimitry Andric     Regs.insert(I, rx);
3450b57cec5SDimitry Andric   }
3460b57cec5SDimitry Andric 
3470b57cec5SDimitry Andric   // doms are now sorted in order of appearance. Try to merge them all, giving
3480b57cec5SDimitry Andric   // priority to the latest ones.
3490b57cec5SDimitry Andric   DomainValue *dv = nullptr;
3500b57cec5SDimitry Andric   while (!Regs.empty()) {
3510b57cec5SDimitry Andric     if (!dv) {
3520b57cec5SDimitry Andric       dv = LiveRegs[Regs.pop_back_val()];
3530b57cec5SDimitry Andric       // Force the first dv to match the current instruction.
3540b57cec5SDimitry Andric       dv->AvailableDomains = dv->getCommonDomains(available);
3550b57cec5SDimitry Andric       assert(dv->AvailableDomains && "Domain should have been filtered");
3560b57cec5SDimitry Andric       continue;
3570b57cec5SDimitry Andric     }
3580b57cec5SDimitry Andric 
3590b57cec5SDimitry Andric     DomainValue *Latest = LiveRegs[Regs.pop_back_val()];
3600b57cec5SDimitry Andric     // Skip already merged values.
3610b57cec5SDimitry Andric     if (Latest == dv || Latest->Next)
3620b57cec5SDimitry Andric       continue;
3630b57cec5SDimitry Andric     if (merge(dv, Latest))
3640b57cec5SDimitry Andric       continue;
3650b57cec5SDimitry Andric 
3660b57cec5SDimitry Andric     // If latest didn't merge, it is useless now. Kill all registers using it.
3670b57cec5SDimitry Andric     for (int i : used) {
3680b57cec5SDimitry Andric       assert(!LiveRegs.empty() && "no space allocated for live registers");
3690b57cec5SDimitry Andric       if (LiveRegs[i] == Latest)
3700b57cec5SDimitry Andric         kill(i);
3710b57cec5SDimitry Andric     }
3720b57cec5SDimitry Andric   }
3730b57cec5SDimitry Andric 
3740b57cec5SDimitry Andric   // dv is the DomainValue we are going to use for this instruction.
3750b57cec5SDimitry Andric   if (!dv) {
3760b57cec5SDimitry Andric     dv = alloc();
3770b57cec5SDimitry Andric     dv->AvailableDomains = available;
3780b57cec5SDimitry Andric   }
3790b57cec5SDimitry Andric   dv->Instrs.push_back(mi);
3800b57cec5SDimitry Andric 
3810b57cec5SDimitry Andric   // Finally set all defs and non-collapsed uses to dv. We must iterate through
3820b57cec5SDimitry Andric   // all the operators, including imp-def ones.
383fe6060f1SDimitry Andric   for (const MachineOperand &mo : mi->operands()) {
3840b57cec5SDimitry Andric     if (!mo.isReg())
3850b57cec5SDimitry Andric       continue;
3860b57cec5SDimitry Andric     for (int rx : regIndices(mo.getReg())) {
3870b57cec5SDimitry Andric       if (!LiveRegs[rx] || (mo.isDef() && LiveRegs[rx] != dv)) {
3880b57cec5SDimitry Andric         kill(rx);
3890b57cec5SDimitry Andric         setLiveReg(rx, dv);
3900b57cec5SDimitry Andric       }
3910b57cec5SDimitry Andric     }
3920b57cec5SDimitry Andric   }
3930b57cec5SDimitry Andric }
3940b57cec5SDimitry Andric 
processBasicBlock(const LoopTraversal::TraversedMBBInfo & TraversedMBB)3950b57cec5SDimitry Andric void ExecutionDomainFix::processBasicBlock(
3960b57cec5SDimitry Andric     const LoopTraversal::TraversedMBBInfo &TraversedMBB) {
3970b57cec5SDimitry Andric   enterBasicBlock(TraversedMBB);
3980b57cec5SDimitry Andric   // If this block is not done, it makes little sense to make any decisions
3990b57cec5SDimitry Andric   // based on clearance information. We need to make a second pass anyway,
4000b57cec5SDimitry Andric   // and by then we'll have better information, so we can avoid doing the work
4010b57cec5SDimitry Andric   // to try and break dependencies now.
4020b57cec5SDimitry Andric   for (MachineInstr &MI : *TraversedMBB.MBB) {
4030b57cec5SDimitry Andric     if (!MI.isDebugInstr()) {
4040b57cec5SDimitry Andric       bool Kill = false;
4050b57cec5SDimitry Andric       if (TraversedMBB.PrimaryPass)
4060b57cec5SDimitry Andric         Kill = visitInstr(&MI);
4070b57cec5SDimitry Andric       processDefs(&MI, Kill);
4080b57cec5SDimitry Andric     }
4090b57cec5SDimitry Andric   }
4100b57cec5SDimitry Andric   leaveBasicBlock(TraversedMBB);
4110b57cec5SDimitry Andric }
4120b57cec5SDimitry Andric 
runOnMachineFunction(MachineFunction & mf)4130b57cec5SDimitry Andric bool ExecutionDomainFix::runOnMachineFunction(MachineFunction &mf) {
4140b57cec5SDimitry Andric   if (skipFunction(mf.getFunction()))
4150b57cec5SDimitry Andric     return false;
4160b57cec5SDimitry Andric   MF = &mf;
4170b57cec5SDimitry Andric   TII = MF->getSubtarget().getInstrInfo();
4180b57cec5SDimitry Andric   TRI = MF->getSubtarget().getRegisterInfo();
4190b57cec5SDimitry Andric   LiveRegs.clear();
4200b57cec5SDimitry Andric   assert(NumRegs == RC->getNumRegs() && "Bad regclass");
4210b57cec5SDimitry Andric 
4220b57cec5SDimitry Andric   LLVM_DEBUG(dbgs() << "********** FIX EXECUTION DOMAIN: "
4230b57cec5SDimitry Andric                     << TRI->getRegClassName(RC) << " **********\n");
4240b57cec5SDimitry Andric 
4250b57cec5SDimitry Andric   // If no relevant registers are used in the function, we can skip it
4260b57cec5SDimitry Andric   // completely.
4270b57cec5SDimitry Andric   bool anyregs = false;
4280b57cec5SDimitry Andric   const MachineRegisterInfo &MRI = mf.getRegInfo();
4290b57cec5SDimitry Andric   for (unsigned Reg : *RC) {
4300b57cec5SDimitry Andric     if (MRI.isPhysRegUsed(Reg)) {
4310b57cec5SDimitry Andric       anyregs = true;
4320b57cec5SDimitry Andric       break;
4330b57cec5SDimitry Andric     }
4340b57cec5SDimitry Andric   }
4350b57cec5SDimitry Andric   if (!anyregs)
4360b57cec5SDimitry Andric     return false;
4370b57cec5SDimitry Andric 
4380b57cec5SDimitry Andric   RDA = &getAnalysis<ReachingDefAnalysis>();
4390b57cec5SDimitry Andric 
4400b57cec5SDimitry Andric   // Initialize the AliasMap on the first use.
4410b57cec5SDimitry Andric   if (AliasMap.empty()) {
4420b57cec5SDimitry Andric     // Given a PhysReg, AliasMap[PhysReg] returns a list of indices into RC and
4430b57cec5SDimitry Andric     // therefore the LiveRegs array.
4440b57cec5SDimitry Andric     AliasMap.resize(TRI->getNumRegs());
4450b57cec5SDimitry Andric     for (unsigned i = 0, e = RC->getNumRegs(); i != e; ++i)
4460b57cec5SDimitry Andric       for (MCRegAliasIterator AI(RC->getRegister(i), TRI, true); AI.isValid();
4470b57cec5SDimitry Andric            ++AI)
4480b57cec5SDimitry Andric         AliasMap[*AI].push_back(i);
4490b57cec5SDimitry Andric   }
4500b57cec5SDimitry Andric 
4510b57cec5SDimitry Andric   // Initialize the MBBOutRegsInfos
4520b57cec5SDimitry Andric   MBBOutRegsInfos.resize(mf.getNumBlockIDs());
4530b57cec5SDimitry Andric 
4540b57cec5SDimitry Andric   // Traverse the basic blocks.
4550b57cec5SDimitry Andric   LoopTraversal Traversal;
4560b57cec5SDimitry Andric   LoopTraversal::TraversalOrder TraversedMBBOrder = Traversal.traverse(mf);
457fe6060f1SDimitry Andric   for (const LoopTraversal::TraversedMBBInfo &TraversedMBB : TraversedMBBOrder)
4580b57cec5SDimitry Andric     processBasicBlock(TraversedMBB);
4590b57cec5SDimitry Andric 
460fe6060f1SDimitry Andric   for (const LiveRegsDVInfo &OutLiveRegs : MBBOutRegsInfos)
461fe6060f1SDimitry Andric     for (DomainValue *OutLiveReg : OutLiveRegs)
4620b57cec5SDimitry Andric       if (OutLiveReg)
4630b57cec5SDimitry Andric         release(OutLiveReg);
464fe6060f1SDimitry Andric 
4650b57cec5SDimitry Andric   MBBOutRegsInfos.clear();
4660b57cec5SDimitry Andric   Avail.clear();
4670b57cec5SDimitry Andric   Allocator.DestroyAll();
4680b57cec5SDimitry Andric 
4690b57cec5SDimitry Andric   return false;
4700b57cec5SDimitry Andric }
471