mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2024-12-14 11:32:34 +00:00
- Two minor improvements to the MachineInstr class to reduce footprint and
overhead: Merge 3 parallel vectors into 1, change regsUsed hash_set to be a bitvector. Sped up LLC a little less than 10% in a debug build! git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@4261 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
4bdb9b7ebe
commit
27a08935ca
@ -12,7 +12,6 @@
|
|||||||
#include "llvm/Target/MachineInstrInfo.h"
|
#include "llvm/Target/MachineInstrInfo.h"
|
||||||
#include "llvm/Annotation.h"
|
#include "llvm/Annotation.h"
|
||||||
#include <Support/iterator>
|
#include <Support/iterator>
|
||||||
#include <Support/hash_set>
|
|
||||||
class Instruction;
|
class Instruction;
|
||||||
|
|
||||||
//---------------------------------------------------------------------------
|
//---------------------------------------------------------------------------
|
||||||
@ -272,12 +271,21 @@ class MachineInstr : public Annotable, // MachineInstrs are annotable
|
|||||||
MachineOpCode opCode; // the opcode
|
MachineOpCode opCode; // the opcode
|
||||||
OpCodeMask opCodeMask; // extra bits for variants of an opcode
|
OpCodeMask opCodeMask; // extra bits for variants of an opcode
|
||||||
std::vector<MachineOperand> operands; // the operands
|
std::vector<MachineOperand> operands; // the operands
|
||||||
std::vector<Value*> implicitRefs; // values implicitly referenced by this
|
|
||||||
std::vector<bool> implicitIsDef; // machine instruction (eg, call args)
|
struct ImplicitRef {
|
||||||
std::vector<bool> implicitIsDefAndUse;
|
Value *Val;
|
||||||
hash_set<int> regsUsed; // all machine registers used for this
|
bool isDef, isDefAndUse;
|
||||||
// instruction, including regs used
|
|
||||||
// to save values across the instr.
|
ImplicitRef(Value *V, bool D, bool DU) : Val(V), isDef(D), isDefAndUse(DU){}
|
||||||
|
};
|
||||||
|
|
||||||
|
// implicitRefs - Values implicitly referenced by this machine instruction
|
||||||
|
// (eg, call args)
|
||||||
|
std::vector<ImplicitRef> implicitRefs;
|
||||||
|
|
||||||
|
// regsUsed - all machine registers used for this instruction, including regs
|
||||||
|
// used to save values across the instruction. This is a bitset of registers.
|
||||||
|
std::vector<bool> regsUsed;
|
||||||
public:
|
public:
|
||||||
/*ctor*/ MachineInstr (MachineOpCode _opCode,
|
/*ctor*/ MachineInstr (MachineOpCode _opCode,
|
||||||
OpCodeMask _opCodeMask = 0x0);
|
OpCodeMask _opCodeMask = 0x0);
|
||||||
@ -324,8 +332,14 @@ public:
|
|||||||
//
|
//
|
||||||
// Information about registers used in this instruction
|
// Information about registers used in this instruction
|
||||||
//
|
//
|
||||||
const hash_set<int>& getRegsUsed () const { return regsUsed; }
|
const std::vector<bool> &getRegsUsed () const { return regsUsed; }
|
||||||
hash_set<int>& getRegsUsed () { return regsUsed; }
|
|
||||||
|
// insertUsedReg - Add a register to the Used registers set...
|
||||||
|
void insertUsedReg(unsigned Reg) {
|
||||||
|
if (Reg >= regsUsed.size())
|
||||||
|
regsUsed.resize(Reg+1);
|
||||||
|
regsUsed[Reg] = true;
|
||||||
|
}
|
||||||
|
|
||||||
//
|
//
|
||||||
// Debugging support
|
// Debugging support
|
||||||
@ -477,41 +491,38 @@ MachineInstr::operandIsDefinedAndUsed(unsigned int i) const
|
|||||||
}
|
}
|
||||||
|
|
||||||
inline bool
|
inline bool
|
||||||
MachineInstr::implicitRefIsDefined(unsigned int i) const
|
MachineInstr::implicitRefIsDefined(unsigned i) const
|
||||||
{
|
{
|
||||||
assert(i < implicitIsDef.size() && "operand out of range!");
|
assert(i < implicitRefs.size() && "operand out of range!");
|
||||||
return implicitIsDef[i];
|
return implicitRefs[i].isDef;
|
||||||
}
|
}
|
||||||
|
|
||||||
inline bool
|
inline bool
|
||||||
MachineInstr::implicitRefIsDefinedAndUsed(unsigned int i) const
|
MachineInstr::implicitRefIsDefinedAndUsed(unsigned i) const
|
||||||
{
|
{
|
||||||
assert(i < implicitIsDefAndUse.size() && "operand out of range!");
|
assert(i < implicitRefs.size() && "operand out of range!");
|
||||||
return implicitIsDefAndUse[i];
|
return implicitRefs[i].isDefAndUse;
|
||||||
}
|
}
|
||||||
|
|
||||||
inline const Value*
|
inline const Value*
|
||||||
MachineInstr::getImplicitRef(unsigned int i) const
|
MachineInstr::getImplicitRef(unsigned i) const
|
||||||
{
|
{
|
||||||
assert(i < implicitRefs.size() && "getImplicitRef() out of range!");
|
assert(i < implicitRefs.size() && "getImplicitRef() out of range!");
|
||||||
return implicitRefs[i];
|
return implicitRefs[i].Val;
|
||||||
}
|
}
|
||||||
|
|
||||||
inline Value*
|
inline Value*
|
||||||
MachineInstr::getImplicitRef(unsigned int i)
|
MachineInstr::getImplicitRef(unsigned i)
|
||||||
{
|
{
|
||||||
assert(i < implicitRefs.size() && "getImplicitRef() out of range!");
|
assert(i < implicitRefs.size() && "getImplicitRef() out of range!");
|
||||||
return implicitRefs[i];
|
return implicitRefs[i].Val;
|
||||||
}
|
}
|
||||||
|
|
||||||
inline void
|
inline void
|
||||||
MachineInstr::addImplicitRef(Value* val,
|
MachineInstr::addImplicitRef(Value* val,
|
||||||
bool isDef,
|
bool isDef,
|
||||||
bool isDefAndUse)
|
bool isDefAndUse) {
|
||||||
{
|
implicitRefs.push_back(ImplicitRef(val, isDef, isDefAndUse));
|
||||||
implicitRefs.push_back(val);
|
|
||||||
implicitIsDef.push_back(isDef);
|
|
||||||
implicitIsDefAndUse.push_back(isDefAndUse);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
inline void
|
inline void
|
||||||
@ -521,9 +532,9 @@ MachineInstr::setImplicitRef(unsigned int i,
|
|||||||
bool isDefAndUse)
|
bool isDefAndUse)
|
||||||
{
|
{
|
||||||
assert(i < implicitRefs.size() && "setImplicitRef() out of range!");
|
assert(i < implicitRefs.size() && "setImplicitRef() out of range!");
|
||||||
implicitRefs[i] = val;
|
implicitRefs[i].Val = val;
|
||||||
implicitIsDef[i] = isDef;
|
implicitRefs[i].isDef = isDef;
|
||||||
implicitIsDefAndUse[i] = isDefAndUse;
|
implicitRefs[i].isDefAndUse = isDefAndUse;
|
||||||
}
|
}
|
||||||
|
|
||||||
inline void
|
inline void
|
||||||
|
@ -82,14 +82,14 @@ MachineInstr::SetMachineOperandReg(unsigned int i,
|
|||||||
operands[i].markDef();
|
operands[i].markDef();
|
||||||
if (isDefAndUse)
|
if (isDefAndUse)
|
||||||
operands[i].markDefAndUse();
|
operands[i].markDefAndUse();
|
||||||
regsUsed.insert(regNum);
|
insertUsedReg(regNum);
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
MachineInstr::SetRegForOperand(unsigned i, int regNum)
|
MachineInstr::SetRegForOperand(unsigned i, int regNum)
|
||||||
{
|
{
|
||||||
operands[i].setRegForValue(regNum);
|
operands[i].setRegForValue(regNum);
|
||||||
regsUsed.insert(regNum);
|
insertUsedReg(regNum);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -111,10 +111,10 @@ MachineInstr::substituteValue(const Value* oldVal, Value* newVal, bool defsOnly)
|
|||||||
|
|
||||||
// Subsitute implicit refs
|
// Subsitute implicit refs
|
||||||
for (unsigned i=0, N=implicitRefs.size(); i < N; ++i)
|
for (unsigned i=0, N=implicitRefs.size(); i < N; ++i)
|
||||||
if (implicitRefs[i] == oldVal)
|
if (getImplicitRef(i) == oldVal)
|
||||||
if (!defsOnly || implicitRefIsDefined(i))
|
if (!defsOnly || implicitRefIsDefined(i))
|
||||||
{
|
{
|
||||||
implicitRefs[i] = newVal;
|
implicitRefs[i].Val = newVal;
|
||||||
++numSubst;
|
++numSubst;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -691,10 +691,10 @@ void PhyRegAlloc::insertCode4SpilledLR(const LiveRange *LR,
|
|||||||
int scratchReg = -1;
|
int scratchReg = -1;
|
||||||
if (MRI.regTypeNeedsScratchReg(RegType, scratchRegType))
|
if (MRI.regTypeNeedsScratchReg(RegType, scratchRegType))
|
||||||
{
|
{
|
||||||
scratchReg = this->getUsableUniRegAtMI(scratchRegType, &LVSetBef,
|
scratchReg = getUsableUniRegAtMI(scratchRegType, &LVSetBef,
|
||||||
MInst, MIBef, MIAft);
|
MInst, MIBef, MIAft);
|
||||||
assert(scratchReg != MRI.getInvalidRegNum());
|
assert(scratchReg != MRI.getInvalidRegNum());
|
||||||
MInst->getRegsUsed().insert(scratchReg);
|
MInst->insertUsedReg(scratchReg);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!isDef || isDefAndUse) {
|
if (!isDef || isDefAndUse) {
|
||||||
@ -774,7 +774,7 @@ int PhyRegAlloc::getUsableUniRegAtMI(const int RegType,
|
|||||||
// of copying it to memory and back. But we have to mark the
|
// of copying it to memory and back. But we have to mark the
|
||||||
// register as used by this instruction, so it does not get used
|
// register as used by this instruction, so it does not get used
|
||||||
// as a scratch reg. by another operand or anyone else.
|
// as a scratch reg. by another operand or anyone else.
|
||||||
MInst->getRegsUsed().insert(scratchReg);
|
MInst->insertUsedReg(scratchReg);
|
||||||
MRI.cpReg2RegMI(MIBef, RegU, scratchReg, RegType);
|
MRI.cpReg2RegMI(MIBef, RegU, scratchReg, RegType);
|
||||||
MRI.cpReg2RegMI(MIAft, scratchReg, RegU, RegType);
|
MRI.cpReg2RegMI(MIAft, scratchReg, RegU, RegType);
|
||||||
}
|
}
|
||||||
@ -874,12 +874,11 @@ void PhyRegAlloc::setRelRegsUsedByThisInst(RegClass *RC,
|
|||||||
// Add the registers already marked as used by the instruction.
|
// Add the registers already marked as used by the instruction.
|
||||||
// This should include any scratch registers that are used to save
|
// This should include any scratch registers that are used to save
|
||||||
// values across the instruction (e.g., for saving state register values).
|
// values across the instruction (e.g., for saving state register values).
|
||||||
const hash_set<int>& regsUsed = MInst->getRegsUsed();
|
const vector<bool> ®sUsed = MInst->getRegsUsed();
|
||||||
for (hash_set<int>::const_iterator SI=regsUsed.begin(), SE=regsUsed.end();
|
for (unsigned i = 0, e = regsUsed.size(); i != e; ++i)
|
||||||
SI != SE; ++SI)
|
if (regsUsed[i]) {
|
||||||
{
|
|
||||||
unsigned classId = 0;
|
unsigned classId = 0;
|
||||||
int classRegNum = MRI.getClassRegNum(*SI, classId);
|
int classRegNum = MRI.getClassRegNum(i, classId);
|
||||||
if (RC->getID() == classId)
|
if (RC->getID() == classId)
|
||||||
{
|
{
|
||||||
assert(classRegNum < (int) IsColorUsedArr.size() &&
|
assert(classRegNum < (int) IsColorUsedArr.size() &&
|
||||||
|
@ -691,10 +691,10 @@ void PhyRegAlloc::insertCode4SpilledLR(const LiveRange *LR,
|
|||||||
int scratchReg = -1;
|
int scratchReg = -1;
|
||||||
if (MRI.regTypeNeedsScratchReg(RegType, scratchRegType))
|
if (MRI.regTypeNeedsScratchReg(RegType, scratchRegType))
|
||||||
{
|
{
|
||||||
scratchReg = this->getUsableUniRegAtMI(scratchRegType, &LVSetBef,
|
scratchReg = getUsableUniRegAtMI(scratchRegType, &LVSetBef,
|
||||||
MInst, MIBef, MIAft);
|
MInst, MIBef, MIAft);
|
||||||
assert(scratchReg != MRI.getInvalidRegNum());
|
assert(scratchReg != MRI.getInvalidRegNum());
|
||||||
MInst->getRegsUsed().insert(scratchReg);
|
MInst->insertUsedReg(scratchReg);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!isDef || isDefAndUse) {
|
if (!isDef || isDefAndUse) {
|
||||||
@ -774,7 +774,7 @@ int PhyRegAlloc::getUsableUniRegAtMI(const int RegType,
|
|||||||
// of copying it to memory and back. But we have to mark the
|
// of copying it to memory and back. But we have to mark the
|
||||||
// register as used by this instruction, so it does not get used
|
// register as used by this instruction, so it does not get used
|
||||||
// as a scratch reg. by another operand or anyone else.
|
// as a scratch reg. by another operand or anyone else.
|
||||||
MInst->getRegsUsed().insert(scratchReg);
|
MInst->insertUsedReg(scratchReg);
|
||||||
MRI.cpReg2RegMI(MIBef, RegU, scratchReg, RegType);
|
MRI.cpReg2RegMI(MIBef, RegU, scratchReg, RegType);
|
||||||
MRI.cpReg2RegMI(MIAft, scratchReg, RegU, RegType);
|
MRI.cpReg2RegMI(MIAft, scratchReg, RegU, RegType);
|
||||||
}
|
}
|
||||||
@ -874,12 +874,11 @@ void PhyRegAlloc::setRelRegsUsedByThisInst(RegClass *RC,
|
|||||||
// Add the registers already marked as used by the instruction.
|
// Add the registers already marked as used by the instruction.
|
||||||
// This should include any scratch registers that are used to save
|
// This should include any scratch registers that are used to save
|
||||||
// values across the instruction (e.g., for saving state register values).
|
// values across the instruction (e.g., for saving state register values).
|
||||||
const hash_set<int>& regsUsed = MInst->getRegsUsed();
|
const vector<bool> ®sUsed = MInst->getRegsUsed();
|
||||||
for (hash_set<int>::const_iterator SI=regsUsed.begin(), SE=regsUsed.end();
|
for (unsigned i = 0, e = regsUsed.size(); i != e; ++i)
|
||||||
SI != SE; ++SI)
|
if (regsUsed[i]) {
|
||||||
{
|
|
||||||
unsigned classId = 0;
|
unsigned classId = 0;
|
||||||
int classRegNum = MRI.getClassRegNum(*SI, classId);
|
int classRegNum = MRI.getClassRegNum(i, classId);
|
||||||
if (RC->getID() == classId)
|
if (RC->getID() == classId)
|
||||||
{
|
{
|
||||||
assert(classRegNum < (int) IsColorUsedArr.size() &&
|
assert(classRegNum < (int) IsColorUsedArr.size() &&
|
||||||
|
@ -663,7 +663,7 @@ UltraSparcRegInfo::InitializeOutgoingArg(MachineInstr* CallMI,
|
|||||||
{
|
{
|
||||||
isArgInReg = true;
|
isArgInReg = true;
|
||||||
UniArgReg = (unsigned) UniArgRegOrNone;
|
UniArgReg = (unsigned) UniArgRegOrNone;
|
||||||
CallMI->getRegsUsed().insert(UniArgReg); // mark the reg as used
|
CallMI->insertUsedReg(UniArgReg); // mark the reg as used
|
||||||
}
|
}
|
||||||
|
|
||||||
if (LR->hasColor()) {
|
if (LR->hasColor()) {
|
||||||
@ -788,7 +788,7 @@ void UltraSparcRegInfo::colorCallArgs(MachineInstr *CallMI,
|
|||||||
UniRetReg = getUnifiedRegNum(RegClassID, CorrectCol);
|
UniRetReg = getUnifiedRegNum(RegClassID, CorrectCol);
|
||||||
|
|
||||||
// Mark the register as used by this instruction
|
// Mark the register as used by this instruction
|
||||||
CallMI->getRegsUsed().insert(UniRetReg);
|
CallMI->insertUsedReg(UniRetReg);
|
||||||
|
|
||||||
// if the LR received the correct color, NOTHING to do
|
// if the LR received the correct color, NOTHING to do
|
||||||
recvCorrectColor = RetValLR->hasColor()? RetValLR->getColor() == CorrectCol
|
recvCorrectColor = RetValLR->hasColor()? RetValLR->getColor() == CorrectCol
|
||||||
@ -1026,7 +1026,7 @@ void UltraSparcRegInfo::colorRetValue(MachineInstr *RetMI,
|
|||||||
unsigned UniRetReg = getUnifiedRegNum(RegClassID, CorrectCol);
|
unsigned UniRetReg = getUnifiedRegNum(RegClassID, CorrectCol);
|
||||||
|
|
||||||
// Mark the register as used by this instruction
|
// Mark the register as used by this instruction
|
||||||
RetMI->getRegsUsed().insert(UniRetReg);
|
RetMI->insertUsedReg(UniRetReg);
|
||||||
|
|
||||||
// if the LR received the correct color, NOTHING to do
|
// if the LR received the correct color, NOTHING to do
|
||||||
|
|
||||||
@ -1433,7 +1433,7 @@ UltraSparcRegInfo::insertCallerSavingCode(vector<MachineInstr*>& instrnsBefore,
|
|||||||
scratchReg = PRA.getUsableUniRegAtMI(scratchRegType, &LVSetBef,
|
scratchReg = PRA.getUsableUniRegAtMI(scratchRegType, &LVSetBef,
|
||||||
CallMI, AdIBef, AdIAft);
|
CallMI, AdIBef, AdIAft);
|
||||||
assert(scratchReg != getInvalidRegNum());
|
assert(scratchReg != getInvalidRegNum());
|
||||||
CallMI->getRegsUsed().insert(scratchReg);
|
CallMI->insertUsedReg(scratchReg);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (AdIBef.size() > 0)
|
if (AdIBef.size() > 0)
|
||||||
@ -1461,7 +1461,7 @@ UltraSparcRegInfo::insertCallerSavingCode(vector<MachineInstr*>& instrnsBefore,
|
|||||||
scratchReg = PRA.getUsableUniRegAtMI(scratchRegType, &LVSetAft,
|
scratchReg = PRA.getUsableUniRegAtMI(scratchRegType, &LVSetAft,
|
||||||
CallMI, AdIBef, AdIAft);
|
CallMI, AdIBef, AdIAft);
|
||||||
assert(scratchReg != getInvalidRegNum());
|
assert(scratchReg != getInvalidRegNum());
|
||||||
CallMI->getRegsUsed().insert(scratchReg);
|
CallMI->insertUsedReg(scratchReg);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (AdIBef.size() > 0)
|
if (AdIBef.size() > 0)
|
||||||
|
Loading…
Reference in New Issue
Block a user