Skip to content

Commit bab3f23

Browse files
author
Reed Kotler
committedMay 1, 2014
Add basic functionality for assignment of ints.
This creates a lot of core infrastructure in which to add, with little effort, quite a bit more to mips fast-isel Test Plan: simplestore.ll Reviewers: dsanders Reviewed By: dsanders Differential Revision: http://reviews.llvm.org/D3527 llvm-svn: 207790
1 parent 06900a0 commit bab3f23

File tree

2 files changed

+182
-1
lines changed

2 files changed

+182
-1
lines changed
 

‎llvm/lib/Target/Mips/MipsFastISel.cpp

+167-1
Original file line numberDiff line numberDiff line change
@@ -3,10 +3,12 @@
33

44
#include "llvm/CodeGen/FunctionLoweringInfo.h"
55
#include "llvm/CodeGen/FastISel.h"
6-
76
#include "llvm/CodeGen/MachineInstrBuilder.h"
7+
#include "llvm/IR/GlobalAlias.h"
8+
#include "llvm/IR/GlobalVariable.h"
89
#include "llvm/Target/TargetInstrInfo.h"
910
#include "llvm/Target/TargetLibraryInfo.h"
11+
#include "MipsRegisterInfo.h"
1012
#include "MipsISelLowering.h"
1113
#include "MipsMachineFunction.h"
1214
#include "MipsSubtarget.h"
@@ -15,6 +17,21 @@ using namespace llvm;
1517

1618
namespace {
1719

20+
// All possible address modes.
21+
typedef struct Address {
22+
enum { RegBase, FrameIndexBase } BaseType;
23+
24+
union {
25+
unsigned Reg;
26+
int FI;
27+
} Base;
28+
29+
int64_t Offset;
30+
31+
// Innocuous defaults for our address.
32+
Address() : BaseType(RegBase), Offset(0) { Base.Reg = 0; }
33+
} Address;
34+
1835
class MipsFastISel final : public FastISel {
1936

2037
/// Subtarget - Keep a pointer to the MipsSubtarget around so that we can
@@ -46,10 +63,120 @@ class MipsFastISel final : public FastISel {
4663
}
4764

4865
bool TargetSelectInstruction(const Instruction *I) override;
66+
unsigned TargetMaterializeConstant(const Constant *C) override;
67+
68+
bool ComputeAddress(const Value *Obj, Address &Addr);
4969

70+
private:
71+
bool EmitStore(MVT VT, unsigned SrcReg, Address &Addr,
72+
unsigned Alignment = 0);
5073
bool SelectRet(const Instruction *I);
74+
bool SelectStore(const Instruction *I);
75+
76+
bool isTypeLegal(Type *Ty, MVT &VT);
77+
bool isLoadTypeLegal(Type *Ty, MVT &VT);
78+
79+
unsigned MaterializeFP(const ConstantFP *CFP, MVT VT);
80+
unsigned MaterializeGV(const GlobalValue *GV, MVT VT);
81+
unsigned MaterializeInt(const Constant *C, MVT VT);
5182
};
5283

84+
bool MipsFastISel::isTypeLegal(Type *Ty, MVT &VT) {
85+
EVT evt = TLI.getValueType(Ty, true);
86+
// Only handle simple types.
87+
if (evt == MVT::Other || !evt.isSimple())
88+
return false;
89+
VT = evt.getSimpleVT();
90+
91+
// Handle all legal types, i.e. a register that will directly hold this
92+
// value.
93+
return TLI.isTypeLegal(VT);
94+
}
95+
96+
bool MipsFastISel::isLoadTypeLegal(Type *Ty, MVT &VT) {
97+
if (isTypeLegal(Ty, VT))
98+
return true;
99+
// We will extend this in a later patch:
100+
// If this is a type than can be sign or zero-extended to a basic operation
101+
// go ahead and accept it now.
102+
return false;
103+
}
104+
105+
bool MipsFastISel::ComputeAddress(const Value *Obj, Address &Addr) {
106+
// This construct looks a big awkward but it is how other ports handle this
107+
// and as this function is more fully completed, these cases which
108+
// return false will have additional code in them.
109+
//
110+
if (isa<Instruction>(Obj))
111+
return false;
112+
else if (isa<ConstantExpr>(Obj))
113+
return false;
114+
Addr.Base.Reg = getRegForValue(Obj);
115+
return Addr.Base.Reg != 0;
116+
}
117+
118+
// Materialize a constant into a register, and return the register
119+
// number (or zero if we failed to handle it).
120+
unsigned MipsFastISel::TargetMaterializeConstant(const Constant *C) {
121+
EVT CEVT = TLI.getValueType(C->getType(), true);
122+
123+
// Only handle simple types.
124+
if (!CEVT.isSimple())
125+
return 0;
126+
MVT VT = CEVT.getSimpleVT();
127+
128+
if (const ConstantFP *CFP = dyn_cast<ConstantFP>(C))
129+
return MaterializeFP(CFP, VT);
130+
else if (const GlobalValue *GV = dyn_cast<GlobalValue>(C))
131+
return MaterializeGV(GV, VT);
132+
else if (isa<ConstantInt>(C))
133+
return MaterializeInt(C, VT);
134+
135+
return 0;
136+
}
137+
138+
bool MipsFastISel::EmitStore(MVT VT, unsigned SrcReg, Address &Addr,
139+
unsigned Alignment) {
140+
//
141+
// more cases will be handled here in following patches.
142+
//
143+
if (VT != MVT::i32)
144+
return false;
145+
BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, DbgLoc, TII.get(Mips::SW))
146+
.addReg(SrcReg)
147+
.addReg(Addr.Base.Reg)
148+
.addImm(Addr.Offset);
149+
return true;
150+
}
151+
152+
bool MipsFastISel::SelectStore(const Instruction *I) {
153+
Value *Op0 = I->getOperand(0);
154+
unsigned SrcReg = 0;
155+
156+
// Atomic stores need special handling.
157+
if (cast<StoreInst>(I)->isAtomic())
158+
return false;
159+
160+
// Verify we have a legal type before going any further.
161+
MVT VT;
162+
if (!isLoadTypeLegal(I->getOperand(0)->getType(), VT))
163+
return false;
164+
165+
// Get the value to be stored into a register.
166+
SrcReg = getRegForValue(Op0);
167+
if (SrcReg == 0)
168+
return false;
169+
170+
// See if we can handle this address.
171+
Address Addr;
172+
if (!ComputeAddress(I->getOperand(1), Addr))
173+
return false;
174+
175+
if (!EmitStore(VT, SrcReg, Addr, cast<StoreInst>(I)->getAlignment()))
176+
return false;
177+
return true;
178+
}
179+
53180
bool MipsFastISel::SelectRet(const Instruction *I) {
54181
const ReturnInst *Ret = cast<ReturnInst>(I);
55182

@@ -69,13 +196,52 @@ bool MipsFastISel::TargetSelectInstruction(const Instruction *I) {
69196
switch (I->getOpcode()) {
70197
default:
71198
break;
199+
case Instruction::Store:
200+
return SelectStore(I);
72201
case Instruction::Ret:
73202
return SelectRet(I);
74203
}
75204
return false;
76205
}
77206
}
78207

