mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2024-12-13 20:32:21 +00:00
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:
parent
d6397eba23
commit
c7908037d8
@ -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;
|
||||||
|
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
@ -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.
|
||||||
|
@ -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
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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:
|
||||||
|
Loading…
Reference in New Issue
Block a user