mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2024-12-15 20:29:48 +00:00
Thumb2 IT blocks are fairly expensive. When there are multiple selects using
the same condition, it's important to make sure they are scheduled together to avoid forming multiple IT blocks. I'm adding a pre-regalloc pass that forms IT blocks early (by re-scheduling instructions and split basic blocks) to attempt to fix this. This is not turned on by default since I am not sure this is the right fix. Another issue is llvm selects are modeled as two-address conditional moves. This can be very bad when the copies before the conditional moves are not coalesced away. Teach IT formation pass to move the copies above the IT block (when legal) to avoid breaking the IT block. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@105669 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
6c060dbf84
commit
d847124211
@ -105,7 +105,7 @@ FunctionPass *createARMExpandPseudoPass();
|
||||
FunctionPass *createARMConstantIslandPass();
|
||||
FunctionPass *createNEONPreAllocPass();
|
||||
FunctionPass *createNEONMoveFixPass();
|
||||
FunctionPass *createThumb2ITBlockPass();
|
||||
FunctionPass *createThumb2ITBlockPass(bool PreAlloc = false);
|
||||
FunctionPass *createThumb2SizeReductionPass();
|
||||
|
||||
extern Target TheARMTarget, TheThumbTarget;
|
||||
|
@ -16,11 +16,17 @@
|
||||
#include "ARM.h"
|
||||
#include "llvm/PassManager.h"
|
||||
#include "llvm/CodeGen/Passes.h"
|
||||
#include "llvm/Support/CommandLine.h"
|
||||
#include "llvm/Support/FormattedStream.h"
|
||||
#include "llvm/Target/TargetOptions.h"
|
||||
#include "llvm/Target/TargetRegistry.h"
|
||||
using namespace llvm;
|
||||
|
||||
static cl::opt<bool>
|
||||
EarlyITBlockFormation("thumb2-early-it-blocks", cl::Hidden,
|
||||
cl::desc("Form IT blocks earlt, before register allocation"),
|
||||
cl::init(false));
|
||||
|
||||
static MCAsmInfo *createMCAsmInfo(const Target &T, StringRef TT) {
|
||||
Triple TheTriple(TT);
|
||||
switch (TheTriple.getOS()) {
|
||||
@ -98,6 +104,10 @@ bool ARMBaseTargetMachine::addPreRegAlloc(PassManagerBase &PM,
|
||||
// FIXME: temporarily disabling load / store optimization pass for Thumb1.
|
||||
if (OptLevel != CodeGenOpt::None && !Subtarget.isThumb1Only())
|
||||
PM.add(createARMLoadStoreOptimizationPass(true));
|
||||
|
||||
if (OptLevel != CodeGenOpt::None && Subtarget.isThumb2() &&
|
||||
EarlyITBlockFormation)
|
||||
PM.add(createThumb2ITBlockPass(true));
|
||||
return true;
|
||||
}
|
||||
|
||||
|
@ -14,15 +14,21 @@
|
||||
#include "llvm/CodeGen/MachineInstr.h"
|
||||
#include "llvm/CodeGen/MachineInstrBuilder.h"
|
||||
#include "llvm/CodeGen/MachineFunctionPass.h"
|
||||
#include "llvm/ADT/SmallSet.h"
|
||||
#include "llvm/ADT/Statistic.h"
|
||||
using namespace llvm;
|
||||
|
||||
STATISTIC(NumITs, "Number of IT blocks inserted");
|
||||
STATISTIC(NumITs, "Number of IT blocks inserted");
|
||||
STATISTIC(NumMovedInsts, "Number of predicated instructions moved");
|
||||
|
||||
namespace {
|
||||
struct Thumb2ITBlockPass : public MachineFunctionPass {
|
||||
class Thumb2ITBlockPass : public MachineFunctionPass {
|
||||
bool PreRegAlloc;
|
||||
|
||||
public:
|
||||
static char ID;
|
||||
Thumb2ITBlockPass() : MachineFunctionPass(&ID) {}
|
||||
Thumb2ITBlockPass(bool PreRA) :
|
||||
MachineFunctionPass(&ID), PreRegAlloc(PreRA) {}
|
||||
|
||||
const Thumb2InstrInfo *TII;
|
||||
ARMFunctionInfo *AFI;
|
||||
@ -34,7 +40,19 @@ namespace {
|
||||
}
|
||||
|
||||
private:
|
||||
bool MoveCPSRUseUp(MachineBasicBlock &MBB,
|
||||
MachineBasicBlock::iterator MBBI,
|
||||
MachineBasicBlock::iterator E,
|
||||
unsigned PredReg,
|
||||
ARMCC::CondCodes CC, ARMCC::CondCodes OCC,
|
||||
bool &Done);
|
||||
|
||||
void FindITBlockRanges(MachineBasicBlock &MBB,
|
||||
SmallVector<MachineInstr*,4> &FirstUses,
|
||||
SmallVector<MachineInstr*,4> &LastUses);
|
||||
bool InsertITBlock(MachineInstr *First, MachineInstr *Last);
|
||||
bool InsertITBlocks(MachineBasicBlock &MBB);
|
||||
bool InsertITInstructions(MachineBasicBlock &MBB);
|
||||
};
|
||||
char Thumb2ITBlockPass::ID = 0;
|
||||
}
|
||||
@ -43,27 +61,255 @@ static ARMCC::CondCodes getPredicate(const MachineInstr *MI, unsigned &PredReg){
|
||||
unsigned Opc = MI->getOpcode();
|
||||
if (Opc == ARM::tBcc || Opc == ARM::t2Bcc)
|
||||
return ARMCC::AL;
|
||||
return llvm::getInstrPredicate(MI, PredReg);
|
||||
|
||||
int PIdx = MI->findFirstPredOperandIdx();
|
||||
if (PIdx == -1) {
|
||||
PredReg = 0;
|
||||
return ARMCC::AL;
|
||||
}
|
||||
|
||||
PredReg = MI->getOperand(PIdx+1).getReg();
|
||||
return (ARMCC::CondCodes)MI->getOperand(PIdx).getImm();
|
||||
}
|
||||
|
||||
bool
|
||||
Thumb2ITBlockPass::MoveCPSRUseUp(MachineBasicBlock &MBB,
|
||||
MachineBasicBlock::iterator MBBI,
|
||||
MachineBasicBlock::iterator E,
|
||||
unsigned PredReg,
|
||||
ARMCC::CondCodes CC, ARMCC::CondCodes OCC,
|
||||
bool &Done) {
|
||||
SmallSet<unsigned, 4> Defs, Uses;
|
||||
MachineBasicBlock::iterator I = MBBI;
|
||||
// Look for next CPSR use by scanning up to 4 instructions.
|
||||
for (unsigned i = 0; i < 4; ++i) {
|
||||
MachineInstr *MI = &*I;
|
||||
unsigned MPredReg = 0;
|
||||
ARMCC::CondCodes MCC = getPredicate(MI, MPredReg);
|
||||
if (MCC != ARMCC::AL) {
|
||||
if (MPredReg != PredReg || (MCC != CC && MCC != OCC))
|
||||
return false;
|
||||
|
||||
// Check if the instruction is using any register that's defined
|
||||
// below the previous predicated instruction. Also return false if
|
||||
// it defines any register which is used in between.
|
||||
for (unsigned i = 0, e = MI->getNumOperands(); i != e; ++i) {
|
||||
const MachineOperand &MO = MI->getOperand(i);
|
||||
if (!MO.isReg())
|
||||
continue;
|
||||
unsigned Reg = MO.getReg();
|
||||
if (!Reg)
|
||||
continue;
|
||||
if (MO.isDef()) {
|
||||
if (Reg == PredReg || Uses.count(Reg))
|
||||
return false;
|
||||
} else {
|
||||
if (Defs.count(Reg))
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
Done = (I == E);
|
||||
MBB.remove(MI);
|
||||
MBB.insert(MBBI, MI);
|
||||
++NumMovedInsts;
|
||||
return true;
|
||||
}
|
||||
|
||||
for (unsigned i = 0, e = MI->getNumOperands(); i != e; ++i) {
|
||||
const MachineOperand &MO = MI->getOperand(i);
|
||||
if (!MO.isReg())
|
||||
continue;
|
||||
unsigned Reg = MO.getReg();
|
||||
if (!Reg)
|
||||
continue;
|
||||
if (MO.isDef()) {
|
||||
if (Reg == PredReg)
|
||||
return false;
|
||||
Defs.insert(Reg);
|
||||
} else
|
||||
Uses.insert(Reg);
|
||||
}
|
||||
|
||||
if (I == E)
|
||||
break;
|
||||
++I;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
static bool isCPSRLiveout(MachineBasicBlock &MBB) {
|
||||
for (MachineBasicBlock::succ_iterator I = MBB.succ_begin(),
|
||||
E = MBB.succ_end(); I != E; ++I) {
|
||||
if ((*I)->isLiveIn(ARM::CPSR))
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
void Thumb2ITBlockPass::FindITBlockRanges(MachineBasicBlock &MBB,
|
||||
SmallVector<MachineInstr*,4> &FirstUses,
|
||||
SmallVector<MachineInstr*,4> &LastUses) {
|
||||
bool SeenUse = false;
|
||||
MachineOperand *LastDef = 0;
|
||||
MachineOperand *LastUse = 0;
|
||||
MachineBasicBlock::iterator MBBI = MBB.begin(), E = MBB.end();
|
||||
while (MBBI != E) {
|
||||
MachineInstr *MI = &*MBBI;
|
||||
++MBBI;
|
||||
|
||||
MachineOperand *Def = 0;
|
||||
MachineOperand *Use = 0;
|
||||
for (unsigned i = 0, e = MI->getNumOperands(); i != e; ++i) {
|
||||
MachineOperand &MO = MI->getOperand(i);
|
||||
if (!MO.isReg() || MO.getReg() != ARM::CPSR)
|
||||
continue;
|
||||
if (MO.isDef()) {
|
||||
assert(Def == 0 && "Multiple defs of CPSR?");
|
||||
Def = &MO;
|
||||
} else {
|
||||
assert(Use == 0 && "Multiple uses of CPSR?");
|
||||
Use = &MO;
|
||||
}
|
||||
}
|
||||
|
||||
if (Use) {
|
||||
LastUse = Use;
|
||||
if (!SeenUse) {
|
||||
FirstUses.push_back(MI);
|
||||
SeenUse = true;
|
||||
}
|
||||
}
|
||||
if (Def) {
|
||||
if (LastUse) {
|
||||
LastUses.push_back(LastUse->getParent());
|
||||
LastUse = 0;
|
||||
}
|
||||
LastDef = Def;
|
||||
SeenUse = false;
|
||||
}
|
||||
}
|
||||
|
||||
if (LastUse) {
|
||||
// Is the last use a kill?
|
||||
if (isCPSRLiveout(MBB))
|
||||
LastUses.push_back(0);
|
||||
else
|
||||
LastUses.push_back(LastUse->getParent());
|
||||
}
|
||||
}
|
||||
|
||||
bool Thumb2ITBlockPass::InsertITBlock(MachineInstr *First, MachineInstr *Last) {
|
||||
if (First == Last)
|
||||
return false;
|
||||
|
||||
bool Modified = false;
|
||||
MachineBasicBlock *MBB = First->getParent();
|
||||
MachineBasicBlock::iterator MBBI = First;
|
||||
MachineBasicBlock::iterator E = Last;
|
||||
|
||||
if (First->getDesc().isBranch() || First->getDesc().isReturn())
|
||||
return false;
|
||||
|
||||
unsigned PredReg = 0;
|
||||
ARMCC::CondCodes CC = getPredicate(First, PredReg);
|
||||
if (CC == ARMCC::AL)
|
||||
return Modified;
|
||||
|
||||
// Move uses of the CPSR together if possible.
|
||||
ARMCC::CondCodes OCC = ARMCC::getOppositeCondition(CC);
|
||||
|
||||
do {
|
||||
++MBBI;
|
||||
if (MBBI->getDesc().isBranch() || MBBI->getDesc().isReturn())
|
||||
return Modified;
|
||||
MachineInstr *NMI = &*MBBI;
|
||||
unsigned NPredReg = 0;
|
||||
ARMCC::CondCodes NCC = getPredicate(NMI, NPredReg);
|
||||
if (NCC != CC && NCC != OCC) {
|
||||
if (NCC != ARMCC::AL)
|
||||
return Modified;
|
||||
assert(MBBI != E);
|
||||
bool Done = false;
|
||||
if (!MoveCPSRUseUp(*MBB, MBBI, E, PredReg, CC, OCC, Done))
|
||||
return Modified;
|
||||
Modified = true;
|
||||
if (Done)
|
||||
MBBI = E;
|
||||
}
|
||||
} while (MBBI != E);
|
||||
|
||||
// Insert a new block for consecutive predicated instructions.
|
||||
MachineFunction *MF = MBB->getParent();
|
||||
MachineBasicBlock *NewMBB = MF->CreateMachineBasicBlock(MBB->getBasicBlock());
|
||||
MachineFunction::iterator Pos = MBB;
|
||||
MF->insert(++Pos, NewMBB);
|
||||
|
||||
// Move all the successors of this block to the specified block.
|
||||
NewMBB->transferSuccessors(MBB);
|
||||
|
||||
// Add an edge from CurMBB to NewMBB for the fall-through.
|
||||
MBB->addSuccessor(NewMBB);
|
||||
NewMBB->splice(NewMBB->end(), MBB, ++MBBI, MBB->end());
|
||||
return true;
|
||||
}
|
||||
|
||||
bool Thumb2ITBlockPass::InsertITBlocks(MachineBasicBlock &MBB) {
|
||||
SmallVector<MachineInstr*, 4> FirstUses;
|
||||
SmallVector<MachineInstr*, 4> LastUses;
|
||||
FindITBlockRanges(MBB, FirstUses, LastUses);
|
||||
assert(FirstUses.size() == LastUses.size() && "Incorrect range information!");
|
||||
|
||||
bool Modified = false;
|
||||
for (unsigned i = 0, e = FirstUses.size(); i != e; ++i) {
|
||||
if (LastUses[i] == 0)
|
||||
// Must be the last pair where CPSR is live out of the block.
|
||||
return Modified;
|
||||
Modified |= InsertITBlock(FirstUses[i], LastUses[i]);
|
||||
}
|
||||
return Modified;
|
||||
}
|
||||
|
||||
static void TrackDefUses(MachineInstr *MI, SmallSet<unsigned, 4> &Defs,
|
||||
SmallSet<unsigned, 4> &Uses) {
|
||||
for (unsigned i = 0, e = MI->getNumOperands(); i != e; ++i) {
|
||||
MachineOperand &MO = MI->getOperand(i);
|
||||
if (!MO.isReg())
|
||||
continue;
|
||||
unsigned Reg = MO.getReg();
|
||||
if (!Reg)
|
||||
continue;
|
||||
if (MO.isDef())
|
||||
Defs.insert(Reg);
|
||||
else
|
||||
Uses.insert(Reg);
|
||||
}
|
||||
}
|
||||
|
||||
bool Thumb2ITBlockPass::InsertITInstructions(MachineBasicBlock &MBB) {
|
||||
bool Modified = false;
|
||||
|
||||
SmallSet<unsigned, 4> Defs;
|
||||
SmallSet<unsigned, 4> Uses;
|
||||
MachineBasicBlock::iterator MBBI = MBB.begin(), E = MBB.end();
|
||||
while (MBBI != E) {
|
||||
MachineInstr *MI = &*MBBI;
|
||||
DebugLoc dl = MI->getDebugLoc();
|
||||
unsigned PredReg = 0;
|
||||
ARMCC::CondCodes CC = getPredicate(MI, PredReg);
|
||||
|
||||
if (CC == ARMCC::AL) {
|
||||
++MBBI;
|
||||
continue;
|
||||
}
|
||||
|
||||
Defs.clear();
|
||||
Uses.clear();
|
||||
TrackDefUses(MI, Defs, Uses);
|
||||
|
||||
// Insert an IT instruction.
|
||||
MachineInstrBuilder MIB = BuildMI(MBB, MBBI, dl, TII->get(ARM::t2IT))
|
||||
.addImm(CC);
|
||||
MachineBasicBlock::iterator InsertPos = MIB;
|
||||
++MBBI;
|
||||
|
||||
// Finalize IT mask.
|
||||
@ -75,17 +321,37 @@ bool Thumb2ITBlockPass::InsertITBlocks(MachineBasicBlock &MBB) {
|
||||
(!MI->getDesc().isBranch() && !MI->getDesc().isReturn()) ; ++MBBI) {
|
||||
if (MBBI->isDebugValue())
|
||||
continue;
|
||||
|
||||
MachineInstr *NMI = &*MBBI;
|
||||
MI = NMI;
|
||||
DebugLoc ndl = NMI->getDebugLoc();
|
||||
|
||||
unsigned NPredReg = 0;
|
||||
ARMCC::CondCodes NCC = getPredicate(NMI, NPredReg);
|
||||
if (NCC == CC || NCC == OCC)
|
||||
Mask |= (NCC & 1) << Pos;
|
||||
else
|
||||
else {
|
||||
unsigned SrcReg, DstReg, SrcSubIdx, DstSubIdx;
|
||||
if (NCC == ARMCC::AL &&
|
||||
TII->isMoveInstr(*NMI, SrcReg, DstReg, SrcSubIdx, DstSubIdx)) {
|
||||
assert(SrcSubIdx == 0 && DstSubIdx == 0 &&
|
||||
"Sub-register indices still around?");
|
||||
// llvm models select's as two-address instructions. That means a copy
|
||||
// is inserted before a t2MOVccr, etc. If the copy is scheduled in
|
||||
// between selects we would end up creating multiple IT blocks.
|
||||
if (!Uses.count(DstReg) && !Defs.count(SrcReg)) {
|
||||
--MBBI;
|
||||
MBB.remove(NMI);
|
||||
MBB.insert(InsertPos, NMI);
|
||||
++NumMovedInsts;
|
||||
continue;
|
||||
}
|
||||
}
|
||||
break;
|
||||
}
|
||||
TrackDefUses(NMI, Defs, Uses);
|
||||
--Pos;
|
||||
}
|
||||
|
||||
Mask |= (1 << Pos);
|
||||
// Tag along (firstcond[0] << 4) with the mask.
|
||||
Mask |= (CC & 1) << 4;
|
||||
@ -106,10 +372,13 @@ bool Thumb2ITBlockPass::runOnMachineFunction(MachineFunction &Fn) {
|
||||
return false;
|
||||
|
||||
bool Modified = false;
|
||||
for (MachineFunction::iterator MFI = Fn.begin(), E = Fn.end(); MFI != E;
|
||||
++MFI) {
|
||||
for (MachineFunction::iterator MFI = Fn.begin(), E = Fn.end(); MFI != E; ) {
|
||||
MachineBasicBlock &MBB = *MFI;
|
||||
Modified |= InsertITBlocks(MBB);
|
||||
++MFI;
|
||||
if (PreRegAlloc)
|
||||
Modified |= InsertITBlocks(MBB);
|
||||
else
|
||||
Modified |= InsertITInstructions(MBB);
|
||||
}
|
||||
|
||||
return Modified;
|
||||
@ -117,6 +386,6 @@ bool Thumb2ITBlockPass::runOnMachineFunction(MachineFunction &Fn) {
|
||||
|
||||
/// createThumb2ITBlockPass - Returns an instance of the Thumb2 IT blocks
|
||||
/// insertion pass.
|
||||
FunctionPass *llvm::createThumb2ITBlockPass() {
|
||||
return new Thumb2ITBlockPass();
|
||||
FunctionPass *llvm::createThumb2ITBlockPass(bool PreAlloc) {
|
||||
return new Thumb2ITBlockPass(PreAlloc);
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user