2001-07-21 12:41:50 +00:00
|
|
|
// $Id$
|
|
|
|
//***************************************************************************
|
|
|
|
// File:
|
|
|
|
// MachineInstr.cpp
|
|
|
|
//
|
|
|
|
// Purpose:
|
|
|
|
//
|
|
|
|
//
|
|
|
|
// Strategy:
|
|
|
|
//
|
|
|
|
// History:
|
|
|
|
// 7/2/01 - Vikram Adve - Created
|
|
|
|
//**************************************************************************/
|
|
|
|
|
2001-08-28 23:02:39 +00:00
|
|
|
|
2001-09-07 17:18:30 +00:00
|
|
|
#include "llvm/CodeGen/MachineInstr.h"
|
2001-09-18 12:56:28 +00:00
|
|
|
#include "llvm/Target/MachineRegInfo.h"
|
2001-08-28 23:02:39 +00:00
|
|
|
#include "llvm/Method.h"
|
2001-07-21 12:41:50 +00:00
|
|
|
#include "llvm/Instruction.h"
|
2001-08-28 23:02:39 +00:00
|
|
|
|
2001-07-21 12:41:50 +00:00
|
|
|
|
2001-10-18 22:40:02 +00:00
|
|
|
|
2001-07-21 12:41:50 +00:00
|
|
|
//************************ Class Implementations **************************/
|
|
|
|
|
2001-07-31 21:49:28 +00:00
|
|
|
// Constructor for instructions with fixed #operands (nearly all)
|
2001-07-21 12:41:50 +00:00
|
|
|
MachineInstr::MachineInstr(MachineOpCode _opCode,
|
|
|
|
OpCodeMask _opCodeMask)
|
|
|
|
: opCode(_opCode),
|
|
|
|
opCodeMask(_opCodeMask),
|
2001-07-28 04:06:37 +00:00
|
|
|
operands(TargetInstrDescriptors[_opCode].numOperands)
|
2001-07-31 21:49:28 +00:00
|
|
|
{
|
|
|
|
assert(TargetInstrDescriptors[_opCode].numOperands >= 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Constructor for instructions with variable #operands
|
|
|
|
MachineInstr::MachineInstr(MachineOpCode _opCode,
|
|
|
|
unsigned numOperands,
|
|
|
|
OpCodeMask _opCodeMask)
|
|
|
|
: opCode(_opCode),
|
|
|
|
opCodeMask(_opCodeMask),
|
|
|
|
operands(numOperands)
|
2001-07-21 12:41:50 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
MachineInstr::SetMachineOperand(unsigned int i,
|
|
|
|
MachineOperand::MachineOperandType operandType,
|
2001-08-07 20:16:52 +00:00
|
|
|
Value* _val, bool isdef=false)
|
2001-07-21 12:41:50 +00:00
|
|
|
{
|
2001-07-28 04:06:37 +00:00
|
|
|
assert(i < operands.size());
|
2001-07-21 12:41:50 +00:00
|
|
|
operands[i].Initialize(operandType, _val);
|
2001-08-13 16:32:45 +00:00
|
|
|
operands[i].isDef = isdef ||
|
2001-09-18 12:56:28 +00:00
|
|
|
TargetInstrDescriptors[opCode].resultPos == (int) i;
|
2001-07-21 12:41:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
MachineInstr::SetMachineOperand(unsigned int i,
|
|
|
|
MachineOperand::MachineOperandType operandType,
|
2001-08-07 20:16:52 +00:00
|
|
|
int64_t intValue, bool isdef=false)
|
2001-07-21 12:41:50 +00:00
|
|
|
{
|
2001-07-28 04:06:37 +00:00
|
|
|
assert(i < operands.size());
|
2001-07-21 12:41:50 +00:00
|
|
|
operands[i].InitializeConst(operandType, intValue);
|
2001-08-13 16:32:45 +00:00
|
|
|
operands[i].isDef = isdef ||
|
2001-09-18 12:56:28 +00:00
|
|
|
TargetInstrDescriptors[opCode].resultPos == (int) i;
|
2001-07-21 12:41:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
MachineInstr::SetMachineOperand(unsigned int i,
|
2001-11-05 03:56:02 +00:00
|
|
|
int regNum, bool isdef=false)
|
2001-07-21 12:41:50 +00:00
|
|
|
{
|
2001-07-28 04:06:37 +00:00
|
|
|
assert(i < operands.size());
|
2001-07-21 12:41:50 +00:00
|
|
|
operands[i].InitializeReg(regNum);
|
2001-08-13 16:32:45 +00:00
|
|
|
operands[i].isDef = isdef ||
|
2001-09-18 12:56:28 +00:00
|
|
|
TargetInstrDescriptors[opCode].resultPos == (int) i;
|
2001-07-21 12:41:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2001-08-07 21:01:23 +00:00
|
|
|
MachineInstr::dump(unsigned int indent) const
|
2001-07-21 12:41:50 +00:00
|
|
|
{
|
|
|
|
for (unsigned i=0; i < indent; i++)
|
|
|
|
cout << " ";
|
|
|
|
|
|
|
|
cout << *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
ostream&
|
|
|
|
operator<< (ostream& os, const MachineInstr& minstr)
|
|
|
|
{
|
2001-07-28 04:06:37 +00:00
|
|
|
os << TargetInstrDescriptors[minstr.opCode].opCodeString;
|
2001-07-21 12:41:50 +00:00
|
|
|
|
|
|
|
for (unsigned i=0, N=minstr.getNumOperands(); i < N; i++)
|
|
|
|
os << "\t" << minstr.getOperand(i);
|
|
|
|
|
2001-07-28 04:06:37 +00:00
|
|
|
#undef DEBUG_VAL_OP_ITERATOR
|
|
|
|
#ifdef DEBUG_VAL_OP_ITERATOR
|
|
|
|
os << endl << "\tValue operands are: ";
|
|
|
|
for (MachineInstr::val_op_const_iterator vo(&minstr); ! vo.done(); ++vo)
|
|
|
|
{
|
|
|
|
const Value* val = *vo;
|
|
|
|
os << val << (vo.isDef()? "(def), " : ", ");
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2001-10-18 22:40:02 +00:00
|
|
|
|
|
|
|
|
|
|
|
#if 1
|
|
|
|
// code for printing implict references
|
|
|
|
|
|
|
|
unsigned NumOfImpRefs = minstr.getNumImplicitRefs();
|
|
|
|
if( NumOfImpRefs > 0 ) {
|
|
|
|
|
|
|
|
os << "\tImplicit:";
|
|
|
|
|
|
|
|
for(unsigned z=0; z < NumOfImpRefs; z++) {
|
|
|
|
os << minstr.getImplicitRef(z);
|
|
|
|
cout << "\t";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2001-10-17 23:57:50 +00:00
|
|
|
os << endl;
|
|
|
|
|
2001-07-21 12:41:50 +00:00
|
|
|
return os;
|
|
|
|
}
|
|
|
|
|
2001-09-18 12:56:28 +00:00
|
|
|
static inline ostream&
|
|
|
|
OutputOperand(ostream &os, const MachineOperand &mop)
|
|
|
|
{
|
|
|
|
switch (mop.getOperandType())
|
|
|
|
{
|
|
|
|
case MachineOperand::MO_CCRegister:
|
|
|
|
case MachineOperand::MO_VirtualRegister:
|
|
|
|
return os << "(val " << mop.getVRegValue() << ")";
|
|
|
|
case MachineOperand::MO_MachineRegister:
|
|
|
|
return os << "(" << mop.getMachineRegNum() << ")";
|
|
|
|
default:
|
|
|
|
assert(0 && "Unknown operand type");
|
|
|
|
return os;
|
|
|
|
}
|
2001-09-09 22:26:29 +00:00
|
|
|
}
|
2001-07-21 12:41:50 +00:00
|
|
|
|
2001-09-09 22:26:29 +00:00
|
|
|
|
2001-09-18 12:56:28 +00:00
|
|
|
ostream&
|
|
|
|
operator<<(ostream &os, const MachineOperand &mop)
|
|
|
|
{
|
|
|
|
switch(mop.opType)
|
|
|
|
{
|
|
|
|
case MachineOperand::MO_VirtualRegister:
|
|
|
|
case MachineOperand::MO_MachineRegister:
|
|
|
|
os << "%reg";
|
|
|
|
return OutputOperand(os, mop);
|
|
|
|
case MachineOperand::MO_CCRegister:
|
|
|
|
os << "%ccreg";
|
|
|
|
return OutputOperand(os, mop);
|
|
|
|
case MachineOperand::MO_SignExtendedImmed:
|
|
|
|
return os << mop.immedVal;
|
|
|
|
case MachineOperand::MO_UnextendedImmed:
|
|
|
|
return os << mop.immedVal;
|
|
|
|
case MachineOperand::MO_PCRelativeDisp:
|
2001-09-30 23:44:19 +00:00
|
|
|
{
|
|
|
|
const Value* opVal = mop.getVRegValue();
|
2001-10-01 20:11:19 +00:00
|
|
|
bool isLabel = isa<Method>(opVal) || isa<BasicBlock>(opVal);
|
2001-09-30 23:44:19 +00:00
|
|
|
return os << "%disp("
|
|
|
|
<< (isLabel? "label " : "addr-of-val ")
|
|
|
|
<< opVal << ")";
|
|
|
|
}
|
2001-09-18 12:56:28 +00:00
|
|
|
default:
|
|
|
|
assert(0 && "Unrecognized operand type");
|
|
|
|
break;
|
|
|
|
}
|
2001-09-09 22:26:29 +00:00
|
|
|
|
2001-07-21 12:41:50 +00:00
|
|
|
return os;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2001-08-28 23:02:39 +00:00
|
|
|
void
|
2001-10-22 13:51:33 +00:00
|
|
|
MachineCodeForMethod::putLocalVarAtOffsetFromFP(const Value* local,
|
|
|
|
int offset,
|
|
|
|
unsigned int size)
|
|
|
|
{
|
|
|
|
offsetsFromFP[local] = offset;
|
|
|
|
incrementAutomaticVarsSize(size);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
MachineCodeForMethod::putLocalVarAtOffsetFromSP(const Value* local,
|
|
|
|
int offset,
|
|
|
|
unsigned int size)
|
|
|
|
{
|
|
|
|
offsetsFromSP[local] = offset;
|
|
|
|
incrementAutomaticVarsSize(size);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
MachineCodeForMethod::getOffsetFromFP(const Value* local) const
|
|
|
|
{
|
|
|
|
hash_map<const Value*, int>::const_iterator pair = offsetsFromFP.find(local);
|
|
|
|
assert(pair != offsetsFromFP.end() && "Offset from FP unknown for Value");
|
|
|
|
return (*pair).second;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
MachineCodeForMethod::getOffsetFromSP(const Value* local) const
|
|
|
|
{
|
|
|
|
hash_map<const Value*, int>::const_iterator pair = offsetsFromSP.find(local);
|
|
|
|
assert(pair != offsetsFromSP.end() && "Offset from SP unknown for Value");
|
|
|
|
return (*pair).second;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
MachineCodeForMethod::dump() const
|
2001-09-15 19:07:45 +00:00
|
|
|
{
|
|
|
|
cout << "\n" << method->getReturnType()
|
|
|
|
<< " \"" << method->getName() << "\"" << endl;
|
|
|
|
|
|
|
|
for (Method::const_iterator BI = method->begin(); BI != method->end(); ++BI)
|
|
|
|
{
|
|
|
|
BasicBlock* bb = *BI;
|
|
|
|
cout << "\n"
|
|
|
|
<< (bb->hasName()? bb->getName() : "Label")
|
|
|
|
<< " (" << bb << ")" << ":"
|
|
|
|
<< endl;
|
|
|
|
|
|
|
|
MachineCodeForBasicBlock& mvec = bb->getMachineInstrVec();
|
|
|
|
for (unsigned i=0; i < mvec.size(); i++)
|
2001-10-17 23:57:50 +00:00
|
|
|
cout << "\t" << *mvec[i];
|
2001-09-15 19:07:45 +00:00
|
|
|
}
|
|
|
|
cout << endl << "End method \"" << method->getName() << "\""
|
|
|
|
<< endl << endl;
|
|
|
|
}
|