~louis/ubuntu/trusty/clamav/lp799623_fix_logrotate

« back to all changes in this revision

Viewing changes to libclamav/c++/llvm/lib/Target/ARM/ARMAddressingModes.h

  • Committer: Bazaar Package Importer
  • Author(s): Scott Kitterman
  • Date: 2010-03-12 11:30:04 UTC
  • mfrom: (0.41.1 upstream)
  • Revision ID: james.westby@ubuntu.com-20100312113004-b0fop4bkycszdd0z
Tags: 0.96~rc1+dfsg-0ubuntu1
* New upstream RC - FFE (LP: #537636):
  - Add OfficialDatabaseOnly option to clamav-base.postinst.in
  - Add LocalSocketGroup option to clamav-base.postinst.in
  - Add LocalSocketMode option to clamav-base.postinst.in
  - Add CrossFilesystems option to clamav-base.postinst.in
  - Add ClamukoScannerCount option to clamav-base.postinst.in
  - Add BytecodeSecurity opiton to clamav-base.postinst.in
  - Add DetectionStatsHostID option to clamav-freshclam.postinst.in
  - Add Bytecode option to clamav-freshclam.postinst.in
  - Add MilterSocketGroup option to clamav-milter.postinst.in
  - Add MilterSocketMode option to clamav-milter.postinst.in
  - Add ReportHostname option to clamav-milter.postinst.in
  - Bump libclamav SO version to 6.1.0 in libclamav6.install
  - Drop clamdmon from clamav.examples (no longer shipped by upstream)
  - Drop libclamav.a from libclamav-dev.install (not built by upstream)
  - Update SO version for lintian override for libclamav6
  - Add new Bytecode Testing Tool, usr/bin/clambc, to clamav.install
  - Add build-depends on python and python-setuptools for new test suite
  - Update debian/copyright for the embedded copy of llvm (using the system
    llvm is not currently feasible)

Show diffs side-by-side

added added

removed removed

Lines of Context:
 
1
//===- ARMAddressingModes.h - ARM Addressing Modes --------------*- C++ -*-===//
 
2
//
 
3
//                     The LLVM Compiler Infrastructure
 
4
//
 
5
// This file is distributed under the University of Illinois Open Source
 
6
// License. See LICENSE.TXT for details.
 
7
//
 
8
//===----------------------------------------------------------------------===//
 
9
//
 
10
// This file contains the ARM addressing mode implementation stuff.
 
11
//
 
12
//===----------------------------------------------------------------------===//
 
13
 
 
14
#ifndef LLVM_TARGET_ARM_ARMADDRESSINGMODES_H
 
15
#define LLVM_TARGET_ARM_ARMADDRESSINGMODES_H
 
16
 
 
17
#include "llvm/CodeGen/SelectionDAGNodes.h"
 
18
#include "llvm/Support/MathExtras.h"
 
19
#include <cassert>
 
20
 
 
21
namespace llvm {
 
22
 
 
23
/// ARM_AM - ARM Addressing Mode Stuff
 
24
namespace ARM_AM {
 
25
  enum ShiftOpc {
 
26
    no_shift = 0,
 
27
    asr,
 
28
    lsl,
 
29
    lsr,
 
30
    ror,
 
31
    rrx
 
32
  };
 
33
 
 
34
  enum AddrOpc {
 
35
    add = '+', sub = '-'
 
36
  };
 
37
 
 
38
  static inline const char *getShiftOpcStr(ShiftOpc Op) {
 
39
    switch (Op) {
 
40
    default: assert(0 && "Unknown shift opc!");
 
41
    case ARM_AM::asr: return "asr";
 
42
    case ARM_AM::lsl: return "lsl";
 
43
    case ARM_AM::lsr: return "lsr";
 
44
    case ARM_AM::ror: return "ror";
 
45
    case ARM_AM::rrx: return "rrx";
 
46
    }
 
47
  }
 
48
 
 
49
  static inline ShiftOpc getShiftOpcForNode(SDValue N) {
 
50
    switch (N.getOpcode()) {
 
51
    default:          return ARM_AM::no_shift;
 
52
    case ISD::SHL:    return ARM_AM::lsl;
 
53
    case ISD::SRL:    return ARM_AM::lsr;
 
54
    case ISD::SRA:    return ARM_AM::asr;
 
55
    case ISD::ROTR:   return ARM_AM::ror;
 
56
    //case ISD::ROTL:  // Only if imm -> turn into ROTR.
 
57
    // Can't handle RRX here, because it would require folding a flag into
 
58
    // the addressing mode.  :(  This causes us to miss certain things.
 
59
    //case ARMISD::RRX: return ARM_AM::rrx;
 
60
    }
 
61
  }
 
62
 
 
63
  enum AMSubMode {
 
64
    bad_am_submode = 0,
 
65
    ia,
 
66
    ib,
 
67
    da,
 
68
    db
 
69
  };
 
70
 
 
71
  static inline const char *getAMSubModeStr(AMSubMode Mode) {
 
72
    switch (Mode) {
 
73
    default: assert(0 && "Unknown addressing sub-mode!");
 
74
    case ARM_AM::ia: return "ia";
 
75
    case ARM_AM::ib: return "ib";
 
76
    case ARM_AM::da: return "da";
 
77
    case ARM_AM::db: return "db";
 
78
    }
 
79
  }
 
80
 
 
81
  static inline const char *getAMSubModeAltStr(AMSubMode Mode, bool isLD) {
 
82
    switch (Mode) {
 
83
    default: assert(0 && "Unknown addressing sub-mode!");
 
84
    case ARM_AM::ia: return isLD ? "fd" : "ea";
 
85
    case ARM_AM::ib: return isLD ? "ed" : "fa";
 
86
    case ARM_AM::da: return isLD ? "fa" : "ed";
 
87
    case ARM_AM::db: return isLD ? "ea" : "fd";
 
88
    }
 
89
  }
 
90
 
 
91
  /// rotr32 - Rotate a 32-bit unsigned value right by a specified # bits.
 
92
  ///
 
93
  static inline unsigned rotr32(unsigned Val, unsigned Amt) {
 
94
    assert(Amt < 32 && "Invalid rotate amount");
 
95
    return (Val >> Amt) | (Val << ((32-Amt)&31));
 
96
  }
 
97
 
 
98
  /// rotl32 - Rotate a 32-bit unsigned value left by a specified # bits.
 
99
  ///
 
100
  static inline unsigned rotl32(unsigned Val, unsigned Amt) {
 
101
    assert(Amt < 32 && "Invalid rotate amount");
 
102
    return (Val << Amt) | (Val >> ((32-Amt)&31));
 
103
  }
 
104
 
 
105
  //===--------------------------------------------------------------------===//
 
106
  // Addressing Mode #1: shift_operand with registers
 
107
  //===--------------------------------------------------------------------===//
 
108
  //
 
109
  // This 'addressing mode' is used for arithmetic instructions.  It can
 
110
  // represent things like:
 
111
  //   reg
 
112
  //   reg [asr|lsl|lsr|ror|rrx] reg
 
113
  //   reg [asr|lsl|lsr|ror|rrx] imm
 
114
  //
 
115
  // This is stored three operands [rega, regb, opc].  The first is the base
 
116
  // reg, the second is the shift amount (or reg0 if not present or imm).  The
 
117
  // third operand encodes the shift opcode and the imm if a reg isn't present.
 
118
  //
 
119
  static inline unsigned getSORegOpc(ShiftOpc ShOp, unsigned Imm) {
 
120
    return ShOp | (Imm << 3);
 
121
  }
 
122
  static inline unsigned getSORegOffset(unsigned Op) {
 
123
    return Op >> 3;
 
124
  }
 
125
  static inline ShiftOpc getSORegShOp(unsigned Op) {
 
126
    return (ShiftOpc)(Op & 7);
 
127
  }
 
128
 
 
129
  /// getSOImmValImm - Given an encoded imm field for the reg/imm form, return
 
130
  /// the 8-bit imm value.
 
131
  static inline unsigned getSOImmValImm(unsigned Imm) {
 
132
    return Imm & 0xFF;
 
133
  }
 
134
  /// getSOImmValRot - Given an encoded imm field for the reg/imm form, return
 
135
  /// the rotate amount.
 
136
  static inline unsigned getSOImmValRot(unsigned Imm) {
 
137
    return (Imm >> 8) * 2;
 
138
  }
 
139
 
 
140
  /// getSOImmValRotate - Try to handle Imm with an immediate shifter operand,
 
141
  /// computing the rotate amount to use.  If this immediate value cannot be
 
142
  /// handled with a single shifter-op, determine a good rotate amount that will
 
143
  /// take a maximal chunk of bits out of the immediate.
 
144
  static inline unsigned getSOImmValRotate(unsigned Imm) {
 
145
    // 8-bit (or less) immediates are trivially shifter_operands with a rotate
 
146
    // of zero.
 
147
    if ((Imm & ~255U) == 0) return 0;
 
148
 
 
149
    // Use CTZ to compute the rotate amount.
 
150
    unsigned TZ = CountTrailingZeros_32(Imm);
 
151
 
 
152
    // Rotate amount must be even.  Something like 0x200 must be rotated 8 bits,
 
153
    // not 9.
 
154
    unsigned RotAmt = TZ & ~1;
 
155
 
 
156
    // If we can handle this spread, return it.
 
157
    if ((rotr32(Imm, RotAmt) & ~255U) == 0)
 
158
      return (32-RotAmt)&31;  // HW rotates right, not left.
 
159
 
 
160
    // For values like 0xF000000F, we should skip the first run of ones, then
 
161
    // retry the hunt.
 
162
    if (Imm & 1) {
 
163
      unsigned TrailingOnes = CountTrailingZeros_32(~Imm);
 
164
      if (TrailingOnes != 32) {  // Avoid overflow on 0xFFFFFFFF
 
165
        // Restart the search for a high-order bit after the initial seconds of
 
166
        // ones.
 
167
        unsigned TZ2 = CountTrailingZeros_32(Imm & ~((1 << TrailingOnes)-1));
 
168
 
 
169
        // Rotate amount must be even.
 
170
        unsigned RotAmt2 = TZ2 & ~1;
 
171
 
 
172
        // If this fits, use it.
 
173
        if (RotAmt2 != 32 && (rotr32(Imm, RotAmt2) & ~255U) == 0)
 
174
          return (32-RotAmt2)&31;  // HW rotates right, not left.
 
175
      }
 
176
    }
 
177
 
 
178
    // Otherwise, we have no way to cover this span of bits with a single
 
179
    // shifter_op immediate.  Return a chunk of bits that will be useful to
 
180
    // handle.
 
181
    return (32-RotAmt)&31;  // HW rotates right, not left.
 
182
  }
 
183
 
 
184
  /// getSOImmVal - Given a 32-bit immediate, if it is something that can fit
 
185
  /// into an shifter_operand immediate operand, return the 12-bit encoding for
 
186
  /// it.  If not, return -1.
 
187
  static inline int getSOImmVal(unsigned Arg) {
 
188
    // 8-bit (or less) immediates are trivially shifter_operands with a rotate
 
189
    // of zero.
 
190
    if ((Arg & ~255U) == 0) return Arg;
 
191
 
 
192
    unsigned RotAmt = getSOImmValRotate(Arg);
 
193
 
 
194
    // If this cannot be handled with a single shifter_op, bail out.
 
195
    if (rotr32(~255U, RotAmt) & Arg)
 
196
      return -1;
 
197
 
 
198
    // Encode this correctly.
 
199
    return rotl32(Arg, RotAmt) | ((RotAmt>>1) << 8);
 
200
  }
 
201
 
 
202
  /// isSOImmTwoPartVal - Return true if the specified value can be obtained by
 
203
  /// or'ing together two SOImmVal's.
 
204
  static inline bool isSOImmTwoPartVal(unsigned V) {
 
205
    // If this can be handled with a single shifter_op, bail out.
 
206
    V = rotr32(~255U, getSOImmValRotate(V)) & V;
 
207
    if (V == 0)
 
208
      return false;
 
209
 
 
210
    // If this can be handled with two shifter_op's, accept.
 
211
    V = rotr32(~255U, getSOImmValRotate(V)) & V;
 
212
    return V == 0;
 
213
  }
 
214
 
 
215
  /// getSOImmTwoPartFirst - If V is a value that satisfies isSOImmTwoPartVal,
 
216
  /// return the first chunk of it.
 
217
  static inline unsigned getSOImmTwoPartFirst(unsigned V) {
 
218
    return rotr32(255U, getSOImmValRotate(V)) & V;
 
219
  }
 
220
 
 
221
  /// getSOImmTwoPartSecond - If V is a value that satisfies isSOImmTwoPartVal,
 
222
  /// return the second chunk of it.
 
223
  static inline unsigned getSOImmTwoPartSecond(unsigned V) {
 
224
    // Mask out the first hunk.
 
225
    V = rotr32(~255U, getSOImmValRotate(V)) & V;
 
226
 
 
227
    // Take what's left.
 
228
    assert(V == (rotr32(255U, getSOImmValRotate(V)) & V));
 
229
    return V;
 
230
  }
 
231
 
 
232
  /// getThumbImmValShift - Try to handle Imm with a 8-bit immediate followed
 
233
  /// by a left shift. Returns the shift amount to use.
 
234
  static inline unsigned getThumbImmValShift(unsigned Imm) {
 
235
    // 8-bit (or less) immediates are trivially immediate operand with a shift
 
236
    // of zero.
 
237
    if ((Imm & ~255U) == 0) return 0;
 
238
 
 
239
    // Use CTZ to compute the shift amount.
 
240
    return CountTrailingZeros_32(Imm);
 
241
  }
 
242
 
 
243
  /// isThumbImmShiftedVal - Return true if the specified value can be obtained
 
244
  /// by left shifting a 8-bit immediate.
 
245
  static inline bool isThumbImmShiftedVal(unsigned V) {
 
246
    // If this can be handled with
 
247
    V = (~255U << getThumbImmValShift(V)) & V;
 
248
    return V == 0;
 
249
  }
 
250
 
 
251
  /// getThumbImm16ValShift - Try to handle Imm with a 16-bit immediate followed
 
252
  /// by a left shift. Returns the shift amount to use.
 
253
  static inline unsigned getThumbImm16ValShift(unsigned Imm) {
 
254
    // 16-bit (or less) immediates are trivially immediate operand with a shift
 
255
    // of zero.
 
256
    if ((Imm & ~65535U) == 0) return 0;
 
257
 
 
258
    // Use CTZ to compute the shift amount.
 
259
    return CountTrailingZeros_32(Imm);
 
260
  }
 
261
 
 
262
  /// isThumbImm16ShiftedVal - Return true if the specified value can be
 
263
  /// obtained by left shifting a 16-bit immediate.
 
264
  static inline bool isThumbImm16ShiftedVal(unsigned V) {
 
265
    // If this can be handled with
 
266
    V = (~65535U << getThumbImm16ValShift(V)) & V;
 
267
    return V == 0;
 
268
  }
 
269
 
 
270
  /// getThumbImmNonShiftedVal - If V is a value that satisfies
 
271
  /// isThumbImmShiftedVal, return the non-shiftd value.
 
272
  static inline unsigned getThumbImmNonShiftedVal(unsigned V) {
 
273
    return V >> getThumbImmValShift(V);
 
274
  }
 
275
 
 
276
 
 
277
  /// getT2SOImmValSplat - Return the 12-bit encoded representation
 
278
  /// if the specified value can be obtained by splatting the low 8 bits
 
279
  /// into every other byte or every byte of a 32-bit value. i.e.,
 
280
  ///     00000000 00000000 00000000 abcdefgh    control = 0
 
281
  ///     00000000 abcdefgh 00000000 abcdefgh    control = 1
 
282
  ///     abcdefgh 00000000 abcdefgh 00000000    control = 2
 
283
  ///     abcdefgh abcdefgh abcdefgh abcdefgh    control = 3
 
284
  /// Return -1 if none of the above apply.
 
285
  /// See ARM Reference Manual A6.3.2.
 
286
  static inline int getT2SOImmValSplatVal(unsigned V) {
 
287
    unsigned u, Vs, Imm;
 
288
    // control = 0
 
289
    if ((V & 0xffffff00) == 0)
 
290
      return V;
 
291
 
 
292
    // If the value is zeroes in the first byte, just shift those off
 
293
    Vs = ((V & 0xff) == 0) ? V >> 8 : V;
 
294
    // Any passing value only has 8 bits of payload, splatted across the word
 
295
    Imm = Vs & 0xff;
 
296
    // Likewise, any passing values have the payload splatted into the 3rd byte
 
297
    u = Imm | (Imm << 16);
 
298
 
 
299
    // control = 1 or 2
 
300
    if (Vs == u)
 
301
      return (((Vs == V) ? 1 : 2) << 8) | Imm;
 
302
 
 
303
    // control = 3
 
304
    if (Vs == (u | (u << 8)))
 
305
      return (3 << 8) | Imm;
 
306
 
 
307
    return -1;
 
308
  }
 
309
 
 
310
  /// getT2SOImmValRotateVal - Return the 12-bit encoded representation if the
 
311
  /// specified value is a rotated 8-bit value. Return -1 if no rotation
 
312
  /// encoding is possible.
 
313
  /// See ARM Reference Manual A6.3.2.
 
314
  static inline int getT2SOImmValRotateVal(unsigned V) {
 
315
    unsigned RotAmt = CountLeadingZeros_32(V);
 
316
    if (RotAmt >= 24)
 
317
      return -1;
 
318
 
 
319
    // If 'Arg' can be handled with a single shifter_op return the value.
 
320
    if ((rotr32(0xff000000U, RotAmt) & V) == V)
 
321
      return (rotr32(V, 24 - RotAmt) & 0x7f) | ((RotAmt + 8) << 7);
 
322
 
 
323
    return -1;
 
324
  }
 
325
 
 
326
  /// getT2SOImmVal - Given a 32-bit immediate, if it is something that can fit
 
327
  /// into a Thumb-2 shifter_operand immediate operand, return the 12-bit
 
328
  /// encoding for it.  If not, return -1.
 
329
  /// See ARM Reference Manual A6.3.2.
 
330
  static inline int getT2SOImmVal(unsigned Arg) {
 
331
    // If 'Arg' is an 8-bit splat, then get the encoded value.
 
332
    int Splat = getT2SOImmValSplatVal(Arg);
 
333
    if (Splat != -1)
 
334
      return Splat;
 
335
 
 
336
    // If 'Arg' can be handled with a single shifter_op return the value.
 
337
    int Rot = getT2SOImmValRotateVal(Arg);
 
338
    if (Rot != -1)
 
339
      return Rot;
 
340
 
 
341
    return -1;
 
342
  }
 
343
 
 
344
  static inline unsigned getT2SOImmValRotate(unsigned V) {
 
345
    if ((V & ~255U) == 0) return 0;
 
346
    // Use CTZ to compute the rotate amount.
 
347
    unsigned RotAmt = CountTrailingZeros_32(V);
 
348
    return (32 - RotAmt) & 31;
 
349
  }
 
350
 
 
351
  static inline bool isT2SOImmTwoPartVal (unsigned Imm) {
 
352
    unsigned V = Imm;
 
353
    // Passing values can be any combination of splat values and shifter
 
354
    // values. If this can be handled with a single shifter or splat, bail
 
355
    // out. Those should be handled directly, not with a two-part val.
 
356
    if (getT2SOImmValSplatVal(V) != -1)
 
357
      return false;
 
358
    V = rotr32 (~255U, getT2SOImmValRotate(V)) & V;
 
359
    if (V == 0)
 
360
      return false;
 
361
 
 
362
    // If this can be handled as an immediate, accept.
 
363
    if (getT2SOImmVal(V) != -1) return true;
 
364
 
 
365
    // Likewise, try masking out a splat value first.
 
366
    V = Imm;
 
367
    if (getT2SOImmValSplatVal(V & 0xff00ff00U) != -1)
 
368
      V &= ~0xff00ff00U;
 
369
    else if (getT2SOImmValSplatVal(V & 0x00ff00ffU) != -1)
 
370
      V &= ~0x00ff00ffU;
 
371
    // If what's left can be handled as an immediate, accept.
 
372
    if (getT2SOImmVal(V) != -1) return true;
 
373
 
 
374
    // Otherwise, do not accept.
 
375
    return false;
 
376
  }
 
377
 
 
378
  static inline unsigned getT2SOImmTwoPartFirst(unsigned Imm) {
 
379
    assert (isT2SOImmTwoPartVal(Imm) &&
 
380
            "Immedate cannot be encoded as two part immediate!");
 
381
    // Try a shifter operand as one part
 
382
    unsigned V = rotr32 (~255, getT2SOImmValRotate(Imm)) & Imm;
 
383
    // If the rest is encodable as an immediate, then return it.
 
384
    if (getT2SOImmVal(V) != -1) return V;
 
385
 
 
386
    // Try masking out a splat value first.
 
387
    if (getT2SOImmValSplatVal(Imm & 0xff00ff00U) != -1)
 
388
      return Imm & 0xff00ff00U;
 
389
 
 
390
    // The other splat is all that's left as an option.
 
391
    assert (getT2SOImmValSplatVal(Imm & 0x00ff00ffU) != -1);
 
392
    return Imm & 0x00ff00ffU;
 
393
  }
 
394
 
 
395
  static inline unsigned getT2SOImmTwoPartSecond(unsigned Imm) {
 
396
    // Mask out the first hunk
 
397
    Imm ^= getT2SOImmTwoPartFirst(Imm);
 
398
    // Return what's left
 
399
    assert (getT2SOImmVal(Imm) != -1 &&
 
400
            "Unable to encode second part of T2 two part SO immediate");
 
401
    return Imm;
 
402
  }
 
403
 
 
404
 
 
405
  //===--------------------------------------------------------------------===//
 
406
  // Addressing Mode #2
 
407
  //===--------------------------------------------------------------------===//
 
408
  //
 
409
  // This is used for most simple load/store instructions.
 
410
  //
 
411
  // addrmode2 := reg +/- reg shop imm
 
412
  // addrmode2 := reg +/- imm12
 
413
  //
 
414
  // The first operand is always a Reg.  The second operand is a reg if in
 
415
  // reg/reg form, otherwise it's reg#0.  The third field encodes the operation
 
416
  // in bit 12, the immediate in bits 0-11, and the shift op in 13-15.
 
417
  //
 
418
  // If this addressing mode is a frame index (before prolog/epilog insertion
 
419
  // and code rewriting), this operand will have the form:  FI#, reg0, <offs>
 
420
  // with no shift amount for the frame offset.
 
421
  //
 
422
  static inline unsigned getAM2Opc(AddrOpc Opc, unsigned Imm12, ShiftOpc SO) {
 
423
    assert(Imm12 < (1 << 12) && "Imm too large!");
 
424
    bool isSub = Opc == sub;
 
425
    return Imm12 | ((int)isSub << 12) | (SO << 13);
 
426
  }
 
427
  static inline unsigned getAM2Offset(unsigned AM2Opc) {
 
428
    return AM2Opc & ((1 << 12)-1);
 
429
  }
 
430
  static inline AddrOpc getAM2Op(unsigned AM2Opc) {
 
431
    return ((AM2Opc >> 12) & 1) ? sub : add;
 
432
  }
 
433
  static inline ShiftOpc getAM2ShiftOpc(unsigned AM2Opc) {
 
434
    return (ShiftOpc)(AM2Opc >> 13);
 
435
  }
 
436
 
 
437
 
 
438
  //===--------------------------------------------------------------------===//
 
439
  // Addressing Mode #3
 
440
  //===--------------------------------------------------------------------===//
 
441
  //
 
442
  // This is used for sign-extending loads, and load/store-pair instructions.
 
443
  //
 
444
  // addrmode3 := reg +/- reg
 
445
  // addrmode3 := reg +/- imm8
 
446
  //
 
447
  // The first operand is always a Reg.  The second operand is a reg if in
 
448
  // reg/reg form, otherwise it's reg#0.  The third field encodes the operation
 
449
  // in bit 8, the immediate in bits 0-7.
 
450
 
 
451
  /// getAM3Opc - This function encodes the addrmode3 opc field.
 
452
  static inline unsigned getAM3Opc(AddrOpc Opc, unsigned char Offset) {
 
453
    bool isSub = Opc == sub;
 
454
    return ((int)isSub << 8) | Offset;
 
455
  }
 
456
  static inline unsigned char getAM3Offset(unsigned AM3Opc) {
 
457
    return AM3Opc & 0xFF;
 
458
  }
 
459
  static inline AddrOpc getAM3Op(unsigned AM3Opc) {
 
460
    return ((AM3Opc >> 8) & 1) ? sub : add;
 
461
  }
 
462
 
 
463
  //===--------------------------------------------------------------------===//
 
464
  // Addressing Mode #4
 
465
  //===--------------------------------------------------------------------===//
 
466
  //
 
467
  // This is used for load / store multiple instructions.
 
468
  //
 
469
  // addrmode4 := reg, <mode>
 
470
  //
 
471
  // The four modes are:
 
472
  //    IA - Increment after
 
473
  //    IB - Increment before
 
474
  //    DA - Decrement after
 
475
  //    DB - Decrement before
 
476
  //
 
477
  // If the 4th bit (writeback)is set, then the base register is updated after
 
478
  // the memory transfer.
 
479
 
 
480
  static inline AMSubMode getAM4SubMode(unsigned Mode) {
 
481
    return (AMSubMode)(Mode & 0x7);
 
482
  }
 
483
 
 
484
  static inline unsigned getAM4ModeImm(AMSubMode SubMode, bool WB = false) {
 
485
    return (int)SubMode | ((int)WB << 3);
 
486
  }
 
487
 
 
488
  static inline bool getAM4WBFlag(unsigned Mode) {
 
489
    return (Mode >> 3) & 1;
 
490
  }
 
491
 
 
492
  //===--------------------------------------------------------------------===//
 
493
  // Addressing Mode #5
 
494
  //===--------------------------------------------------------------------===//
 
495
  //
 
496
  // This is used for coprocessor instructions, such as FP load/stores.
 
497
  //
 
498
  // addrmode5 := reg +/- imm8*4
 
499
  //
 
500
  // The first operand is always a Reg.  The second operand encodes the
 
501
  // operation in bit 8 and the immediate in bits 0-7.
 
502
  //
 
503
  // This is also used for FP load/store multiple ops. The second operand
 
504
  // encodes the writeback mode in bit 8 and the number of registers (or 2
 
505
  // times the number of registers for DPR ops) in bits 0-7. In addition,
 
506
  // bits 9-11 encode one of the following two sub-modes:
 
507
  //
 
508
  //    IA - Increment after
 
509
  //    DB - Decrement before
 
510
 
 
511
  /// getAM5Opc - This function encodes the addrmode5 opc field.
 
512
  static inline unsigned getAM5Opc(AddrOpc Opc, unsigned char Offset) {
 
513
    bool isSub = Opc == sub;
 
514
    return ((int)isSub << 8) | Offset;
 
515
  }
 
516
  static inline unsigned char getAM5Offset(unsigned AM5Opc) {
 
517
    return AM5Opc & 0xFF;
 
518
  }
 
519
  static inline AddrOpc getAM5Op(unsigned AM5Opc) {
 
520
    return ((AM5Opc >> 8) & 1) ? sub : add;
 
521
  }
 
522
 
 
523
  /// getAM5Opc - This function encodes the addrmode5 opc field for VLDM and
 
524
  /// VSTM instructions.
 
525
  static inline unsigned getAM5Opc(AMSubMode SubMode, bool WB,
 
526
                                   unsigned char Offset) {
 
527
    assert((SubMode == ia || SubMode == db) &&
 
528
           "Illegal addressing mode 5 sub-mode!");
 
529
    return ((int)SubMode << 9) | ((int)WB << 8) | Offset;
 
530
  }
 
531
  static inline AMSubMode getAM5SubMode(unsigned AM5Opc) {
 
532
    return (AMSubMode)((AM5Opc >> 9) & 0x7);
 
533
  }
 
534
  static inline bool getAM5WBFlag(unsigned AM5Opc) {
 
535
    return ((AM5Opc >> 8) & 1);
 
536
  }
 
537
 
 
538
  //===--------------------------------------------------------------------===//
 
539
  // Addressing Mode #6
 
540
  //===--------------------------------------------------------------------===//
 
541
  //
 
542
  // This is used for NEON load / store instructions.
 
543
  //
 
544
  // addrmode6 := reg with optional writeback and alignment
 
545
  //
 
546
  // This is stored in four operands [regaddr, regupdate, opc, align].  The
 
547
  // first is the address register.  The second register holds the value of
 
548
  // a post-access increment for writeback or reg0 if no writeback or if the
 
549
  // writeback increment is the size of the memory access.  The third
 
550
  // operand encodes whether there is writeback to the address register. The
 
551
  // fourth operand is the value of the alignment specifier to use or zero if
 
552
  // no explicit alignment.
 
553
 
 
554
  static inline unsigned getAM6Opc(bool WB = false) {
 
555
    return (int)WB;
 
556
  }
 
557
 
 
558
  static inline bool getAM6WBFlag(unsigned Mode) {
 
559
    return Mode & 1;
 
560
  }
 
561
 
 
562
} // end namespace ARM_AM
 
563
} // end namespace llvm
 
564
 
 
565
#endif
 
566