mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-07-21 00:24:42 +00:00
Updated version of r96634 (which was reverted due to failing 176.gcc and
126.gcc nightly tests. These failures uncovered latent bugs that machine DCE could remove one half of a stack adjust down/up pair, causing PEI to assert. This update fixes that, and the tests now pass. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@96822 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
@@ -587,6 +587,17 @@ public:
|
|||||||
return !hasFP(MF);
|
return !hasFP(MF);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// canSimplifyCallFramePseudos - When possible, it's best to simplify the
|
||||||
|
/// call frame pseudo ops before doing frame index elimination. This is
|
||||||
|
/// possible only when frame index references between the pseudos won't
|
||||||
|
/// need adjusted for the call frame adjustments. Normally, that's true
|
||||||
|
/// if the function has a reserved call frame or a frame pointer. Some
|
||||||
|
/// targets (Thumb2, for example) may have more complicated criteria,
|
||||||
|
/// however, and can override this behavior.
|
||||||
|
virtual bool canSimplifyCallFramePseudos(MachineFunction &MF) const {
|
||||||
|
return hasReservedCallFrame(MF) || hasFP(MF);
|
||||||
|
}
|
||||||
|
|
||||||
/// hasReservedSpillSlot - Return true if target has reserved a spill slot in
|
/// hasReservedSpillSlot - Return true if target has reserved a spill slot in
|
||||||
/// the stack frame of the given function for the specified register. e.g. On
|
/// the stack frame of the given function for the specified register. e.g. On
|
||||||
/// x86, if the frame register is required, the first fixed stack object is
|
/// x86, if the frame register is required, the first fixed stack object is
|
||||||
|
@@ -175,9 +175,10 @@ void PEI::calculateCallsInformation(MachineFunction &Fn) {
|
|||||||
MachineBasicBlock::iterator I = *i;
|
MachineBasicBlock::iterator I = *i;
|
||||||
|
|
||||||
// If call frames are not being included as part of the stack frame, and
|
// If call frames are not being included as part of the stack frame, and
|
||||||
// there is no dynamic allocation (therefore referencing frame slots off
|
// the target doesn't indicate otherwise, remove the call frame pseudos
|
||||||
// sp), leave the pseudo ops alone. We'll eliminate them later.
|
// here. The sub/add sp instruction pairs are still inserted, but we don't
|
||||||
if (RegInfo->hasReservedCallFrame(Fn) || RegInfo->hasFP(Fn))
|
// need to track the SP adjustment for frame index elimination.
|
||||||
|
if (RegInfo->canSimplifyCallFramePseudos(Fn))
|
||||||
RegInfo->eliminateCallFramePseudoInstr(Fn, *I->getParent(), I);
|
RegInfo->eliminateCallFramePseudoInstr(Fn, *I->getParent(), I);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -1085,6 +1085,16 @@ hasReservedCallFrame(MachineFunction &MF) const {
|
|||||||
return !MF.getFrameInfo()->hasVarSizedObjects();
|
return !MF.getFrameInfo()->hasVarSizedObjects();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// canSimplifyCallFramePseudos - If there is a reserved call frame, the
|
||||||
|
// call frame pseudos can be simplified. Unlike most targets, having a FP
|
||||||
|
// is not sufficient here since we still may reference some objects via SP
|
||||||
|
// even when FP is available in Thumb2 mode.
|
||||||
|
bool ARMBaseRegisterInfo::
|
||||||
|
canSimplifyCallFramePseudos(MachineFunction &MF) const {
|
||||||
|
ARMFunctionInfo *AFI = MF.getInfo<ARMFunctionInfo>();
|
||||||
|
return hasReservedCallFrame(MF) || (AFI->isThumb1OnlyFunction() && hasFP(MF));
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
emitSPUpdate(bool isARM,
|
emitSPUpdate(bool isARM,
|
||||||
MachineBasicBlock &MBB, MachineBasicBlock::iterator &MBBI,
|
MachineBasicBlock &MBB, MachineBasicBlock::iterator &MBBI,
|
||||||
|
@@ -138,6 +138,7 @@ public:
|
|||||||
virtual bool requiresFrameIndexScavenging(const MachineFunction &MF) const;
|
virtual bool requiresFrameIndexScavenging(const MachineFunction &MF) const;
|
||||||
|
|
||||||
virtual bool hasReservedCallFrame(MachineFunction &MF) const;
|
virtual bool hasReservedCallFrame(MachineFunction &MF) const;
|
||||||
|
virtual bool canSimplifyCallFramePseudos(MachineFunction &MF) const;
|
||||||
|
|
||||||
virtual void eliminateCallFramePseudoInstr(MachineFunction &MF,
|
virtual void eliminateCallFramePseudoInstr(MachineFunction &MF,
|
||||||
MachineBasicBlock &MBB,
|
MachineBasicBlock &MBB,
|
||||||
|
@@ -635,7 +635,10 @@ PseudoInst<(outs), (ins cpinst_operand:$instid, cpinst_operand:$cpidx,
|
|||||||
i32imm:$size), NoItinerary,
|
i32imm:$size), NoItinerary,
|
||||||
"${instid:label} ${cpidx:cpentry}", []>;
|
"${instid:label} ${cpidx:cpentry}", []>;
|
||||||
|
|
||||||
let Defs = [SP], Uses = [SP] in {
|
// FIXME: Marking these as hasSideEffects is necessary to prevent machine DCE
|
||||||
|
// from removing one half of the matched pairs. That breaks PEI, which assumes
|
||||||
|
// these will always be in pairs, and asserts if it finds otherwise. Better way?
|
||||||
|
let Defs = [SP], Uses = [SP], hasSideEffects = 1 in {
|
||||||
def ADJCALLSTACKUP :
|
def ADJCALLSTACKUP :
|
||||||
PseudoInst<(outs), (ins i32imm:$amt1, i32imm:$amt2, pred:$p), NoItinerary,
|
PseudoInst<(outs), (ins i32imm:$amt1, i32imm:$amt2, pred:$p), NoItinerary,
|
||||||
"@ ADJCALLSTACKUP $amt1",
|
"@ ADJCALLSTACKUP $amt1",
|
||||||
|
@@ -120,7 +120,10 @@ def t_addrmode_sp : Operand<i32>,
|
|||||||
// Miscellaneous Instructions.
|
// Miscellaneous Instructions.
|
||||||
//
|
//
|
||||||
|
|
||||||
let Defs = [SP], Uses = [SP] in {
|
// FIXME: Marking these as hasSideEffects is necessary to prevent machine DCE
|
||||||
|
// from removing one half of the matched pairs. That breaks PEI, which assumes
|
||||||
|
// these will always be in pairs, and asserts if it finds otherwise. Better way?
|
||||||
|
let Defs = [SP], Uses = [SP], hasSideEffects = 1 in {
|
||||||
def tADJCALLSTACKUP :
|
def tADJCALLSTACKUP :
|
||||||
PseudoInst<(outs), (ins i32imm:$amt1, i32imm:$amt2), NoItinerary,
|
PseudoInst<(outs), (ins i32imm:$amt1, i32imm:$amt2), NoItinerary,
|
||||||
"@ tADJCALLSTACKUP $amt1",
|
"@ tADJCALLSTACKUP $amt1",
|
||||||
|
Reference in New Issue
Block a user