aboutsummaryrefslogtreecommitdiff
path: root/lib/CodeGen/LiveIntervalAnalysis.cpp
diff options
context:
space:
mode:
authorDale Johannesen <dalej@apple.com>2008-09-17 21:13:11 +0000
committerDale Johannesen <dalej@apple.com>2008-09-17 21:13:11 +0000
commit91aac1015e6714d959801dd8d60f55a72827dc4d (patch)
treec8440b8031a0fe3d7c63fcf9145ffc930759c547 /lib/CodeGen/LiveIntervalAnalysis.cpp
parent870e4bef419b1bd3e5ee05673975f1c05198b612 (diff)
Add a bit to mark operands of asm's that conflict
with an earlyclobber operand elsewhere. Propagate this bit and the earlyclobber bit through SDISel. Change linear-scan RA not to allocate regs in a way that conflicts with an earlyclobber. See also comments. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@56290 91177308-0d34-0410-b5e6-96231b3b80d8
Diffstat (limited to 'lib/CodeGen/LiveIntervalAnalysis.cpp')
-rw-r--r--lib/CodeGen/LiveIntervalAnalysis.cpp82
1 files changed, 80 insertions, 2 deletions
diff --git a/lib/CodeGen/LiveIntervalAnalysis.cpp b/lib/CodeGen/LiveIntervalAnalysis.cpp
index 577400c674..cb83194d97 100644
--- a/lib/CodeGen/LiveIntervalAnalysis.cpp
+++ b/lib/CodeGen/LiveIntervalAnalysis.cpp
@@ -673,7 +673,10 @@ exit:
/// registers. for some ordering of the machine instructions [1,N] a
/// live interval is an interval [i, j) where 1 <= i <= j < N for
/// which a variable is live
-void LiveIntervals::computeIntervals() {
+void LiveIntervals::computeIntervals() {
+ AsmsThatEarlyClobber.clear();
+ AsmsWithEarlyClobberConflict.clear();
+
DOUT << "********** COMPUTING LIVE INTERVALS **********\n"
<< "********** Function: "
<< ((Value*)mf_->getFunction())->getName() << '\n';
@@ -710,8 +713,17 @@ void LiveIntervals::computeIntervals() {
for (int i = MI->getNumOperands() - 1; i >= 0; --i) {
MachineOperand &MO = MI->getOperand(i);
// handle register defs - build intervals
- if (MO.isRegister() && MO.getReg() && MO.isDef())
+ if (MO.isRegister() && MO.getReg() && MO.isDef()) {
handleRegisterDef(MBB, MI, MIIndex, MO, i);
+ if (MO.isEarlyClobber()) {
+ AsmsThatEarlyClobber.insert(std::make_pair(MO.getReg(), MI));
+ }
+ }
+ if (MO.isRegister() && !MO.isDef() &&
+ MO.getReg() && TargetRegisterInfo::isVirtualRegister(MO.getReg()) &&
+ MO.overlapsEarlyClobber()) {
+ AsmsWithEarlyClobberConflict.insert(std::make_pair(MO.getReg(), MI));
+ }
}
MIIndex += InstrSlots::NUM;
@@ -740,6 +752,72 @@ bool LiveIntervals::findLiveInMBBs(const LiveRange &LR,
return ResVal;
}
+/// noEarlyclobberConflict - see whether virtual reg VReg has a conflict with
+/// hard reg HReg because of earlyclobbers.
+///
+/// Earlyclobber operands may not be assigned the same register as
+/// each other, or as earlyclobber-conflict operands (i.e. those that
+/// are non-earlyclobbered inputs to an asm that also has earlyclobbers).
+///
+/// Thus there are two cases to check for:
+/// 1. VReg is an earlyclobber-conflict register and HReg is an earlyclobber
+/// register in some asm that also has VReg as an input.
+/// 2. VReg is an earlyclobber register and HReg is an earlyclobber-conflict
+/// input elsewhere in some asm.
+/// In both cases HReg can be assigned by the user, or assigned early in
+/// register allocation.
+///
+/// Dropping the distinction between earlyclobber and earlyclobber-conflict,
+/// keeping only one multimap, looks promising, but two earlyclobber-conflict
+/// operands may be assigned the same register if they happen to contain the
+/// same value, and that implementation would prevent this.
+///
+bool LiveIntervals::noEarlyclobberConflict(unsigned VReg, VirtRegMap &vrm,
+ unsigned HReg) {
+ typedef std::multimap<unsigned, MachineInstr*>::iterator It;
+
+ // Short circuit the most common case.
+ if (AsmsWithEarlyClobberConflict.size()!=0) {
+ std::pair<It, It> x = AsmsWithEarlyClobberConflict.equal_range(VReg);
+ for (It I = x.first; I!=x.second; I++) {
+ MachineInstr* MI = I->second;
+ for (int i = MI->getNumOperands() - 1; i >= 0; --i) {
+ MachineOperand &MO = MI->getOperand(i);
+ if (MO.isRegister() && MO.isEarlyClobber()) {
+ unsigned PhysReg = MO.getReg();
+ if (PhysReg && TargetRegisterInfo::isVirtualRegister(PhysReg)) {
+ if (!vrm.hasPhys(PhysReg))
+ continue;
+ PhysReg = vrm.getPhys(PhysReg);
+ }
+ if (PhysReg==HReg)
+ return false;
+ }
+ }
+ }
+ }
+ // Short circuit the most common case.
+ if (AsmsThatEarlyClobber.size()!=0) {
+ std::pair<It, It> x = AsmsThatEarlyClobber.equal_range(VReg);
+ for (It I = x.first; I!=x.second; I++) {
+ MachineInstr* MI = I->second;
+ for (int i = MI->getNumOperands() - 1; i >= 0; --i) {
+ MachineOperand &MO = MI->getOperand(i);
+ if (MO.isRegister() && MO.overlapsEarlyClobber()) {
+ unsigned PhysReg = MO.getReg();
+ if (PhysReg && TargetRegisterInfo::isVirtualRegister(PhysReg)) {
+ if (!vrm.hasPhys(PhysReg))
+ continue;
+ PhysReg = vrm.getPhys(PhysReg);
+ }
+ if (PhysReg==HReg)
+ return false;
+ }
+ }
+ }
+ }
+ return true;
+}
LiveInterval* LiveIntervals::createInterval(unsigned reg) {
float Weight = TargetRegisterInfo::isPhysicalRegister(reg) ?