208+
unsigned MipsFastISel::MaterializeFP(const ConstantFP *CFP, MVT VT) {
209+
return 0;
210+
}
211+
212+
unsigned MipsFastISel::MaterializeGV(const GlobalValue *GV, MVT VT) {
213+
// For now 32-bit only.
214+
if (VT != MVT::i32)
215+
return 0;
216+
const TargetRegisterClass *RC = &Mips::GPR32RegClass;
217+
unsigned DestReg = createResultReg(RC);
218+
const GlobalVariable *GVar = dyn_cast<GlobalVariable>(GV);
219+
bool IsThreadLocal = GVar && GVar->isThreadLocal();
220+
// TLS not supported at this time.
221+
if (IsThreadLocal)
222+
return 0;
223+
BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, DbgLoc, TII.get(Mips::LW), DestReg)
224+
.addReg(MFI->getGlobalBaseReg())
225+
.addGlobalAddress(GV, 0, MipsII::MO_GOT);
226+
return DestReg;
227+
}
228+
unsigned MipsFastISel::MaterializeInt(const Constant *C, MVT VT) {
229+
if (VT != MVT::i32)
230+
return 0;
231+
const TargetRegisterClass *RC = &Mips::GPR32RegClass;
232+
// If the constant is in range, use a load-immediate.
233+
const ConstantInt *CI = cast<ConstantInt>(C);
234+
if (isInt<16>(CI->getSExtValue())) {
235+
unsigned Opc = Mips::ADDiu;
236+
unsigned ImmReg = createResultReg(RC);
237+
BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, DbgLoc, TII.get(Opc), ImmReg)
238+
.addReg(Mips::ZERO)
239+
.addImm(CI->getSExtValue());
240+
return ImmReg;
241+
}
242+
return 0;
243+
}
244+
79245
namespace llvm {
80246
FastISel *Mips::createFastISel(FunctionLoweringInfo &funcInfo,
81247
const TargetLibraryInfo *libInfo) {
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
; RUN: llc -march=mipsel -relocation-model=pic -O0 -mips-fast-isel -fast-isel-abort -mcpu=mips32r2 \
2+
; RUN: < %s | FileCheck %s
3+
4+
@abcd = external global i32
5+
6+
; Function Attrs: nounwind
7+
define void @foo() {
8+
entry:
9+
store i32 12345, i32* @abcd, align 4
10+
; CHECK: addiu $[[REG1:[0-9]+]], $zero, 12345
11+
; CHECK: lw $[[REG2:[0-9]+]], %got(abcd)(${{[0-9]+}})
12+
; CHECK: sw $[[REG1]], 0($[[REG2]])
13+
ret void
14+
}
15+

0 commit comments

Comments
 (0)
Please sign in to comment.