Reapply r161633-161634 "Partition use lists so defs always come before uses.""

No changes to these patches, MRI needed to be notified when changing
uses into defs and vice versa.

git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@161644 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Jakob Stoklund Olesen 2012-08-10 00:21:30 +00:00
parent d6397eba23
commit c7908037d8
5 changed files with 61 additions and 54 deletions

View File

@ -150,7 +150,7 @@ private:
struct { // For MO_Register. struct { // For MO_Register.
// Register number is in SmallContents.RegNo. // Register number is in SmallContents.RegNo.
MachineOperand **Prev; // Access list for register. MachineOperand *Prev; // Access list for register. See MRI.
MachineOperand *Next; MachineOperand *Next;
} Reg; } Reg;

View File

@ -468,10 +468,6 @@ public:
const TargetRegisterInfo &TRI, const TargetRegisterInfo &TRI,
const TargetInstrInfo &TII); const TargetInstrInfo &TII);
private:
void HandleVRegListReallocation();
public:
/// defusechain_iterator - This class provides iterator support for machine /// defusechain_iterator - This class provides iterator support for machine
/// operands in the function that use or define a specific register. If /// operands in the function that use or define a specific register. If
/// ReturnUses is true it returns uses of registers, if ReturnDefs is true it /// ReturnUses is true it returns uses of registers, if ReturnDefs is true it
@ -517,11 +513,20 @@ public:
assert(Op && "Cannot increment end iterator!"); assert(Op && "Cannot increment end iterator!");
Op = getNextOperandForReg(Op); Op = getNextOperandForReg(Op);
// All defs come before the uses, so stop def_iterator early.
if (!ReturnUses) {
if (Op) {
if (Op->isUse())
Op = 0;
else
assert(!Op->isDebug() && "Can't have debug defs");
}
} else {
// If this is an operand we don't care about, skip it. // If this is an operand we don't care about, skip it.
while (Op && ((!ReturnUses && Op->isUse()) || while (Op && ((!ReturnDefs && Op->isDef()) ||
(!ReturnDefs && Op->isDef()) ||
(SkipDebug && Op->isDebug()))) (SkipDebug && Op->isDebug())))
Op = getNextOperandForReg(Op); Op = getNextOperandForReg(Op);
}
return *this; return *this;
} }

View File

