2007-06-08 17:18:56 +00:00
|
|
|
//===-- SimpleRegisterCoalescing.cpp - Register Coalescing ----------------===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
2007-12-29 20:36:04 +00:00
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
2007-06-08 17:18:56 +00:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file implements a simple register coalescing pass that attempts to
|
|
|
|
// aggressively coalesce every register copy that it can.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2007-07-31 22:37:44 +00:00
|
|
|
#define DEBUG_TYPE "regcoalescing"
|
2007-11-05 17:41:38 +00:00
|
|
|
#include "SimpleRegisterCoalescing.h"
|
2007-06-08 17:18:56 +00:00
|
|
|
#include "VirtRegMap.h"
|
2007-11-05 17:41:38 +00:00
|
|
|
#include "llvm/CodeGen/LiveIntervalAnalysis.h"
|
2007-06-08 17:18:56 +00:00
|
|
|
#include "llvm/Value.h"
|
|
|
|
#include "llvm/CodeGen/LiveVariables.h"
|
|
|
|
#include "llvm/CodeGen/MachineFrameInfo.h"
|
|
|
|
#include "llvm/CodeGen/MachineInstr.h"
|
2007-12-11 02:09:15 +00:00
|
|
|
#include "llvm/CodeGen/MachineLoopInfo.h"
|
2007-12-31 04:13:23 +00:00
|
|
|
#include "llvm/CodeGen/MachineRegisterInfo.h"
|
2007-06-08 17:18:56 +00:00
|
|
|
#include "llvm/CodeGen/Passes.h"
|
2007-09-06 16:18:45 +00:00
|
|
|
#include "llvm/CodeGen/RegisterCoalescer.h"
|
2007-06-08 17:18:56 +00:00
|
|
|
#include "llvm/Target/TargetInstrInfo.h"
|
|
|
|
#include "llvm/Target/TargetMachine.h"
|
|
|
|
#include "llvm/Support/CommandLine.h"
|
|
|
|
#include "llvm/Support/Debug.h"
|
|
|
|
#include "llvm/ADT/SmallSet.h"
|
|
|
|
#include "llvm/ADT/Statistic.h"
|
|
|
|
#include "llvm/ADT/STLExtras.h"
|
|
|
|
#include <algorithm>
|
|
|
|
#include <cmath>
|
|
|
|
using namespace llvm;
|
|
|
|
|
|
|
|
STATISTIC(numJoins , "Number of interval joins performed");
|
2008-02-13 03:01:43 +00:00
|
|
|
STATISTIC(numCommutes , "Number of instruction commuting performed");
|
|
|
|
STATISTIC(numExtends , "Number of copies extended");
|
2007-06-08 17:18:56 +00:00
|
|
|
STATISTIC(numPeep , "Number of identity moves eliminated after coalescing");
|
|
|
|
STATISTIC(numAborts , "Number of times interval joining aborted");
|
|
|
|
|
|
|
|
char SimpleRegisterCoalescing::ID = 0;
|
|
|
|
namespace {
|
|
|
|
static cl::opt<bool>
|
|
|
|
EnableJoining("join-liveintervals",
|
2007-07-09 12:00:59 +00:00
|
|
|
cl::desc("Coalesce copies (default=true)"),
|
2007-06-08 17:18:56 +00:00
|
|
|
cl::init(true));
|
|
|
|
|
2007-11-06 08:52:21 +00:00
|
|
|
static cl::opt<bool>
|
|
|
|
NewHeuristic("new-coalescer-heuristic",
|
|
|
|
cl::desc("Use new coalescer heuristic"),
|
|
|
|
cl::init(false));
|
|
|
|
|
2007-06-08 17:18:56 +00:00
|
|
|
RegisterPass<SimpleRegisterCoalescing>
|
2007-08-05 18:45:33 +00:00
|
|
|
X("simple-register-coalescing", "Simple Register Coalescing");
|
2007-09-06 16:18:45 +00:00
|
|
|
|
|
|
|
// Declare that we implement the RegisterCoalescer interface
|
|
|
|
RegisterAnalysisGroup<RegisterCoalescer, true/*The Default*/> V(X);
|
2007-06-08 17:18:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const PassInfo *llvm::SimpleRegisterCoalescingID = X.getPassInfo();
|
|
|
|
|
|
|
|
void SimpleRegisterCoalescing::getAnalysisUsage(AnalysisUsage &AU) const {
|
|
|
|
AU.addPreserved<LiveIntervals>();
|
2008-01-04 20:54:55 +00:00
|
|
|
AU.addPreserved<MachineLoopInfo>();
|
|
|
|
AU.addPreservedID(MachineDominatorsID);
|
2007-06-08 17:18:56 +00:00
|
|
|
AU.addPreservedID(PHIEliminationID);
|
|
|
|
AU.addPreservedID(TwoAddressInstructionPassID);
|
|
|
|
AU.addRequired<LiveVariables>();
|
|
|
|
AU.addRequired<LiveIntervals>();
|
2007-12-11 02:09:15 +00:00
|
|
|
AU.addRequired<MachineLoopInfo>();
|
2007-06-08 17:18:56 +00:00
|
|
|
MachineFunctionPass::getAnalysisUsage(AU);
|
|
|
|
}
|
|
|
|
|
2007-07-09 12:00:59 +00:00
|
|
|
/// AdjustCopiesBackFrom - We found a non-trivially-coalescable copy with IntA
|
2007-06-08 17:18:56 +00:00
|
|
|
/// being the source and IntB being the dest, thus this defines a value number
|
|
|
|
/// in IntB. If the source value number (in IntA) is defined by a copy from B,
|
|
|
|
/// see if we can merge these two pieces of B into a single value number,
|
|
|
|
/// eliminating a copy. For example:
|
|
|
|
///
|
|
|
|
/// A3 = B0
|
|
|
|
/// ...
|
|
|
|
/// B1 = A3 <- this copy
|
|
|
|
///
|
|
|
|
/// In this case, B0 can be extended to where the B1 copy lives, allowing the B1
|
|
|
|
/// value number to be replaced with B0 (which simplifies the B liveinterval).
|
|
|
|
///
|
|
|
|
/// This returns true if an interval was modified.
|
|
|
|
///
|
2008-01-04 08:59:18 +00:00
|
|
|
bool SimpleRegisterCoalescing::AdjustCopiesBackFrom(LiveInterval &IntA,
|
|
|
|
LiveInterval &IntB,
|
|
|
|
MachineInstr *CopyMI) {
|
2007-06-08 17:18:56 +00:00
|
|
|
unsigned CopyIdx = li_->getDefIndex(li_->getInstructionIndex(CopyMI));
|
|
|
|
|
|
|
|
// BValNo is a value number in B that is defined by a copy from A. 'B3' in
|
|
|
|
// the example above.
|
|
|
|
LiveInterval::iterator BLR = IntB.FindLiveRangeContaining(CopyIdx);
|
2008-04-16 18:48:43 +00:00
|
|
|
if (BLR == IntB.end()) // Should never happen!
|
|
|
|
return false;
|
2007-08-29 20:45:00 +00:00
|
|
|
VNInfo *BValNo = BLR->valno;
|
2007-06-08 17:18:56 +00:00
|
|
|
|
|
|
|
// Get the location that B is defined at. Two options: either this value has
|
|
|
|
// an unknown definition point or it is defined at CopyIdx. If unknown, we
|
|
|
|
// can't process it.
|
2008-02-15 18:24:29 +00:00
|
|
|
if (!BValNo->copy) return false;
|
|
|
|
assert(BValNo->def == CopyIdx && "Copy doesn't define the value?");
|
2007-06-08 17:18:56 +00:00
|
|
|
|
2008-02-13 03:01:43 +00:00
|
|
|
// AValNo is the value number in A that defines the copy, A3 in the example.
|
|
|
|
LiveInterval::iterator ALR = IntA.FindLiveRangeContaining(CopyIdx-1);
|
2008-04-16 18:48:43 +00:00
|
|
|
if (ALR == IntA.end()) // Should never happen!
|
|
|
|
return false;
|
2008-02-13 03:01:43 +00:00
|
|
|
VNInfo *AValNo = ALR->valno;
|
2007-06-08 17:18:56 +00:00
|
|
|
|
2008-02-13 03:01:43 +00:00
|
|
|
// If AValNo is defined as a copy from IntB, we can potentially process this.
|
2007-06-08 17:18:56 +00:00
|
|
|
// Get the instruction that defines this value number.
|
2008-02-15 18:24:29 +00:00
|
|
|
unsigned SrcReg = li_->getVNInfoSourceReg(AValNo);
|
2007-06-08 17:18:56 +00:00
|
|
|
if (!SrcReg) return false; // Not defined by a copy.
|
|
|
|
|
|
|
|
// If the value number is not defined by a copy instruction, ignore it.
|
2008-02-15 18:24:29 +00:00
|
|
|
|
2007-06-08 17:18:56 +00:00
|
|
|
// If the source register comes from an interval other than IntB, we can't
|
|
|
|
// handle this.
|
2008-02-15 18:24:29 +00:00
|
|
|
if (SrcReg != IntB.reg) return false;
|
2007-06-08 17:18:56 +00:00
|
|
|
|
|
|
|
// Get the LiveRange in IntB that this value number starts with.
|
2007-08-29 20:45:00 +00:00
|
|
|
LiveInterval::iterator ValLR = IntB.FindLiveRangeContaining(AValNo->def-1);
|
2008-04-16 18:48:43 +00:00
|
|
|
if (ValLR == IntB.end()) // Should never happen!
|
|
|
|
return false;
|
2007-06-08 17:18:56 +00:00
|
|
|
|
|
|
|
// Make sure that the end of the live range is inside the same block as
|
|
|
|
// CopyMI.
|
|
|
|
MachineInstr *ValLREndInst = li_->getInstructionFromIndex(ValLR->end-1);
|
|
|
|
if (!ValLREndInst ||
|
|
|
|
ValLREndInst->getParent() != CopyMI->getParent()) return false;
|
|
|
|
|
|
|
|
// Okay, we now know that ValLR ends in the same block that the CopyMI
|
|
|
|
// live-range starts. If there are no intervening live ranges between them in
|
|
|
|
// IntB, we can merge them.
|
|
|
|
if (ValLR+1 != BLR) return false;
|
2007-08-14 23:19:28 +00:00
|
|
|
|
|
|
|
// If a live interval is a physical register, conservatively check if any
|
|
|
|
// of its sub-registers is overlapping the live interval of the virtual
|
|
|
|
// register. If so, do not coalesce.
|
2008-02-10 18:45:23 +00:00
|
|
|
if (TargetRegisterInfo::isPhysicalRegister(IntB.reg) &&
|
|
|
|
*tri_->getSubRegisters(IntB.reg)) {
|
|
|
|
for (const unsigned* SR = tri_->getSubRegisters(IntB.reg); *SR; ++SR)
|
2007-08-14 23:19:28 +00:00
|
|
|
if (li_->hasInterval(*SR) && IntA.overlaps(li_->getInterval(*SR))) {
|
|
|
|
DOUT << "Interfere with sub-register ";
|
2008-02-10 18:45:23 +00:00
|
|
|
DEBUG(li_->getInterval(*SR).print(DOUT, tri_));
|
2007-08-14 23:19:28 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2007-06-08 17:18:56 +00:00
|
|
|
|
2008-02-10 18:45:23 +00:00
|
|
|
DOUT << "\nExtending: "; IntB.print(DOUT, tri_);
|
2007-06-08 17:18:56 +00:00
|
|
|
|
2007-08-07 23:49:57 +00:00
|
|
|
unsigned FillerStart = ValLR->end, FillerEnd = BLR->start;
|
2007-06-08 17:18:56 +00:00
|
|
|
// We are about to delete CopyMI, so need to remove it as the 'instruction
|
2007-08-07 23:49:57 +00:00
|
|
|
// that defines this value #'. Update the the valnum with the new defining
|
|
|
|
// instruction #.
|
2008-02-15 18:24:29 +00:00
|
|
|
BValNo->def = FillerStart;
|
|
|
|
BValNo->copy = NULL;
|
2007-06-08 17:18:56 +00:00
|
|
|
|
|
|
|
// Okay, we can merge them. We need to insert a new liverange:
|
|
|
|
// [ValLR.end, BLR.begin) of either value number, then we merge the
|
|
|
|
// two value numbers.
|
|
|
|
IntB.addRange(LiveRange(FillerStart, FillerEnd, BValNo));
|
|
|
|
|
|
|
|
// If the IntB live range is assigned to a physical register, and if that
|
|
|
|
// physreg has aliases,
|
2008-02-10 18:45:23 +00:00
|
|
|
if (TargetRegisterInfo::isPhysicalRegister(IntB.reg)) {
|
2007-06-08 17:18:56 +00:00
|
|
|
// Update the liveintervals of sub-registers.
|
2008-02-10 18:45:23 +00:00
|
|
|
for (const unsigned *AS = tri_->getSubRegisters(IntB.reg); *AS; ++AS) {
|
2007-06-08 17:18:56 +00:00
|
|
|
LiveInterval &AliasLI = li_->getInterval(*AS);
|
|
|
|
AliasLI.addRange(LiveRange(FillerStart, FillerEnd,
|
2007-09-05 21:46:51 +00:00
|
|
|
AliasLI.getNextValue(FillerStart, 0, li_->getVNInfoAllocator())));
|
2007-06-08 17:18:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Okay, merge "B1" into the same value number as "B0".
|
2007-08-29 20:45:00 +00:00
|
|
|
if (BValNo != ValLR->valno)
|
|
|
|
IntB.MergeValueNumberInto(BValNo, ValLR->valno);
|
2008-02-10 18:45:23 +00:00
|
|
|
DOUT << " result = "; IntB.print(DOUT, tri_);
|
2007-06-08 17:18:56 +00:00
|
|
|
DOUT << "\n";
|
|
|
|
|
|
|
|
// If the source instruction was killing the source register before the
|
|
|
|
// merge, unset the isKill marker given the live range has been extended.
|
|
|
|
int UIdx = ValLREndInst->findRegisterUseOperandIdx(IntB.reg, true);
|
|
|
|
if (UIdx != -1)
|
2007-12-30 21:56:09 +00:00
|
|
|
ValLREndInst->getOperand(UIdx).setIsKill(false);
|
2008-02-13 03:01:43 +00:00
|
|
|
|
|
|
|
++numExtends;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2008-02-16 02:32:17 +00:00
|
|
|
/// HasOtherReachingDefs - Return true if there are definitions of IntB
|
|
|
|
/// other than BValNo val# that can reach uses of AValno val# of IntA.
|
|
|
|
bool SimpleRegisterCoalescing::HasOtherReachingDefs(LiveInterval &IntA,
|
|
|
|
LiveInterval &IntB,
|
|
|
|
VNInfo *AValNo,
|
|
|
|
VNInfo *BValNo) {
|
|
|
|
for (LiveInterval::iterator AI = IntA.begin(), AE = IntA.end();
|
|
|
|
AI != AE; ++AI) {
|
|
|
|
if (AI->valno != AValNo) continue;
|
|
|
|
LiveInterval::Ranges::iterator BI =
|
|
|
|
std::upper_bound(IntB.ranges.begin(), IntB.ranges.end(), AI->start);
|
|
|
|
if (BI != IntB.ranges.begin())
|
|
|
|
--BI;
|
|
|
|
for (; BI != IntB.ranges.end() && AI->end >= BI->start; ++BI) {
|
|
|
|
if (BI->valno == BValNo)
|
|
|
|
continue;
|
|
|
|
if (BI->start <= AI->start && BI->end > AI->start)
|
|
|
|
return true;
|
|
|
|
if (BI->start > AI->start && BI->start < AI->end)
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2008-02-13 03:01:43 +00:00
|
|
|
/// RemoveCopyByCommutingDef - We found a non-trivially-coalescable copy with IntA
|
|
|
|
/// being the source and IntB being the dest, thus this defines a value number
|
|
|
|
/// in IntB. If the source value number (in IntA) is defined by a commutable
|
|
|
|
/// instruction and its other operand is coalesced to the copy dest register,
|
|
|
|
/// see if we can transform the copy into a noop by commuting the definition. For
|
|
|
|
/// example,
|
|
|
|
///
|
|
|
|
/// A3 = op A2 B0<kill>
|
|
|
|
/// ...
|
|
|
|
/// B1 = A3 <- this copy
|
|
|
|
/// ...
|
|
|
|
/// = op A3 <- more uses
|
|
|
|
///
|
|
|
|
/// ==>
|
|
|
|
///
|
|
|
|
/// B2 = op B0 A2<kill>
|
|
|
|
/// ...
|
|
|
|
/// B1 = B2 <- now an identify copy
|
|
|
|
/// ...
|
|
|
|
/// = op B2 <- more uses
|
|
|
|
///
|
|
|
|
/// This returns true if an interval was modified.
|
|
|
|
///
|
|
|
|
bool SimpleRegisterCoalescing::RemoveCopyByCommutingDef(LiveInterval &IntA,
|
|
|
|
LiveInterval &IntB,
|
|
|
|
MachineInstr *CopyMI) {
|
|
|
|
unsigned CopyIdx = li_->getDefIndex(li_->getInstructionIndex(CopyMI));
|
|
|
|
|
2008-02-18 18:56:31 +00:00
|
|
|
// FIXME: For now, only eliminate the copy by commuting its def when the
|
|
|
|
// source register is a virtual register. We want to guard against cases
|
|
|
|
// where the copy is a back edge copy and commuting the def lengthen the
|
|
|
|
// live interval of the source register to the entire loop.
|
|
|
|
if (TargetRegisterInfo::isPhysicalRegister(IntA.reg))
|
2008-02-18 08:40:53 +00:00
|
|
|
return false;
|
|
|
|
|
2008-02-15 18:24:29 +00:00
|
|
|
// BValNo is a value number in B that is defined by a copy from A. 'B3' in
|
2008-02-13 03:01:43 +00:00
|
|
|
// the example above.
|
|
|
|
LiveInterval::iterator BLR = IntB.FindLiveRangeContaining(CopyIdx);
|
2008-04-16 18:48:43 +00:00
|
|
|
if (BLR == IntB.end()) // Should never happen!
|
|
|
|
return false;
|
2008-02-13 03:01:43 +00:00
|
|
|
VNInfo *BValNo = BLR->valno;
|
|
|
|
|
|
|
|
// Get the location that B is defined at. Two options: either this value has
|
|
|
|
// an unknown definition point or it is defined at CopyIdx. If unknown, we
|
|
|
|
// can't process it.
|
2008-02-15 18:24:29 +00:00
|
|
|
if (!BValNo->copy) return false;
|
2008-02-13 03:01:43 +00:00
|
|
|
assert(BValNo->def == CopyIdx && "Copy doesn't define the value?");
|
2007-06-08 17:18:56 +00:00
|
|
|
|
2008-02-13 03:01:43 +00:00
|
|
|
// AValNo is the value number in A that defines the copy, A3 in the example.
|
|
|
|
LiveInterval::iterator ALR = IntA.FindLiveRangeContaining(CopyIdx-1);
|
2008-04-16 18:48:43 +00:00
|
|
|
if (ALR == IntA.end()) // Should never happen!
|
|
|
|
return false;
|
2008-02-13 03:01:43 +00:00
|
|
|
VNInfo *AValNo = ALR->valno;
|
2008-02-13 08:41:08 +00:00
|
|
|
// If other defs can reach uses of this def, then it's not safe to perform
|
|
|
|
// the optimization.
|
|
|
|
if (AValNo->def == ~0U || AValNo->def == ~1U || AValNo->hasPHIKill)
|
2008-02-13 03:01:43 +00:00
|
|
|
return false;
|
|
|
|
MachineInstr *DefMI = li_->getInstructionFromIndex(AValNo->def);
|
|
|
|
const TargetInstrDesc &TID = DefMI->getDesc();
|
2008-02-15 18:24:29 +00:00
|
|
|
unsigned NewDstIdx;
|
|
|
|
if (!TID.isCommutable() ||
|
|
|
|
!tii_->CommuteChangesDestination(DefMI, NewDstIdx))
|
2008-02-13 03:01:43 +00:00
|
|
|
return false;
|
|
|
|
|
2008-02-15 18:24:29 +00:00
|
|
|
MachineOperand &NewDstMO = DefMI->getOperand(NewDstIdx);
|
|
|
|
unsigned NewReg = NewDstMO.getReg();
|
|
|
|
if (NewReg != IntB.reg || !NewDstMO.isKill())
|
2008-02-13 03:01:43 +00:00
|
|
|
return false;
|
|
|
|
|
|
|
|
// Make sure there are no other definitions of IntB that would reach the
|
|
|
|
// uses which the new definition can reach.
|
2008-02-16 02:32:17 +00:00
|
|
|
if (HasOtherReachingDefs(IntA, IntB, AValNo, BValNo))
|
|
|
|
return false;
|
2008-02-13 03:01:43 +00:00
|
|
|
|
2008-03-26 19:03:01 +00:00
|
|
|
// If some of the uses of IntA.reg is already coalesced away, return false.
|
|
|
|
// It's not possible to determine whether it's safe to perform the coalescing.
|
|
|
|
for (MachineRegisterInfo::use_iterator UI = mri_->use_begin(IntA.reg),
|
|
|
|
UE = mri_->use_end(); UI != UE; ++UI) {
|
|
|
|
MachineInstr *UseMI = &*UI;
|
|
|
|
unsigned UseIdx = li_->getInstructionIndex(UseMI);
|
|
|
|
LiveInterval::iterator ULR = IntA.FindLiveRangeContaining(UseIdx);
|
2008-04-16 18:48:43 +00:00
|
|
|
if (ULR == IntA.end())
|
|
|
|
continue;
|
2008-03-26 19:03:01 +00:00
|
|
|
if (ULR->valno == AValNo && JoinedCopies.count(UseMI))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2008-02-13 09:56:03 +00:00
|
|
|
// At this point we have decided that it is legal to do this
|
|
|
|
// transformation. Start by commuting the instruction.
|
2008-02-13 03:01:43 +00:00
|
|
|
MachineBasicBlock *MBB = DefMI->getParent();
|
|
|
|
MachineInstr *NewMI = tii_->commuteInstruction(DefMI);
|
2008-02-16 02:32:17 +00:00
|
|
|
if (!NewMI)
|
|
|
|
return false;
|
2008-02-13 03:01:43 +00:00
|
|
|
if (NewMI != DefMI) {
|
|
|
|
li_->ReplaceMachineInstrInMaps(DefMI, NewMI);
|
|
|
|
MBB->insert(DefMI, NewMI);
|
|
|
|
MBB->erase(DefMI);
|
|
|
|
}
|
2008-03-05 00:59:57 +00:00
|
|
|
unsigned OpIdx = NewMI->findRegisterUseOperandIdx(IntA.reg, false);
|
2008-02-13 03:01:43 +00:00
|
|
|
NewMI->getOperand(OpIdx).setIsKill();
|
|
|
|
|
|
|
|
bool BHasPHIKill = BValNo->hasPHIKill;
|
|
|
|
SmallVector<VNInfo*, 4> BDeadValNos;
|
|
|
|
SmallVector<unsigned, 4> BKills;
|
|
|
|
std::map<unsigned, unsigned> BExtend;
|
2008-03-10 08:11:32 +00:00
|
|
|
|
|
|
|
// If ALR and BLR overlaps and end of BLR extends beyond end of ALR, e.g.
|
|
|
|
// A = or A, B
|
|
|
|
// ...
|
|
|
|
// B = A
|
|
|
|
// ...
|
|
|
|
// C = A<kill>
|
|
|
|
// ...
|
|
|
|
// = B
|
|
|
|
//
|
|
|
|
// then do not add kills of A to the newly created B interval.
|
|
|
|
bool Extended = BLR->end > ALR->end && ALR->end != ALR->start;
|
|
|
|
if (Extended)
|
|
|
|
BExtend[ALR->end] = BLR->end;
|
|
|
|
|
|
|
|
// Update uses of IntA of the specific Val# with IntB.
|
2008-02-13 03:01:43 +00:00
|
|
|
for (MachineRegisterInfo::use_iterator UI = mri_->use_begin(IntA.reg),
|
|
|
|
UE = mri_->use_end(); UI != UE;) {
|
|
|
|
MachineOperand &UseMO = UI.getOperand();
|
2008-02-13 09:56:03 +00:00
|
|
|
MachineInstr *UseMI = &*UI;
|
2008-02-13 03:01:43 +00:00
|
|
|
++UI;
|
2008-02-13 09:56:03 +00:00
|
|
|
if (JoinedCopies.count(UseMI))
|
2008-03-26 19:03:01 +00:00
|
|
|
continue;
|
2008-02-13 03:01:43 +00:00
|
|
|
unsigned UseIdx = li_->getInstructionIndex(UseMI);
|
|
|
|
LiveInterval::iterator ULR = IntA.FindLiveRangeContaining(UseIdx);
|
2008-04-16 18:48:43 +00:00
|
|
|
if (ULR == IntA.end() || ULR->valno != AValNo)
|
2008-02-13 03:01:43 +00:00
|
|
|
continue;
|
|
|
|
UseMO.setReg(NewReg);
|
2008-02-13 09:56:03 +00:00
|
|
|
if (UseMI == CopyMI)
|
|
|
|
continue;
|
2008-03-10 08:11:32 +00:00
|
|
|
if (UseMO.isKill()) {
|
|
|
|
if (Extended)
|
|
|
|
UseMO.setIsKill(false);
|
|
|
|
else
|
|
|
|
BKills.push_back(li_->getUseIndex(UseIdx)+1);
|
|
|
|
}
|
2008-02-13 09:56:03 +00:00
|
|
|
unsigned SrcReg, DstReg;
|
|
|
|
if (!tii_->isMoveInstr(*UseMI, SrcReg, DstReg))
|
|
|
|
continue;
|
2008-02-15 18:24:29 +00:00
|
|
|
if (DstReg == IntB.reg) {
|
2008-02-13 09:56:03 +00:00
|
|
|
// This copy will become a noop. If it's defining a new val#,
|
|
|
|
// remove that val# as well. However this live range is being
|
|
|
|
// extended to the end of the existing live range defined by the copy.
|
|
|
|
unsigned DefIdx = li_->getDefIndex(UseIdx);
|
2008-04-16 18:48:43 +00:00
|
|
|
const LiveRange *DLR = IntB.getLiveRangeContaining(DefIdx);
|
2008-02-13 09:56:03 +00:00
|
|
|
BHasPHIKill |= DLR->valno->hasPHIKill;
|
|
|
|
assert(DLR->valno->def == DefIdx);
|
|
|
|
BDeadValNos.push_back(DLR->valno);
|
|
|
|
BExtend[DLR->start] = DLR->end;
|
|
|
|
JoinedCopies.insert(UseMI);
|
|
|
|
// If this is a kill but it's going to be removed, the last use
|
|
|
|
// of the same val# is the new kill.
|
2008-03-10 08:11:32 +00:00
|
|
|
if (UseMO.isKill())
|
2008-02-13 09:56:03 +00:00
|
|
|
BKills.pop_back();
|
2008-02-13 03:01:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// We need to insert a new liverange: [ALR.start, LastUse). It may be we can
|
|
|
|
// simply extend BLR if CopyMI doesn't end the range.
|
|
|
|
DOUT << "\nExtending: "; IntB.print(DOUT, tri_);
|
|
|
|
|
|
|
|
IntB.removeValNo(BValNo);
|
|
|
|
for (unsigned i = 0, e = BDeadValNos.size(); i != e; ++i)
|
|
|
|
IntB.removeValNo(BDeadValNos[i]);
|
2008-03-19 02:26:36 +00:00
|
|
|
VNInfo *ValNo = IntB.getNextValue(AValNo->def, 0, li_->getVNInfoAllocator());
|
2008-02-13 03:01:43 +00:00
|
|
|
for (LiveInterval::iterator AI = IntA.begin(), AE = IntA.end();
|
|
|
|
AI != AE; ++AI) {
|
|
|
|
if (AI->valno != AValNo) continue;
|
|
|
|
unsigned End = AI->end;
|
|
|
|
std::map<unsigned, unsigned>::iterator EI = BExtend.find(End);
|
|
|
|
if (EI != BExtend.end())
|
|
|
|
End = EI->second;
|
|
|
|
IntB.addRange(LiveRange(AI->start, End, ValNo));
|
|
|
|
}
|
|
|
|
IntB.addKills(ValNo, BKills);
|
|
|
|
ValNo->hasPHIKill = BHasPHIKill;
|
|
|
|
|
|
|
|
DOUT << " result = "; IntB.print(DOUT, tri_);
|
|
|
|
DOUT << "\n";
|
|
|
|
|
|
|
|
DOUT << "\nShortening: "; IntA.print(DOUT, tri_);
|
|
|
|
IntA.removeValNo(AValNo);
|
|
|
|
DOUT << " result = "; IntA.print(DOUT, tri_);
|
|
|
|
DOUT << "\n";
|
|
|
|
|
|
|
|
++numCommutes;
|
2007-06-08 17:18:56 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2007-11-06 08:52:21 +00:00
|
|
|
/// isBackEdgeCopy - Returns true if CopyMI is a back edge copy.
|
|
|
|
///
|
|
|
|
bool SimpleRegisterCoalescing::isBackEdgeCopy(MachineInstr *CopyMI,
|
2008-04-09 20:57:25 +00:00
|
|
|
unsigned DstReg) const {
|
2007-11-06 08:52:21 +00:00
|
|
|
MachineBasicBlock *MBB = CopyMI->getParent();
|
2007-12-11 02:09:15 +00:00
|
|
|
const MachineLoop *L = loopInfo->getLoopFor(MBB);
|
2007-11-06 08:52:21 +00:00
|
|
|
if (!L)
|
|
|
|
return false;
|
2007-12-11 02:09:15 +00:00
|
|
|
if (MBB != L->getLoopLatch())
|
2007-11-06 08:52:21 +00:00
|
|
|
return false;
|
|
|
|
|
|
|
|
LiveInterval &LI = li_->getInterval(DstReg);
|
|
|
|
unsigned DefIdx = li_->getInstructionIndex(CopyMI);
|
|
|
|
LiveInterval::const_iterator DstLR =
|
|
|
|
LI.FindLiveRangeContaining(li_->getDefIndex(DefIdx));
|
|
|
|
if (DstLR == LI.end())
|
|
|
|
return false;
|
2008-02-13 03:01:43 +00:00
|
|
|
unsigned KillIdx = li_->getInstructionIndex(&MBB->back()) + InstrSlots::NUM;
|
|
|
|
if (DstLR->valno->kills.size() == 1 &&
|
|
|
|
DstLR->valno->kills[0] == KillIdx && DstLR->valno->hasPHIKill)
|
2007-11-06 08:52:21 +00:00
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2008-02-15 18:24:29 +00:00
|
|
|
/// UpdateRegDefsUses - Replace all defs and uses of SrcReg to DstReg and
|
|
|
|
/// update the subregister number if it is not zero. If DstReg is a
|
|
|
|
/// physical register and the existing subregister number of the def / use
|
|
|
|
/// being updated is not zero, make sure to set it to the correct physical
|
|
|
|
/// subregister.
|
|
|
|
void
|
|
|
|
SimpleRegisterCoalescing::UpdateRegDefsUses(unsigned SrcReg, unsigned DstReg,
|
|
|
|
unsigned SubIdx) {
|
|
|
|
bool DstIsPhys = TargetRegisterInfo::isPhysicalRegister(DstReg);
|
|
|
|
if (DstIsPhys && SubIdx) {
|
|
|
|
// Figure out the real physical register we are updating with.
|
|
|
|
DstReg = tri_->getSubReg(DstReg, SubIdx);
|
|
|
|
SubIdx = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (MachineRegisterInfo::reg_iterator I = mri_->reg_begin(SrcReg),
|
|
|
|
E = mri_->reg_end(); I != E; ) {
|
|
|
|
MachineOperand &O = I.getOperand();
|
2008-03-21 19:09:30 +00:00
|
|
|
MachineInstr *UseMI = &*I;
|
2008-02-15 18:24:29 +00:00
|
|
|
++I;
|
2008-04-17 00:06:42 +00:00
|
|
|
unsigned OldSubIdx = O.getSubReg();
|
2008-02-15 18:24:29 +00:00
|
|
|
if (DstIsPhys) {
|
|
|
|
unsigned UseDstReg = DstReg;
|
2008-04-17 00:06:42 +00:00
|
|
|
if (OldSubIdx)
|
|
|
|
UseDstReg = tri_->getSubReg(DstReg, OldSubIdx);
|
2008-02-15 18:24:29 +00:00
|
|
|
O.setReg(UseDstReg);
|
|
|
|
O.setSubReg(0);
|
|
|
|
} else {
|
2008-02-26 08:03:41 +00:00
|
|
|
// Sub-register indexes goes from small to large. e.g.
|
2008-04-18 19:25:26 +00:00
|
|
|
// RAX: 1 -> AL, 2 -> AX, 3 -> EAX
|
|
|
|
// EAX: 1 -> AL, 2 -> AX
|
2008-02-26 08:03:41 +00:00
|
|
|
// So RAX's sub-register 2 is AX, RAX's sub-regsiter 3 is EAX, whose
|
|
|
|
// sub-register 2 is also AX.
|
|
|
|
if (SubIdx && OldSubIdx && SubIdx != OldSubIdx)
|
|
|
|
assert(OldSubIdx < SubIdx && "Conflicting sub-register index!");
|
|
|
|
else if (SubIdx)
|
2008-02-15 18:24:29 +00:00
|
|
|
O.setSubReg(SubIdx);
|
2008-03-21 19:09:30 +00:00
|
|
|
// Remove would-be duplicated kill marker.
|
|
|
|
if (O.isKill() && UseMI->killsRegister(DstReg))
|
|
|
|
O.setIsKill(false);
|
2008-02-15 18:24:29 +00:00
|
|
|
O.setReg(DstReg);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-04-09 20:57:25 +00:00
|
|
|
/// RemoveDeadImpDef - Remove implicit_def instructions which are "re-defining"
|
|
|
|
/// registers due to insert_subreg coalescing. e.g.
|
|
|
|
/// r1024 = op
|
|
|
|
/// r1025 = implicit_def
|
|
|
|
/// r1025 = insert_subreg r1025, r1024
|
|
|
|
/// = op r1025
|
|
|
|
/// =>
|
|
|
|
/// r1025 = op
|
|
|
|
/// r1025 = implicit_def
|
|
|
|
/// r1025 = insert_subreg r1025, r1025
|
|
|
|
/// = op r1025
|
|
|
|
void
|
|
|
|
SimpleRegisterCoalescing::RemoveDeadImpDef(unsigned Reg, LiveInterval &LI) {
|
|
|
|
for (MachineRegisterInfo::reg_iterator I = mri_->reg_begin(Reg),
|
|
|
|
E = mri_->reg_end(); I != E; ) {
|
|
|
|
MachineOperand &O = I.getOperand();
|
|
|
|
MachineInstr *DefMI = &*I;
|
|
|
|
++I;
|
|
|
|
if (!O.isDef())
|
|
|
|
continue;
|
|
|
|
if (DefMI->getOpcode() != TargetInstrInfo::IMPLICIT_DEF)
|
|
|
|
continue;
|
|
|
|
if (!LI.liveBeforeAndAt(li_->getInstructionIndex(DefMI)))
|
|
|
|
continue;
|
|
|
|
li_->RemoveMachineInstrFromMaps(DefMI);
|
|
|
|
DefMI->eraseFromParent();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-03-10 08:11:32 +00:00
|
|
|
/// RemoveUnnecessaryKills - Remove kill markers that are no longer accurate
|
|
|
|
/// due to live range lengthening as the result of coalescing.
|
|
|
|
void SimpleRegisterCoalescing::RemoveUnnecessaryKills(unsigned Reg,
|
|
|
|
LiveInterval &LI) {
|
|
|
|
for (MachineRegisterInfo::use_iterator UI = mri_->use_begin(Reg),
|
|
|
|
UE = mri_->use_end(); UI != UE; ++UI) {
|
|
|
|
MachineOperand &UseMO = UI.getOperand();
|
|
|
|
if (UseMO.isKill()) {
|
|
|
|
MachineInstr *UseMI = UseMO.getParent();
|
|
|
|
unsigned SReg, DReg;
|
|
|
|
if (!tii_->isMoveInstr(*UseMI, SReg, DReg))
|
|
|
|
continue;
|
|
|
|
unsigned UseIdx = li_->getUseIndex(li_->getInstructionIndex(UseMI));
|
|
|
|
if (JoinedCopies.count(UseMI))
|
|
|
|
continue;
|
2008-04-16 18:48:43 +00:00
|
|
|
const LiveRange *UI = LI.getLiveRangeContaining(UseIdx);
|
2008-03-10 08:11:32 +00:00
|
|
|
if (!LI.isKill(UI->valno, UseIdx+1))
|
|
|
|
UseMO.setIsKill(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-03-18 08:26:47 +00:00
|
|
|
/// removeRange - Wrapper for LiveInterval::removeRange. This removes a range
|
|
|
|
/// from a physical register live interval as well as from the live intervals
|
|
|
|
/// of its sub-registers.
|
|
|
|
static void removeRange(LiveInterval &li, unsigned Start, unsigned End,
|
|
|
|
LiveIntervals *li_, const TargetRegisterInfo *tri_) {
|
|
|
|
li.removeRange(Start, End, true);
|
|
|
|
if (TargetRegisterInfo::isPhysicalRegister(li.reg)) {
|
|
|
|
for (const unsigned* SR = tri_->getSubRegisters(li.reg); *SR; ++SR) {
|
|
|
|
if (!li_->hasInterval(*SR))
|
|
|
|
continue;
|
|
|
|
LiveInterval &sli = li_->getInterval(*SR);
|
|
|
|
unsigned RemoveEnd = Start;
|
|
|
|
while (RemoveEnd != End) {
|
|
|
|
LiveInterval::iterator LR = sli.FindLiveRangeContaining(Start);
|
|
|
|
if (LR == sli.end())
|
|
|
|
break;
|
|
|
|
RemoveEnd = (LR->end < End) ? LR->end : End;
|
|
|
|
sli.removeRange(Start, RemoveEnd, true);
|
|
|
|
Start = RemoveEnd;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// removeIntervalIfEmpty - Check if the live interval of a physical register
|
|
|
|
/// is empty, if so remove it and also remove the empty intervals of its
|
2008-04-16 20:24:25 +00:00
|
|
|
/// sub-registers. Return true if live interval is removed.
|
|
|
|
static bool removeIntervalIfEmpty(LiveInterval &li, LiveIntervals *li_,
|
2008-03-18 08:26:47 +00:00
|
|
|
const TargetRegisterInfo *tri_) {
|
|
|
|
if (li.empty()) {
|
|
|
|
if (TargetRegisterInfo::isPhysicalRegister(li.reg))
|
|
|
|
for (const unsigned* SR = tri_->getSubRegisters(li.reg); *SR; ++SR) {
|
|
|
|
if (!li_->hasInterval(*SR))
|
|
|
|
continue;
|
|
|
|
LiveInterval &sli = li_->getInterval(*SR);
|
|
|
|
if (sli.empty())
|
|
|
|
li_->removeInterval(*SR);
|
|
|
|
}
|
2008-04-16 01:22:28 +00:00
|
|
|
li_->removeInterval(li.reg);
|
2008-04-16 20:24:25 +00:00
|
|
|
return true;
|
2008-03-18 08:26:47 +00:00
|
|
|
}
|
2008-04-16 20:24:25 +00:00
|
|
|
return false;
|
2008-03-18 08:26:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// ShortenDeadCopyLiveRange - Shorten a live range defined by a dead copy.
|
2008-04-16 20:24:25 +00:00
|
|
|
/// Return true if live interval is removed.
|
|
|
|
bool SimpleRegisterCoalescing::ShortenDeadCopyLiveRange(LiveInterval &li,
|
2008-03-05 22:09:42 +00:00
|
|
|
MachineInstr *CopyMI) {
|
|
|
|
unsigned CopyIdx = li_->getInstructionIndex(CopyMI);
|
|
|
|
LiveInterval::iterator MLR =
|
|
|
|
li.FindLiveRangeContaining(li_->getDefIndex(CopyIdx));
|
2008-03-18 08:26:47 +00:00
|
|
|
if (MLR == li.end())
|
2008-04-16 20:24:25 +00:00
|
|
|
return false; // Already removed by ShortenDeadCopySrcLiveRange.
|
2008-03-05 22:09:42 +00:00
|
|
|
unsigned RemoveStart = MLR->start;
|
|
|
|
unsigned RemoveEnd = MLR->end;
|
2008-03-18 08:26:47 +00:00
|
|
|
// Remove the liverange that's defined by this.
|
|
|
|
if (RemoveEnd == li_->getDefIndex(CopyIdx)+1) {
|
|
|
|
removeRange(li, RemoveStart, RemoveEnd, li_, tri_);
|
2008-04-16 20:24:25 +00:00
|
|
|
return removeIntervalIfEmpty(li, li_, tri_);
|
2008-03-18 08:26:47 +00:00
|
|
|
}
|
2008-04-16 20:24:25 +00:00
|
|
|
return false;
|
2008-03-18 08:26:47 +00:00
|
|
|
}
|
|
|
|
|
2008-03-26 20:15:49 +00:00
|
|
|
/// PropagateDeadness - Propagate the dead marker to the instruction which
|
|
|
|
/// defines the val#.
|
|
|
|
static void PropagateDeadness(LiveInterval &li, MachineInstr *CopyMI,
|
|
|
|
unsigned &LRStart, LiveIntervals *li_,
|
|
|
|
const TargetRegisterInfo* tri_) {
|
|
|
|
MachineInstr *DefMI =
|
|
|
|
li_->getInstructionFromIndex(li_->getDefIndex(LRStart));
|
|
|
|
if (DefMI && DefMI != CopyMI) {
|
|
|
|
int DeadIdx = DefMI->findRegisterDefOperandIdx(li.reg, false, tri_);
|
|
|
|
if (DeadIdx != -1) {
|
|
|
|
DefMI->getOperand(DeadIdx).setIsDead();
|
|
|
|
// A dead def should have a single cycle interval.
|
|
|
|
++LRStart;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-04-18 19:22:23 +00:00
|
|
|
/// isSameOrFallThroughBB - Return true if MBB == SuccMBB or MBB simply
|
|
|
|
/// fallthoughs to SuccMBB.
|
|
|
|
static bool isSameOrFallThroughBB(MachineBasicBlock *MBB,
|
|
|
|
MachineBasicBlock *SuccMBB,
|
|
|
|
const TargetInstrInfo *tii_) {
|
|
|
|
if (MBB == SuccMBB)
|
|
|
|
return true;
|
|
|
|
MachineBasicBlock *TBB = 0, *FBB = 0;
|
|
|
|
std::vector<MachineOperand> Cond;
|
|
|
|
return !tii_->AnalyzeBranch(*MBB, TBB, FBB, Cond) && !TBB && !FBB &&
|
|
|
|
MBB->isSuccessor(SuccMBB);
|
|
|
|
}
|
|
|
|
|
2008-04-17 05:20:39 +00:00
|
|
|
/// ShortenDeadCopySrcLiveRange - Shorten a live range as it's artificially
|
|
|
|
/// extended by a dead copy. Mark the last use (if any) of the val# as kill as
|
|
|
|
/// ends the live range there. If there isn't another use, then this live range
|
|
|
|
/// is dead. Return true if live interval is removed.
|
2008-04-16 20:24:25 +00:00
|
|
|
bool
|
2008-03-18 08:26:47 +00:00
|
|
|
SimpleRegisterCoalescing::ShortenDeadCopySrcLiveRange(LiveInterval &li,
|
|
|
|
MachineInstr *CopyMI) {
|
|
|
|
unsigned CopyIdx = li_->getInstructionIndex(CopyMI);
|
|
|
|
if (CopyIdx == 0) {
|
|
|
|
// FIXME: special case: function live in. It can be a general case if the
|
|
|
|
// first instruction index starts at > 0 value.
|
|
|
|
assert(TargetRegisterInfo::isPhysicalRegister(li.reg));
|
|
|
|
// Live-in to the function but dead. Remove it from entry live-in set.
|
2008-04-24 09:06:33 +00:00
|
|
|
if (mf_->begin()->isLiveIn(li.reg))
|
|
|
|
mf_->begin()->removeLiveIn(li.reg);
|
2008-04-16 18:48:43 +00:00
|
|
|
const LiveRange *LR = li.getLiveRangeContaining(CopyIdx);
|
2008-03-18 08:26:47 +00:00
|
|
|
removeRange(li, LR->start, LR->end, li_, tri_);
|
2008-04-16 20:24:25 +00:00
|
|
|
return removeIntervalIfEmpty(li, li_, tri_);
|
2008-03-18 08:26:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
LiveInterval::iterator LR = li.FindLiveRangeContaining(CopyIdx-1);
|
|
|
|
if (LR == li.end())
|
|
|
|
// Livein but defined by a phi.
|
2008-04-16 20:24:25 +00:00
|
|
|
return false;
|
2008-03-18 08:26:47 +00:00
|
|
|
|
|
|
|
unsigned RemoveStart = LR->start;
|
|
|
|
unsigned RemoveEnd = li_->getDefIndex(CopyIdx)+1;
|
|
|
|
if (LR->end > RemoveEnd)
|
|
|
|
// More uses past this copy? Nothing to do.
|
2008-04-16 20:24:25 +00:00
|
|
|
return false;
|
2008-03-18 08:26:47 +00:00
|
|
|
|
2008-04-18 19:22:23 +00:00
|
|
|
MachineBasicBlock *CopyMBB = CopyMI->getParent();
|
|
|
|
unsigned MBBStart = li_->getMBBStartIdx(CopyMBB);
|
2008-03-05 22:09:42 +00:00
|
|
|
unsigned LastUseIdx;
|
2008-04-10 23:48:35 +00:00
|
|
|
MachineOperand *LastUse = lastRegisterUse(LR->start, CopyIdx-1, li.reg,
|
|
|
|
LastUseIdx);
|
2008-03-05 22:09:42 +00:00
|
|
|
if (LastUse) {
|
2008-04-18 19:22:23 +00:00
|
|
|
MachineInstr *LastUseMI = LastUse->getParent();
|
|
|
|
if (!isSameOrFallThroughBB(LastUseMI->getParent(), CopyMBB, tii_)) {
|
|
|
|
// r1024 = op
|
|
|
|
// ...
|
|
|
|
// BB1:
|
|
|
|
// = r1024
|
|
|
|
//
|
|
|
|
// BB2:
|
|
|
|
// r1025<dead> = r1024<kill>
|
|
|
|
if (MBBStart < LR->end)
|
|
|
|
removeRange(li, MBBStart, LR->end, li_, tri_);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2008-03-18 08:26:47 +00:00
|
|
|
// There are uses before the copy, just shorten the live range to the end
|
|
|
|
// of last use.
|
2008-03-05 22:09:42 +00:00
|
|
|
LastUse->setIsKill();
|
2008-03-18 08:26:47 +00:00
|
|
|
removeRange(li, li_->getDefIndex(LastUseIdx), LR->end, li_, tri_);
|
|
|
|
unsigned SrcReg, DstReg;
|
|
|
|
if (tii_->isMoveInstr(*LastUseMI, SrcReg, DstReg) &&
|
|
|
|
DstReg == li.reg) {
|
|
|
|
// Last use is itself an identity code.
|
|
|
|
int DeadIdx = LastUseMI->findRegisterDefOperandIdx(li.reg, false, tri_);
|
|
|
|
LastUseMI->getOperand(DeadIdx).setIsDead();
|
|
|
|
}
|
2008-04-16 20:24:25 +00:00
|
|
|
return false;
|
2008-03-05 22:09:42 +00:00
|
|
|
}
|
2008-03-18 08:26:47 +00:00
|
|
|
|
|
|
|
// Is it livein?
|
|
|
|
if (LR->start <= MBBStart && LR->end > MBBStart) {
|
|
|
|
if (LR->start == 0) {
|
|
|
|
assert(TargetRegisterInfo::isPhysicalRegister(li.reg));
|
|
|
|
// Live-in to the function but dead. Remove it from entry live-in set.
|
|
|
|
mf_->begin()->removeLiveIn(li.reg);
|
|
|
|
}
|
|
|
|
// FIXME: Shorten intervals in BBs that reaches this BB.
|
|
|
|
}
|
|
|
|
|
2008-03-26 20:15:49 +00:00
|
|
|
if (LR->valno->def == RemoveStart)
|
|
|
|
// If the def MI defines the val#, propagate the dead marker.
|
|
|
|
PropagateDeadness(li, CopyMI, RemoveStart, li_, tri_);
|
|
|
|
|
|
|
|
removeRange(li, RemoveStart, LR->end, li_, tri_);
|
2008-04-16 20:24:25 +00:00
|
|
|
return removeIntervalIfEmpty(li, li_, tri_);
|
2008-03-05 22:09:42 +00:00
|
|
|
}
|
|
|
|
|
2008-04-09 20:57:25 +00:00
|
|
|
/// CanCoalesceWithImpDef - Returns true if the specified copy instruction
|
|
|
|
/// from an implicit def to another register can be coalesced away.
|
|
|
|
bool SimpleRegisterCoalescing::CanCoalesceWithImpDef(MachineInstr *CopyMI,
|
|
|
|
LiveInterval &li,
|
|
|
|
LiveInterval &ImpLi) const{
|
|
|
|
if (!CopyMI->killsRegister(ImpLi.reg))
|
|
|
|
return false;
|
|
|
|
unsigned CopyIdx = li_->getDefIndex(li_->getInstructionIndex(CopyMI));
|
|
|
|
LiveInterval::iterator LR = li.FindLiveRangeContaining(CopyIdx);
|
|
|
|
if (LR == li.end())
|
|
|
|
return false;
|
|
|
|
if (LR->valno->hasPHIKill)
|
|
|
|
return false;
|
|
|
|
if (LR->valno->def != CopyIdx)
|
|
|
|
return false;
|
|
|
|
// Make sure all of val# uses are copies.
|
|
|
|
for (MachineRegisterInfo::use_iterator UI = mri_->use_begin(li.reg),
|
|
|
|
UE = mri_->use_end(); UI != UE;) {
|
|
|
|
MachineInstr *UseMI = &*UI;
|
|
|
|
++UI;
|
|
|
|
if (JoinedCopies.count(UseMI))
|
|
|
|
continue;
|
|
|
|
unsigned UseIdx = li_->getUseIndex(li_->getInstructionIndex(UseMI));
|
|
|
|
LiveInterval::iterator ULR = li.FindLiveRangeContaining(UseIdx);
|
2008-04-16 18:48:43 +00:00
|
|
|
if (ULR == li.end() || ULR->valno != LR->valno)
|
2008-04-09 20:57:25 +00:00
|
|
|
continue;
|
|
|
|
// If the use is not a use, then it's not safe to coalesce the move.
|
|
|
|
unsigned SrcReg, DstReg;
|
|
|
|
if (!tii_->isMoveInstr(*UseMI, SrcReg, DstReg)) {
|
|
|
|
if (UseMI->getOpcode() == TargetInstrInfo::INSERT_SUBREG &&
|
|
|
|
UseMI->getOperand(1).getReg() == li.reg)
|
|
|
|
continue;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// RemoveCopiesFromValNo - The specified value# is defined by an implicit
|
|
|
|
/// def and it is being removed. Turn all copies from this value# into
|
|
|
|
/// identity copies so they will be removed.
|
|
|
|
void SimpleRegisterCoalescing::RemoveCopiesFromValNo(LiveInterval &li,
|
|
|
|
VNInfo *VNI) {
|
2008-04-10 23:48:35 +00:00
|
|
|
MachineInstr *ImpDef = NULL;
|
|
|
|
MachineOperand *LastUse = NULL;
|
|
|
|
unsigned LastUseIdx = li_->getUseIndex(VNI->def);
|
|
|
|
for (MachineRegisterInfo::reg_iterator RI = mri_->reg_begin(li.reg),
|
|
|
|
RE = mri_->reg_end(); RI != RE;) {
|
|
|
|
MachineOperand *MO = &RI.getOperand();
|
|
|
|
MachineInstr *MI = &*RI;
|
|
|
|
++RI;
|
|
|
|
if (MO->isDef()) {
|
|
|
|
if (MI->getOpcode() == TargetInstrInfo::IMPLICIT_DEF) {
|
|
|
|
assert(!ImpDef && "Multiple implicit_def defining same register?");
|
|
|
|
ImpDef = MI;
|
|
|
|
}
|
2008-04-09 20:57:25 +00:00
|
|
|
continue;
|
2008-04-10 23:48:35 +00:00
|
|
|
}
|
|
|
|
if (JoinedCopies.count(MI))
|
|
|
|
continue;
|
|
|
|
unsigned UseIdx = li_->getUseIndex(li_->getInstructionIndex(MI));
|
2008-04-09 20:57:25 +00:00
|
|
|
LiveInterval::iterator ULR = li.FindLiveRangeContaining(UseIdx);
|
2008-04-16 18:48:43 +00:00
|
|
|
if (ULR == li.end() || ULR->valno != VNI)
|
2008-04-09 20:57:25 +00:00
|
|
|
continue;
|
|
|
|
// If the use is a copy, turn it into an identity copy.
|
|
|
|
unsigned SrcReg, DstReg;
|
2008-04-10 23:48:35 +00:00
|
|
|
if (tii_->isMoveInstr(*MI, SrcReg, DstReg) && SrcReg == li.reg) {
|
|
|
|
// Each use MI may have multiple uses of this register. Change them all.
|
|
|
|
for (unsigned i = 0, e = MI->getNumOperands(); i != e; ++i) {
|
|
|
|
MachineOperand &MO = MI->getOperand(i);
|
|
|
|
if (MO.isReg() && MO.getReg() == li.reg)
|
|
|
|
MO.setReg(DstReg);
|
|
|
|
}
|
|
|
|
JoinedCopies.insert(MI);
|
|
|
|
} else if (UseIdx > LastUseIdx) {
|
|
|
|
LastUseIdx = UseIdx;
|
|
|
|
LastUse = MO;
|
2008-04-10 18:38:47 +00:00
|
|
|
}
|
2008-04-10 23:48:35 +00:00
|
|
|
}
|
|
|
|
if (LastUse)
|
|
|
|
LastUse->setIsKill();
|
|
|
|
else {
|
|
|
|
// Remove dead implicit_def.
|
|
|
|
li_->RemoveMachineInstrFromMaps(ImpDef);
|
|
|
|
ImpDef->eraseFromParent();
|
2008-04-09 20:57:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static unsigned getMatchingSuperReg(unsigned Reg, unsigned SubIdx,
|
|
|
|
const TargetRegisterClass *RC,
|
|
|
|
const TargetRegisterInfo* TRI) {
|
|
|
|
for (const unsigned *SRs = TRI->getSuperRegisters(Reg);
|
|
|
|
unsigned SR = *SRs; ++SRs)
|
|
|
|
if (Reg == TRI->getSubReg(SR, SubIdx) && RC->contains(SR))
|
|
|
|
return SR;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-06-08 17:18:56 +00:00
|
|
|
/// JoinCopy - Attempt to join intervals corresponding to SrcReg/DstReg,
|
|
|
|
/// which are the src/dst of the copy instruction CopyMI. This returns true
|
2007-11-01 06:22:48 +00:00
|
|
|
/// if the copy was successfully coalesced away. If it is not currently
|
|
|
|
/// possible to coalesce this interval, but it may be possible if other
|
|
|
|
/// things get coalesced, then it returns true by reference in 'Again'.
|
2008-02-13 03:01:43 +00:00
|
|
|
bool SimpleRegisterCoalescing::JoinCopy(CopyRec &TheCopy, bool &Again) {
|
2007-11-06 08:52:21 +00:00
|
|
|
MachineInstr *CopyMI = TheCopy.MI;
|
|
|
|
|
|
|
|
Again = false;
|
|
|
|
if (JoinedCopies.count(CopyMI))
|
|
|
|
return false; // Already done.
|
|
|
|
|
2007-06-08 17:18:56 +00:00
|
|
|
DOUT << li_->getInstructionIndex(CopyMI) << '\t' << *CopyMI;
|
|
|
|
|
2008-02-15 18:24:29 +00:00
|
|
|
unsigned SrcReg;
|
|
|
|
unsigned DstReg;
|
|
|
|
bool isExtSubReg = CopyMI->getOpcode() == TargetInstrInfo::EXTRACT_SUBREG;
|
2008-04-09 20:57:25 +00:00
|
|
|
bool isInsSubReg = CopyMI->getOpcode() == TargetInstrInfo::INSERT_SUBREG;
|
2008-02-15 18:24:29 +00:00
|
|
|
unsigned SubIdx = 0;
|
|
|
|
if (isExtSubReg) {
|
|
|
|
DstReg = CopyMI->getOperand(0).getReg();
|
|
|
|
SrcReg = CopyMI->getOperand(1).getReg();
|
2008-04-09 20:57:25 +00:00
|
|
|
} else if (isInsSubReg) {
|
|
|
|
if (CopyMI->getOperand(2).getSubReg()) {
|
|
|
|
DOUT << "\tSource of insert_subreg is already coalesced "
|
|
|
|
<< "to another register.\n";
|
|
|
|
return false; // Not coalescable.
|
|
|
|
}
|
|
|
|
DstReg = CopyMI->getOperand(0).getReg();
|
|
|
|
SrcReg = CopyMI->getOperand(2).getReg();
|
2008-02-15 18:24:29 +00:00
|
|
|
} else if (!tii_->isMoveInstr(*CopyMI, SrcReg, DstReg)) {
|
|
|
|
assert(0 && "Unrecognized copy instruction!");
|
|
|
|
return false;
|
2008-02-13 03:01:43 +00:00
|
|
|
}
|
|
|
|
|
2007-06-08 17:18:56 +00:00
|
|
|
// If they are already joined we continue.
|
2008-02-15 18:24:29 +00:00
|
|
|
if (SrcReg == DstReg) {
|
2007-07-09 12:00:59 +00:00
|
|
|
DOUT << "\tCopy already coalesced.\n";
|
2007-11-01 06:22:48 +00:00
|
|
|
return false; // Not coalescable.
|
2007-06-08 17:18:56 +00:00
|
|
|
}
|
|
|
|
|
2008-02-15 18:24:29 +00:00
|
|
|
bool SrcIsPhys = TargetRegisterInfo::isPhysicalRegister(SrcReg);
|
|
|
|
bool DstIsPhys = TargetRegisterInfo::isPhysicalRegister(DstReg);
|
2007-06-08 17:18:56 +00:00
|
|
|
|
|
|
|
// If they are both physical registers, we cannot join them.
|
|
|
|
if (SrcIsPhys && DstIsPhys) {
|
2007-07-09 12:00:59 +00:00
|
|
|
DOUT << "\tCan not coalesce physregs.\n";
|
2007-11-01 06:22:48 +00:00
|
|
|
return false; // Not coalescable.
|
2007-06-08 17:18:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// We only join virtual registers with allocatable physical registers.
|
2008-02-15 18:24:29 +00:00
|
|
|
if (SrcIsPhys && !allocatableRegs_[SrcReg]) {
|
2007-06-08 17:18:56 +00:00
|
|
|
DOUT << "\tSrc reg is unallocatable physreg.\n";
|
2007-11-01 06:22:48 +00:00
|
|
|
return false; // Not coalescable.
|
2007-06-08 17:18:56 +00:00
|
|
|
}
|
2008-02-15 18:24:29 +00:00
|
|
|
if (DstIsPhys && !allocatableRegs_[DstReg]) {
|
2007-06-08 17:18:56 +00:00
|
|
|
DOUT << "\tDst reg is unallocatable physreg.\n";
|
2007-11-01 06:22:48 +00:00
|
|
|
return false; // Not coalescable.
|
2007-06-08 17:18:56 +00:00
|
|
|
}
|
2007-10-12 08:50:34 +00:00
|
|
|
|
|
|
|
unsigned RealDstReg = 0;
|
2008-04-09 20:57:25 +00:00
|
|
|
unsigned RealSrcReg = 0;
|
|
|
|
if (isExtSubReg || isInsSubReg) {
|
|
|
|
SubIdx = CopyMI->getOperand(isExtSubReg ? 2 : 3).getImm();
|
|
|
|
if (SrcIsPhys && isExtSubReg) {
|
2007-10-12 08:50:34 +00:00
|
|
|
// r1024 = EXTRACT_SUBREG EAX, 0 then r1024 is really going to be
|
|
|
|
// coalesced with AX.
|
2008-04-17 00:06:42 +00:00
|
|
|
unsigned DstSubIdx = CopyMI->getOperand(0).getSubReg();
|
2008-04-17 07:58:04 +00:00
|
|
|
if (DstSubIdx) {
|
|
|
|
// r1024<2> = EXTRACT_SUBREG EAX, 2. Then r1024 has already been
|
|
|
|
// coalesced to a larger register so the subreg indices cancel out.
|
|
|
|
if (DstSubIdx != SubIdx) {
|
|
|
|
DOUT << "\t Sub-register indices mismatch.\n";
|
|
|
|
return false; // Not coalescable.
|
|
|
|
}
|
|
|
|
} else
|
2008-04-17 00:06:42 +00:00
|
|
|
SrcReg = tri_->getSubReg(SrcReg, SubIdx);
|
2008-02-15 18:24:29 +00:00
|
|
|
SubIdx = 0;
|
2008-04-09 20:57:25 +00:00
|
|
|
} else if (DstIsPhys && isInsSubReg) {
|
|
|
|
// EAX = INSERT_SUBREG EAX, r1024, 0
|
2008-04-17 00:06:42 +00:00
|
|
|
unsigned SrcSubIdx = CopyMI->getOperand(2).getSubReg();
|
2008-04-17 07:58:04 +00:00
|
|
|
if (SrcSubIdx) {
|
|
|
|
// EAX = INSERT_SUBREG EAX, r1024<2>, 2 Then r1024 has already been
|
|
|
|
// coalesced to a larger register so the subreg indices cancel out.
|
|
|
|
if (SrcSubIdx != SubIdx) {
|
|
|
|
DOUT << "\t Sub-register indices mismatch.\n";
|
|
|
|
return false; // Not coalescable.
|
|
|
|
}
|
|
|
|
} else
|
|
|
|
DstReg = tri_->getSubReg(DstReg, SubIdx);
|
2008-04-09 20:57:25 +00:00
|
|
|
SubIdx = 0;
|
|
|
|
} else if ((DstIsPhys && isExtSubReg) || (SrcIsPhys && isInsSubReg)) {
|
2007-10-12 08:50:34 +00:00
|
|
|
// If this is a extract_subreg where dst is a physical register, e.g.
|
|
|
|
// cl = EXTRACT_SUBREG reg1024, 1
|
|
|
|
// then create and update the actual physical register allocated to RHS.
|
2008-04-09 20:57:25 +00:00
|
|
|
// Ditto for
|
|
|
|
// reg1024 = INSERT_SUBREG r1024, cl, 1
|
2008-04-17 07:58:04 +00:00
|
|
|
if (CopyMI->getOperand(1).getSubReg()) {
|
|
|
|
DOUT << "\tSrc of extract_ / insert_subreg already coalesced with reg"
|
|
|
|
<< " of a super-class.\n";
|
|
|
|
return false; // Not coalescable.
|
|
|
|
}
|
2008-04-09 20:57:25 +00:00
|
|
|
const TargetRegisterClass *RC =
|
|
|
|
mri_->getRegClass(isExtSubReg ? SrcReg : DstReg);
|
|
|
|
if (isExtSubReg) {
|
|
|
|
RealDstReg = getMatchingSuperReg(DstReg, SubIdx, RC, tri_);
|
|
|
|
assert(RealDstReg && "Invalid extra_subreg instruction!");
|
|
|
|
} else {
|
|
|
|
RealSrcReg = getMatchingSuperReg(SrcReg, SubIdx, RC, tri_);
|
|
|
|
assert(RealSrcReg && "Invalid extra_subreg instruction!");
|
2007-10-12 08:50:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// For this type of EXTRACT_SUBREG, conservatively
|
|
|
|
// check if the live interval of the source register interfere with the
|
|
|
|
// actual super physical register we are trying to coalesce with.
|
2008-04-09 20:57:25 +00:00
|
|
|
unsigned PhysReg = isExtSubReg ? RealDstReg : RealSrcReg;
|
|
|
|
LiveInterval &RHS = li_->getInterval(isExtSubReg ? SrcReg : DstReg);
|
|
|
|
if (li_->hasInterval(PhysReg) &&
|
|
|
|
RHS.overlaps(li_->getInterval(PhysReg))) {
|
2007-10-12 08:50:34 +00:00
|
|
|
DOUT << "Interfere with register ";
|
2008-04-09 20:57:25 +00:00
|
|
|
DEBUG(li_->getInterval(PhysReg).print(DOUT, tri_));
|
2007-11-01 06:22:48 +00:00
|
|
|
return false; // Not coalescable
|
2007-10-12 08:50:34 +00:00
|
|
|
}
|
2008-04-09 20:57:25 +00:00
|
|
|
for (const unsigned* SR = tri_->getSubRegisters(PhysReg); *SR; ++SR)
|
2007-10-12 08:50:34 +00:00
|
|
|
if (li_->hasInterval(*SR) && RHS.overlaps(li_->getInterval(*SR))) {
|
|
|
|
DOUT << "Interfere with sub-register ";
|
2008-02-10 18:45:23 +00:00
|
|
|
DEBUG(li_->getInterval(*SR).print(DOUT, tri_));
|
2007-11-01 06:22:48 +00:00
|
|
|
return false; // Not coalescable
|
2007-10-12 08:50:34 +00:00
|
|
|
}
|
2008-02-15 18:24:29 +00:00
|
|
|
SubIdx = 0;
|
2007-11-01 06:22:48 +00:00
|
|
|
} else {
|
2008-04-17 07:58:04 +00:00
|
|
|
unsigned OldSubIdx = isExtSubReg ? CopyMI->getOperand(0).getSubReg()
|
|
|
|
: CopyMI->getOperand(2).getSubReg();
|
|
|
|
if (OldSubIdx) {
|
2008-04-29 01:41:44 +00:00
|
|
|
if (OldSubIdx == SubIdx && !differingRegisterClasses(SrcReg, DstReg))
|
2008-04-17 07:58:04 +00:00
|
|
|
// r1024<2> = EXTRACT_SUBREG r1025, 2. Then r1024 has already been
|
|
|
|
// coalesced to a larger register so the subreg indices cancel out.
|
2008-04-29 01:41:44 +00:00
|
|
|
// Also check if the other larger register is of the same register
|
|
|
|
// class as the would be resulting register.
|
2008-04-17 07:58:04 +00:00
|
|
|
SubIdx = 0;
|
|
|
|
else {
|
|
|
|
DOUT << "\t Sub-register indices mismatch.\n";
|
|
|
|
return false; // Not coalescable.
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (SubIdx) {
|
|
|
|
unsigned LargeReg = isExtSubReg ? SrcReg : DstReg;
|
|
|
|
unsigned SmallReg = isExtSubReg ? DstReg : SrcReg;
|
|
|
|
unsigned LargeRegSize =
|
|
|
|
li_->getInterval(LargeReg).getSize() / InstrSlots::NUM;
|
|
|
|
unsigned SmallRegSize =
|
|
|
|
li_->getInterval(SmallReg).getSize() / InstrSlots::NUM;
|
|
|
|
const TargetRegisterClass *RC = mri_->getRegClass(SmallReg);
|
|
|
|
unsigned Threshold = allocatableRCRegs_[RC].count();
|
|
|
|
// Be conservative. If both sides are virtual registers, do not coalesce
|
|
|
|
// if this will cause a high use density interval to target a smaller
|
|
|
|
// set of registers.
|
|
|
|
if (SmallRegSize > Threshold || LargeRegSize > Threshold) {
|
|
|
|
LiveVariables::VarInfo &svi = lv_->getVarInfo(LargeReg);
|
|
|
|
LiveVariables::VarInfo &dvi = lv_->getVarInfo(SmallReg);
|
|
|
|
if ((float)dvi.NumUses / SmallRegSize <
|
|
|
|
(float)svi.NumUses / LargeRegSize) {
|
|
|
|
Again = true; // May be possible to coalesce later.
|
|
|
|
return false;
|
|
|
|
}
|
2007-11-01 06:22:48 +00:00
|
|
|
}
|
|
|
|
}
|
2007-10-12 08:50:34 +00:00
|
|
|
}
|
2008-02-15 18:24:29 +00:00
|
|
|
} else if (differingRegisterClasses(SrcReg, DstReg)) {
|
|
|
|
// FIXME: What if the resul of a EXTRACT_SUBREG is then coalesced
|
|
|
|
// with another? If it's the resulting destination register, then
|
|
|
|
// the subidx must be propagated to uses (but only those defined
|
|
|
|
// by the EXTRACT_SUBREG). If it's being coalesced into another
|
|
|
|
// register, it should be safe because register is assumed to have
|
|
|
|
// the register class of the super-register.
|
|
|
|
|
2007-10-12 08:50:34 +00:00
|
|
|
// If they are not of the same register class, we cannot join them.
|
2007-06-08 17:18:56 +00:00
|
|
|
DOUT << "\tSrc/Dest are different register classes.\n";
|
2007-10-12 08:50:34 +00:00
|
|
|
// Allow the coalescer to try again in case either side gets coalesced to
|
|
|
|
// a physical register that's compatible with the other side. e.g.
|
|
|
|
// r1024 = MOV32to32_ r1025
|
|
|
|
// but later r1024 is assigned EAX then r1025 may be coalesced with EAX.
|
2007-11-01 06:22:48 +00:00
|
|
|
Again = true; // May be possible to coalesce later.
|
2007-10-12 08:50:34 +00:00
|
|
|
return false;
|
2007-06-08 17:18:56 +00:00
|
|
|
}
|
|
|
|
|
2008-02-15 18:24:29 +00:00
|
|
|
LiveInterval &SrcInt = li_->getInterval(SrcReg);
|
|
|
|
LiveInterval &DstInt = li_->getInterval(DstReg);
|
|
|
|
assert(SrcInt.reg == SrcReg && DstInt.reg == DstReg &&
|
2007-06-08 17:18:56 +00:00
|
|
|
"Register mapping is horribly broken!");
|
|
|
|
|
2008-02-10 18:45:23 +00:00
|
|
|
DOUT << "\t\tInspecting "; SrcInt.print(DOUT, tri_);
|
|
|
|
DOUT << " and "; DstInt.print(DOUT, tri_);
|
2007-06-08 17:18:56 +00:00
|
|
|
DOUT << ": ";
|
|
|
|
|
2008-03-18 08:26:47 +00:00
|
|
|
// Check if it is necessary to propagate "isDead" property.
|
2008-04-09 20:57:25 +00:00
|
|
|
if (!isExtSubReg && !isInsSubReg) {
|
|
|
|
MachineOperand *mopd = CopyMI->findRegisterDefOperand(DstReg, false);
|
|
|
|
bool isDead = mopd->isDead();
|
|
|
|
|
|
|
|
// We need to be careful about coalescing a source physical register with a
|
|
|
|
// virtual register. Once the coalescing is done, it cannot be broken and
|
|
|
|
// these are not spillable! If the destination interval uses are far away,
|
|
|
|
// think twice about coalescing them!
|
|
|
|
if (!isDead && (SrcIsPhys || DstIsPhys)) {
|
|
|
|
LiveInterval &JoinVInt = SrcIsPhys ? DstInt : SrcInt;
|
|
|
|
unsigned JoinVReg = SrcIsPhys ? DstReg : SrcReg;
|
|
|
|
unsigned JoinPReg = SrcIsPhys ? SrcReg : DstReg;
|
|
|
|
const TargetRegisterClass *RC = mri_->getRegClass(JoinVReg);
|
|
|
|
unsigned Threshold = allocatableRCRegs_[RC].count() * 2;
|
|
|
|
if (TheCopy.isBackEdge)
|
|
|
|
Threshold *= 2; // Favors back edge copies.
|
|
|
|
|
|
|
|
// If the virtual register live interval is long but it has low use desity,
|
|
|
|
// do not join them, instead mark the physical register as its allocation
|
|
|
|
// preference.
|
|
|
|
unsigned Length = JoinVInt.getSize() / InstrSlots::NUM;
|
|
|
|
LiveVariables::VarInfo &vi = lv_->getVarInfo(JoinVReg);
|
|
|
|
if (Length > Threshold &&
|
|
|
|
(((float)vi.NumUses / Length) < (1.0 / Threshold))) {
|
|
|
|
JoinVInt.preference = JoinPReg;
|
|
|
|
++numAborts;
|
|
|
|
DOUT << "\tMay tie down a physical register, abort!\n";
|
|
|
|
Again = true; // May be possible to coalesce later.
|
|
|
|
return false;
|
|
|
|
}
|
2007-06-08 17:18:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Okay, attempt to join these two intervals. On failure, this returns false.
|
|
|
|
// Otherwise, if one of the intervals being joined is a physreg, this method
|
|
|
|
// always canonicalizes DstInt to be it. The output "SrcInt" will not have
|
|
|
|
// been modified, so we can use this information below to update aliases.
|
2007-08-28 08:28:51 +00:00
|
|
|
bool Swapped = false;
|
2008-04-03 16:41:54 +00:00
|
|
|
// If SrcInt is implicitly defined, it's safe to coalesce.
|
|
|
|
bool isEmpty = SrcInt.empty();
|
2008-04-09 20:57:25 +00:00
|
|
|
if (isEmpty && !CanCoalesceWithImpDef(CopyMI, DstInt, SrcInt)) {
|
2008-04-03 16:41:54 +00:00
|
|
|
// Only coalesce an empty interval (defined by implicit_def) with
|
2008-04-09 20:57:25 +00:00
|
|
|
// another interval which has a valno defined by the CopyMI and the CopyMI
|
|
|
|
// is a kill of the implicit def.
|
2008-04-03 16:41:54 +00:00
|
|
|
DOUT << "Not profitable!\n";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isEmpty && !JoinIntervals(DstInt, SrcInt, Swapped)) {
|
2007-07-09 12:00:59 +00:00
|
|
|
// Coalescing failed.
|
2007-06-08 17:18:56 +00:00
|
|
|
|
|
|
|
// If we can eliminate the copy without merging the live ranges, do so now.
|
2008-04-09 20:57:25 +00:00
|
|
|
if (!isExtSubReg && !isInsSubReg &&
|
2008-02-13 03:01:43 +00:00
|
|
|
(AdjustCopiesBackFrom(SrcInt, DstInt, CopyMI) ||
|
|
|
|
RemoveCopyByCommutingDef(SrcInt, DstInt, CopyMI))) {
|
2007-11-06 08:52:21 +00:00
|
|
|
JoinedCopies.insert(CopyMI);
|
2007-06-08 17:18:56 +00:00
|
|
|
return true;
|
2007-11-06 08:52:21 +00:00
|
|
|
}
|
2008-02-13 03:01:43 +00:00
|
|
|
|
2007-06-08 17:18:56 +00:00
|
|
|
// Otherwise, we are unable to join the intervals.
|
|
|
|
DOUT << "Interference!\n";
|
2007-11-01 06:22:48 +00:00
|
|
|
Again = true; // May be possible to coalesce later.
|
2007-06-08 17:18:56 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2007-08-28 08:28:51 +00:00
|
|
|
LiveInterval *ResSrcInt = &SrcInt;
|
|
|
|
LiveInterval *ResDstInt = &DstInt;
|
|
|
|
if (Swapped) {
|
2008-02-15 18:24:29 +00:00
|
|
|
std::swap(SrcReg, DstReg);
|
2007-08-28 08:28:51 +00:00
|
|
|
std::swap(ResSrcInt, ResDstInt);
|
|
|
|
}
|
2008-02-15 18:24:29 +00:00
|
|
|
assert(TargetRegisterInfo::isVirtualRegister(SrcReg) &&
|
2007-06-08 17:18:56 +00:00
|
|
|
"LiveInterval::join didn't work right!");
|
|
|
|
|
|
|
|
// If we're about to merge live ranges into a physical register live range,
|
|
|
|
// we have to update any aliased register's live ranges to indicate that they
|
|
|
|
// have clobbered values for this range.
|
2008-02-15 18:24:29 +00:00
|
|
|
if (TargetRegisterInfo::isPhysicalRegister(DstReg)) {
|
2007-10-12 08:50:34 +00:00
|
|
|
// If this is a extract_subreg where dst is a physical register, e.g.
|
|
|
|
// cl = EXTRACT_SUBREG reg1024, 1
|
|
|
|
// then create and update the actual physical register allocated to RHS.
|
2008-04-09 20:57:25 +00:00
|
|
|
if (RealDstReg || RealSrcReg) {
|
|
|
|
LiveInterval &RealInt =
|
|
|
|
li_->getOrCreateInterval(RealDstReg ? RealDstReg : RealSrcReg);
|
2007-10-15 18:33:50 +00:00
|
|
|
SmallSet<const VNInfo*, 4> CopiedValNos;
|
|
|
|
for (LiveInterval::Ranges::const_iterator I = ResSrcInt->ranges.begin(),
|
|
|
|
E = ResSrcInt->ranges.end(); I != E; ++I) {
|
2008-04-16 18:48:43 +00:00
|
|
|
const LiveRange *DstLR = ResDstInt->getLiveRangeContaining(I->start);
|
|
|
|
assert(DstLR && "Invalid joined interval!");
|
2007-10-15 18:33:50 +00:00
|
|
|
const VNInfo *DstValNo = DstLR->valno;
|
|
|
|
if (CopiedValNos.insert(DstValNo)) {
|
2008-04-09 20:57:25 +00:00
|
|
|
VNInfo *ValNo = RealInt.getNextValue(DstValNo->def, DstValNo->copy,
|
|
|
|
li_->getVNInfoAllocator());
|
2007-11-29 09:49:23 +00:00
|
|
|
ValNo->hasPHIKill = DstValNo->hasPHIKill;
|
2008-04-09 20:57:25 +00:00
|
|
|
RealInt.addKills(ValNo, DstValNo->kills);
|
|
|
|
RealInt.MergeValueInAsValue(*ResDstInt, DstValNo, ValNo);
|
2007-10-15 18:33:50 +00:00
|
|
|
}
|
2007-10-14 10:08:34 +00:00
|
|
|
}
|
2008-04-09 20:57:25 +00:00
|
|
|
|
|
|
|
DstReg = RealDstReg ? RealDstReg : RealSrcReg;
|
2007-10-12 08:50:34 +00:00
|
|
|
}
|
|
|
|
|
2007-06-08 17:18:56 +00:00
|
|
|
// Update the liveintervals of sub-registers.
|
2008-02-15 18:24:29 +00:00
|
|
|
for (const unsigned *AS = tri_->getSubRegisters(DstReg); *AS; ++AS)
|
2007-10-12 08:50:34 +00:00
|
|
|
li_->getOrCreateInterval(*AS).MergeInClobberRanges(*ResSrcInt,
|
2007-09-05 21:46:51 +00:00
|
|
|
li_->getVNInfoAllocator());
|
2007-06-08 17:18:56 +00:00
|
|
|
} else {
|
|
|
|
// Merge use info if the destination is a virtual register.
|
2008-02-15 18:24:29 +00:00
|
|
|
LiveVariables::VarInfo& dVI = lv_->getVarInfo(DstReg);
|
|
|
|
LiveVariables::VarInfo& sVI = lv_->getVarInfo(SrcReg);
|
2007-06-08 17:18:56 +00:00
|
|
|
dVI.NumUses += sVI.NumUses;
|
|
|
|
}
|
|
|
|
|
2008-02-15 18:24:29 +00:00
|
|
|
// If this is a EXTRACT_SUBREG, make sure the result of coalescing is the
|
|
|
|
// larger super-register.
|
2008-04-09 20:57:25 +00:00
|
|
|
if ((isExtSubReg || isInsSubReg) && !SrcIsPhys && !DstIsPhys) {
|
|
|
|
if ((isExtSubReg && !Swapped) || (isInsSubReg && Swapped)) {
|
2007-10-12 08:50:34 +00:00
|
|
|
ResSrcInt->Copy(*ResDstInt, li_->getVNInfoAllocator());
|
2008-02-15 18:24:29 +00:00
|
|
|
std::swap(SrcReg, DstReg);
|
2007-10-12 08:50:34 +00:00
|
|
|
std::swap(ResSrcInt, ResDstInt);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-11-06 08:52:21 +00:00
|
|
|
if (NewHeuristic) {
|
2008-02-15 18:24:29 +00:00
|
|
|
// Add all copies that define val# in the source interval into the queue.
|
2007-11-06 08:52:21 +00:00
|
|
|
for (LiveInterval::const_vni_iterator i = ResSrcInt->vni_begin(),
|
|
|
|
e = ResSrcInt->vni_end(); i != e; ++i) {
|
|
|
|
const VNInfo *vni = *i;
|
2008-02-15 18:24:29 +00:00
|
|
|
if (!vni->def || vni->def == ~1U || vni->def == ~0U)
|
|
|
|
continue;
|
|
|
|
MachineInstr *CopyMI = li_->getInstructionFromIndex(vni->def);
|
|
|
|
unsigned NewSrcReg, NewDstReg;
|
|
|
|
if (CopyMI &&
|
|
|
|
JoinedCopies.count(CopyMI) == 0 &&
|
|
|
|
tii_->isMoveInstr(*CopyMI, NewSrcReg, NewDstReg)) {
|
|
|
|
unsigned LoopDepth = loopInfo->getLoopDepth(CopyMI->getParent());
|
|
|
|
JoinQueue->push(CopyRec(CopyMI, LoopDepth,
|
|
|
|
isBackEdgeCopy(CopyMI, DstReg)));
|
2007-11-06 08:52:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-02-15 18:24:29 +00:00
|
|
|
// Remember to delete the copy instruction.
|
|
|
|
JoinedCopies.insert(CopyMI);
|
|
|
|
|
2008-03-10 08:11:32 +00:00
|
|
|
// Some live range has been lengthened due to colaescing, eliminate the
|
|
|
|
// unnecessary kills.
|
|
|
|
RemoveUnnecessaryKills(SrcReg, *ResDstInt);
|
|
|
|
if (TargetRegisterInfo::isVirtualRegister(DstReg))
|
|
|
|
RemoveUnnecessaryKills(DstReg, *ResDstInt);
|
|
|
|
|
2008-02-15 18:24:29 +00:00
|
|
|
// SrcReg is guarateed to be the register whose live interval that is
|
2007-07-18 23:34:48 +00:00
|
|
|
// being merged.
|
2008-02-15 18:24:29 +00:00
|
|
|
li_->removeInterval(SrcReg);
|
2008-04-09 20:57:25 +00:00
|
|
|
if (isInsSubReg)
|
|
|
|
// Avoid:
|
|
|
|
// r1024 = op
|
|
|
|
// r1024 = implicit_def
|
|
|
|
// ...
|
|
|
|
// = r1024
|
|
|
|
RemoveDeadImpDef(DstReg, *ResDstInt);
|
2008-02-15 18:24:29 +00:00
|
|
|
UpdateRegDefsUses(SrcReg, DstReg, SubIdx);
|
2007-06-08 17:18:56 +00:00
|
|
|
|
2008-04-03 16:41:54 +00:00
|
|
|
if (isEmpty) {
|
|
|
|
// Now the copy is being coalesced away, the val# previously defined
|
|
|
|
// by the copy is being defined by an IMPLICIT_DEF which defines a zero
|
|
|
|
// length interval. Remove the val#.
|
|
|
|
unsigned CopyIdx = li_->getDefIndex(li_->getInstructionIndex(CopyMI));
|
2008-04-16 18:48:43 +00:00
|
|
|
const LiveRange *LR = ResDstInt->getLiveRangeContaining(CopyIdx);
|
2008-04-03 16:41:54 +00:00
|
|
|
VNInfo *ImpVal = LR->valno;
|
|
|
|
assert(ImpVal->def == CopyIdx);
|
2008-04-09 20:57:25 +00:00
|
|
|
unsigned NextDef = LR->end;
|
|
|
|
RemoveCopiesFromValNo(*ResDstInt, ImpVal);
|
2008-04-03 16:41:54 +00:00
|
|
|
ResDstInt->removeValNo(ImpVal);
|
2008-04-09 20:57:25 +00:00
|
|
|
LR = ResDstInt->FindLiveRangeContaining(NextDef);
|
|
|
|
if (LR != ResDstInt->end() && LR->valno->def == NextDef) {
|
|
|
|
// Special case: vr1024 = implicit_def
|
|
|
|
// vr1024 = insert_subreg vr1024, vr1025, c
|
|
|
|
// The insert_subreg becomes a "copy" that defines a val# which can itself
|
|
|
|
// be coalesced away.
|
|
|
|
MachineInstr *DefMI = li_->getInstructionFromIndex(NextDef);
|
|
|
|
if (DefMI->getOpcode() == TargetInstrInfo::INSERT_SUBREG)
|
|
|
|
LR->valno->copy = DefMI;
|
|
|
|
}
|
2008-04-03 16:41:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
DOUT << "\n\t\tJoined. Result = "; ResDstInt->print(DOUT, tri_);
|
|
|
|
DOUT << "\n";
|
|
|
|
|
2007-06-08 17:18:56 +00:00
|
|
|
++numJoins;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// ComputeUltimateVN - Assuming we are going to join two live intervals,
|
|
|
|
/// compute what the resultant value numbers for each value in the input two
|
|
|
|
/// ranges will be. This is complicated by copies between the two which can
|
|
|
|
/// and will commonly cause multiple value numbers to be merged into one.
|
|
|
|
///
|
|
|
|
/// VN is the value number that we're trying to resolve. InstDefiningValue
|
|
|
|
/// keeps track of the new InstDefiningValue assignment for the result
|
|
|
|
/// LiveInterval. ThisFromOther/OtherFromThis are sets that keep track of
|
|
|
|
/// whether a value in this or other is a copy from the opposite set.
|
|
|
|
/// ThisValNoAssignments/OtherValNoAssignments keep track of value #'s that have
|
|
|
|
/// already been assigned.
|
|
|
|
///
|
|
|
|
/// ThisFromOther[x] - If x is defined as a copy from the other interval, this
|
|
|
|
/// contains the value number the copy is from.
|
|
|
|
///
|
2007-08-29 20:45:00 +00:00
|
|
|
static unsigned ComputeUltimateVN(VNInfo *VNI,
|
|
|
|
SmallVector<VNInfo*, 16> &NewVNInfo,
|
2007-08-31 21:23:06 +00:00
|
|
|
DenseMap<VNInfo*, VNInfo*> &ThisFromOther,
|
|
|
|
DenseMap<VNInfo*, VNInfo*> &OtherFromThis,
|
2007-06-08 17:18:56 +00:00
|
|
|
SmallVector<int, 16> &ThisValNoAssignments,
|
2007-08-29 20:45:00 +00:00
|
|
|
SmallVector<int, 16> &OtherValNoAssignments) {
|
|
|
|
unsigned VN = VNI->id;
|
|
|
|
|
2007-06-08 17:18:56 +00:00
|
|
|
// If the VN has already been computed, just return it.
|
|
|
|
if (ThisValNoAssignments[VN] >= 0)
|
|
|
|
return ThisValNoAssignments[VN];
|
|
|
|
// assert(ThisValNoAssignments[VN] != -2 && "Cyclic case?");
|
2007-08-29 20:45:00 +00:00
|
|
|
|
2007-06-08 17:18:56 +00:00
|
|
|
// If this val is not a copy from the other val, then it must be a new value
|
|
|
|
// number in the destination.
|
2007-08-31 21:23:06 +00:00
|
|
|
DenseMap<VNInfo*, VNInfo*>::iterator I = ThisFromOther.find(VNI);
|
2007-08-31 08:04:17 +00:00
|
|
|
if (I == ThisFromOther.end()) {
|
2007-08-29 20:45:00 +00:00
|
|
|
NewVNInfo.push_back(VNI);
|
|
|
|
return ThisValNoAssignments[VN] = NewVNInfo.size()-1;
|
2007-06-08 17:18:56 +00:00
|
|
|
}
|
2007-08-31 08:04:17 +00:00
|
|
|
VNInfo *OtherValNo = I->second;
|
2007-06-08 17:18:56 +00:00
|
|
|
|
|
|
|
// Otherwise, this *is* a copy from the RHS. If the other side has already
|
|
|
|
// been computed, return it.
|
2007-08-29 20:45:00 +00:00
|
|
|
if (OtherValNoAssignments[OtherValNo->id] >= 0)
|
|
|
|
return ThisValNoAssignments[VN] = OtherValNoAssignments[OtherValNo->id];
|
2007-06-08 17:18:56 +00:00
|
|
|
|
|
|
|
// Mark this value number as currently being computed, then ask what the
|
|
|
|
// ultimate value # of the other value is.
|
|
|
|
ThisValNoAssignments[VN] = -2;
|
|
|
|
unsigned UltimateVN =
|
2007-08-29 20:45:00 +00:00
|
|
|
ComputeUltimateVN(OtherValNo, NewVNInfo, OtherFromThis, ThisFromOther,
|
|
|
|
OtherValNoAssignments, ThisValNoAssignments);
|
2007-06-08 17:18:56 +00:00
|
|
|
return ThisValNoAssignments[VN] = UltimateVN;
|
|
|
|
}
|
|
|
|
|
2007-08-29 20:45:00 +00:00
|
|
|
static bool InVector(VNInfo *Val, const SmallVector<VNInfo*, 8> &V) {
|
2007-06-08 17:18:56 +00:00
|
|
|
return std::find(V.begin(), V.end(), Val) != V.end();
|
|
|
|
}
|
|
|
|
|
2008-04-09 20:57:25 +00:00
|
|
|
/// RangeIsDefinedByCopyFromReg - Return true if the specified live range of
|
|
|
|
/// the specified live interval is defined by a copy from the specified
|
|
|
|
/// register.
|
|
|
|
bool SimpleRegisterCoalescing::RangeIsDefinedByCopyFromReg(LiveInterval &li,
|
|
|
|
LiveRange *LR,
|
|
|
|
unsigned Reg) {
|
|
|
|
unsigned SrcReg = li_->getVNInfoSourceReg(LR->valno);
|
|
|
|
if (SrcReg == Reg)
|
|
|
|
return true;
|
|
|
|
if (LR->valno->def == ~0U &&
|
|
|
|
TargetRegisterInfo::isPhysicalRegister(li.reg) &&
|
|
|
|
*tri_->getSuperRegisters(li.reg)) {
|
|
|
|
// It's a sub-register live interval, we may not have precise information.
|
|
|
|
// Re-compute it.
|
|
|
|
MachineInstr *DefMI = li_->getInstructionFromIndex(LR->start);
|
|
|
|
unsigned SrcReg, DstReg;
|
|
|
|
if (tii_->isMoveInstr(*DefMI, SrcReg, DstReg) &&
|
|
|
|
DstReg == li.reg && SrcReg == Reg) {
|
|
|
|
// Cache computed info.
|
|
|
|
LR->valno->def = LR->start;
|
|
|
|
LR->valno->copy = DefMI;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2007-06-08 17:18:56 +00:00
|
|
|
/// SimpleJoin - Attempt to joint the specified interval into this one. The
|
|
|
|
/// caller of this method must guarantee that the RHS only contains a single
|
|
|
|
/// value number and that the RHS is not defined by a copy from this
|
|
|
|
/// interval. This returns false if the intervals are not joinable, or it
|
|
|
|
/// joins them and returns true.
|
2008-01-04 08:59:18 +00:00
|
|
|
bool SimpleRegisterCoalescing::SimpleJoin(LiveInterval &LHS, LiveInterval &RHS){
|
2007-06-08 17:18:56 +00:00
|
|
|
assert(RHS.containsOneValue());
|
|
|
|
|
|
|
|
// Some number (potentially more than one) value numbers in the current
|
|
|
|
// interval may be defined as copies from the RHS. Scan the overlapping
|
|
|
|
// portions of the LHS and RHS, keeping track of this and looking for
|
|
|
|
// overlapping live ranges that are NOT defined as copies. If these exist, we
|
2007-07-09 12:00:59 +00:00
|
|
|
// cannot coalesce.
|
2007-06-08 17:18:56 +00:00
|
|
|
|
|
|
|
LiveInterval::iterator LHSIt = LHS.begin(), LHSEnd = LHS.end();
|
|
|
|
LiveInterval::iterator RHSIt = RHS.begin(), RHSEnd = RHS.end();
|
|
|
|
|
|
|
|
if (LHSIt->start < RHSIt->start) {
|
|
|
|
LHSIt = std::upper_bound(LHSIt, LHSEnd, RHSIt->start);
|
|
|
|
if (LHSIt != LHS.begin()) --LHSIt;
|
|
|
|
} else if (RHSIt->start < LHSIt->start) {
|
|
|
|
RHSIt = std::upper_bound(RHSIt, RHSEnd, LHSIt->start);
|
|
|
|
if (RHSIt != RHS.begin()) --RHSIt;
|
|
|
|
}
|
|
|
|
|
2007-08-29 20:45:00 +00:00
|
|
|
SmallVector<VNInfo*, 8> EliminatedLHSVals;
|
2007-06-08 17:18:56 +00:00
|
|
|
|
|
|
|
while (1) {
|
|
|
|
// Determine if these live intervals overlap.
|
|
|
|
bool Overlaps = false;
|
|
|
|
if (LHSIt->start <= RHSIt->start)
|
|
|
|
Overlaps = LHSIt->end > RHSIt->start;
|
|
|
|
else
|
|
|
|
Overlaps = RHSIt->end > LHSIt->start;
|
|
|
|
|
|
|
|
// If the live intervals overlap, there are two interesting cases: if the
|
|
|
|
// LHS interval is defined by a copy from the RHS, it's ok and we record
|
|
|
|
// that the LHS value # is the same as the RHS. If it's not, then we cannot
|
2007-07-09 12:00:59 +00:00
|
|
|
// coalesce these live ranges and we bail out.
|
2007-06-08 17:18:56 +00:00
|
|
|
if (Overlaps) {
|
|
|
|
// If we haven't already recorded that this value # is safe, check it.
|
2007-08-29 20:45:00 +00:00
|
|
|
if (!InVector(LHSIt->valno, EliminatedLHSVals)) {
|
2007-06-08 17:18:56 +00:00
|
|
|
// Copy from the RHS?
|
2008-04-09 20:57:25 +00:00
|
|
|
if (!RangeIsDefinedByCopyFromReg(LHS, LHSIt, RHS.reg))
|
2007-06-08 17:18:56 +00:00
|
|
|
return false; // Nope, bail out.
|
|
|
|
|
2007-08-29 20:45:00 +00:00
|
|
|
EliminatedLHSVals.push_back(LHSIt->valno);
|
2007-06-08 17:18:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// We know this entire LHS live range is okay, so skip it now.
|
|
|
|
if (++LHSIt == LHSEnd) break;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (LHSIt->end < RHSIt->end) {
|
|
|
|
if (++LHSIt == LHSEnd) break;
|
|
|
|
} else {
|
|
|
|
// One interesting case to check here. It's possible that we have
|
|
|
|
// something like "X3 = Y" which defines a new value number in the LHS,
|
|
|
|
// and is the last use of this liverange of the RHS. In this case, we
|
2007-07-09 12:00:59 +00:00
|
|
|
// want to notice this copy (so that it gets coalesced away) even though
|
2007-06-08 17:18:56 +00:00
|
|
|
// the live ranges don't actually overlap.
|
|
|
|
if (LHSIt->start == RHSIt->end) {
|
2007-08-29 20:45:00 +00:00
|
|
|
if (InVector(LHSIt->valno, EliminatedLHSVals)) {
|
2007-06-08 17:18:56 +00:00
|
|
|
// We already know that this value number is going to be merged in
|
2007-07-09 12:00:59 +00:00
|
|
|
// if coalescing succeeds. Just skip the liverange.
|
2007-06-08 17:18:56 +00:00
|
|
|
if (++LHSIt == LHSEnd) break;
|
|
|
|
} else {
|
|
|
|
// Otherwise, if this is a copy from the RHS, mark it as being merged
|
|
|
|
// in.
|
2008-04-09 20:57:25 +00:00
|
|
|
if (RangeIsDefinedByCopyFromReg(LHS, LHSIt, RHS.reg)) {
|
2007-08-29 20:45:00 +00:00
|
|
|
EliminatedLHSVals.push_back(LHSIt->valno);
|
2007-06-08 17:18:56 +00:00
|
|
|
|
|
|
|
// We know this entire LHS live range is okay, so skip it now.
|
|
|
|
if (++LHSIt == LHSEnd) break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (++RHSIt == RHSEnd) break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-07-09 12:00:59 +00:00
|
|
|
// If we got here, we know that the coalescing will be successful and that
|
2007-06-08 17:18:56 +00:00
|
|
|
// the value numbers in EliminatedLHSVals will all be merged together. Since
|
|
|
|
// the most common case is that EliminatedLHSVals has a single number, we
|
|
|
|
// optimize for it: if there is more than one value, we merge them all into
|
|
|
|
// the lowest numbered one, then handle the interval as if we were merging
|
|
|
|
// with one value number.
|
2007-08-29 20:45:00 +00:00
|
|
|
VNInfo *LHSValNo;
|
2007-06-08 17:18:56 +00:00
|
|
|
if (EliminatedLHSVals.size() > 1) {
|
|
|
|
// Loop through all the equal value numbers merging them into the smallest
|
|
|
|
// one.
|
2007-08-29 20:45:00 +00:00
|
|
|
VNInfo *Smallest = EliminatedLHSVals[0];
|
2007-06-08 17:18:56 +00:00
|
|
|
for (unsigned i = 1, e = EliminatedLHSVals.size(); i != e; ++i) {
|
2007-08-29 20:45:00 +00:00
|
|
|
if (EliminatedLHSVals[i]->id < Smallest->id) {
|
2007-06-08 17:18:56 +00:00
|
|
|
// Merge the current notion of the smallest into the smaller one.
|
|
|
|
LHS.MergeValueNumberInto(Smallest, EliminatedLHSVals[i]);
|
|
|
|
Smallest = EliminatedLHSVals[i];
|
|
|
|
} else {
|
|
|
|
// Merge into the smallest.
|
|
|
|
LHS.MergeValueNumberInto(EliminatedLHSVals[i], Smallest);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
LHSValNo = Smallest;
|
2008-04-09 20:57:25 +00:00
|
|
|
} else if (EliminatedLHSVals.empty()) {
|
|
|
|
if (TargetRegisterInfo::isPhysicalRegister(LHS.reg) &&
|
|
|
|
*tri_->getSuperRegisters(LHS.reg))
|
|
|
|
// Imprecise sub-register information. Can't handle it.
|
|
|
|
return false;
|
|
|
|
assert(0 && "No copies from the RHS?");
|
2007-06-08 17:18:56 +00:00
|
|
|
} else {
|
|
|
|
LHSValNo = EliminatedLHSVals[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
// Okay, now that there is a single LHS value number that we're merging the
|
|
|
|
// RHS into, update the value number info for the LHS to indicate that the
|
|
|
|
// value number is defined where the RHS value number was.
|
2007-09-05 21:46:51 +00:00
|
|
|
const VNInfo *VNI = RHS.getValNumInfo(0);
|
2008-02-15 18:24:29 +00:00
|
|
|
LHSValNo->def = VNI->def;
|
|
|
|
LHSValNo->copy = VNI->copy;
|
2007-06-08 17:18:56 +00:00
|
|
|
|
|
|
|
// Okay, the final step is to loop over the RHS live intervals, adding them to
|
|
|
|
// the LHS.
|
2007-11-29 09:49:23 +00:00
|
|
|
LHSValNo->hasPHIKill |= VNI->hasPHIKill;
|
2007-09-05 21:46:51 +00:00
|
|
|
LHS.addKills(LHSValNo, VNI->kills);
|
2007-08-14 01:56:58 +00:00
|
|
|
LHS.MergeRangesInAsValue(RHS, LHSValNo);
|
2007-06-08 17:18:56 +00:00
|
|
|
LHS.weight += RHS.weight;
|
|
|
|
if (RHS.preference && !LHS.preference)
|
|
|
|
LHS.preference = RHS.preference;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// JoinIntervals - Attempt to join these two intervals. On failure, this
|
|
|
|
/// returns false. Otherwise, if one of the intervals being joined is a
|
|
|
|
/// physreg, this method always canonicalizes LHS to be it. The output
|
|
|
|
/// "RHS" will not have been modified, so we can use this information
|
|
|
|
/// below to update aliases.
|
2007-08-28 08:28:51 +00:00
|
|
|
bool SimpleRegisterCoalescing::JoinIntervals(LiveInterval &LHS,
|
|
|
|
LiveInterval &RHS, bool &Swapped) {
|
2007-06-08 17:18:56 +00:00
|
|
|
// Compute the final value assignment, assuming that the live ranges can be
|
2007-07-09 12:00:59 +00:00
|
|
|
// coalesced.
|
2007-06-08 17:18:56 +00:00
|
|
|
SmallVector<int, 16> LHSValNoAssignments;
|
|
|
|
SmallVector<int, 16> RHSValNoAssignments;
|
2007-08-31 21:23:06 +00:00
|
|
|
DenseMap<VNInfo*, VNInfo*> LHSValsDefinedFromRHS;
|
|
|
|
DenseMap<VNInfo*, VNInfo*> RHSValsDefinedFromLHS;
|
2007-08-29 20:45:00 +00:00
|
|
|
SmallVector<VNInfo*, 16> NewVNInfo;
|
2007-06-08 17:18:56 +00:00
|
|
|
|
|
|
|
// If a live interval is a physical register, conservatively check if any
|
|
|
|
// of its sub-registers is overlapping the live interval of the virtual
|
|
|
|
// register. If so, do not coalesce.
|
2008-02-10 18:45:23 +00:00
|
|
|
if (TargetRegisterInfo::isPhysicalRegister(LHS.reg) &&
|
|
|
|
*tri_->getSubRegisters(LHS.reg)) {
|
|
|
|
for (const unsigned* SR = tri_->getSubRegisters(LHS.reg); *SR; ++SR)
|
2007-06-08 17:18:56 +00:00
|
|
|
if (li_->hasInterval(*SR) && RHS.overlaps(li_->getInterval(*SR))) {
|
|
|
|
DOUT << "Interfere with sub-register ";
|
2008-02-10 18:45:23 +00:00
|
|
|
DEBUG(li_->getInterval(*SR).print(DOUT, tri_));
|
2007-06-08 17:18:56 +00:00
|
|
|
return false;
|
|
|
|
}
|
2008-02-10 18:45:23 +00:00
|
|
|
} else if (TargetRegisterInfo::isPhysicalRegister(RHS.reg) &&
|
|
|
|
*tri_->getSubRegisters(RHS.reg)) {
|
|
|
|
for (const unsigned* SR = tri_->getSubRegisters(RHS.reg); *SR; ++SR)
|
2007-06-08 17:18:56 +00:00
|
|
|
if (li_->hasInterval(*SR) && LHS.overlaps(li_->getInterval(*SR))) {
|
|
|
|
DOUT << "Interfere with sub-register ";
|
2008-02-10 18:45:23 +00:00
|
|
|
DEBUG(li_->getInterval(*SR).print(DOUT, tri_));
|
2007-06-08 17:18:56 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Compute ultimate value numbers for the LHS and RHS values.
|
|
|
|
if (RHS.containsOneValue()) {
|
|
|
|
// Copies from a liveinterval with a single value are simple to handle and
|
|
|
|
// very common, handle the special case here. This is important, because
|
|
|
|
// often RHS is small and LHS is large (e.g. a physreg).
|
|
|
|
|
|
|
|
// Find out if the RHS is defined as a copy from some value in the LHS.
|
2007-08-11 00:59:19 +00:00
|
|
|
int RHSVal0DefinedFromLHS = -1;
|
2007-06-08 17:18:56 +00:00
|
|
|
int RHSValID = -1;
|
2007-08-29 20:45:00 +00:00
|
|
|
VNInfo *RHSValNoInfo = NULL;
|
2007-09-05 21:46:51 +00:00
|
|
|
VNInfo *RHSValNoInfo0 = RHS.getValNumInfo(0);
|
2008-02-15 18:24:29 +00:00
|
|
|
unsigned RHSSrcReg = li_->getVNInfoSourceReg(RHSValNoInfo0);
|
|
|
|
if ((RHSSrcReg == 0 || RHSSrcReg != LHS.reg)) {
|
2007-06-08 17:18:56 +00:00
|
|
|
// If RHS is not defined as a copy from the LHS, we can use simpler and
|
2007-07-09 12:00:59 +00:00
|
|
|
// faster checks to see if the live ranges are coalescable. This joiner
|
2007-06-08 17:18:56 +00:00
|
|
|
// can't swap the LHS/RHS intervals though.
|
2008-02-10 18:45:23 +00:00
|
|
|
if (!TargetRegisterInfo::isPhysicalRegister(RHS.reg)) {
|
2007-06-08 17:18:56 +00:00
|
|
|
return SimpleJoin(LHS, RHS);
|
|
|
|
} else {
|
2007-08-31 08:04:17 +00:00
|
|
|
RHSValNoInfo = RHSValNoInfo0;
|
2007-06-08 17:18:56 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// It was defined as a copy from the LHS, find out what value # it is.
|
2007-08-31 08:04:17 +00:00
|
|
|
RHSValNoInfo = LHS.getLiveRangeContaining(RHSValNoInfo0->def-1)->valno;
|
2007-08-29 20:45:00 +00:00
|
|
|
RHSValID = RHSValNoInfo->id;
|
2007-08-11 00:59:19 +00:00
|
|
|
RHSVal0DefinedFromLHS = RHSValID;
|
2007-06-08 17:18:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
LHSValNoAssignments.resize(LHS.getNumValNums(), -1);
|
|
|
|
RHSValNoAssignments.resize(RHS.getNumValNums(), -1);
|
2007-08-29 20:45:00 +00:00
|
|
|
NewVNInfo.resize(LHS.getNumValNums(), NULL);
|
2007-06-08 17:18:56 +00:00
|
|
|
|
|
|
|
// Okay, *all* of the values in LHS that are defined as a copy from RHS
|
|
|
|
// should now get updated.
|
2007-08-29 20:45:00 +00:00
|
|
|
for (LiveInterval::vni_iterator i = LHS.vni_begin(), e = LHS.vni_end();
|
|
|
|
i != e; ++i) {
|
|
|
|
VNInfo *VNI = *i;
|
|
|
|
unsigned VN = VNI->id;
|
2008-02-15 18:24:29 +00:00
|
|
|
if (unsigned LHSSrcReg = li_->getVNInfoSourceReg(VNI)) {
|
|
|
|
if (LHSSrcReg != RHS.reg) {
|
2007-06-08 17:18:56 +00:00
|
|
|
// If this is not a copy from the RHS, its value number will be
|
2007-07-09 12:00:59 +00:00
|
|
|
// unmodified by the coalescing.
|
2007-08-29 20:45:00 +00:00
|
|
|
NewVNInfo[VN] = VNI;
|
2007-06-08 17:18:56 +00:00
|
|
|
LHSValNoAssignments[VN] = VN;
|
|
|
|
} else if (RHSValID == -1) {
|
|
|
|
// Otherwise, it is a copy from the RHS, and we don't already have a
|
|
|
|
// value# for it. Keep the current value number, but remember it.
|
|
|
|
LHSValNoAssignments[VN] = RHSValID = VN;
|
2007-08-29 20:45:00 +00:00
|
|
|
NewVNInfo[VN] = RHSValNoInfo;
|
2007-08-31 08:04:17 +00:00
|
|
|
LHSValsDefinedFromRHS[VNI] = RHSValNoInfo0;
|
2007-06-08 17:18:56 +00:00
|
|
|
} else {
|
|
|
|
// Otherwise, use the specified value #.
|
|
|
|
LHSValNoAssignments[VN] = RHSValID;
|
2007-08-29 20:45:00 +00:00
|
|
|
if (VN == (unsigned)RHSValID) { // Else this val# is dead.
|
|
|
|
NewVNInfo[VN] = RHSValNoInfo;
|
2007-08-31 08:04:17 +00:00
|
|
|
LHSValsDefinedFromRHS[VNI] = RHSValNoInfo0;
|
2007-08-11 00:59:19 +00:00
|
|
|
}
|
2007-06-08 17:18:56 +00:00
|
|
|
}
|
|
|
|
} else {
|
2007-08-29 20:45:00 +00:00
|
|
|
NewVNInfo[VN] = VNI;
|
2007-06-08 17:18:56 +00:00
|
|
|
LHSValNoAssignments[VN] = VN;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(RHSValID != -1 && "Didn't find value #?");
|
|
|
|
RHSValNoAssignments[0] = RHSValID;
|
2007-08-11 00:59:19 +00:00
|
|
|
if (RHSVal0DefinedFromLHS != -1) {
|
2007-09-01 02:03:17 +00:00
|
|
|
// This path doesn't go through ComputeUltimateVN so just set
|
|
|
|
// it to anything.
|
|
|
|
RHSValsDefinedFromLHS[RHSValNoInfo0] = (VNInfo*)1;
|
2007-08-11 00:59:19 +00:00
|
|
|
}
|
2007-06-08 17:18:56 +00:00
|
|
|
} else {
|
|
|
|
// Loop over the value numbers of the LHS, seeing if any are defined from
|
|
|
|
// the RHS.
|
2007-08-29 20:45:00 +00:00
|
|
|
for (LiveInterval::vni_iterator i = LHS.vni_begin(), e = LHS.vni_end();
|
|
|
|
i != e; ++i) {
|
|
|
|
VNInfo *VNI = *i;
|
2008-02-15 18:24:29 +00:00
|
|
|
if (VNI->def == ~1U || VNI->copy == 0) // Src not defined by a copy?
|
2007-06-08 17:18:56 +00:00
|
|
|
continue;
|
|
|
|
|
|
|
|
// DstReg is known to be a register in the LHS interval. If the src is
|
|
|
|
// from the RHS interval, we can use its value #.
|
2008-02-15 18:24:29 +00:00
|
|
|
if (li_->getVNInfoSourceReg(VNI) != RHS.reg)
|
2007-06-08 17:18:56 +00:00
|
|
|
continue;
|
|
|
|
|
|
|
|
// Figure out the value # from the RHS.
|
2008-01-04 08:59:18 +00:00
|
|
|
LHSValsDefinedFromRHS[VNI]=RHS.getLiveRangeContaining(VNI->def-1)->valno;
|
2007-06-08 17:18:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Loop over the value numbers of the RHS, seeing if any are defined from
|
|
|
|
// the LHS.
|
2007-08-29 20:45:00 +00:00
|
|
|
for (LiveInterval::vni_iterator i = RHS.vni_begin(), e = RHS.vni_end();
|
|
|
|
i != e; ++i) {
|
|
|
|
VNInfo *VNI = *i;
|
2008-02-15 18:24:29 +00:00
|
|
|
if (VNI->def == ~1U || VNI->copy == 0) // Src not defined by a copy?
|
2007-06-08 17:18:56 +00:00
|
|
|
continue;
|
|
|
|
|
|
|
|
// DstReg is known to be a register in the RHS interval. If the src is
|
|
|
|
// from the LHS interval, we can use its value #.
|
2008-02-15 18:24:29 +00:00
|
|
|
if (li_->getVNInfoSourceReg(VNI) != LHS.reg)
|
2007-06-08 17:18:56 +00:00
|
|
|
continue;
|
|
|
|
|
|
|
|
// Figure out the value # from the LHS.
|
2008-01-04 08:59:18 +00:00
|
|
|
RHSValsDefinedFromLHS[VNI]=LHS.getLiveRangeContaining(VNI->def-1)->valno;
|
2007-06-08 17:18:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
LHSValNoAssignments.resize(LHS.getNumValNums(), -1);
|
|
|
|
RHSValNoAssignments.resize(RHS.getNumValNums(), -1);
|
2007-08-29 20:45:00 +00:00
|
|
|
NewVNInfo.reserve(LHS.getNumValNums() + RHS.getNumValNums());
|
2007-06-08 17:18:56 +00:00
|
|
|
|
2007-08-29 20:45:00 +00:00
|
|
|
for (LiveInterval::vni_iterator i = LHS.vni_begin(), e = LHS.vni_end();
|
|
|
|
i != e; ++i) {
|
|
|
|
VNInfo *VNI = *i;
|
|
|
|
unsigned VN = VNI->id;
|
|
|
|
if (LHSValNoAssignments[VN] >= 0 || VNI->def == ~1U)
|
2007-06-08 17:18:56 +00:00
|
|
|
continue;
|
2007-08-29 20:45:00 +00:00
|
|
|
ComputeUltimateVN(VNI, NewVNInfo,
|
2007-06-08 17:18:56 +00:00
|
|
|
LHSValsDefinedFromRHS, RHSValsDefinedFromLHS,
|
2007-08-29 20:45:00 +00:00
|
|
|
LHSValNoAssignments, RHSValNoAssignments);
|
2007-06-08 17:18:56 +00:00
|
|
|
}
|
2007-08-29 20:45:00 +00:00
|
|
|
for (LiveInterval::vni_iterator i = RHS.vni_begin(), e = RHS.vni_end();
|
|
|
|
i != e; ++i) {
|
|
|
|
VNInfo *VNI = *i;
|
|
|
|
unsigned VN = VNI->id;
|
|
|
|
if (RHSValNoAssignments[VN] >= 0 || VNI->def == ~1U)
|
2007-06-08 17:18:56 +00:00
|
|
|
continue;
|
|
|
|
// If this value number isn't a copy from the LHS, it's a new number.
|
2007-08-31 08:04:17 +00:00
|
|
|
if (RHSValsDefinedFromLHS.find(VNI) == RHSValsDefinedFromLHS.end()) {
|
2007-08-29 20:45:00 +00:00
|
|
|
NewVNInfo.push_back(VNI);
|
|
|
|
RHSValNoAssignments[VN] = NewVNInfo.size()-1;
|
2007-06-08 17:18:56 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2007-08-29 20:45:00 +00:00
|
|
|
ComputeUltimateVN(VNI, NewVNInfo,
|
2007-06-08 17:18:56 +00:00
|
|
|
RHSValsDefinedFromLHS, LHSValsDefinedFromRHS,
|
2007-08-29 20:45:00 +00:00
|
|
|
RHSValNoAssignments, LHSValNoAssignments);
|
2007-06-08 17:18:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Armed with the mappings of LHS/RHS values to ultimate values, walk the
|
2007-07-09 12:00:59 +00:00
|
|
|
// interval lists to see if these intervals are coalescable.
|
2007-06-08 17:18:56 +00:00
|
|
|
LiveInterval::const_iterator I = LHS.begin();
|
|
|
|
LiveInterval::const_iterator IE = LHS.end();
|
|
|
|
LiveInterval::const_iterator J = RHS.begin();
|
|
|
|
LiveInterval::const_iterator JE = RHS.end();
|
|
|
|
|
|
|
|
// Skip ahead until the first place of potential sharing.
|
|
|
|
if (I->start < J->start) {
|
|
|
|
I = std::upper_bound(I, IE, J->start);
|
|
|
|
if (I != LHS.begin()) --I;
|
|
|
|
} else if (J->start < I->start) {
|
|
|
|
J = std::upper_bound(J, JE, I->start);
|
|
|
|
if (J != RHS.begin()) --J;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
// Determine if these two live ranges overlap.
|
|
|
|
bool Overlaps;
|
|
|
|
if (I->start < J->start) {
|
|
|
|
Overlaps = I->end > J->start;
|
|
|
|
} else {
|
|
|
|
Overlaps = J->end > I->start;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If so, check value # info to determine if they are really different.
|
|
|
|
if (Overlaps) {
|
|
|
|
// If the live range overlap will map to the same value number in the
|
2007-07-09 12:00:59 +00:00
|
|
|
// result liverange, we can still coalesce them. If not, we can't.
|
2007-08-29 20:45:00 +00:00
|
|
|
if (LHSValNoAssignments[I->valno->id] !=
|
|
|
|
RHSValNoAssignments[J->valno->id])
|
2007-06-08 17:18:56 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (I->end < J->end) {
|
|
|
|
++I;
|
|
|
|
if (I == IE) break;
|
|
|
|
} else {
|
|
|
|
++J;
|
|
|
|
if (J == JE) break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-10-14 10:08:34 +00:00
|
|
|
// Update kill info. Some live ranges are extended due to copy coalescing.
|
|
|
|
for (DenseMap<VNInfo*, VNInfo*>::iterator I = LHSValsDefinedFromRHS.begin(),
|
|
|
|
E = LHSValsDefinedFromRHS.end(); I != E; ++I) {
|
|
|
|
VNInfo *VNI = I->first;
|
|
|
|
unsigned LHSValID = LHSValNoAssignments[VNI->id];
|
|
|
|
LiveInterval::removeKill(NewVNInfo[LHSValID], VNI->def);
|
2007-11-29 09:49:23 +00:00
|
|
|
NewVNInfo[LHSValID]->hasPHIKill |= VNI->hasPHIKill;
|
2007-10-14 10:08:34 +00:00
|
|
|
RHS.addKills(NewVNInfo[LHSValID], VNI->kills);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update kill info. Some live ranges are extended due to copy coalescing.
|
|
|
|
for (DenseMap<VNInfo*, VNInfo*>::iterator I = RHSValsDefinedFromLHS.begin(),
|
|
|
|
E = RHSValsDefinedFromLHS.end(); I != E; ++I) {
|
|
|
|
VNInfo *VNI = I->first;
|
|
|
|
unsigned RHSValID = RHSValNoAssignments[VNI->id];
|
|
|
|
LiveInterval::removeKill(NewVNInfo[RHSValID], VNI->def);
|
2007-11-29 09:49:23 +00:00
|
|
|
NewVNInfo[RHSValID]->hasPHIKill |= VNI->hasPHIKill;
|
2007-10-14 10:08:34 +00:00
|
|
|
LHS.addKills(NewVNInfo[RHSValID], VNI->kills);
|
|
|
|
}
|
|
|
|
|
2007-07-09 12:00:59 +00:00
|
|
|
// If we get here, we know that we can coalesce the live ranges. Ask the
|
|
|
|
// intervals to coalesce themselves now.
|
2007-08-28 08:28:51 +00:00
|
|
|
if ((RHS.ranges.size() > LHS.ranges.size() &&
|
2008-02-10 18:45:23 +00:00
|
|
|
TargetRegisterInfo::isVirtualRegister(LHS.reg)) ||
|
|
|
|
TargetRegisterInfo::isPhysicalRegister(RHS.reg)) {
|
2007-08-29 20:45:00 +00:00
|
|
|
RHS.join(LHS, &RHSValNoAssignments[0], &LHSValNoAssignments[0], NewVNInfo);
|
2007-08-28 08:28:51 +00:00
|
|
|
Swapped = true;
|
|
|
|
} else {
|
2007-08-29 20:45:00 +00:00
|
|
|
LHS.join(RHS, &LHSValNoAssignments[0], &RHSValNoAssignments[0], NewVNInfo);
|
2007-08-28 08:28:51 +00:00
|
|
|
Swapped = false;
|
|
|
|
}
|
2007-06-08 17:18:56 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
// DepthMBBCompare - Comparison predicate that sort first based on the loop
|
|
|
|
// depth of the basic block (the unsigned), and then on the MBB number.
|
|
|
|
struct DepthMBBCompare {
|
|
|
|
typedef std::pair<unsigned, MachineBasicBlock*> DepthMBBPair;
|
|
|
|
bool operator()(const DepthMBBPair &LHS, const DepthMBBPair &RHS) const {
|
|
|
|
if (LHS.first > RHS.first) return true; // Deeper loops first
|
|
|
|
return LHS.first == RHS.first &&
|
|
|
|
LHS.second->getNumber() < RHS.second->getNumber();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2007-11-06 08:52:21 +00:00
|
|
|
/// getRepIntervalSize - Returns the size of the interval that represents the
|
|
|
|
/// specified register.
|
|
|
|
template<class SF>
|
|
|
|
unsigned JoinPriorityQueue<SF>::getRepIntervalSize(unsigned Reg) {
|
|
|
|
return Rc->getRepIntervalSize(Reg);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// CopyRecSort::operator - Join priority queue sorting function.
|
|
|
|
///
|
|
|
|
bool CopyRecSort::operator()(CopyRec left, CopyRec right) const {
|
|
|
|
// Inner loops first.
|
|
|
|
if (left.LoopDepth > right.LoopDepth)
|
|
|
|
return false;
|
2008-02-15 18:24:29 +00:00
|
|
|
else if (left.LoopDepth == right.LoopDepth)
|
2007-11-06 08:52:21 +00:00
|
|
|
if (left.isBackEdge && !right.isBackEdge)
|
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2007-07-09 12:00:59 +00:00
|
|
|
void SimpleRegisterCoalescing::CopyCoalesceInMBB(MachineBasicBlock *MBB,
|
2007-10-16 08:04:24 +00:00
|
|
|
std::vector<CopyRec> &TryAgain) {
|
2007-06-08 17:18:56 +00:00
|
|
|
DOUT << ((Value*)MBB->getBasicBlock())->getName() << ":\n";
|
2007-11-06 08:52:21 +00:00
|
|
|
|
2007-10-16 08:04:24 +00:00
|
|
|
std::vector<CopyRec> VirtCopies;
|
|
|
|
std::vector<CopyRec> PhysCopies;
|
2008-04-09 20:57:25 +00:00
|
|
|
std::vector<CopyRec> ImpDefCopies;
|
2007-12-11 02:09:15 +00:00
|
|
|
unsigned LoopDepth = loopInfo->getLoopDepth(MBB);
|
2007-06-08 17:18:56 +00:00
|
|
|
for (MachineBasicBlock::iterator MII = MBB->begin(), E = MBB->end();
|
|
|
|
MII != E;) {
|
|
|
|
MachineInstr *Inst = MII++;
|
|
|
|
|
2007-10-12 08:50:34 +00:00
|
|
|
// If this isn't a copy nor a extract_subreg, we can't join intervals.
|
2007-06-08 17:18:56 +00:00
|
|
|
unsigned SrcReg, DstReg;
|
2007-10-12 08:50:34 +00:00
|
|
|
if (Inst->getOpcode() == TargetInstrInfo::EXTRACT_SUBREG) {
|
|
|
|
DstReg = Inst->getOperand(0).getReg();
|
|
|
|
SrcReg = Inst->getOperand(1).getReg();
|
2008-04-09 20:57:25 +00:00
|
|
|
} else if (Inst->getOpcode() == TargetInstrInfo::INSERT_SUBREG) {
|
|
|
|
DstReg = Inst->getOperand(0).getReg();
|
|
|
|
SrcReg = Inst->getOperand(2).getReg();
|
2007-10-12 08:50:34 +00:00
|
|
|
} else if (!tii_->isMoveInstr(*Inst, SrcReg, DstReg))
|
|
|
|
continue;
|
2007-10-16 08:04:24 +00:00
|
|
|
|
2008-02-15 18:24:29 +00:00
|
|
|
bool SrcIsPhys = TargetRegisterInfo::isPhysicalRegister(SrcReg);
|
|
|
|
bool DstIsPhys = TargetRegisterInfo::isPhysicalRegister(DstReg);
|
2007-11-06 08:52:21 +00:00
|
|
|
if (NewHeuristic) {
|
2008-02-15 18:24:29 +00:00
|
|
|
JoinQueue->push(CopyRec(Inst, LoopDepth, isBackEdgeCopy(Inst, DstReg)));
|
2007-11-06 08:52:21 +00:00
|
|
|
} else {
|
2008-04-09 20:57:25 +00:00
|
|
|
if (li_->hasInterval(SrcReg) && li_->getInterval(SrcReg).empty())
|
|
|
|
ImpDefCopies.push_back(CopyRec(Inst, 0, false));
|
|
|
|
else if (SrcIsPhys || DstIsPhys)
|
2008-02-15 18:24:29 +00:00
|
|
|
PhysCopies.push_back(CopyRec(Inst, 0, false));
|
2007-11-06 08:52:21 +00:00
|
|
|
else
|
2008-02-15 18:24:29 +00:00
|
|
|
VirtCopies.push_back(CopyRec(Inst, 0, false));
|
2007-11-06 08:52:21 +00:00
|
|
|
}
|
2007-10-16 08:04:24 +00:00
|
|
|
}
|
|
|
|
|
2007-11-06 08:52:21 +00:00
|
|
|
if (NewHeuristic)
|
|
|
|
return;
|
|
|
|
|
2008-04-09 20:57:25 +00:00
|
|
|
// Try coalescing implicit copies first, followed by copies to / from
|
|
|
|
// physical registers, then finally copies from virtual registers to
|
|
|
|
// virtual registers.
|
|
|
|
for (unsigned i = 0, e = ImpDefCopies.size(); i != e; ++i) {
|
|
|
|
CopyRec &TheCopy = ImpDefCopies[i];
|
|
|
|
bool Again = false;
|
|
|
|
if (!JoinCopy(TheCopy, Again))
|
|
|
|
if (Again)
|
|
|
|
TryAgain.push_back(TheCopy);
|
|
|
|
}
|
2007-10-16 08:04:24 +00:00
|
|
|
for (unsigned i = 0, e = PhysCopies.size(); i != e; ++i) {
|
|
|
|
CopyRec &TheCopy = PhysCopies[i];
|
2007-11-01 06:22:48 +00:00
|
|
|
bool Again = false;
|
2007-11-06 08:52:21 +00:00
|
|
|
if (!JoinCopy(TheCopy, Again))
|
2007-11-01 06:22:48 +00:00
|
|
|
if (Again)
|
|
|
|
TryAgain.push_back(TheCopy);
|
2007-10-16 08:04:24 +00:00
|
|
|
}
|
|
|
|
for (unsigned i = 0, e = VirtCopies.size(); i != e; ++i) {
|
|
|
|
CopyRec &TheCopy = VirtCopies[i];
|
2007-11-01 06:22:48 +00:00
|
|
|
bool Again = false;
|
2007-11-06 08:52:21 +00:00
|
|
|
if (!JoinCopy(TheCopy, Again))
|
2007-11-01 06:22:48 +00:00
|
|
|
if (Again)
|
|
|
|
TryAgain.push_back(TheCopy);
|
2007-06-08 17:18:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void SimpleRegisterCoalescing::joinIntervals() {
|
|
|
|
DOUT << "********** JOINING INTERVALS ***********\n";
|
|
|
|
|
2007-11-06 08:52:21 +00:00
|
|
|
if (NewHeuristic)
|
|
|
|
JoinQueue = new JoinPriorityQueue<CopyRecSort>(this);
|
|
|
|
|
2007-06-08 17:18:56 +00:00
|
|
|
std::vector<CopyRec> TryAgainList;
|
2007-11-06 08:52:21 +00:00
|
|
|
if (loopInfo->begin() == loopInfo->end()) {
|
2007-06-08 17:18:56 +00:00
|
|
|
// If there are no loops in the function, join intervals in function order.
|
|
|
|
for (MachineFunction::iterator I = mf_->begin(), E = mf_->end();
|
|
|
|
I != E; ++I)
|
2007-10-16 08:04:24 +00:00
|
|
|
CopyCoalesceInMBB(I, TryAgainList);
|
2007-06-08 17:18:56 +00:00
|
|
|
} else {
|
|
|
|
// Otherwise, join intervals in inner loops before other intervals.
|
|
|
|
// Unfortunately we can't just iterate over loop hierarchy here because
|
|
|
|
// there may be more MBB's than BB's. Collect MBB's for sorting.
|
|
|
|
|
|
|
|
// Join intervals in the function prolog first. We want to join physical
|
|
|
|
// registers with virtual registers before the intervals got too long.
|
|
|
|
std::vector<std::pair<unsigned, MachineBasicBlock*> > MBBs;
|
2007-12-11 02:09:15 +00:00
|
|
|
for (MachineFunction::iterator I = mf_->begin(), E = mf_->end();I != E;++I){
|
|
|
|
MachineBasicBlock *MBB = I;
|
|
|
|
MBBs.push_back(std::make_pair(loopInfo->getLoopDepth(MBB), I));
|
|
|
|
}
|
2007-06-08 17:18:56 +00:00
|
|
|
|
|
|
|
// Sort by loop depth.
|
|
|
|
std::sort(MBBs.begin(), MBBs.end(), DepthMBBCompare());
|
|
|
|
|
|
|
|
// Finally, join intervals in loop nest order.
|
|
|
|
for (unsigned i = 0, e = MBBs.size(); i != e; ++i)
|
2007-10-16 08:04:24 +00:00
|
|
|
CopyCoalesceInMBB(MBBs[i].second, TryAgainList);
|
2007-06-08 17:18:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Joining intervals can allow other intervals to be joined. Iteratively join
|
|
|
|
// until we make no progress.
|
2007-11-06 08:52:21 +00:00
|
|
|
if (NewHeuristic) {
|
|
|
|
SmallVector<CopyRec, 16> TryAgain;
|
|
|
|
bool ProgressMade = true;
|
|
|
|
while (ProgressMade) {
|
|
|
|
ProgressMade = false;
|
|
|
|
while (!JoinQueue->empty()) {
|
|
|
|
CopyRec R = JoinQueue->pop();
|
2007-11-01 06:22:48 +00:00
|
|
|
bool Again = false;
|
2007-11-06 08:52:21 +00:00
|
|
|
bool Success = JoinCopy(R, Again);
|
|
|
|
if (Success)
|
2007-11-01 06:22:48 +00:00
|
|
|
ProgressMade = true;
|
2007-11-06 08:52:21 +00:00
|
|
|
else if (Again)
|
|
|
|
TryAgain.push_back(R);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ProgressMade) {
|
|
|
|
while (!TryAgain.empty()) {
|
|
|
|
JoinQueue->push(TryAgain.back());
|
|
|
|
TryAgain.pop_back();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
bool ProgressMade = true;
|
|
|
|
while (ProgressMade) {
|
|
|
|
ProgressMade = false;
|
|
|
|
|
|
|
|
for (unsigned i = 0, e = TryAgainList.size(); i != e; ++i) {
|
|
|
|
CopyRec &TheCopy = TryAgainList[i];
|
|
|
|
if (TheCopy.MI) {
|
|
|
|
bool Again = false;
|
|
|
|
bool Success = JoinCopy(TheCopy, Again);
|
|
|
|
if (Success || !Again) {
|
|
|
|
TheCopy.MI = 0; // Mark this one as done.
|
|
|
|
ProgressMade = true;
|
|
|
|
}
|
2007-11-01 06:22:48 +00:00
|
|
|
}
|
2007-06-08 17:18:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-11-06 08:52:21 +00:00
|
|
|
if (NewHeuristic)
|
2008-02-15 18:24:29 +00:00
|
|
|
delete JoinQueue;
|
2007-06-08 17:18:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Return true if the two specified registers belong to different register
|
|
|
|
/// classes. The registers may be either phys or virt regs.
|
|
|
|
bool SimpleRegisterCoalescing::differingRegisterClasses(unsigned RegA,
|
2007-10-12 08:50:34 +00:00
|
|
|
unsigned RegB) const {
|
2007-06-08 17:18:56 +00:00
|
|
|
|
|
|
|
// Get the register classes for the first reg.
|
2008-02-10 18:45:23 +00:00
|
|
|
if (TargetRegisterInfo::isPhysicalRegister(RegA)) {
|
|
|
|
assert(TargetRegisterInfo::isVirtualRegister(RegB) &&
|
2007-06-08 17:18:56 +00:00
|
|
|
"Shouldn't consider two physregs!");
|
2008-02-15 18:24:29 +00:00
|
|
|
return !mri_->getRegClass(RegB)->contains(RegA);
|
2007-06-08 17:18:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Compare against the regclass for the second reg.
|
2008-02-15 18:24:29 +00:00
|
|
|
const TargetRegisterClass *RegClass = mri_->getRegClass(RegA);
|
2008-02-10 18:45:23 +00:00
|
|
|
if (TargetRegisterInfo::isVirtualRegister(RegB))
|
2008-02-15 18:24:29 +00:00
|
|
|
return RegClass != mri_->getRegClass(RegB);
|
2007-06-08 17:18:56 +00:00
|
|
|
else
|
|
|
|
return !RegClass->contains(RegB);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// lastRegisterUse - Returns the last use of the specific register between
|
2008-02-15 18:24:29 +00:00
|
|
|
/// cycles Start and End or NULL if there are no uses.
|
|
|
|
MachineOperand *
|
2007-12-31 04:13:23 +00:00
|
|
|
SimpleRegisterCoalescing::lastRegisterUse(unsigned Start, unsigned End,
|
2008-02-15 18:24:29 +00:00
|
|
|
unsigned Reg, unsigned &UseIdx) const{
|
|
|
|
UseIdx = 0;
|
|
|
|
if (TargetRegisterInfo::isVirtualRegister(Reg)) {
|
|
|
|
MachineOperand *LastUse = NULL;
|
|
|
|
for (MachineRegisterInfo::use_iterator I = mri_->use_begin(Reg),
|
|
|
|
E = mri_->use_end(); I != E; ++I) {
|
|
|
|
MachineOperand &Use = I.getOperand();
|
|
|
|
MachineInstr *UseMI = Use.getParent();
|
2008-03-25 02:02:19 +00:00
|
|
|
unsigned SrcReg, DstReg;
|
|
|
|
if (tii_->isMoveInstr(*UseMI, SrcReg, DstReg) && SrcReg == DstReg)
|
|
|
|
// Ignore identity copies.
|
|
|
|
continue;
|
2008-02-15 18:24:29 +00:00
|
|
|
unsigned Idx = li_->getInstructionIndex(UseMI);
|
|
|
|
if (Idx >= Start && Idx < End && Idx >= UseIdx) {
|
|
|
|
LastUse = &Use;
|
|
|
|
UseIdx = Idx;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return LastUse;
|
|
|
|
}
|
|
|
|
|
2007-06-08 17:18:56 +00:00
|
|
|
int e = (End-1) / InstrSlots::NUM * InstrSlots::NUM;
|
|
|
|
int s = Start;
|
|
|
|
while (e >= s) {
|
|
|
|
// Skip deleted instructions
|
|
|
|
MachineInstr *MI = li_->getInstructionFromIndex(e);
|
|
|
|
while ((e - InstrSlots::NUM) >= s && !MI) {
|
|
|
|
e -= InstrSlots::NUM;
|
|
|
|
MI = li_->getInstructionFromIndex(e);
|
|
|
|
}
|
|
|
|
if (e < s || MI == NULL)
|
|
|
|
return NULL;
|
|
|
|
|
2008-03-25 02:02:19 +00:00
|
|
|
// Ignore identity copies.
|
|
|
|
unsigned SrcReg, DstReg;
|
|
|
|
if (!(tii_->isMoveInstr(*MI, SrcReg, DstReg) && SrcReg == DstReg))
|
|
|
|
for (unsigned i = 0, NumOps = MI->getNumOperands(); i != NumOps; ++i) {
|
|
|
|
MachineOperand &Use = MI->getOperand(i);
|
|
|
|
if (Use.isRegister() && Use.isUse() && Use.getReg() &&
|
|
|
|
tri_->regsOverlap(Use.getReg(), Reg)) {
|
|
|
|
UseIdx = e;
|
|
|
|
return &Use;
|
|
|
|
}
|
2007-06-08 17:18:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
e -= InstrSlots::NUM;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void SimpleRegisterCoalescing::printRegName(unsigned reg) const {
|
2008-02-10 18:45:23 +00:00
|
|
|
if (TargetRegisterInfo::isPhysicalRegister(reg))
|
2008-02-26 21:47:57 +00:00
|
|
|
cerr << tri_->getName(reg);
|
2007-06-08 17:18:56 +00:00
|
|
|
else
|
|
|
|
cerr << "%reg" << reg;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SimpleRegisterCoalescing::releaseMemory() {
|
2007-11-06 08:52:21 +00:00
|
|
|
JoinedCopies.clear();
|
2007-06-08 17:18:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static bool isZeroLengthInterval(LiveInterval *li) {
|
|
|
|
for (LiveInterval::Ranges::const_iterator
|
|
|
|
i = li->ranges.begin(), e = li->ranges.end(); i != e; ++i)
|
|
|
|
if (i->end - i->start > LiveIntervals::InstrSlots::NUM)
|
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2008-04-03 16:41:54 +00:00
|
|
|
/// TurnCopyIntoImpDef - If source of the specified copy is an implicit def,
|
|
|
|
/// turn the copy into an implicit def.
|
|
|
|
bool
|
|
|
|
SimpleRegisterCoalescing::TurnCopyIntoImpDef(MachineBasicBlock::iterator &I,
|
|
|
|
MachineBasicBlock *MBB,
|
|
|
|
unsigned DstReg, unsigned SrcReg) {
|
|
|
|
MachineInstr *CopyMI = &*I;
|
|
|
|
unsigned CopyIdx = li_->getDefIndex(li_->getInstructionIndex(CopyMI));
|
|
|
|
if (!li_->hasInterval(SrcReg))
|
|
|
|
return false;
|
|
|
|
LiveInterval &SrcInt = li_->getInterval(SrcReg);
|
|
|
|
if (!SrcInt.empty())
|
|
|
|
return false;
|
2008-04-09 01:30:15 +00:00
|
|
|
if (!li_->hasInterval(DstReg))
|
|
|
|
return false;
|
2008-04-03 16:41:54 +00:00
|
|
|
LiveInterval &DstInt = li_->getInterval(DstReg);
|
2008-04-16 18:48:43 +00:00
|
|
|
const LiveRange *DstLR = DstInt.getLiveRangeContaining(CopyIdx);
|
2008-04-03 16:41:54 +00:00
|
|
|
DstInt.removeValNo(DstLR->valno);
|
|
|
|
CopyMI->setDesc(tii_->get(TargetInstrInfo::IMPLICIT_DEF));
|
|
|
|
for (int i = CopyMI->getNumOperands() - 1, e = 0; i > e; --i)
|
|
|
|
CopyMI->RemoveOperand(i);
|
|
|
|
bool NoUse = mri_->use_begin(SrcReg) == mri_->use_end();
|
|
|
|
if (NoUse) {
|
|
|
|
for (MachineRegisterInfo::reg_iterator I = mri_->reg_begin(SrcReg),
|
|
|
|
E = mri_->reg_end(); I != E; ) {
|
|
|
|
assert(I.getOperand().isDef());
|
|
|
|
MachineInstr *DefMI = &*I;
|
|
|
|
++I;
|
|
|
|
// The implicit_def source has no other uses, delete it.
|
|
|
|
assert(DefMI->getOpcode() == TargetInstrInfo::IMPLICIT_DEF);
|
|
|
|
li_->RemoveMachineInstrFromMaps(DefMI);
|
|
|
|
DefMI->eraseFromParent();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
++I;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-06-08 17:18:56 +00:00
|
|
|
bool SimpleRegisterCoalescing::runOnMachineFunction(MachineFunction &fn) {
|
|
|
|
mf_ = &fn;
|
2008-02-13 03:01:43 +00:00
|
|
|
mri_ = &fn.getRegInfo();
|
2007-06-08 17:18:56 +00:00
|
|
|
tm_ = &fn.getTarget();
|
2008-02-10 18:45:23 +00:00
|
|
|
tri_ = tm_->getRegisterInfo();
|
2007-06-08 17:18:56 +00:00
|
|
|
tii_ = tm_->getInstrInfo();
|
|
|
|
li_ = &getAnalysis<LiveIntervals>();
|
|
|
|
lv_ = &getAnalysis<LiveVariables>();
|
2007-12-11 02:09:15 +00:00
|
|
|
loopInfo = &getAnalysis<MachineLoopInfo>();
|
2007-06-08 17:18:56 +00:00
|
|
|
|
|
|
|
DOUT << "********** SIMPLE REGISTER COALESCING **********\n"
|
|
|
|
<< "********** Function: "
|
|
|
|
<< ((Value*)mf_->getFunction())->getName() << '\n';
|
|
|
|
|
2008-02-10 18:45:23 +00:00
|
|
|
allocatableRegs_ = tri_->getAllocatableSet(fn);
|
|
|
|
for (TargetRegisterInfo::regclass_iterator I = tri_->regclass_begin(),
|
|
|
|
E = tri_->regclass_end(); I != E; ++I)
|
2008-01-04 08:59:18 +00:00
|
|
|
allocatableRCRegs_.insert(std::make_pair(*I,
|
2008-02-10 18:45:23 +00:00
|
|
|
tri_->getAllocatableSet(fn, *I)));
|
2007-06-08 17:18:56 +00:00
|
|
|
|
2007-07-09 12:00:59 +00:00
|
|
|
// Join (coalesce) intervals if requested.
|
2007-06-08 17:18:56 +00:00
|
|
|
if (EnableJoining) {
|
|
|
|
joinIntervals();
|
|
|
|
DOUT << "********** INTERVALS POST JOINING **********\n";
|
2008-01-04 08:59:18 +00:00
|
|
|
for (LiveIntervals::iterator I = li_->begin(), E = li_->end(); I != E; ++I){
|
2008-02-10 18:45:23 +00:00
|
|
|
I->second.print(DOUT, tri_);
|
2007-06-08 17:18:56 +00:00
|
|
|
DOUT << "\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-02-15 18:24:29 +00:00
|
|
|
// Perform a final pass over the instructions and compute spill weights
|
|
|
|
// and remove identity moves.
|
2007-06-08 17:18:56 +00:00
|
|
|
for (MachineFunction::iterator mbbi = mf_->begin(), mbbe = mf_->end();
|
|
|
|
mbbi != mbbe; ++mbbi) {
|
|
|
|
MachineBasicBlock* mbb = mbbi;
|
2007-12-11 02:09:15 +00:00
|
|
|
unsigned loopDepth = loopInfo->getLoopDepth(mbb);
|
2007-06-08 17:18:56 +00:00
|
|
|
|
|
|
|
for (MachineBasicBlock::iterator mii = mbb->begin(), mie = mbb->end();
|
|
|
|
mii != mie; ) {
|
2008-04-24 09:06:33 +00:00
|
|
|
MachineInstr *MI = mii;
|
|
|
|
unsigned SrcReg, DstReg;
|
|
|
|
if (JoinedCopies.count(MI)) {
|
|
|
|
// Delete all coalesced copies.
|
|
|
|
if (!tii_->isMoveInstr(*MI, SrcReg, DstReg)) {
|
|
|
|
assert((MI->getOpcode() == TargetInstrInfo::EXTRACT_SUBREG ||
|
|
|
|
MI->getOpcode() == TargetInstrInfo::INSERT_SUBREG) &&
|
|
|
|
"Unrecognized copy instruction");
|
|
|
|
DstReg = MI->getOperand(0).getReg();
|
|
|
|
}
|
|
|
|
if (MI->registerDefIsDead(DstReg)) {
|
|
|
|
LiveInterval &li = li_->getInterval(DstReg);
|
|
|
|
if (!ShortenDeadCopySrcLiveRange(li, MI))
|
|
|
|
ShortenDeadCopyLiveRange(li, MI);
|
|
|
|
}
|
|
|
|
li_->RemoveMachineInstrFromMaps(MI);
|
|
|
|
mii = mbbi->erase(mii);
|
|
|
|
++numPeep;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the move will be an identity move delete it
|
|
|
|
bool isMove = tii_->isMoveInstr(*mii, SrcReg, DstReg);
|
|
|
|
if (isMove && SrcReg == DstReg) {
|
|
|
|
if (li_->hasInterval(SrcReg)) {
|
|
|
|
LiveInterval &RegInt = li_->getInterval(SrcReg);
|
2008-03-18 08:26:47 +00:00
|
|
|
// If def of this move instruction is dead, remove its live range
|
|
|
|
// from the dstination register's live interval.
|
2008-04-24 09:06:33 +00:00
|
|
|
if (mii->registerDefIsDead(DstReg)) {
|
2008-04-16 20:24:25 +00:00
|
|
|
if (!ShortenDeadCopySrcLiveRange(RegInt, mii))
|
|
|
|
ShortenDeadCopyLiveRange(RegInt, mii);
|
2008-03-18 08:26:47 +00:00
|
|
|
}
|
|
|
|
}
|
2007-06-08 17:18:56 +00:00
|
|
|
li_->RemoveMachineInstrFromMaps(mii);
|
|
|
|
mii = mbbi->erase(mii);
|
|
|
|
++numPeep;
|
2008-04-24 09:06:33 +00:00
|
|
|
} else if (!isMove || !TurnCopyIntoImpDef(mii, mbb, DstReg, SrcReg)) {
|
2007-06-08 17:18:56 +00:00
|
|
|
SmallSet<unsigned, 4> UniqueUses;
|
|
|
|
for (unsigned i = 0, e = mii->getNumOperands(); i != e; ++i) {
|
|
|
|
const MachineOperand &mop = mii->getOperand(i);
|
|
|
|
if (mop.isRegister() && mop.getReg() &&
|
2008-02-10 18:45:23 +00:00
|
|
|
TargetRegisterInfo::isVirtualRegister(mop.getReg())) {
|
2008-02-15 18:24:29 +00:00
|
|
|
unsigned reg = mop.getReg();
|
2007-06-08 17:18:56 +00:00
|
|
|
// Multiple uses of reg by the same instruction. It should not
|
|
|
|
// contribute to spill weight again.
|
|
|
|
if (UniqueUses.count(reg) != 0)
|
|
|
|
continue;
|
|
|
|
LiveInterval &RegInt = li_->getInterval(reg);
|
Live interval splitting:
When a live interval is being spilled, rather than creating short, non-spillable
intervals for every def / use, split the interval at BB boundaries. That is, for
every BB where the live interval is defined or used, create a new interval that
covers all the defs and uses in the BB.
This is designed to eliminate one common problem: multiple reloads of the same
value in a single basic block. Note, it does *not* decrease the number of spills
since no copies are inserted so the split intervals are *connected* through
spill and reloads (or rematerialization). The newly created intervals can be
spilled again, in that case, since it does not span multiple basic blocks, it's
spilled in the usual manner. However, it can reuse the same stack slot as the
previously split interval.
This is currently controlled by -split-intervals-at-bb.
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@44198 91177308-0d34-0410-b5e6-96231b3b80d8
2007-11-17 00:40:40 +00:00
|
|
|
RegInt.weight +=
|
|
|
|
li_->getSpillWeight(mop.isDef(), mop.isUse(), loopDepth);
|
2007-06-08 17:18:56 +00:00
|
|
|
UniqueUses.insert(reg);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
++mii;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (LiveIntervals::iterator I = li_->begin(), E = li_->end(); I != E; ++I) {
|
|
|
|
LiveInterval &LI = I->second;
|
2008-02-10 18:45:23 +00:00
|
|
|
if (TargetRegisterInfo::isVirtualRegister(LI.reg)) {
|
2007-06-08 17:18:56 +00:00
|
|
|
// If the live interval length is essentially zero, i.e. in every live
|
|
|
|
// range the use follows def immediately, it doesn't make sense to spill
|
|
|
|
// it and hope it will be easier to allocate for this li.
|
|
|
|
if (isZeroLengthInterval(&LI))
|
|
|
|
LI.weight = HUGE_VALF;
|
2007-12-06 00:01:56 +00:00
|
|
|
else {
|
|
|
|
bool isLoad = false;
|
2008-02-09 08:36:28 +00:00
|
|
|
if (li_->isReMaterializable(LI, isLoad)) {
|
2007-12-06 00:01:56 +00:00
|
|
|
// If all of the definitions of the interval are re-materializable,
|
|
|
|
// it is a preferred candidate for spilling. If non of the defs are
|
|
|
|
// loads, then it's potentially very cheap to re-materialize.
|
|
|
|
// FIXME: this gets much more complicated once we support non-trivial
|
|
|
|
// re-materialization.
|
|
|
|
if (isLoad)
|
|
|
|
LI.weight *= 0.9F;
|
|
|
|
else
|
|
|
|
LI.weight *= 0.5F;
|
|
|
|
}
|
|
|
|
}
|
2007-06-08 17:18:56 +00:00
|
|
|
|
|
|
|
// Slightly prefer live interval that has been assigned a preferred reg.
|
|
|
|
if (LI.preference)
|
|
|
|
LI.weight *= 1.01F;
|
|
|
|
|
|
|
|
// Divide the weight of the interval by its size. This encourages
|
|
|
|
// spilling of intervals that are large and have few uses, and
|
|
|
|
// discourages spilling of small intervals with many uses.
|
|
|
|
LI.weight /= LI.getSize();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
DEBUG(dump());
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// print - Implement the dump method.
|
|
|
|
void SimpleRegisterCoalescing::print(std::ostream &O, const Module* m) const {
|
|
|
|
li_->print(O, m);
|
|
|
|
}
|
2007-09-06 16:18:45 +00:00
|
|
|
|
|
|
|
RegisterCoalescer* llvm::createSimpleRegisterCoalescer() {
|
|
|
|
return new SimpleRegisterCoalescing();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make sure that anything that uses RegisterCoalescer pulls in this file...
|
|
|
|
DEFINING_FILE_FOR(SimpleRegisterCoalescing)
|