add the memri memory operand

this makes it possible for ldr instructions with non-zero immediate


git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@29103 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Rafael Espindola 2006-07-11 11:36:48 +00:00
parent 4a8aadd1f5
commit a4e64359aa
4 changed files with 41 additions and 13 deletions

View File

@ -58,6 +58,12 @@ namespace {
return "ARM Assembly Printer"; return "ARM Assembly Printer";
} }
void printMemRegImm(const MachineInstr *MI, unsigned OpNo) {
printOperand(MI, OpNo + 1);
O << ", ";
printOperand(MI, OpNo);
}
void printOperand(const MachineInstr *MI, int opNum); void printOperand(const MachineInstr *MI, int opNum);
void printMemOperand(const MachineInstr *MI, int opNum, void printMemOperand(const MachineInstr *MI, int opNum,
const char *Modifier = 0); const char *Modifier = 0);

View File

@ -164,7 +164,7 @@ public:
void Select(SDOperand &Result, SDOperand Op); void Select(SDOperand &Result, SDOperand Op);
virtual void InstructionSelectBasicBlock(SelectionDAG &DAG); virtual void InstructionSelectBasicBlock(SelectionDAG &DAG);
bool SelectAddrReg(SDOperand N, SDOperand &Base); bool SelectAddrRegImm(SDOperand N, SDOperand &Offset, SDOperand &Base);
// Include the pieces autogenerated from the target description. // Include the pieces autogenerated from the target description.
#include "ARMGenDAGISel.inc" #include "ARMGenDAGISel.inc"
@ -183,7 +183,10 @@ void ARMDAGToDAGISel::InstructionSelectBasicBlock(SelectionDAG &DAG) {
ScheduleAndEmitDAG(DAG); ScheduleAndEmitDAG(DAG);
} }
bool ARMDAGToDAGISel::SelectAddrReg(SDOperand N, SDOperand &Base) { //register plus/minus 12 bit offset
bool ARMDAGToDAGISel::SelectAddrRegImm(SDOperand N, SDOperand &Offset,
SDOperand &Base) {
Offset = CurDAG->getTargetConstant(0, MVT::i32);
if (FrameIndexSDNode *FI = dyn_cast<FrameIndexSDNode>(N)) { if (FrameIndexSDNode *FI = dyn_cast<FrameIndexSDNode>(N)) {
Base = CurDAG->getTargetFrameIndex(FI->getIndex(), N.getValueType()); Base = CurDAG->getTargetFrameIndex(FI->getIndex(), N.getValueType());
} }

View File

@ -12,9 +12,18 @@
// //
//===----------------------------------------------------------------------===// //===----------------------------------------------------------------------===//
// Address operands
def memri : Operand<iPTR> {
let PrintMethod = "printMemRegImm";
let NumMIOperands = 2;
let MIOperandInfo = (ops i32imm, ptr_rc);
}
// Define ARM specific addressing mode. // Define ARM specific addressing mode.
//register or frame index //register plus/minus 12 bit offset
def raddr : ComplexPattern<iPTR, 1, "SelectAddrReg", []>; def iaddr : ComplexPattern<iPTR, 2, "SelectAddrRegImm", []>;
//register plus scaled register
//def raddr : ComplexPattern<iPTR, 2, "SelectAddrRegReg", []>;
//===----------------------------------------------------------------------===// //===----------------------------------------------------------------------===//
// Instructions // Instructions
@ -42,9 +51,9 @@ def ADJCALLSTACKDOWN : InstARM<(ops i32imm:$amt),
def bxr: InstARM<(ops IntRegs:$dst), "bx $dst", [(brind IntRegs:$dst)]>; def bxr: InstARM<(ops IntRegs:$dst), "bx $dst", [(brind IntRegs:$dst)]>;
def ldr : InstARM<(ops IntRegs:$dst, IntRegs:$addr), def ldr : InstARM<(ops IntRegs:$dst, memri:$addr),
"ldr $dst, [$addr]", "ldr $dst, [$addr]",
[(set IntRegs:$dst, (load raddr:$addr))]>; [(set IntRegs:$dst, (load iaddr:$addr))]>;
def str : InstARM<(ops IntRegs:$src, IntRegs:$addr), def str : InstARM<(ops IntRegs:$src, IntRegs:$addr),
"str $src, [$addr]", "str $src, [$addr]",

View File

@ -83,23 +83,33 @@ ARMRegisterInfo::eliminateFrameIndex(MachineBasicBlock::iterator II) const {
assert (MI.getOpcode() == ARM::ldr); assert (MI.getOpcode() == ARM::ldr);
unsigned FrameIdx = 1; unsigned FrameIdx = 2;
unsigned OffIdx = 1;
int FrameIndex = MI.getOperand(FrameIdx).getFrameIndex(); int FrameIndex = MI.getOperand(FrameIdx).getFrameIndex();
int Offset = MF.getFrameInfo()->getObjectOffset(FrameIndex); int Offset = MF.getFrameInfo()->getObjectOffset(FrameIndex);
assert (MI.getOperand(OffIdx).getImmedValue() == 0);
unsigned StackSize = MF.getFrameInfo()->getStackSize(); unsigned StackSize = MF.getFrameInfo()->getStackSize();
Offset += StackSize; Offset += StackSize;
// Insert a set of r12 with the full address assert (Offset >= 0);
// r12 = r13 + offset if (Offset < 4096) {
MachineBasicBlock *MBB2 = MI.getParent(); // Replace the FrameIndex with r13
BuildMI(*MBB2, II, ARM::addri, 2, ARM::R12).addReg(ARM::R13).addImm(Offset); MI.getOperand(FrameIdx).ChangeToRegister(ARM::R13);
// Replace the ldr offset with Offset
MI.getOperand(OffIdx).ChangeToImmediate(Offset);
} else {
// Insert a set of r12 with the full address
// r12 = r13 + offset
MachineBasicBlock *MBB2 = MI.getParent();
BuildMI(*MBB2, II, ARM::addri, 2, ARM::R12).addReg(ARM::R13).addImm(Offset);
// Replace the FrameIndex with r12 // Replace the FrameIndex with r12
MI.getOperand(FrameIdx).ChangeToRegister(ARM::R12); MI.getOperand(FrameIdx).ChangeToRegister(ARM::R12);
}
} }
void ARMRegisterInfo:: void ARMRegisterInfo::