1dff0c46cSDimitry Andric //===-- InterferenceCache.cpp - Caching per-block interference ---------*--===//
23b0f4066SDimitry Andric //
33b0f4066SDimitry Andric //                     The LLVM Compiler Infrastructure
43b0f4066SDimitry Andric //
53b0f4066SDimitry Andric // This file is distributed under the University of Illinois Open Source
63b0f4066SDimitry Andric // License. See LICENSE.TXT for details.
73b0f4066SDimitry Andric //
83b0f4066SDimitry Andric //===----------------------------------------------------------------------===//
93b0f4066SDimitry Andric //
103b0f4066SDimitry Andric // InterferenceCache remembers per-block interference in LiveIntervalUnions.
113b0f4066SDimitry Andric //
123b0f4066SDimitry Andric //===----------------------------------------------------------------------===//
133b0f4066SDimitry Andric 
143b0f4066SDimitry Andric #define DEBUG_TYPE "regalloc"
153b0f4066SDimitry Andric #include "InterferenceCache.h"
163b0f4066SDimitry Andric #include "llvm/Target/TargetRegisterInfo.h"
1717a519f9SDimitry Andric #include "llvm/Support/ErrorHandling.h"
18dff0c46cSDimitry Andric #include "llvm/CodeGen/LiveIntervalAnalysis.h"
193b0f4066SDimitry Andric 
203b0f4066SDimitry Andric using namespace llvm;
213b0f4066SDimitry Andric 
226122f3e6SDimitry Andric // Static member used for null interference cursors.
236122f3e6SDimitry Andric InterferenceCache::BlockInterference InterferenceCache::Cursor::NoInterference;
246122f3e6SDimitry Andric 
253b0f4066SDimitry Andric void InterferenceCache::init(MachineFunction *mf,
263b0f4066SDimitry Andric                              LiveIntervalUnion *liuarray,
273b0f4066SDimitry Andric                              SlotIndexes *indexes,
28dff0c46cSDimitry Andric                              LiveIntervals *lis,
293b0f4066SDimitry Andric                              const TargetRegisterInfo *tri) {
303b0f4066SDimitry Andric   MF = mf;
313b0f4066SDimitry Andric   LIUArray = liuarray;
323b0f4066SDimitry Andric   TRI = tri;
333b0f4066SDimitry Andric   PhysRegEntries.assign(TRI->getNumRegs(), 0);
343b0f4066SDimitry Andric   for (unsigned i = 0; i != CacheEntries; ++i)
35dff0c46cSDimitry Andric     Entries[i].clear(mf, indexes, lis);
363b0f4066SDimitry Andric }
373b0f4066SDimitry Andric 
383b0f4066SDimitry Andric InterferenceCache::Entry *InterferenceCache::get(unsigned PhysReg) {
393b0f4066SDimitry Andric   unsigned E = PhysRegEntries[PhysReg];
403b0f4066SDimitry Andric   if (E < CacheEntries && Entries[E].getPhysReg() == PhysReg) {
413b0f4066SDimitry Andric     if (!Entries[E].valid(LIUArray, TRI))
423b0f4066SDimitry Andric       Entries[E].revalidate();
433b0f4066SDimitry Andric     return &Entries[E];
443b0f4066SDimitry Andric   }
453b0f4066SDimitry Andric   // No valid entry exists, pick the next round-robin entry.
463b0f4066SDimitry Andric   E = RoundRobin;
473b0f4066SDimitry Andric   if (++RoundRobin == CacheEntries)
483b0f4066SDimitry Andric     RoundRobin = 0;
4917a519f9SDimitry Andric   for (unsigned i = 0; i != CacheEntries; ++i) {
5017a519f9SDimitry Andric     // Skip entries that are in use.
5117a519f9SDimitry Andric     if (Entries[E].hasRefs()) {
5217a519f9SDimitry Andric       if (++E == CacheEntries)
5317a519f9SDimitry Andric         E = 0;
5417a519f9SDimitry Andric       continue;
5517a519f9SDimitry Andric     }
563b0f4066SDimitry Andric     Entries[E].reset(PhysReg, LIUArray, TRI, MF);
573b0f4066SDimitry Andric     PhysRegEntries[PhysReg] = E;
583b0f4066SDimitry Andric     return &Entries[E];
593b0f4066SDimitry Andric   }
6017a519f9SDimitry Andric   llvm_unreachable("Ran out of interference cache entries.");
6117a519f9SDimitry Andric }
623b0f4066SDimitry Andric 
633b0f4066SDimitry Andric /// revalidate - LIU contents have changed, update tags.
643b0f4066SDimitry Andric void InterferenceCache::Entry::revalidate() {
653b0f4066SDimitry Andric   // Invalidate all block entries.
663b0f4066SDimitry Andric   ++Tag;
673b0f4066SDimitry Andric   // Invalidate all iterators.
683b0f4066SDimitry Andric   PrevPos = SlotIndex();
693b0f4066SDimitry Andric   for (unsigned i = 0, e = Aliases.size(); i != e; ++i)
703b0f4066SDimitry Andric     Aliases[i].second = Aliases[i].first->getTag();
713b0f4066SDimitry Andric }
723b0f4066SDimitry Andric 
733b0f4066SDimitry Andric void InterferenceCache::Entry::reset(unsigned physReg,
743b0f4066SDimitry Andric                                      LiveIntervalUnion *LIUArray,
753b0f4066SDimitry Andric                                      const TargetRegisterInfo *TRI,
763b0f4066SDimitry Andric                                      const MachineFunction *MF) {
7717a519f9SDimitry Andric   assert(!hasRefs() && "Cannot reset cache entry with references");
783b0f4066SDimitry Andric   // LIU's changed, invalidate cache.
793b0f4066SDimitry Andric   ++Tag;
803b0f4066SDimitry Andric   PhysReg = physReg;
813b0f4066SDimitry Andric   Blocks.resize(MF->getNumBlockIDs());
823b0f4066SDimitry Andric   Aliases.clear();
83dff0c46cSDimitry Andric   for (const uint16_t *AS = TRI->getOverlaps(PhysReg); *AS; ++AS) {
843b0f4066SDimitry Andric     LiveIntervalUnion *LIU = LIUArray + *AS;
853b0f4066SDimitry Andric     Aliases.push_back(std::make_pair(LIU, LIU->getTag()));
863b0f4066SDimitry Andric   }
873b0f4066SDimitry Andric 
883b0f4066SDimitry Andric   // Reset iterators.
893b0f4066SDimitry Andric   PrevPos = SlotIndex();
903b0f4066SDimitry Andric   unsigned e = Aliases.size();
913b0f4066SDimitry Andric   Iters.resize(e);
923b0f4066SDimitry Andric   for (unsigned i = 0; i != e; ++i)
933b0f4066SDimitry Andric     Iters[i].setMap(Aliases[i].first->getMap());
943b0f4066SDimitry Andric }
953b0f4066SDimitry Andric 
963b0f4066SDimitry Andric bool InterferenceCache::Entry::valid(LiveIntervalUnion *LIUArray,
973b0f4066SDimitry Andric                                      const TargetRegisterInfo *TRI) {
983b0f4066SDimitry Andric   unsigned i = 0, e = Aliases.size();
99dff0c46cSDimitry Andric   for (const uint16_t *AS = TRI->getOverlaps(PhysReg); *AS; ++AS, ++i) {
1003b0f4066SDimitry Andric     LiveIntervalUnion *LIU = LIUArray + *AS;
1013b0f4066SDimitry Andric     if (i == e ||  Aliases[i].first != LIU)
1023b0f4066SDimitry Andric       return false;
1033b0f4066SDimitry Andric     if (LIU->changedSince(Aliases[i].second))
1043b0f4066SDimitry Andric       return false;
1053b0f4066SDimitry Andric   }
1063b0f4066SDimitry Andric   return i == e;
1073b0f4066SDimitry Andric }
1083b0f4066SDimitry Andric 
1093b0f4066SDimitry Andric void InterferenceCache::Entry::update(unsigned MBBNum) {
1103b0f4066SDimitry Andric   SlotIndex Start, Stop;
1113b0f4066SDimitry Andric   tie(Start, Stop) = Indexes->getMBBRange(MBBNum);
1123b0f4066SDimitry Andric 
1133b0f4066SDimitry Andric   // Use advanceTo only when possible.
1143b0f4066SDimitry Andric   if (PrevPos != Start) {
1153b0f4066SDimitry Andric     if (!PrevPos.isValid() || Start < PrevPos)
1163b0f4066SDimitry Andric       for (unsigned i = 0, e = Iters.size(); i != e; ++i)
1173b0f4066SDimitry Andric         Iters[i].find(Start);
1183b0f4066SDimitry Andric     else
1193b0f4066SDimitry Andric       for (unsigned i = 0, e = Iters.size(); i != e; ++i)
1203b0f4066SDimitry Andric         Iters[i].advanceTo(Start);
1213b0f4066SDimitry Andric     PrevPos = Start;
1223b0f4066SDimitry Andric   }
1233b0f4066SDimitry Andric 
1243b0f4066SDimitry Andric   MachineFunction::const_iterator MFI = MF->getBlockNumbered(MBBNum);
1253b0f4066SDimitry Andric   BlockInterference *BI = &Blocks[MBBNum];
126dff0c46cSDimitry Andric   ArrayRef<SlotIndex> RegMaskSlots;
127dff0c46cSDimitry Andric   ArrayRef<const uint32_t*> RegMaskBits;
1283b0f4066SDimitry Andric   for (;;) {
1293b0f4066SDimitry Andric     BI->Tag = Tag;
1303b0f4066SDimitry Andric     BI->First = BI->Last = SlotIndex();
1313b0f4066SDimitry Andric 
1323b0f4066SDimitry Andric     // Check for first interference.
1333b0f4066SDimitry Andric     for (unsigned i = 0, e = Iters.size(); i != e; ++i) {
1343b0f4066SDimitry Andric       Iter &I = Iters[i];
1353b0f4066SDimitry Andric       if (!I.valid())
1363b0f4066SDimitry Andric         continue;
1373b0f4066SDimitry Andric       SlotIndex StartI = I.start();
1383b0f4066SDimitry Andric       if (StartI >= Stop)
1393b0f4066SDimitry Andric         continue;
1403b0f4066SDimitry Andric       if (!BI->First.isValid() || StartI < BI->First)
1413b0f4066SDimitry Andric         BI->First = StartI;
1423b0f4066SDimitry Andric     }
1433b0f4066SDimitry Andric 
144dff0c46cSDimitry Andric     // Also check for register mask interference.
145dff0c46cSDimitry Andric     RegMaskSlots = LIS->getRegMaskSlotsInBlock(MBBNum);
146dff0c46cSDimitry Andric     RegMaskBits = LIS->getRegMaskBitsInBlock(MBBNum);
147dff0c46cSDimitry Andric     SlotIndex Limit = BI->First.isValid() ? BI->First : Stop;
148dff0c46cSDimitry Andric     for (unsigned i = 0, e = RegMaskSlots.size();
149dff0c46cSDimitry Andric          i != e && RegMaskSlots[i] < Limit; ++i)
150dff0c46cSDimitry Andric       if (MachineOperand::clobbersPhysReg(RegMaskBits[i], PhysReg)) {
151dff0c46cSDimitry Andric         // Register mask i clobbers PhysReg before the LIU interference.
152dff0c46cSDimitry Andric         BI->First = RegMaskSlots[i];
153dff0c46cSDimitry Andric         break;
154dff0c46cSDimitry Andric       }
155dff0c46cSDimitry Andric 
1563b0f4066SDimitry Andric     PrevPos = Stop;
1573b0f4066SDimitry Andric     if (BI->First.isValid())
1583b0f4066SDimitry Andric       break;
1593b0f4066SDimitry Andric 
1603b0f4066SDimitry Andric     // No interference in this block? Go ahead and precompute the next block.
1613b0f4066SDimitry Andric     if (++MFI == MF->end())
1623b0f4066SDimitry Andric       return;
1633b0f4066SDimitry Andric     MBBNum = MFI->getNumber();
1643b0f4066SDimitry Andric     BI = &Blocks[MBBNum];
1653b0f4066SDimitry Andric     if (BI->Tag == Tag)
1663b0f4066SDimitry Andric       return;
1673b0f4066SDimitry Andric     tie(Start, Stop) = Indexes->getMBBRange(MBBNum);
1683b0f4066SDimitry Andric   }
1693b0f4066SDimitry Andric 
1703b0f4066SDimitry Andric   // Check for last interference in block.
1713b0f4066SDimitry Andric   for (unsigned i = 0, e = Iters.size(); i != e; ++i) {
1723b0f4066SDimitry Andric     Iter &I = Iters[i];
1733b0f4066SDimitry Andric     if (!I.valid() || I.start() >= Stop)
1743b0f4066SDimitry Andric       continue;
1753b0f4066SDimitry Andric     I.advanceTo(Stop);
1763b0f4066SDimitry Andric     bool Backup = !I.valid() || I.start() >= Stop;
1773b0f4066SDimitry Andric     if (Backup)
1783b0f4066SDimitry Andric       --I;
1793b0f4066SDimitry Andric     SlotIndex StopI = I.stop();
1803b0f4066SDimitry Andric     if (!BI->Last.isValid() || StopI > BI->Last)
1813b0f4066SDimitry Andric       BI->Last = StopI;
1823b0f4066SDimitry Andric     if (Backup)
1833b0f4066SDimitry Andric       ++I;
1843b0f4066SDimitry Andric   }
185dff0c46cSDimitry Andric 
186dff0c46cSDimitry Andric   // Also check for register mask interference.
187dff0c46cSDimitry Andric   SlotIndex Limit = BI->Last.isValid() ? BI->Last : Start;
188dff0c46cSDimitry Andric   for (unsigned i = RegMaskSlots.size();
189dff0c46cSDimitry Andric        i && RegMaskSlots[i-1].getDeadSlot() > Limit; --i)
190dff0c46cSDimitry Andric     if (MachineOperand::clobbersPhysReg(RegMaskBits[i-1], PhysReg)) {
191dff0c46cSDimitry Andric       // Register mask i-1 clobbers PhysReg after the LIU interference.
192dff0c46cSDimitry Andric       // Model the regmask clobber as a dead def.
193dff0c46cSDimitry Andric       BI->Last = RegMaskSlots[i-1].getDeadSlot();
194dff0c46cSDimitry Andric       break;
195dff0c46cSDimitry Andric     }
1963b0f4066SDimitry Andric }
197