20
20
#include < cassert>
21
21
22
22
namespace llvm {
23
-
23
+
24
24
// / ARM_AM - ARM Addressing Mode Stuff
25
25
namespace ARM_AM {
26
26
enum ShiftOpc {
@@ -31,11 +31,11 @@ namespace ARM_AM {
31
31
ror,
32
32
rrx
33
33
};
34
-
34
+
35
35
enum AddrOpc {
36
36
add = ' +' , sub = ' -'
37
37
};
38
-
38
+
39
39
static inline const char *getShiftOpcStr (ShiftOpc Op) {
40
40
switch (Op) {
41
41
default : llvm_unreachable (" Unknown shift opc!" );
@@ -46,7 +46,7 @@ namespace ARM_AM {
46
46
case ARM_AM::rrx: return " rrx" ;
47
47
}
48
48
}
49
-
49
+
50
50
static inline ShiftOpc getShiftOpcForNode (SDValue N) {
51
51
switch (N.getOpcode ()) {
52
52
default : return ARM_AM::no_shift;
@@ -95,14 +95,14 @@ namespace ARM_AM {
95
95
assert (Amt < 32 && " Invalid rotate amount" );
96
96
return (Val >> Amt) | (Val << ((32 -Amt)&31 ));
97
97
}
98
-
98
+
99
99
// / rotl32 - Rotate a 32-bit unsigned value left by a specified # bits.
100
100
// /
101
101
static inline unsigned rotl32 (unsigned Val, unsigned Amt) {
102
102
assert (Amt < 32 && " Invalid rotate amount" );
103
103
return (Val << Amt) | (Val >> ((32 -Amt)&31 ));
104
104
}
105
-
105
+
106
106
// ===--------------------------------------------------------------------===//
107
107
// Addressing Mode #1: shift_operand with registers
108
108
// ===--------------------------------------------------------------------===//
@@ -137,7 +137,7 @@ namespace ARM_AM {
137
137
static inline unsigned getSOImmValRot (unsigned Imm) {
138
138
return (Imm >> 8 ) * 2 ;
139
139
}
140
-
140
+
141
141
// / getSOImmValRotate - Try to handle Imm with an immediate shifter operand,
142
142
// / computing the rotate amount to use. If this immediate value cannot be
143
143
// / handled with a single shifter-op, determine a good rotate amount that will
@@ -146,14 +146,14 @@ namespace ARM_AM {
146
146
// 8-bit (or less) immediates are trivially shifter_operands with a rotate
147
147
// of zero.
148
148
if ((Imm & ~255U ) == 0 ) return 0 ;
149
-
149
+
150
150
// Use CTZ to compute the rotate amount.
151
151
unsigned TZ = CountTrailingZeros_32 (Imm);
152
-
152
+
153
153
// Rotate amount must be even. Something like 0x200 must be rotated 8 bits,
154
154
// not 9.
155
155
unsigned RotAmt = TZ & ~1 ;
156
-
156
+
157
157
// If we can handle this spread, return it.
158
158
if ((rotr32 (Imm, RotAmt) & ~255U ) == 0 )
159
159
return (32 -RotAmt)&31 ; // HW rotates right, not left.
@@ -166,16 +166,16 @@ namespace ARM_AM {
166
166
// Restart the search for a high-order bit after the initial seconds of
167
167
// ones.
168
168
unsigned TZ2 = CountTrailingZeros_32 (Imm & ~((1 << TrailingOnes)-1 ));
169
-
169
+
170
170
// Rotate amount must be even.
171
171
unsigned RotAmt2 = TZ2 & ~1 ;
172
-
172
+
173
173
// If this fits, use it.
174
174
if (RotAmt2 != 32 && (rotr32 (Imm, RotAmt2) & ~255U ) == 0 )
175
175
return (32 -RotAmt2)&31 ; // HW rotates right, not left.
176
176
}
177
177
}
178
-
178
+
179
179
// Otherwise, we have no way to cover this span of bits with a single
180
180
// shifter_op immediate. Return a chunk of bits that will be useful to
181
181
// handle.
@@ -189,30 +189,30 @@ namespace ARM_AM {
189
189
// 8-bit (or less) immediates are trivially shifter_operands with a rotate
190
190
// of zero.
191
191
if ((Arg & ~255U ) == 0 ) return Arg;
192
-
192
+
193
193
unsigned RotAmt = getSOImmValRotate (Arg);
194
194
195
195
// If this cannot be handled with a single shifter_op, bail out.
196
196
if (rotr32 (~255U , RotAmt) & Arg)
197
197
return -1 ;
198
-
198
+
199
199
// Encode this correctly.
200
200
return rotl32 (Arg, RotAmt) | ((RotAmt>>1 ) << 8 );
201
201
}
202
-
202
+
203
203
// / isSOImmTwoPartVal - Return true if the specified value can be obtained by
204
204
// / or'ing together two SOImmVal's.
205
205
static inline bool isSOImmTwoPartVal (unsigned V) {
206
206
// If this can be handled with a single shifter_op, bail out.
207
207
V = rotr32 (~255U , getSOImmValRotate (V)) & V;
208
208
if (V == 0 )
209
209
return false ;
210
-
210
+
211
211
// If this can be handled with two shifter_op's, accept.
212
212
V = rotr32 (~255U , getSOImmValRotate (V)) & V;
213
213
return V == 0 ;
214
214
}
215
-
215
+
216
216
// / getSOImmTwoPartFirst - If V is a value that satisfies isSOImmTwoPartVal,
217
217
// / return the first chunk of it.
218
218
static inline unsigned getSOImmTwoPartFirst (unsigned V) {
@@ -222,14 +222,14 @@ namespace ARM_AM {
222
222
// / getSOImmTwoPartSecond - If V is a value that satisfies isSOImmTwoPartVal,
223
223
// / return the second chunk of it.
224
224
static inline unsigned getSOImmTwoPartSecond (unsigned V) {
225
- // Mask out the first hunk.
225
+ // Mask out the first hunk.
226
226
V = rotr32 (~255U , getSOImmValRotate (V)) & V;
227
-
227
+
228
228
// Take what's left.
229
229
assert (V == (rotr32 (255U , getSOImmValRotate (V)) & V));
230
230
return V;
231
231
}
232
-
232
+
233
233
// / getThumbImmValShift - Try to handle Imm with a 8-bit immediate followed
234
234
// / by a left shift. Returns the shift amount to use.
235
235
static inline unsigned getThumbImmValShift (unsigned Imm) {
@@ -244,7 +244,7 @@ namespace ARM_AM {
244
244
// / isThumbImmShiftedVal - Return true if the specified value can be obtained
245
245
// / by left shifting a 8-bit immediate.
246
246
static inline bool isThumbImmShiftedVal (unsigned V) {
247
- // If this can be handled with
247
+ // If this can be handled with
248
248
V = (~255U << getThumbImmValShift (V)) & V;
249
249
return V == 0 ;
250
250
}
@@ -260,10 +260,10 @@ namespace ARM_AM {
260
260
return CountTrailingZeros_32 (Imm);
261
261
}
262
262
263
- // / isThumbImm16ShiftedVal - Return true if the specified value can be
263
+ // / isThumbImm16ShiftedVal - Return true if the specified value can be
264
264
// / obtained by left shifting a 16-bit immediate.
265
265
static inline bool isThumbImm16ShiftedVal (unsigned V) {
266
- // If this can be handled with
266
+ // If this can be handled with
267
267
V = (~65535U << getThumbImm16ValShift (V)) & V;
268
268
return V == 0 ;
269
269
}
@@ -287,9 +287,9 @@ namespace ARM_AM {
287
287
static inline int getT2SOImmValSplatVal (unsigned V) {
288
288
unsigned u, Vs, Imm;
289
289
// control = 0
290
- if ((V & 0xffffff00 ) == 0 )
290
+ if ((V & 0xffffff00 ) == 0 )
291
291
return V;
292
-
292
+
293
293
// If the value is zeroes in the first byte, just shift those off
294
294
Vs = ((V & 0xff ) == 0 ) ? V >> 8 : V;
295
295
// Any passing value only has 8 bits of payload, splatted across the word
@@ -325,23 +325,23 @@ namespace ARM_AM {
325
325
}
326
326
327
327
// / getT2SOImmVal - Given a 32-bit immediate, if it is something that can fit
328
- // / into a Thumb-2 shifter_operand immediate operand, return the 12-bit
328
+ // / into a Thumb-2 shifter_operand immediate operand, return the 12-bit
329
329
// / encoding for it. If not, return -1.
330
330
// / See ARM Reference Manual A6.3.2.
331
331
static inline int getT2SOImmVal (unsigned Arg) {
332
332
// If 'Arg' is an 8-bit splat, then get the encoded value.
333
333
int Splat = getT2SOImmValSplatVal (Arg);
334
334
if (Splat != -1 )
335
335
return Splat;
336
-
336
+
337
337
// If 'Arg' can be handled with a single shifter_op return the value.
338
338
int Rot = getT2SOImmValRotateVal (Arg);
339
339
if (Rot != -1 )
340
340
return Rot;
341
341
342
342
return -1 ;
343
343
}
344
-
344
+
345
345
346
346
// ===--------------------------------------------------------------------===//
347
347
// Addressing Mode #2
@@ -359,7 +359,7 @@ namespace ARM_AM {
359
359
// If this addressing mode is a frame index (before prolog/epilog insertion
360
360
// and code rewriting), this operand will have the form: FI#, reg0, <offs>
361
361
// with no shift amount for the frame offset.
362
- //
362
+ //
363
363
static inline unsigned getAM2Opc (AddrOpc Opc, unsigned Imm12, ShiftOpc SO) {
364
364
assert (Imm12 < (1 << 12 ) && " Imm too large!" );
365
365
bool isSub = Opc == sub;
@@ -374,8 +374,8 @@ namespace ARM_AM {
374
374
static inline ShiftOpc getAM2ShiftOpc (unsigned AM2Opc) {
375
375
return (ShiftOpc)(AM2Opc >> 13 );
376
376
}
377
-
378
-
377
+
378
+
379
379
// ===--------------------------------------------------------------------===//
380
380
// Addressing Mode #3
381
381
// ===--------------------------------------------------------------------===//
@@ -388,7 +388,7 @@ namespace ARM_AM {
388
388
// The first operand is always a Reg. The second operand is a reg if in
389
389
// reg/reg form, otherwise it's reg#0. The third field encodes the operation
390
390
// in bit 8, the immediate in bits 0-7.
391
-
391
+
392
392
// / getAM3Opc - This function encodes the addrmode3 opc field.
393
393
static inline unsigned getAM3Opc (AddrOpc Opc, unsigned char Offset) {
394
394
bool isSub = Opc == sub;
@@ -400,7 +400,7 @@ namespace ARM_AM {
400
400
static inline AddrOpc getAM3Op (unsigned AM3Opc) {
401
401
return ((AM3Opc >> 8 ) & 1 ) ? sub : add;
402
402
}
403
-
403
+
404
404
// ===--------------------------------------------------------------------===//
405
405
// Addressing Mode #4
406
406
// ===--------------------------------------------------------------------===//
@@ -448,7 +448,7 @@ namespace ARM_AM {
448
448
//
449
449
// IA - Increment after
450
450
// DB - Decrement before
451
-
451
+
452
452
// / getAM5Opc - This function encodes the addrmode5 opc field.
453
453
static inline unsigned getAM5Opc (AddrOpc Opc, unsigned char Offset) {
454
454
bool isSub = Opc == sub;
0 commit comments