mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-01-28 06:32:09 +00:00
Fix PR3486. Fix a bug in code that manually patch physical register live interval after its sub-register is coalesced with a virtual register.
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@64082 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
86fb9fdb20
commit
0a1fcce092
@ -283,6 +283,10 @@ namespace llvm {
|
|||||||
return *I->second;
|
return *I->second;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// dupInterval - Duplicate a live interval. The caller is responsible for
|
||||||
|
/// managing the allocated memory.
|
||||||
|
LiveInterval *dupInterval(LiveInterval *li);
|
||||||
|
|
||||||
/// addLiveRangeToEndOfBlock - Given a register and an instruction,
|
/// addLiveRangeToEndOfBlock - Given a register and an instruction,
|
||||||
/// adds a live range from that instruction to the end of its MBB.
|
/// adds a live range from that instruction to the end of its MBB.
|
||||||
LiveRange addLiveRangeToEndOfBlock(unsigned reg,
|
LiveRange addLiveRangeToEndOfBlock(unsigned reg,
|
||||||
|
@ -824,11 +824,18 @@ bool LiveIntervals::findReachableMBBs(unsigned Start, unsigned End,
|
|||||||
}
|
}
|
||||||
|
|
||||||
LiveInterval* LiveIntervals::createInterval(unsigned reg) {
|
LiveInterval* LiveIntervals::createInterval(unsigned reg) {
|
||||||
float Weight = TargetRegisterInfo::isPhysicalRegister(reg) ?
|
float Weight = TargetRegisterInfo::isPhysicalRegister(reg) ? HUGE_VALF : 0.0F;
|
||||||
HUGE_VALF : 0.0F;
|
|
||||||
return new LiveInterval(reg, Weight);
|
return new LiveInterval(reg, Weight);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// dupInterval - Duplicate a live interval. The caller is responsible for
|
||||||
|
/// managing the allocated memory.
|
||||||
|
LiveInterval* LiveIntervals::dupInterval(LiveInterval *li) {
|
||||||
|
LiveInterval *NewLI = createInterval(li->reg);
|
||||||
|
NewLI->Copy(*li, getVNInfoAllocator());
|
||||||
|
return NewLI;
|
||||||
|
}
|
||||||
|
|
||||||
/// getVNInfoSourceReg - Helper function that parses the specified VNInfo
|
/// getVNInfoSourceReg - Helper function that parses the specified VNInfo
|
||||||
/// copy field and returns the source register that defines it.
|
/// copy field and returns the source register that defines it.
|
||||||
unsigned LiveIntervals::getVNInfoSourceReg(const VNInfo *VNI) const {
|
unsigned LiveIntervals::getVNInfoSourceReg(const VNInfo *VNI) const {
|
||||||
|
@ -1354,6 +1354,15 @@ bool SimpleRegisterCoalescing::JoinCopy(CopyRec &TheCopy, bool &Again) {
|
|||||||
DOUT << " and "; DstInt.print(DOUT, tri_);
|
DOUT << " and "; DstInt.print(DOUT, tri_);
|
||||||
DOUT << ": ";
|
DOUT << ": ";
|
||||||
|
|
||||||
|
// Save a copy of the virtual register live interval. We'll manually
|
||||||
|
// merge this into the "real" physical register live interval this is
|
||||||
|
// coalesced with.
|
||||||
|
LiveInterval *SavedLI = 0;
|
||||||
|
if (RealDstReg)
|
||||||
|
SavedLI = li_->dupInterval(&SrcInt);
|
||||||
|
else if (RealSrcReg)
|
||||||
|
SavedLI = li_->dupInterval(&DstInt);
|
||||||
|
|
||||||
// Check if it is necessary to propagate "isDead" property.
|
// Check if it is necessary to propagate "isDead" property.
|
||||||
if (!isExtSubReg && !isInsSubReg) {
|
if (!isExtSubReg && !isInsSubReg) {
|
||||||
MachineOperand *mopd = CopyMI->findRegisterDefOperand(DstReg, false);
|
MachineOperand *mopd = CopyMI->findRegisterDefOperand(DstReg, false);
|
||||||
@ -1445,21 +1454,17 @@ bool SimpleRegisterCoalescing::JoinCopy(CopyRec &TheCopy, bool &Again) {
|
|||||||
if (RealDstReg || RealSrcReg) {
|
if (RealDstReg || RealSrcReg) {
|
||||||
LiveInterval &RealInt =
|
LiveInterval &RealInt =
|
||||||
li_->getOrCreateInterval(RealDstReg ? RealDstReg : RealSrcReg);
|
li_->getOrCreateInterval(RealDstReg ? RealDstReg : RealSrcReg);
|
||||||
SmallSet<const VNInfo*, 4> CopiedValNos;
|
for (LiveInterval::const_vni_iterator I = SavedLI->vni_begin(),
|
||||||
for (LiveInterval::Ranges::const_iterator I = ResSrcInt->ranges.begin(),
|
E = SavedLI->vni_end(); I != E; ++I) {
|
||||||
E = ResSrcInt->ranges.end(); I != E; ++I) {
|
const VNInfo *ValNo = *I;
|
||||||
const LiveRange *DstLR = ResDstInt->getLiveRangeContaining(I->start);
|
VNInfo *NewValNo = RealInt.getNextValue(ValNo->def, ValNo->copy,
|
||||||
assert(DstLR && "Invalid joined interval!");
|
|
||||||
const VNInfo *DstValNo = DstLR->valno;
|
|
||||||
if (CopiedValNos.insert(DstValNo)) {
|
|
||||||
VNInfo *ValNo = RealInt.getNextValue(DstValNo->def, DstValNo->copy,
|
|
||||||
li_->getVNInfoAllocator());
|
li_->getVNInfoAllocator());
|
||||||
ValNo->hasPHIKill = DstValNo->hasPHIKill;
|
NewValNo->hasPHIKill = ValNo->hasPHIKill;
|
||||||
RealInt.addKills(ValNo, DstValNo->kills);
|
NewValNo->redefByEC = ValNo->redefByEC;
|
||||||
RealInt.MergeValueInAsValue(*ResDstInt, DstValNo, ValNo);
|
RealInt.addKills(NewValNo, ValNo->kills);
|
||||||
|
RealInt.MergeValueInAsValue(*SavedLI, ValNo, NewValNo);
|
||||||
}
|
}
|
||||||
}
|
RealInt.weight += SavedLI->weight;
|
||||||
|
|
||||||
DstReg = RealDstReg ? RealDstReg : RealSrcReg;
|
DstReg = RealDstReg ? RealDstReg : RealSrcReg;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1529,6 +1534,12 @@ bool SimpleRegisterCoalescing::JoinCopy(CopyRec &TheCopy, bool &Again) {
|
|||||||
// being merged.
|
// being merged.
|
||||||
li_->removeInterval(SrcReg);
|
li_->removeInterval(SrcReg);
|
||||||
|
|
||||||
|
// Manually deleted the live interval copy.
|
||||||
|
if (SavedLI) {
|
||||||
|
SavedLI->clear();
|
||||||
|
delete SavedLI;
|
||||||
|
}
|
||||||
|
|
||||||
if (isEmpty) {
|
if (isEmpty) {
|
||||||
// Now the copy is being coalesced away, the val# previously defined
|
// 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
|
// by the copy is being defined by an IMPLICIT_DEF which defines a zero
|
||||||
|
22
test/CodeGen/X86/2009-02-08-CoalescerBug.ll
Normal file
22
test/CodeGen/X86/2009-02-08-CoalescerBug.ll
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
; RUN: llvm-as < %s | llc -march=x86
|
||||||
|
; PR3486
|
||||||
|
|
||||||
|
define i32 @foo(i8 signext %p_26) nounwind {
|
||||||
|
entry:
|
||||||
|
%0 = icmp eq i8 %p_26, 0 ; <i1> [#uses=2]
|
||||||
|
%or.cond = or i1 false, %0 ; <i1> [#uses=2]
|
||||||
|
%iftmp.1.0 = zext i1 %or.cond to i16 ; <i16> [#uses=1]
|
||||||
|
br i1 %0, label %bb.i, label %bar.exit
|
||||||
|
|
||||||
|
bb.i: ; preds = %entry
|
||||||
|
%1 = zext i1 %or.cond to i32 ; <i32> [#uses=1]
|
||||||
|
%2 = sdiv i32 %1, 0 ; <i32> [#uses=1]
|
||||||
|
%3 = trunc i32 %2 to i16 ; <i16> [#uses=1]
|
||||||
|
br label %bar.exit
|
||||||
|
|
||||||
|
bar.exit: ; preds = %bb.i, %entry
|
||||||
|
%4 = phi i16 [ %3, %bb.i ], [ %iftmp.1.0, %entry ] ; <i16> [#uses=1]
|
||||||
|
%5 = trunc i16 %4 to i8 ; <i8> [#uses=1]
|
||||||
|
%6 = sext i8 %5 to i32 ; <i32> [#uses=1]
|
||||||
|
ret i32 %6
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user