[RegisterCoalescer] Refine the terminal rule to still consider the terminal

nodes.
When a node is terminal it is pushed at the end of the list of the copies to
coalesce instead of being completely ignored. In effect, this reduces its
priority over non-terminal nodes.

Because of that, we do not miss the rematerialization opportunities, nor the
copies that can be merged with more complex, than the terminal rule,
interference checks.

Related to PR22768.


git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@233395 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Quentin Colombet 2015-03-27 18:37:15 +00:00
parent 2cee1c9d3c
commit 1ad854adba
2 changed files with 34 additions and 8 deletions

View File

@ -2733,13 +2733,17 @@ bool RegisterCoalescer::applyTerminalRule(const MachineInstr &Copy) const {
return false; return false;
// Check if the destination of this copy has any other affinity. // Check if the destination of this copy has any other affinity.
unsigned DstReg = Copy.getOperand(0).getReg(); unsigned DstReg = Copy.getOperand(0).getReg();
unsigned SrcReg = Copy.getOperand(1).getReg();
if (TargetRegisterInfo::isPhysicalRegister(DstReg) || if (TargetRegisterInfo::isPhysicalRegister(DstReg) ||
// If SrcReg is a physical register, the copy won't be coalesced.
// Ignoring it may have other side effect (like missing
// rematerialization). So keep it.
TargetRegisterInfo::isPhysicalRegister(SrcReg) ||
!isTerminalReg(DstReg, Copy, MRI)) !isTerminalReg(DstReg, Copy, MRI))
return false; return false;
// DstReg is a terminal node. Check if it inteferes with any other // DstReg is a terminal node. Check if it inteferes with any other
// copy involving SrcReg. // copy involving SrcReg.
unsigned SrcReg = Copy.getOperand(1).getReg();
const MachineBasicBlock *OrigBB = Copy.getParent(); const MachineBasicBlock *OrigBB = Copy.getParent();
const LiveInterval &DstLI = LIS->getInterval(DstReg); const LiveInterval &DstLI = LIS->getInterval(DstReg);
for (const MachineInstr &MI : MRI->reg_nodbg_instructions(SrcReg)) { for (const MachineInstr &MI : MRI->reg_nodbg_instructions(SrcReg)) {
@ -2775,25 +2779,45 @@ RegisterCoalescer::copyCoalesceInMBB(MachineBasicBlock *MBB) {
// yet, it might invalidate the iterator. // yet, it might invalidate the iterator.
const unsigned PrevSize = WorkList.size(); const unsigned PrevSize = WorkList.size();
if (JoinGlobalCopies) { if (JoinGlobalCopies) {
SmallVector<MachineInstr*, 2> LocalTerminals;
SmallVector<MachineInstr*, 2> GlobalTerminals;
// Coalesce copies bottom-up to coalesce local defs before local uses. They // Coalesce copies bottom-up to coalesce local defs before local uses. They
// are not inherently easier to resolve, but slightly preferable until we // are not inherently easier to resolve, but slightly preferable until we
// have local live range splitting. In particular this is required by // have local live range splitting. In particular this is required by
// cmp+jmp macro fusion. // cmp+jmp macro fusion.
for (MachineBasicBlock::iterator MII = MBB->begin(), E = MBB->end(); for (MachineBasicBlock::iterator MII = MBB->begin(), E = MBB->end();
MII != E; ++MII) { MII != E; ++MII) {
if (!MII->isCopyLike() || applyTerminalRule(*MII)) if (!MII->isCopyLike())
continue; continue;
if (isLocalCopy(&(*MII), LIS)) bool ApplyTerminalRule = applyTerminalRule(*MII);
LocalWorkList.push_back(&(*MII)); if (isLocalCopy(&(*MII), LIS)) {
else if (ApplyTerminalRule)
WorkList.push_back(&(*MII)); LocalTerminals.push_back(&(*MII));
else
LocalWorkList.push_back(&(*MII));
} else {
if (ApplyTerminalRule)
GlobalTerminals.push_back(&(*MII));
else
WorkList.push_back(&(*MII));
}
} }
// Append the copies evicted by the terminal rule at the end of the list.
LocalWorkList.append(LocalTerminals.begin(), LocalTerminals.end());
WorkList.append(GlobalTerminals.begin(), GlobalTerminals.end());
} }
else { else {
SmallVector<MachineInstr*, 2> Terminals;
for (MachineBasicBlock::iterator MII = MBB->begin(), E = MBB->end(); for (MachineBasicBlock::iterator MII = MBB->begin(), E = MBB->end();
MII != E; ++MII) MII != E; ++MII)
if (MII->isCopyLike() && !applyTerminalRule(*MII)) if (MII->isCopyLike()) {
WorkList.push_back(MII); if (applyTerminalRule(*MII))
Terminals.push_back(&(*MII));
else
WorkList.push_back(MII);
}
// Append the copies evicted by the terminal rule at the end of the list.
WorkList.append(Terminals.begin(), Terminals.end());
} }
// Try coalescing the collected copies immediately, and remove the nulls. // Try coalescing the collected copies immediately, and remove the nulls.
// This prevents the WorkList from getting too large since most copies are // This prevents the WorkList from getting too large since most copies are

View File

@ -1,5 +1,7 @@
; RUN: llc -mtriple=x86_64-apple-darwin8 < %s | FileCheck %s ; RUN: llc -mtriple=x86_64-apple-darwin8 < %s | FileCheck %s
; RUN: llc -mtriple=x86_64-pc-linux < %s | FileCheck %s ; RUN: llc -mtriple=x86_64-pc-linux < %s | FileCheck %s
; RUN: llc -mtriple=x86_64-apple-darwin8 -terminal-rule < %s | FileCheck %s
; RUN: llc -mtriple=x86_64-pc-linux -terminal-rule < %s | FileCheck %s
; CHECK-LABEL: return32 ; CHECK-LABEL: return32
; CHECK-DAG: movq $0, (%rdi) ; CHECK-DAG: movq $0, (%rdi)