It is always good to do a cross-class join when the large register has a tiny interval.

Also avoid division by zero.

llvm-svn: 95917
This commit is contained in:
Jakob Stoklund Olesen 2010-02-11 20:58:45 +00:00
parent 6b88fb6575
commit 8b992fb512

View File

@ -1148,12 +1148,14 @@ SimpleRegisterCoalescing::isWinToJoinCrossClass(unsigned LargeReg,
LiveInterval &SmallInt = li_->getInterval(SmallReg); LiveInterval &SmallInt = li_->getInterval(SmallReg);
unsigned LargeSize = li_->getApproximateInstructionCount(LargeInt); unsigned LargeSize = li_->getApproximateInstructionCount(LargeInt);
unsigned SmallSize = li_->getApproximateInstructionCount(SmallInt); unsigned SmallSize = li_->getApproximateInstructionCount(SmallInt);
if (SmallSize > Threshold || LargeSize > Threshold) if (LargeSize > Threshold) {
if ((float)std::distance(mri_->use_nodbg_begin(SmallReg), unsigned SmallUses = std::distance(mri_->use_nodbg_begin(SmallReg),
mri_->use_nodbg_end()) / SmallSize < mri_->use_nodbg_end());
(float)std::distance(mri_->use_nodbg_begin(LargeReg), unsigned LargeUses = std::distance(mri_->use_nodbg_begin(LargeReg),
mri_->use_nodbg_end()) / LargeSize) mri_->use_nodbg_end());
if (SmallUses*LargeSize < LargeUses*SmallSize)
return false; return false;
}
return true; return true;
} }
@ -1561,7 +1563,10 @@ bool SimpleRegisterCoalescing::JoinCopy(CopyRec &TheCopy, bool &Again) {
(isExtSubReg || DstRC->isASubClass()) && (isExtSubReg || DstRC->isASubClass()) &&
!isWinToJoinCrossClass(LargeReg, SmallReg, !isWinToJoinCrossClass(LargeReg, SmallReg,
allocatableRCRegs_[NewRC].count())) { allocatableRCRegs_[NewRC].count())) {
DEBUG(dbgs() << "\tSrc/Dest are different register classes.\n"); DEBUG(dbgs() << "\tSrc/Dest are different register classes: "
<< SrcRC->getName() << "/"
<< DstRC->getName() << " -> "
<< NewRC->getName() << ".\n");
// Allow the coalescer to try again in case either side gets coalesced to // 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. // a physical register that's compatible with the other side. e.g.
// r1024 = MOV32to32_ r1025 // r1024 = MOV32to32_ r1025