@ -102,17 +102,9 @@ MachineRegisterInfo::createVirtualRegister(const TargetRegisterClass *RegClass){
// New virtual register number. // New virtual register number.
unsigned Reg = TargetRegisterInfo::index2VirtReg(getNumVirtRegs()); unsigned Reg = TargetRegisterInfo::index2VirtReg(getNumVirtRegs());
// Add a reg, but keep track of whether the vector reallocated or not.
const unsigned FirstVirtReg = TargetRegisterInfo::index2VirtReg(0);
void *ArrayBase = getNumVirtRegs() == 0 ? 0 : &VRegInfo[FirstVirtReg];
VRegInfo.grow(Reg); VRegInfo.grow(Reg);
VRegInfo[Reg].first = RegClass; VRegInfo[Reg].first = RegClass;
RegAllocHints.grow(Reg); RegAllocHints.grow(Reg);
if (ArrayBase && &VRegInfo[FirstVirtReg] != ArrayBase)
// The vector reallocated, handle this now.
HandleVRegListReallocation();
return Reg; return Reg;
} }
@ -129,55 +121,65 @@ void MachineRegisterInfo::clearVirtRegs() {
/// Add MO to the linked list of operands for its register. /// Add MO to the linked list of operands for its register.
void MachineRegisterInfo::addRegOperandToUseList(MachineOperand *MO) { void MachineRegisterInfo::addRegOperandToUseList(MachineOperand *MO) {
assert(!MO->isOnRegUseList() && "Already on list"); assert(!MO->isOnRegUseList() && "Already on list");
MachineOperand **Head = &getRegUseDefListHead(MO->getReg()); MachineOperand *&HeadRef = getRegUseDefListHead(MO->getReg());
MachineOperand *const Head = HeadRef;
// For SSA values, we prefer to keep the definition at the start of the list. // Head points to the first list element.
// we do this by skipping over the definition if it is at the head of the // Next is NULL on the last list element.
// list. // Prev pointers are circular, so Head->Prev == Last.
if (*Head && (*Head)->isDef())
Head = &(*Head)->Contents.Reg.Next;
MO->Contents.Reg.Next = *Head; // Head is NULL for an empty list.
if (MO->Contents.Reg.Next) { if (!Head) {
assert(MO->getReg() == MO->Contents.Reg.Next->getReg() && MO->Contents.Reg.Prev = MO;
"Different regs on the same list!"); MO->Contents.Reg.Next = 0;
MO->Contents.Reg.Next->Contents.Reg.Prev = &MO->Contents.Reg.Next; HeadRef = MO;
return;
} }
assert(MO->getReg() == Head->getReg() && "Different regs on the same list!");
MO->Contents.Reg.Prev = Head; // Insert MO between Last and Head in the circular Prev chain.
*Head = MO; MachineOperand *Last = Head->Contents.Reg.Prev;
assert(Last && "Inconsistent use list");
assert(MO->getReg() == Last->getReg() && "Different regs on the same list!");
Head->Contents.Reg.Prev = MO;
MO->Contents.Reg.Prev = Last;
// Def operands always precede uses. This allows def_iterator to stop early.
// Insert def operands at the front, and use operands at the back.
if (MO->isDef()) {
// Insert def at the front.
MO->Contents.Reg.Next = Head;
HeadRef = MO;
} else {
// Insert use at the end.
MO->Contents.Reg.Next = 0;
Last->Contents.Reg.Next = MO;
}
} }
/// Remove MO from its use-def list. /// Remove MO from its use-def list.
void MachineRegisterInfo::removeRegOperandFromUseList(MachineOperand *MO) { void MachineRegisterInfo::removeRegOperandFromUseList(MachineOperand *MO) {
assert(MO->isOnRegUseList() && "Operand not on use list"); assert(MO->isOnRegUseList() && "Operand not on use list");
MachineOperand *&HeadRef = getRegUseDefListHead(MO->getReg());
MachineOperand *const Head = HeadRef;
assert(Head && "List already empty");
// Unlink this from the doubly linked list of operands. // Unlink this from the doubly linked list of operands.
MachineOperand *NextOp = MO->Contents.Reg.Next; MachineOperand *Next = MO->Contents.Reg.Next;
*MO->Contents.Reg.Prev = NextOp; MachineOperand *Prev = MO->Contents.Reg.Prev;
if (NextOp) {
assert(NextOp->getReg() == MO->getReg() && "Corrupt reg use/def chain!"); // Prev links are circular, next link is NULL instead of looping back to Head.
NextOp->Contents.Reg.Prev = MO->Contents.Reg.Prev; if (MO == Head)
} HeadRef = Next;
else
Prev->Contents.Reg.Next = Next;
(Next ? Next : Head)->Contents.Reg.Prev = Prev;
MO->Contents.Reg.Prev = 0; MO->Contents.Reg.Prev = 0;
MO->Contents.Reg.Next = 0; MO->Contents.Reg.Next = 0;
} }
/// HandleVRegListReallocation - We just added a virtual register to the
/// VRegInfo info list and it reallocated. Update the use/def lists info
/// pointers.
void MachineRegisterInfo::HandleVRegListReallocation() {
// The back pointers for the vreg lists point into the previous vector.
// Update them to point to their correct slots.
for (unsigned i = 0, e = getNumVirtRegs(); i != e; ++i) {
unsigned Reg = TargetRegisterInfo::index2VirtReg(i);
MachineOperand *List = VRegInfo[Reg].second;
if (!List) continue;
// Update the back-pointer to be accurate once more.
List->Contents.Reg.Prev = &VRegInfo[Reg].second;
}
}
/// replaceRegWith - Replace all instances of FromReg with ToReg in the /// replaceRegWith - Replace all instances of FromReg with ToReg in the
/// machine function. This is like llvm-level X->replaceAllUsesWith(Y), /// machine function. This is like llvm-level X->replaceAllUsesWith(Y),
/// except that it also changes any definitions of the register as well. /// except that it also changes any definitions of the register as well.

View File

@ -4,7 +4,7 @@ target triple = "msp430-generic-generic"
define i8 @mov(i8 %a, i8 %b) nounwind { define i8 @mov(i8 %a, i8 %b) nounwind {
; CHECK: mov: ; CHECK: mov:
; CHECK: mov.b r14, r15 ; CHECK: mov.{{[bw]}} r14, r15
ret i8 %b ret i8 %b
} }

View File

@ -31,7 +31,7 @@ entry:
; X64: test3: ; X64: test3:
; X64: notl ; X64: notl
; X64: andl ; X64: andl
; X64: shrl %eax ; X64: shrl
; X64: ret ; X64: ret
; X32: test3: ; X32: test3: