diff options
Diffstat (limited to 'llvm/lib/CodeGen')
| -rw-r--r-- | llvm/lib/CodeGen/CMakeLists.txt | 1 | ||||
| -rw-r--r-- | llvm/lib/CodeGen/InterferenceCache.cpp | 139 | ||||
| -rw-r--r-- | llvm/lib/CodeGen/InterferenceCache.h | 159 | ||||
| -rw-r--r-- | llvm/lib/CodeGen/RegAllocGreedy.cpp | 2 | 
4 files changed, 300 insertions, 1 deletions
| diff --git a/llvm/lib/CodeGen/CMakeLists.txt b/llvm/lib/CodeGen/CMakeLists.txt index d7d0e1b3812..2ca3859caf0 100644 --- a/llvm/lib/CodeGen/CMakeLists.txt +++ b/llvm/lib/CodeGen/CMakeLists.txt @@ -19,6 +19,7 @@ add_llvm_library(LLVMCodeGen    GCStrategy.cpp    IfConversion.cpp    InlineSpiller.cpp +  InterferenceCache.cpp    IntrinsicLowering.cpp    LLVMTargetMachine.cpp    LatencyPriorityQueue.cpp diff --git a/llvm/lib/CodeGen/InterferenceCache.cpp b/llvm/lib/CodeGen/InterferenceCache.cpp new file mode 100644 index 00000000000..512b4b3ccc1 --- /dev/null +++ b/llvm/lib/CodeGen/InterferenceCache.cpp @@ -0,0 +1,139 @@ +//===-- InterferenceCache.h - Caching per-block interference ---*- C++ -*--===// +// +//                     The LLVM Compiler Infrastructure +// +// This file is distributed under the University of Illinois Open Source +// License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// +// +// InterferenceCache remembers per-block interference in LiveIntervalUnions. +// +//===----------------------------------------------------------------------===// + +#define DEBUG_TYPE "regalloc" +#include "InterferenceCache.h" +#include "llvm/Target/TargetRegisterInfo.h" + +using namespace llvm; + +void InterferenceCache::init(MachineFunction *mf, +                             LiveIntervalUnion *liuarray, +                             SlotIndexes *indexes, +                            const TargetRegisterInfo *tri) { +  MF = mf; +  LIUArray = liuarray; +  TRI = tri; +  PhysRegEntries.assign(TRI->getNumRegs(), 0); +  for (unsigned i = 0; i != CacheEntries; ++i) +    Entries[i].clear(indexes); +} + +InterferenceCache::Entry *InterferenceCache::get(unsigned PhysReg) { +  unsigned E = PhysRegEntries[PhysReg]; +  if (E < CacheEntries && Entries[E].getPhysReg() == PhysReg) { +    if (!Entries[E].valid(LIUArray, TRI)) +      Entries[E].revalidate(); +    return &Entries[E]; +  } +  // No valid entry exists, pick the next round-robin entry. +  E = RoundRobin; +  if (++RoundRobin == CacheEntries) +    RoundRobin = 0; +  Entries[E].reset(PhysReg, LIUArray, TRI, MF); +  PhysRegEntries[PhysReg] = E; +  return &Entries[E]; +} + +/// revalidate - LIU contents have changed, update tags. +void InterferenceCache::Entry::revalidate() { +  // Invalidate all block entries. +  ++Tag; +  // Invalidate all iterators. +  PrevPos = SlotIndex(); +  for (unsigned i = 0, e = Aliases.size(); i != e; ++i) +    Aliases[i].second = Aliases[i].first->getTag(); +} + +void InterferenceCache::Entry::reset(unsigned physReg, +                                     LiveIntervalUnion *LIUArray, +                                     const TargetRegisterInfo *TRI, +                                     const MachineFunction *MF) { +  // LIU's changed, invalidate cache. +  ++Tag; +  PhysReg = physReg; +  Blocks.resize(MF->getNumBlockIDs()); +  Aliases.clear(); +  for (const unsigned *AS = TRI->getOverlaps(PhysReg); *AS; ++AS) { +    LiveIntervalUnion *LIU = LIUArray + *AS; +    Aliases.push_back(std::make_pair(LIU, LIU->getTag())); +  } + +  // Reset iterators. +  PrevPos = SlotIndex(); +  unsigned e = Aliases.size(); +  Iters.resize(e); +  for (unsigned i = 0; i != e; ++i) +    Iters[i].setMap(Aliases[i].first->getMap()); +} + +bool InterferenceCache::Entry::valid(LiveIntervalUnion *LIUArray, +                                     const TargetRegisterInfo *TRI) { +  unsigned i = 0, e = Aliases.size(); +  for (const unsigned *AS = TRI->getOverlaps(PhysReg); *AS; ++AS, ++i) { +    LiveIntervalUnion *LIU = LIUArray + *AS; +    if (i == e ||  Aliases[i].first != LIU) +      return false; +    if (LIU->changedSince(Aliases[i].second)) +      return false; +  } +  return i == e; +} + +void InterferenceCache::Entry::update(unsigned MBBNum) { +  BlockInterference *BI = &Blocks[MBBNum]; +  BI->Tag = Tag; +  BI->First = BI->Last = SlotIndex(); + +  SlotIndex Start, Stop; +  tie(Start, Stop) = Indexes->getMBBRange(MBBNum); + +  // Use advanceTo only when possible. +  if (!PrevPos.isValid() || Start < PrevPos) +    for (unsigned i = 0, e = Iters.size(); i != e; ++i) +      Iters[i].find(Start); +  else +    for (unsigned i = 0, e = Iters.size(); i != e; ++i) +      Iters[i].advanceTo(Start); +  PrevPos = Start; + +  // Check for first interference. +  for (unsigned i = 0, e = Iters.size(); i != e; ++i) { +    Iter &I = Iters[i]; +    if (!I.valid()) +      continue; +    SlotIndex StartI = I.start(); +    if (StartI >= Stop) +      continue; +    if (!BI->First.isValid() || StartI < BI->First) +      BI->First = StartI; +  } + +  // No interference in block. +  if (!BI->First.isValid()) +    return; + +  // Check for last interference. +  for (unsigned i = 0, e = Iters.size(); i != e; ++i) { +    Iter &I = Iters[i]; +    if (!I.valid() || I.start() >= Stop) +      continue; +    I.advanceTo(Stop); +    if (!I.valid() || I.start() >= Stop) +      --I; +    SlotIndex StopI = I.stop(); +    if (!BI->Last.isValid() || StopI > BI->Last) +      BI->Last = StopI; +  } +  PrevPos = Stop; +} diff --git a/llvm/lib/CodeGen/InterferenceCache.h b/llvm/lib/CodeGen/InterferenceCache.h new file mode 100644 index 00000000000..2e613ae0959 --- /dev/null +++ b/llvm/lib/CodeGen/InterferenceCache.h @@ -0,0 +1,159 @@ +//===-- InterferenceCache.h - Caching per-block interference ---*- C++ -*--===// +// +//                     The LLVM Compiler Infrastructure +// +// This file is distributed under the University of Illinois Open Source +// License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// +// +// InterferenceCache remembers per-block interference in LiveIntervalUnions. +// +//===----------------------------------------------------------------------===// + +#ifndef LLVM_CODEGEN_INTERFERENCECACHE +#define LLVM_CODEGEN_INTERFERENCECACHE + +#include "LiveIntervalUnion.h" + +namespace llvm { + +class InterferenceCache { +  const TargetRegisterInfo *TRI; +  LiveIntervalUnion *LIUArray; +  SlotIndexes *Indexes; +  MachineFunction *MF; + +  /// BlockInterference - information about the interference in a single basic +  /// block. +  struct BlockInterference { +    BlockInterference() : Tag(0) {} +    unsigned Tag; +    SlotIndex First; +    SlotIndex Last; +  }; + +  /// Entry - A cache entry containing interference information for all aliases +  /// of PhysReg in all basic blocks. +  class Entry { +    /// PhysReg - The register currently represented. +    unsigned PhysReg; + +    /// Tag - Cache tag is changed when any of the underlying LiveIntervalUnions +    /// change. +    unsigned Tag; + +    /// Indexes - Mapping block numbers to SlotIndex ranges. +    SlotIndexes *Indexes; + +    /// PrevPos - The previous position the iterators were moved to. +    SlotIndex PrevPos; + +    /// AliasTags - A LiveIntervalUnion pointer and tag for each alias of +    /// PhysReg. +    SmallVector<std::pair<LiveIntervalUnion*, unsigned>, 8> Aliases; + +    typedef LiveIntervalUnion::SegmentIter Iter; + +    /// Iters - an iterator for each alias +    SmallVector<Iter, 8> Iters; + +    /// Blocks - Interference for each block in the function. +    SmallVector<BlockInterference, 8> Blocks; + +    /// update - Recompute Blocks[MBBNum] +    void update(unsigned MBBNum); + +  public: +    Entry() : PhysReg(0), Tag(0), Indexes(0) {} + +    void clear(SlotIndexes *indexes) { +      PhysReg = 0; +      Indexes = indexes; +    } + +    unsigned getPhysReg() const { return PhysReg; } + +    void revalidate(); + +    /// valid - Return true if this is a valid entry for physReg. +    bool valid(LiveIntervalUnion *LIUArray, const TargetRegisterInfo *TRI); + +    /// reset - Initialize entry to represent physReg's aliases. +    void reset(unsigned physReg, +               LiveIntervalUnion *LIUArray, +               const TargetRegisterInfo *TRI, +               const MachineFunction *MF); + +    /// get - Return an up to date BlockInterference. +    BlockInterference *get(unsigned MBBNum) { +      if (Blocks[MBBNum].Tag != Tag) +        update(MBBNum); +      return &Blocks[MBBNum]; +    } +  }; + +  // We don't keep a cache entry for every physical register, that would use too +  // much memory. Instead, a fixed number of cache entries are used in a round- +  // robin manner. +  enum { CacheEntries = 32 }; + +  // Point to an entry for each physreg. The entry pointed to may not be up to +  // date, and it may have been reused for a different physreg. +  SmallVector<unsigned char, 2> PhysRegEntries; + +  // Next round-robin entry to be picked. +  unsigned RoundRobin; + +  // The actual cache entries. +  Entry Entries[CacheEntries]; + +  // get - Get a valid entry for PhysReg. +  Entry *get(unsigned PhysReg); + +public: +  InterferenceCache() : TRI(0), LIUArray(0), Indexes(0), MF(0), RoundRobin(0) {} + +  /// init - Prepare cache for a new function. +  void init(MachineFunction*, LiveIntervalUnion*, SlotIndexes*, +            const TargetRegisterInfo *); + +  /// Cursor - The primary query interface for the block interference cache. +  class Cursor { +    Entry *CacheEntry; +    BlockInterference *Current; +  public: +    /// Cursor - Create a cursor for the interference allocated to PhysReg and +    /// all its aliases. +    Cursor(InterferenceCache &Cache, unsigned PhysReg) +      : CacheEntry(Cache.get(PhysReg)), Current(0) {} + +    /// moveTo - Move cursor to basic block MBBNum. +    void moveToBlock(unsigned MBBNum) { +      Current = CacheEntry->get(MBBNum); +    } + +    /// hasInterference - Return true if the current block has any interference. +    bool hasInterference() { +      return Current->First.isValid(); +    } + +    /// first - Return the starting index of the first interfering range in the +    /// current block. +    SlotIndex first() { +      return Current->First; +    } + +    /// last - Return the ending index of the last interfering range in the +    /// current block. +    SlotIndex last() { +      return Current->Last; +    } +  }; + +  friend class Cursor; +}; + +} // namespace llvm + +#endif diff --git a/llvm/lib/CodeGen/RegAllocGreedy.cpp b/llvm/lib/CodeGen/RegAllocGreedy.cpp index 05387d27272..e2cf71a6504 100644 --- a/llvm/lib/CodeGen/RegAllocGreedy.cpp +++ b/llvm/lib/CodeGen/RegAllocGreedy.cpp @@ -14,7 +14,7 @@  #define DEBUG_TYPE "regalloc"  #include "AllocationOrder.h" -#include "LiveIntervalUnion.h" +#include "InterferenceCache.h"  #include "LiveRangeEdit.h"  #include "RegAllocBase.h"  #include "Spiller.h" | 

