diff options
Diffstat (limited to 'lib/CodeGen/LiveInterval.cpp')
| -rw-r--r-- | lib/CodeGen/LiveInterval.cpp | 148 | 
1 files changed, 95 insertions, 53 deletions
diff --git a/lib/CodeGen/LiveInterval.cpp b/lib/CodeGen/LiveInterval.cpp index 26722a3ca11a..a02a4a6c83a1 100644 --- a/lib/CodeGen/LiveInterval.cpp +++ b/lib/CodeGen/LiveInterval.cpp @@ -23,12 +23,16 @@  #include "llvm/ADT/DenseMap.h"  #include "llvm/ADT/SmallSet.h"  #include "llvm/ADT/STLExtras.h" -#include "llvm/Support/Streams.h" +#include "llvm/Support/raw_ostream.h"  #include "llvm/Target/TargetRegisterInfo.h"  #include <algorithm> -#include <ostream>  using namespace llvm; +// Print a LiveIndex to a raw_ostream. +void LiveIndex::print(raw_ostream &os) const { +  os << (index & ~PHI_BIT); +} +  // An example for liveAt():  //  // this = [1,4), liveAt(0) will return false. The instruction defining this @@ -36,7 +40,7 @@ using namespace llvm;  // variable it represents. This is because slot 1 is used (def slot) and spans  // up to slot 3 (store slot).  // -bool LiveInterval::liveAt(unsigned I) const { +bool LiveInterval::liveAt(LiveIndex I) const {    Ranges::const_iterator r = std::upper_bound(ranges.begin(), ranges.end(), I);    if (r == ranges.begin()) @@ -49,7 +53,7 @@ bool LiveInterval::liveAt(unsigned I) const {  // liveBeforeAndAt - Check if the interval is live at the index and the index  // just before it. If index is liveAt, check if it starts a new live range.  // If it does, then check if the previous live range ends at index-1. -bool LiveInterval::liveBeforeAndAt(unsigned I) const { +bool LiveInterval::liveBeforeAndAt(LiveIndex I) const {    Ranges::const_iterator r = std::upper_bound(ranges.begin(), ranges.end(), I);    if (r == ranges.begin()) @@ -127,7 +131,7 @@ bool LiveInterval::overlapsFrom(const LiveInterval& other,  /// overlaps - Return true if the live interval overlaps a range specified  /// by [Start, End). -bool LiveInterval::overlaps(unsigned Start, unsigned End) const { +bool LiveInterval::overlaps(LiveIndex Start, LiveIndex End) const {    assert(Start < End && "Invalid range");    const_iterator I  = begin();    const_iterator E  = end(); @@ -145,10 +149,10 @@ bool LiveInterval::overlaps(unsigned Start, unsigned End) const {  /// specified by I to end at the specified endpoint.  To do this, we should  /// merge and eliminate all ranges that this will overlap with.  The iterator is  /// not invalidated. -void LiveInterval::extendIntervalEndTo(Ranges::iterator I, unsigned NewEnd) { +void LiveInterval::extendIntervalEndTo(Ranges::iterator I, LiveIndex NewEnd) {    assert(I != ranges.end() && "Not a valid interval!");    VNInfo *ValNo = I->valno; -  unsigned OldEnd = I->end; +  LiveIndex OldEnd = I->end;    // Search for the first interval that we can't merge with.    Ranges::iterator MergeTo = next(I); @@ -163,7 +167,7 @@ void LiveInterval::extendIntervalEndTo(Ranges::iterator I, unsigned NewEnd) {    ranges.erase(next(I), MergeTo);    // Update kill info. -  removeKills(ValNo, OldEnd, I->end-1); +  ValNo->removeKills(OldEnd, I->end.prevSlot_());    // If the newly formed range now touches the range after it and if they have    // the same value number, merge the two ranges into one range. @@ -179,7 +183,7 @@ void LiveInterval::extendIntervalEndTo(Ranges::iterator I, unsigned NewEnd) {  /// specified by I to start at the specified endpoint.  To do this, we should  /// merge and eliminate all ranges that this will overlap with.  LiveInterval::Ranges::iterator -LiveInterval::extendIntervalStartTo(Ranges::iterator I, unsigned NewStart) { +LiveInterval::extendIntervalStartTo(Ranges::iterator I, LiveIndex NewStart) {    assert(I != ranges.end() && "Not a valid interval!");    VNInfo *ValNo = I->valno; @@ -212,7 +216,7 @@ LiveInterval::extendIntervalStartTo(Ranges::iterator I, unsigned NewStart) {  LiveInterval::iterator  LiveInterval::addRangeFrom(LiveRange LR, iterator From) { -  unsigned Start = LR.start, End = LR.end; +  LiveIndex Start = LR.start, End = LR.end;    iterator it = std::upper_bound(From, ranges.end(), Start);    // If the inserted interval starts in the middle or right at the end of @@ -246,7 +250,7 @@ LiveInterval::addRangeFrom(LiveRange LR, iterator From) {            extendIntervalEndTo(it, End);          else if (End < it->end)            // Overlapping intervals, there might have been a kill here. -          removeKill(it->valno, End); +          it->valno->removeKill(End);          return it;        }      } else { @@ -262,33 +266,32 @@ LiveInterval::addRangeFrom(LiveRange LR, iterator From) {    return ranges.insert(it, LR);  } -/// isInOneLiveRange - Return true if the range specified is entirely in the +/// isInOneLiveRange - Return true if the range specified is entirely in   /// a single LiveRange of the live interval. -bool LiveInterval::isInOneLiveRange(unsigned Start, unsigned End) { +bool LiveInterval::isInOneLiveRange(LiveIndex Start, LiveIndex End) {    Ranges::iterator I = std::upper_bound(ranges.begin(), ranges.end(), Start);    if (I == ranges.begin())      return false;    --I; -  return I->contains(Start) && I->contains(End-1); +  return I->containsRange(Start, End);  }  /// removeRange - Remove the specified range from this interval.  Note that  /// the range must be in a single LiveRange in its entirety. -void LiveInterval::removeRange(unsigned Start, unsigned End, +void LiveInterval::removeRange(LiveIndex Start, LiveIndex End,                                 bool RemoveDeadValNo) {    // Find the LiveRange containing this span.    Ranges::iterator I = std::upper_bound(ranges.begin(), ranges.end(), Start);    assert(I != ranges.begin() && "Range is not in interval!");    --I; -  assert(I->contains(Start) && I->contains(End-1) && -         "Range is not entirely in interval!"); +  assert(I->containsRange(Start, End) && "Range is not entirely in interval!");    // If the span we are removing is at the start of the LiveRange, adjust it.    VNInfo *ValNo = I->valno;    if (I->start == Start) {      if (I->end == End) { -      removeKills(I->valno, Start, End); +      ValNo->removeKills(Start, End);        if (RemoveDeadValNo) {          // Check if val# is dead.          bool isDead = true; @@ -322,13 +325,13 @@ void LiveInterval::removeRange(unsigned Start, unsigned End,    // Otherwise if the span we are removing is at the end of the LiveRange,    // adjust the other way.    if (I->end == End) { -    removeKills(ValNo, Start, End); +    ValNo->removeKills(Start, End);      I->end = Start;      return;    }    // Otherwise, we are splitting the LiveRange into two pieces. -  unsigned OldEnd = I->end; +  LiveIndex OldEnd = I->end;    I->end = Start;   // Trim the old interval.    // Insert the new one. @@ -362,11 +365,12 @@ void LiveInterval::removeValNo(VNInfo *ValNo) {  /// scaleNumbering - Renumber VNI and ranges to provide gaps for new  /// instructions.                                                    +  void LiveInterval::scaleNumbering(unsigned factor) {    // Scale ranges.                                                                for (iterator RI = begin(), RE = end(); RI != RE; ++RI) { -    RI->start = InstrSlots::scale(RI->start, factor); -    RI->end = InstrSlots::scale(RI->end, factor); +    RI->start = RI->start.scale(factor); +    RI->end = RI->end.scale(factor);    }    // Scale VNI info.                                                           @@ -374,19 +378,20 @@ void LiveInterval::scaleNumbering(unsigned factor) {      VNInfo *vni = *VNI;      if (vni->isDefAccurate()) -      vni->def = InstrSlots::scale(vni->def, factor); +      vni->def = vni->def.scale(factor);      for (unsigned i = 0; i < vni->kills.size(); ++i) { -      if (vni->kills[i] != 0) -        vni->kills[i] = InstrSlots::scale(vni->kills[i], factor); +      if (!vni->kills[i].isPHIIndex()) +        vni->kills[i] = vni->kills[i].scale(factor);      }    }  } +  /// getLiveRangeContaining - Return the live range that contains the  /// specified index, or null if there is none.  LiveInterval::const_iterator  -LiveInterval::FindLiveRangeContaining(unsigned Idx) const { +LiveInterval::FindLiveRangeContaining(LiveIndex Idx) const {    const_iterator It = std::upper_bound(begin(), end(), Idx);    if (It != ranges.begin()) {      --It; @@ -398,7 +403,7 @@ LiveInterval::FindLiveRangeContaining(unsigned Idx) const {  }  LiveInterval::iterator  -LiveInterval::FindLiveRangeContaining(unsigned Idx) { +LiveInterval::FindLiveRangeContaining(LiveIndex Idx) {    iterator It = std::upper_bound(begin(), end(), Idx);    if (It != begin()) {      --It; @@ -409,17 +414,27 @@ LiveInterval::FindLiveRangeContaining(unsigned Idx) {    return end();  } -/// findDefinedVNInfo - Find the VNInfo that's defined at the specified index -/// (register interval) or defined by the specified register (stack inteval). -VNInfo *LiveInterval::findDefinedVNInfo(unsigned DefIdxOrReg) const { -  VNInfo *VNI = NULL; +/// findDefinedVNInfo - Find the VNInfo defined by the specified +/// index (register interval). +VNInfo *LiveInterval::findDefinedVNInfoForRegInt(LiveIndex Idx) const {    for (LiveInterval::const_vni_iterator i = vni_begin(), e = vni_end(); -       i != e; ++i) -    if ((*i)->def == DefIdxOrReg) { -      VNI = *i; -      break; -    } -  return VNI; +       i != e; ++i) { +    if ((*i)->def == Idx) +      return *i; +  } + +  return 0; +} + +/// findDefinedVNInfo - Find the VNInfo defined by the specified +/// register (stack inteval). +VNInfo *LiveInterval::findDefinedVNInfoForStackInt(unsigned reg) const { +  for (LiveInterval::const_vni_iterator i = vni_begin(), e = vni_end(); +       i != e; ++i) { +    if ((*i)->getReg() == reg) +      return *i; +  } +  return 0;  }  /// join - Join two live intervals (this, and other) together.  This applies @@ -502,7 +517,7 @@ void LiveInterval::join(LiveInterval &Other, const int *LHSValNoAssignments,      InsertPos = addRangeFrom(*I, InsertPos);    } -  weight += Other.weight; +  ComputeJoinedWeight(Other);    // Update regalloc hint if currently there isn't one.    if (TargetRegisterInfo::isVirtualRegister(reg) && @@ -546,7 +561,7 @@ void LiveInterval::MergeValueInAsValue(const LiveInterval &RHS,    for (const_iterator I = RHS.begin(), E = RHS.end(); I != E; ++I) {      if (I->valno != RHSValNo)        continue; -    unsigned Start = I->start, End = I->end; +    LiveIndex Start = I->start, End = I->end;      IP = std::upper_bound(IP, end(), Start);      // If the start of this range overlaps with an existing liverange, trim it.      if (IP != begin() && IP[-1].end > Start) { @@ -622,20 +637,21 @@ void LiveInterval::MergeInClobberRanges(const LiveInterval &Clobbers,      else if (UnusedValNo)        ClobberValNo = UnusedValNo;      else { -      UnusedValNo = ClobberValNo = getNextValue(0, 0, false, VNInfoAllocator); +      UnusedValNo = ClobberValNo = +        getNextValue(LiveIndex(), 0, false, VNInfoAllocator);        ValNoMaps.insert(std::make_pair(I->valno, ClobberValNo));      }      bool Done = false; -    unsigned Start = I->start, End = I->end; +    LiveIndex Start = I->start, End = I->end;      // If a clobber range starts before an existing range and ends after      // it, the clobber range will need to be split into multiple ranges.      // Loop until the entire clobber range is handled.      while (!Done) {        Done = true;        IP = std::upper_bound(IP, end(), Start); -      unsigned SubRangeStart = Start; -      unsigned SubRangeEnd = End; +      LiveIndex SubRangeStart = Start; +      LiveIndex SubRangeEnd = End;        // If the start of this range overlaps with an existing liverange, trim it.        if (IP != begin() && IP[-1].end > SubRangeStart) { @@ -671,11 +687,13 @@ void LiveInterval::MergeInClobberRanges(const LiveInterval &Clobbers,    }  } -void LiveInterval::MergeInClobberRange(unsigned Start, unsigned End, +void LiveInterval::MergeInClobberRange(LiveIndex Start, +                                       LiveIndex End,                                         BumpPtrAllocator &VNInfoAllocator) {    // Find a value # to use for the clobber ranges.  If there is already a value#    // for unknown values, use it. -  VNInfo *ClobberValNo = getNextValue(0, 0, false, VNInfoAllocator); +  VNInfo *ClobberValNo = +    getNextValue(LiveIndex(), 0, false, VNInfoAllocator);    iterator IP = begin();    IP = std::upper_bound(IP, end(), Start); @@ -711,7 +729,7 @@ VNInfo* LiveInterval::MergeValueNumberInto(VNInfo *V1, VNInfo *V2) {    // Make sure V2 is smaller than V1.    if (V1->id < V2->id) { -    copyValNumInfo(V1, V2); +    V1->copyFrom(*V2);      std::swap(V1, V2);    } @@ -788,20 +806,42 @@ void LiveInterval::Copy(const LiveInterval &RHS,  unsigned LiveInterval::getSize() const {    unsigned Sum = 0;    for (const_iterator I = begin(), E = end(); I != E; ++I) -    Sum += I->end - I->start; +    Sum += I->start.distance(I->end);    return Sum;  } -std::ostream& llvm::operator<<(std::ostream& os, const LiveRange &LR) { +/// ComputeJoinedWeight - Set the weight of a live interval Joined +/// after Other has been merged into it. +void LiveInterval::ComputeJoinedWeight(const LiveInterval &Other) { +  // If either of these intervals was spilled, the weight is the +  // weight of the non-spilled interval.  This can only happen with +  // iterative coalescers. + +  if (Other.weight != HUGE_VALF) { +    weight += Other.weight; +  } +  else if (weight == HUGE_VALF && +      !TargetRegisterInfo::isPhysicalRegister(reg)) { +    // Remove this assert if you have an iterative coalescer +    assert(0 && "Joining to spilled interval"); +    weight = Other.weight; +  } +  else { +    // Otherwise the weight stays the same +    // Remove this assert if you have an iterative coalescer +    assert(0 && "Joining from spilled interval"); +  } +} + +raw_ostream& llvm::operator<<(raw_ostream& os, const LiveRange &LR) {    return os << '[' << LR.start << ',' << LR.end << ':' << LR.valno->id << ")";  }  void LiveRange::dump() const { -  cerr << *this << "\n"; +  errs() << *this << "\n";  } -void LiveInterval::print(std::ostream &OS, -                         const TargetRegisterInfo *TRI) const { +void LiveInterval::print(raw_ostream &OS, const TargetRegisterInfo *TRI) const {    if (isStackSlot())      OS << "SS#" << getStackSlotIndex();    else if (TRI && TargetRegisterInfo::isPhysicalRegister(reg)) @@ -841,6 +881,8 @@ void LiveInterval::print(std::ostream &OS,            OS << "-(";            for (unsigned j = 0; j != ee; ++j) {              OS << vni->kills[j]; +            if (vni->kills[j].isPHIIndex()) +              OS << "*";              if (j != ee-1)                OS << " ";            } @@ -857,10 +899,10 @@ void LiveInterval::print(std::ostream &OS,  }  void LiveInterval::dump() const { -  cerr << *this << "\n"; +  errs() << *this << "\n";  } -void LiveRange::print(std::ostream &os) const { +void LiveRange::print(raw_ostream &os) const {    os << *this;  }  | 
