* doc/as.texinfo: Add mention of RX port and inclusion of RX
[binutils/dougsmingw.git] / opcodes / arm-dis.c
blob75512499048d5d05b8dff7d7ff9ed0794b61c66f
1 /* Instruction printing code for the ARM
2 Copyright 1994, 1995, 1996, 1997, 1998, 1999, 2000, 2001, 2002, 2003,
3 2004, 2005, 2006, 2007, 2008, 2009 Free Software Foundation, Inc.
4 Contributed by Richard Earnshaw (rwe@pegasus.esprit.ec.org)
5 Modification by James G. Smith (jsmith@cygnus.co.uk)
7 This file is part of libopcodes.
9 This library is free software; you can redistribute it and/or modify
10 it under the terms of the GNU General Public License as published by
11 the Free Software Foundation; either version 3 of the License, or
12 (at your option) any later version.
14 It is distributed in the hope that it will be useful, but WITHOUT
15 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
16 or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public
17 License for more details.
19 You should have received a copy of the GNU General Public License
20 along with this program; if not, write to the Free Software
21 Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston,
22 MA 02110-1301, USA. */
24 #include "sysdep.h"
26 #include "dis-asm.h"
27 #include "opcode/arm.h"
28 #include "opintl.h"
29 #include "safe-ctype.h"
30 #include "floatformat.h"
32 /* FIXME: This shouldn't be done here. */
33 #include "coff/internal.h"
34 #include "libcoff.h"
35 #include "elf-bfd.h"
36 #include "elf/internal.h"
37 #include "elf/arm.h"
39 /* FIXME: Belongs in global header. */
40 #ifndef strneq
41 #define strneq(a,b,n) (strncmp ((a), (b), (n)) == 0)
42 #endif
44 #ifndef NUM_ELEM
45 #define NUM_ELEM(a) (sizeof (a) / sizeof (a)[0])
46 #endif
48 struct opcode32
50 unsigned long arch; /* Architecture defining this insn. */
51 unsigned long value; /* If arch == 0 then value is a sentinel. */
52 unsigned long mask; /* Recognise insn if (op & mask) == value. */
53 const char * assembler; /* How to disassemble this insn. */
56 struct opcode16
58 unsigned long arch; /* Architecture defining this insn. */
59 unsigned short value, mask; /* Recognise insn if (op&mask)==value. */
60 const char *assembler; /* How to disassemble this insn. */
63 /* print_insn_coprocessor recognizes the following format control codes:
65 %% %
67 %c print condition code (always bits 28-31 in ARM mode)
68 %q print shifter argument
69 %u print condition code (unconditional in ARM mode)
70 %A print address for ldc/stc/ldf/stf instruction
71 %B print vstm/vldm register list
72 %C print vstr/vldr address operand
73 %I print cirrus signed shift immediate: bits 0..3|4..6
74 %F print the COUNT field of a LFM/SFM instruction.
75 %P print floating point precision in arithmetic insn
76 %Q print floating point precision in ldf/stf insn
77 %R print floating point rounding mode
79 %<bitfield>r print as an ARM register
80 %<bitfield>d print the bitfield in decimal
81 %<bitfield>k print immediate for VFPv3 conversion instruction
82 %<bitfield>x print the bitfield in hex
83 %<bitfield>X print the bitfield as 1 hex digit without leading "0x"
84 %<bitfield>f print a floating point constant if >7 else a
85 floating point register
86 %<bitfield>w print as an iWMMXt width field - [bhwd]ss/us
87 %<bitfield>g print as an iWMMXt 64-bit register
88 %<bitfield>G print as an iWMMXt general purpose or control register
89 %<bitfield>D print as a NEON D register
90 %<bitfield>Q print as a NEON Q register
92 %y<code> print a single precision VFP reg.
93 Codes: 0=>Sm, 1=>Sd, 2=>Sn, 3=>multi-list, 4=>Sm pair
94 %z<code> print a double precision VFP reg
95 Codes: 0=>Dm, 1=>Dd, 2=>Dn, 3=>multi-list
97 %<bitfield>'c print specified char iff bitfield is all ones
98 %<bitfield>`c print specified char iff bitfield is all zeroes
99 %<bitfield>?ab... select from array of values in big endian order
101 %L print as an iWMMXt N/M width field.
102 %Z print the Immediate of a WSHUFH instruction.
103 %l like 'A' except use byte offsets for 'B' & 'H'
104 versions.
105 %i print 5-bit immediate in bits 8,3..0
106 (print "32" when 0)
107 %r print register offset address for wldt/wstr instruction. */
109 enum opcode_sentinel_enum
111 SENTINEL_IWMMXT_START = 1,
112 SENTINEL_IWMMXT_END,
113 SENTINEL_GENERIC_START
114 } opcode_sentinels;
116 #define UNDEFINED_INSTRUCTION "undefined instruction %0-31x"
118 /* Common coprocessor opcodes shared between Arm and Thumb-2. */
120 static const struct opcode32 coprocessor_opcodes[] =
122 /* XScale instructions. */
123 {ARM_CEXT_XSCALE, 0x0e200010, 0x0fff0ff0, "mia%c\tacc0, %0-3r, %12-15r"},
124 {ARM_CEXT_XSCALE, 0x0e280010, 0x0fff0ff0, "miaph%c\tacc0, %0-3r, %12-15r"},
125 {ARM_CEXT_XSCALE, 0x0e2c0010, 0x0ffc0ff0, "mia%17'T%17`B%16'T%16`B%c\tacc0, %0-3r, %12-15r"},
126 {ARM_CEXT_XSCALE, 0x0c400000, 0x0ff00fff, "mar%c\tacc0, %12-15r, %16-19r"},
127 {ARM_CEXT_XSCALE, 0x0c500000, 0x0ff00fff, "mra%c\t%12-15r, %16-19r, acc0"},
129 /* Intel Wireless MMX technology instructions. */
130 { 0, SENTINEL_IWMMXT_START, 0, "" },
131 {ARM_CEXT_IWMMXT, 0x0e130130, 0x0f3f0fff, "tandc%22-23w%c\t%12-15r"},
132 {ARM_CEXT_XSCALE, 0x0e400010, 0x0ff00f3f, "tbcst%6-7w%c\t%16-19g, %12-15r"},
133 {ARM_CEXT_XSCALE, 0x0e130170, 0x0f3f0ff8, "textrc%22-23w%c\t%12-15r, #%0-2d"},
134 {ARM_CEXT_XSCALE, 0x0e100070, 0x0f300ff0, "textrm%3?su%22-23w%c\t%12-15r, %16-19g, #%0-2d"},
135 {ARM_CEXT_XSCALE, 0x0e600010, 0x0ff00f38, "tinsr%6-7w%c\t%16-19g, %12-15r, #%0-2d"},
136 {ARM_CEXT_XSCALE, 0x0e000110, 0x0ff00fff, "tmcr%c\t%16-19G, %12-15r"},
137 {ARM_CEXT_XSCALE, 0x0c400000, 0x0ff00ff0, "tmcrr%c\t%0-3g, %12-15r, %16-19r"},
138 {ARM_CEXT_XSCALE, 0x0e2c0010, 0x0ffc0e10, "tmia%17?tb%16?tb%c\t%5-8g, %0-3r, %12-15r"},
139 {ARM_CEXT_XSCALE, 0x0e200010, 0x0fff0e10, "tmia%c\t%5-8g, %0-3r, %12-15r"},
140 {ARM_CEXT_XSCALE, 0x0e280010, 0x0fff0e10, "tmiaph%c\t%5-8g, %0-3r, %12-15r"},
141 {ARM_CEXT_XSCALE, 0x0e100030, 0x0f300fff, "tmovmsk%22-23w%c\t%12-15r, %16-19g"},
142 {ARM_CEXT_XSCALE, 0x0e100110, 0x0ff00ff0, "tmrc%c\t%12-15r, %16-19G"},
143 {ARM_CEXT_XSCALE, 0x0c500000, 0x0ff00ff0, "tmrrc%c\t%12-15r, %16-19r, %0-3g"},
144 {ARM_CEXT_XSCALE, 0x0e130150, 0x0f3f0fff, "torc%22-23w%c\t%12-15r"},
145 {ARM_CEXT_XSCALE, 0x0e120190, 0x0f3f0fff, "torvsc%22-23w%c\t%12-15r"},
146 {ARM_CEXT_XSCALE, 0x0e2001c0, 0x0f300fff, "wabs%22-23w%c\t%12-15g, %16-19g"},
147 {ARM_CEXT_XSCALE, 0x0e0001c0, 0x0f300fff, "wacc%22-23w%c\t%12-15g, %16-19g"},
148 {ARM_CEXT_XSCALE, 0x0e000180, 0x0f000ff0, "wadd%20-23w%c\t%12-15g, %16-19g, %0-3g"},
149 {ARM_CEXT_XSCALE, 0x0e2001a0, 0x0fb00ff0, "waddbhus%22?ml%c\t%12-15g, %16-19g, %0-3g"},
150 {ARM_CEXT_XSCALE, 0x0ea001a0, 0x0ff00ff0, "waddsubhx%c\t%12-15g, %16-19g, %0-3g"},
151 {ARM_CEXT_XSCALE, 0x0e000020, 0x0f800ff0, "waligni%c\t%12-15g, %16-19g, %0-3g, #%20-22d"},
152 {ARM_CEXT_XSCALE, 0x0e800020, 0x0fc00ff0, "walignr%20-21d%c\t%12-15g, %16-19g, %0-3g"},
153 {ARM_CEXT_XSCALE, 0x0e200000, 0x0fe00ff0, "wand%20'n%c\t%12-15g, %16-19g, %0-3g"},
154 {ARM_CEXT_XSCALE, 0x0e800000, 0x0fa00ff0, "wavg2%22?hb%20'r%c\t%12-15g, %16-19g, %0-3g"},
155 {ARM_CEXT_XSCALE, 0x0e400000, 0x0fe00ff0, "wavg4%20'r%c\t%12-15g, %16-19g, %0-3g"},
156 {ARM_CEXT_XSCALE, 0x0e000060, 0x0f300ff0, "wcmpeq%22-23w%c\t%12-15g, %16-19g, %0-3g"},
157 {ARM_CEXT_XSCALE, 0x0e100060, 0x0f100ff0, "wcmpgt%21?su%22-23w%c\t%12-15g, %16-19g, %0-3g"},
158 {ARM_CEXT_XSCALE, 0xfc500100, 0xfe500f00, "wldrd\t%12-15g, %r"},
159 {ARM_CEXT_XSCALE, 0xfc100100, 0xfe500f00, "wldrw\t%12-15G, %A"},
160 {ARM_CEXT_XSCALE, 0x0c100000, 0x0e100e00, "wldr%L%c\t%12-15g, %l"},
161 {ARM_CEXT_XSCALE, 0x0e400100, 0x0fc00ff0, "wmac%21?su%20'z%c\t%12-15g, %16-19g, %0-3g"},
162 {ARM_CEXT_XSCALE, 0x0e800100, 0x0fc00ff0, "wmadd%21?su%20'x%c\t%12-15g, %16-19g, %0-3g"},
163 {ARM_CEXT_XSCALE, 0x0ec00100, 0x0fd00ff0, "wmadd%21?sun%c\t%12-15g, %16-19g, %0-3g"},
164 {ARM_CEXT_XSCALE, 0x0e000160, 0x0f100ff0, "wmax%21?su%22-23w%c\t%12-15g, %16-19g, %0-3g"},
165 {ARM_CEXT_XSCALE, 0x0e000080, 0x0f100fe0, "wmerge%c\t%12-15g, %16-19g, %0-3g, #%21-23d"},
166 {ARM_CEXT_XSCALE, 0x0e0000a0, 0x0f800ff0, "wmia%21?tb%20?tb%22'n%c\t%12-15g, %16-19g, %0-3g"},
167 {ARM_CEXT_XSCALE, 0x0e800120, 0x0f800ff0, "wmiaw%21?tb%20?tb%22'n%c\t%12-15g, %16-19g, %0-3g"},
168 {ARM_CEXT_XSCALE, 0x0e100160, 0x0f100ff0, "wmin%21?su%22-23w%c\t%12-15g, %16-19g, %0-3g"},
169 {ARM_CEXT_XSCALE, 0x0e000100, 0x0fc00ff0, "wmul%21?su%20?ml%23'r%c\t%12-15g, %16-19g, %0-3g"},
170 {ARM_CEXT_XSCALE, 0x0ed00100, 0x0fd00ff0, "wmul%21?sumr%c\t%12-15g, %16-19g, %0-3g"},
171 {ARM_CEXT_XSCALE, 0x0ee000c0, 0x0fe00ff0, "wmulwsm%20`r%c\t%12-15g, %16-19g, %0-3g"},
172 {ARM_CEXT_XSCALE, 0x0ec000c0, 0x0fe00ff0, "wmulwum%20`r%c\t%12-15g, %16-19g, %0-3g"},
173 {ARM_CEXT_XSCALE, 0x0eb000c0, 0x0ff00ff0, "wmulwl%c\t%12-15g, %16-19g, %0-3g"},
174 {ARM_CEXT_XSCALE, 0x0e8000a0, 0x0f800ff0, "wqmia%21?tb%20?tb%22'n%c\t%12-15g, %16-19g, %0-3g"},
175 {ARM_CEXT_XSCALE, 0x0e100080, 0x0fd00ff0, "wqmulm%21'r%c\t%12-15g, %16-19g, %0-3g"},
176 {ARM_CEXT_XSCALE, 0x0ec000e0, 0x0fd00ff0, "wqmulwm%21'r%c\t%12-15g, %16-19g, %0-3g"},
177 {ARM_CEXT_XSCALE, 0x0e000000, 0x0ff00ff0, "wor%c\t%12-15g, %16-19g, %0-3g"},
178 {ARM_CEXT_XSCALE, 0x0e000080, 0x0f000ff0, "wpack%20-23w%c\t%12-15g, %16-19g, %0-3g"},
179 {ARM_CEXT_XSCALE, 0xfe300040, 0xff300ef0, "wror%22-23w\t%12-15g, %16-19g, #%i"},
180 {ARM_CEXT_XSCALE, 0x0e300040, 0x0f300ff0, "wror%22-23w%c\t%12-15g, %16-19g, %0-3g"},
181 {ARM_CEXT_XSCALE, 0x0e300140, 0x0f300ff0, "wror%22-23wg%c\t%12-15g, %16-19g, %0-3G"},
182 {ARM_CEXT_XSCALE, 0x0e000120, 0x0fa00ff0, "wsad%22?hb%20'z%c\t%12-15g, %16-19g, %0-3g"},
183 {ARM_CEXT_XSCALE, 0x0e0001e0, 0x0f000ff0, "wshufh%c\t%12-15g, %16-19g, #%Z"},
184 {ARM_CEXT_XSCALE, 0xfe100040, 0xff300ef0, "wsll%22-23w\t%12-15g, %16-19g, #%i"},
185 {ARM_CEXT_XSCALE, 0x0e100040, 0x0f300ff0, "wsll%22-23w%8'g%c\t%12-15g, %16-19g, %0-3g"},
186 {ARM_CEXT_XSCALE, 0x0e100148, 0x0f300ffc, "wsll%22-23w%8'g%c\t%12-15g, %16-19g, %0-3G"},
187 {ARM_CEXT_XSCALE, 0xfe000040, 0xff300ef0, "wsra%22-23w\t%12-15g, %16-19g, #%i"},
188 {ARM_CEXT_XSCALE, 0x0e000040, 0x0f300ff0, "wsra%22-23w%8'g%c\t%12-15g, %16-19g, %0-3g"},
189 {ARM_CEXT_XSCALE, 0x0e000148, 0x0f300ffc, "wsra%22-23w%8'g%c\t%12-15g, %16-19g, %0-3G"},
190 {ARM_CEXT_XSCALE, 0xfe200040, 0xff300ef0, "wsrl%22-23w\t%12-15g, %16-19g, #%i"},
191 {ARM_CEXT_XSCALE, 0x0e200040, 0x0f300ff0, "wsrl%22-23w%8'g%c\t%12-15g, %16-19g, %0-3g"},
192 {ARM_CEXT_XSCALE, 0x0e200148, 0x0f300ffc, "wsrl%22-23w%8'g%c\t%12-15g, %16-19g, %0-3G"},
193 {ARM_CEXT_XSCALE, 0xfc400100, 0xfe500f00, "wstrd\t%12-15g, %r"},
194 {ARM_CEXT_XSCALE, 0xfc000100, 0xfe500f00, "wstrw\t%12-15G, %A"},
195 {ARM_CEXT_XSCALE, 0x0c000000, 0x0e100e00, "wstr%L%c\t%12-15g, %l"},
196 {ARM_CEXT_XSCALE, 0x0e0001a0, 0x0f000ff0, "wsub%20-23w%c\t%12-15g, %16-19g, %0-3g"},
197 {ARM_CEXT_XSCALE, 0x0ed001c0, 0x0ff00ff0, "wsubaddhx%c\t%12-15g, %16-19g, %0-3g"},
198 {ARM_CEXT_XSCALE, 0x0e1001c0, 0x0f300ff0, "wabsdiff%22-23w%c\t%12-15g, %16-19g, %0-3g"},
199 {ARM_CEXT_XSCALE, 0x0e0000c0, 0x0fd00fff, "wunpckeh%21?sub%c\t%12-15g, %16-19g"},
200 {ARM_CEXT_XSCALE, 0x0e4000c0, 0x0fd00fff, "wunpckeh%21?suh%c\t%12-15g, %16-19g"},
201 {ARM_CEXT_XSCALE, 0x0e8000c0, 0x0fd00fff, "wunpckeh%21?suw%c\t%12-15g, %16-19g"},
202 {ARM_CEXT_XSCALE, 0x0e0000e0, 0x0f100fff, "wunpckel%21?su%22-23w%c\t%12-15g, %16-19g"},
203 {ARM_CEXT_XSCALE, 0x0e1000c0, 0x0f300ff0, "wunpckih%22-23w%c\t%12-15g, %16-19g, %0-3g"},
204 {ARM_CEXT_XSCALE, 0x0e1000e0, 0x0f300ff0, "wunpckil%22-23w%c\t%12-15g, %16-19g, %0-3g"},
205 {ARM_CEXT_XSCALE, 0x0e100000, 0x0ff00ff0, "wxor%c\t%12-15g, %16-19g, %0-3g"},
206 { 0, SENTINEL_IWMMXT_END, 0, "" },
208 /* Floating point coprocessor (FPA) instructions. */
209 {FPU_FPA_EXT_V1, 0x0e000100, 0x0ff08f10, "adf%c%P%R\t%12-14f, %16-18f, %0-3f"},
210 {FPU_FPA_EXT_V1, 0x0e100100, 0x0ff08f10, "muf%c%P%R\t%12-14f, %16-18f, %0-3f"},
211 {FPU_FPA_EXT_V1, 0x0e200100, 0x0ff08f10, "suf%c%P%R\t%12-14f, %16-18f, %0-3f"},
212 {FPU_FPA_EXT_V1, 0x0e300100, 0x0ff08f10, "rsf%c%P%R\t%12-14f, %16-18f, %0-3f"},
213 {FPU_FPA_EXT_V1, 0x0e400100, 0x0ff08f10, "dvf%c%P%R\t%12-14f, %16-18f, %0-3f"},
214 {FPU_FPA_EXT_V1, 0x0e500100, 0x0ff08f10, "rdf%c%P%R\t%12-14f, %16-18f, %0-3f"},
215 {FPU_FPA_EXT_V1, 0x0e600100, 0x0ff08f10, "pow%c%P%R\t%12-14f, %16-18f, %0-3f"},
216 {FPU_FPA_EXT_V1, 0x0e700100, 0x0ff08f10, "rpw%c%P%R\t%12-14f, %16-18f, %0-3f"},
217 {FPU_FPA_EXT_V1, 0x0e800100, 0x0ff08f10, "rmf%c%P%R\t%12-14f, %16-18f, %0-3f"},
218 {FPU_FPA_EXT_V1, 0x0e900100, 0x0ff08f10, "fml%c%P%R\t%12-14f, %16-18f, %0-3f"},
219 {FPU_FPA_EXT_V1, 0x0ea00100, 0x0ff08f10, "fdv%c%P%R\t%12-14f, %16-18f, %0-3f"},
220 {FPU_FPA_EXT_V1, 0x0eb00100, 0x0ff08f10, "frd%c%P%R\t%12-14f, %16-18f, %0-3f"},
221 {FPU_FPA_EXT_V1, 0x0ec00100, 0x0ff08f10, "pol%c%P%R\t%12-14f, %16-18f, %0-3f"},
222 {FPU_FPA_EXT_V1, 0x0e008100, 0x0ff08f10, "mvf%c%P%R\t%12-14f, %0-3f"},
223 {FPU_FPA_EXT_V1, 0x0e108100, 0x0ff08f10, "mnf%c%P%R\t%12-14f, %0-3f"},
224 {FPU_FPA_EXT_V1, 0x0e208100, 0x0ff08f10, "abs%c%P%R\t%12-14f, %0-3f"},
225 {FPU_FPA_EXT_V1, 0x0e308100, 0x0ff08f10, "rnd%c%P%R\t%12-14f, %0-3f"},
226 {FPU_FPA_EXT_V1, 0x0e408100, 0x0ff08f10, "sqt%c%P%R\t%12-14f, %0-3f"},
227 {FPU_FPA_EXT_V1, 0x0e508100, 0x0ff08f10, "log%c%P%R\t%12-14f, %0-3f"},
228 {FPU_FPA_EXT_V1, 0x0e608100, 0x0ff08f10, "lgn%c%P%R\t%12-14f, %0-3f"},
229 {FPU_FPA_EXT_V1, 0x0e708100, 0x0ff08f10, "exp%c%P%R\t%12-14f, %0-3f"},
230 {FPU_FPA_EXT_V1, 0x0e808100, 0x0ff08f10, "sin%c%P%R\t%12-14f, %0-3f"},
231 {FPU_FPA_EXT_V1, 0x0e908100, 0x0ff08f10, "cos%c%P%R\t%12-14f, %0-3f"},
232 {FPU_FPA_EXT_V1, 0x0ea08100, 0x0ff08f10, "tan%c%P%R\t%12-14f, %0-3f"},
233 {FPU_FPA_EXT_V1, 0x0eb08100, 0x0ff08f10, "asn%c%P%R\t%12-14f, %0-3f"},
234 {FPU_FPA_EXT_V1, 0x0ec08100, 0x0ff08f10, "acs%c%P%R\t%12-14f, %0-3f"},
235 {FPU_FPA_EXT_V1, 0x0ed08100, 0x0ff08f10, "atn%c%P%R\t%12-14f, %0-3f"},
236 {FPU_FPA_EXT_V1, 0x0ee08100, 0x0ff08f10, "urd%c%P%R\t%12-14f, %0-3f"},
237 {FPU_FPA_EXT_V1, 0x0ef08100, 0x0ff08f10, "nrm%c%P%R\t%12-14f, %0-3f"},
238 {FPU_FPA_EXT_V1, 0x0e000110, 0x0ff00f1f, "flt%c%P%R\t%16-18f, %12-15r"},
239 {FPU_FPA_EXT_V1, 0x0e100110, 0x0fff0f98, "fix%c%R\t%12-15r, %0-2f"},
240 {FPU_FPA_EXT_V1, 0x0e200110, 0x0fff0fff, "wfs%c\t%12-15r"},
241 {FPU_FPA_EXT_V1, 0x0e300110, 0x0fff0fff, "rfs%c\t%12-15r"},
242 {FPU_FPA_EXT_V1, 0x0e400110, 0x0fff0fff, "wfc%c\t%12-15r"},
243 {FPU_FPA_EXT_V1, 0x0e500110, 0x0fff0fff, "rfc%c\t%12-15r"},
244 {FPU_FPA_EXT_V1, 0x0e90f110, 0x0ff8fff0, "cmf%c\t%16-18f, %0-3f"},
245 {FPU_FPA_EXT_V1, 0x0eb0f110, 0x0ff8fff0, "cnf%c\t%16-18f, %0-3f"},
246 {FPU_FPA_EXT_V1, 0x0ed0f110, 0x0ff8fff0, "cmfe%c\t%16-18f, %0-3f"},
247 {FPU_FPA_EXT_V1, 0x0ef0f110, 0x0ff8fff0, "cnfe%c\t%16-18f, %0-3f"},
248 {FPU_FPA_EXT_V1, 0x0c000100, 0x0e100f00, "stf%c%Q\t%12-14f, %A"},
249 {FPU_FPA_EXT_V1, 0x0c100100, 0x0e100f00, "ldf%c%Q\t%12-14f, %A"},
250 {FPU_FPA_EXT_V2, 0x0c000200, 0x0e100f00, "sfm%c\t%12-14f, %F, %A\t; (stc%22'l%c %8-11d, cr%12-15d, %A)"},
251 {FPU_FPA_EXT_V2, 0x0c100200, 0x0e100f00, "lfm%c\t%12-14f, %F, %A\t; (ldc%22'l%c %8-11d, cr%12-15d, %A)"},
253 /* Register load/store. */
254 {FPU_VFP_EXT_V1xD | FPU_NEON_EXT_V1, 0x0d2d0b00, 0x0fbf0f01, "vpush%c\t%B"},
255 {FPU_VFP_EXT_V1xD | FPU_NEON_EXT_V1, 0x0d200b00, 0x0fb00f01, "vstmdb%c\t%16-19r!, %B"},
256 {FPU_VFP_EXT_V1xD | FPU_NEON_EXT_V1, 0x0d300b00, 0x0fb00f01, "vldmdb%c\t%16-19r!, %B"},
257 {FPU_VFP_EXT_V1xD | FPU_NEON_EXT_V1, 0x0c800b00, 0x0f900f01, "vstmia%c\t%16-19r%21'!, %B"},
258 {FPU_VFP_EXT_V1xD | FPU_NEON_EXT_V1, 0x0cbd0b00, 0x0fbf0f01, "vpop%c\t%B"},
259 {FPU_VFP_EXT_V1xD | FPU_NEON_EXT_V1, 0x0c900b00, 0x0f900f01, "vldmia%c\t%16-19r%21'!, %B"},
260 {FPU_VFP_EXT_V1xD | FPU_NEON_EXT_V1, 0x0d000b00, 0x0f300f00, "vstr%c\t%12-15,22D, %C"},
261 {FPU_VFP_EXT_V1xD | FPU_NEON_EXT_V1, 0x0d100b00, 0x0f300f00, "vldr%c\t%12-15,22D, %C"},
262 {FPU_VFP_EXT_V1xD, 0x0d2d0a00, 0x0fbf0f00, "vpush%c\t%y3"},
263 {FPU_VFP_EXT_V1xD, 0x0d200a00, 0x0fb00f00, "vstmdb%c\t%16-19r!, %y3"},
264 {FPU_VFP_EXT_V1xD, 0x0d300a00, 0x0fb00f00, "vldmdb%c\t%16-19r!, %y3"},
265 {FPU_VFP_EXT_V1xD, 0x0c800a00, 0x0f900f00, "vstmia%c\t%16-19r%21'!, %y3"},
266 {FPU_VFP_EXT_V1xD, 0x0cbd0a00, 0x0fbf0f00, "vpop%c\t%y3"},
267 {FPU_VFP_EXT_V1xD, 0x0c900a00, 0x0f900f00, "vldmia%c\t%16-19r%21'!, %y3"},
268 {FPU_VFP_EXT_V1xD, 0x0d000a00, 0x0f300f00, "vstr%c\t%y1, %A"},
269 {FPU_VFP_EXT_V1xD, 0x0d100a00, 0x0f300f00, "vldr%c\t%y1, %A"},
271 {FPU_VFP_EXT_V1xD, 0x0d200b01, 0x0fb00f01, "fstmdbx%c\t%16-19r!, %z3\t;@ Deprecated"},
272 {FPU_VFP_EXT_V1xD, 0x0d300b01, 0x0fb00f01, "fldmdbx%c\t%16-19r!, %z3\t;@ Deprecated"},
273 {FPU_VFP_EXT_V1xD, 0x0c800b01, 0x0f900f01, "fstmiax%c\t%16-19r%21'!, %z3\t;@ Deprecated"},
274 {FPU_VFP_EXT_V1xD, 0x0c900b01, 0x0f900f01, "fldmiax%c\t%16-19r%21'!, %z3\t;@ Deprecated"},
276 /* Data transfer between ARM and NEON registers. */
277 {FPU_NEON_EXT_V1, 0x0e800b10, 0x0ff00f70, "vdup%c.32\t%16-19,7D, %12-15r"},
278 {FPU_NEON_EXT_V1, 0x0e800b30, 0x0ff00f70, "vdup%c.16\t%16-19,7D, %12-15r"},
279 {FPU_NEON_EXT_V1, 0x0ea00b10, 0x0ff00f70, "vdup%c.32\t%16-19,7Q, %12-15r"},
280 {FPU_NEON_EXT_V1, 0x0ea00b30, 0x0ff00f70, "vdup%c.16\t%16-19,7Q, %12-15r"},
281 {FPU_NEON_EXT_V1, 0x0ec00b10, 0x0ff00f70, "vdup%c.8\t%16-19,7D, %12-15r"},
282 {FPU_NEON_EXT_V1, 0x0ee00b10, 0x0ff00f70, "vdup%c.8\t%16-19,7Q, %12-15r"},
283 {FPU_NEON_EXT_V1, 0x0c400b10, 0x0ff00fd0, "vmov%c\t%0-3,5D, %12-15r, %16-19r"},
284 {FPU_NEON_EXT_V1, 0x0c500b10, 0x0ff00fd0, "vmov%c\t%12-15r, %16-19r, %0-3,5D"},
285 {FPU_NEON_EXT_V1, 0x0e000b10, 0x0fd00f70, "vmov%c.32\t%16-19,7D[%21d], %12-15r"},
286 {FPU_NEON_EXT_V1, 0x0e100b10, 0x0f500f70, "vmov%c.32\t%12-15r, %16-19,7D[%21d]"},
287 {FPU_NEON_EXT_V1, 0x0e000b30, 0x0fd00f30, "vmov%c.16\t%16-19,7D[%6,21d], %12-15r"},
288 {FPU_NEON_EXT_V1, 0x0e100b30, 0x0f500f30, "vmov%c.%23?us16\t%12-15r, %16-19,7D[%6,21d]"},
289 {FPU_NEON_EXT_V1, 0x0e400b10, 0x0fd00f10, "vmov%c.8\t%16-19,7D[%5,6,21d], %12-15r"},
290 {FPU_NEON_EXT_V1, 0x0e500b10, 0x0f500f10, "vmov%c.%23?us8\t%12-15r, %16-19,7D[%5,6,21d]"},
291 /* Half-precision conversion instructions. */
292 {FPU_NEON_FP16, 0x0eb20a40, 0x0fbf0f50, "vcvt%7?tb%c.f32.f16\t%y1, %y0"},
293 {FPU_NEON_FP16, 0x0eb30a40, 0x0fbf0f50, "vcvt%7?tb%c.f16.f32\t%y1, %y0"},
295 /* Floating point coprocessor (VFP) instructions. */
296 {FPU_VFP_EXT_V1xD, 0x0ee00a10, 0x0fff0fff, "vmsr%c\tfpsid, %12-15r"},
297 {FPU_VFP_EXT_V1xD, 0x0ee10a10, 0x0fff0fff, "vmsr%c\tfpscr, %12-15r"},
298 {FPU_VFP_EXT_V1xD, 0x0ee60a10, 0x0fff0fff, "vmsr%c\tmvfr1, %12-15r"},
299 {FPU_VFP_EXT_V1xD, 0x0ee70a10, 0x0fff0fff, "vmsr%c\tmvfr0, %12-15r"},
300 {FPU_VFP_EXT_V1xD, 0x0ee80a10, 0x0fff0fff, "vmsr%c\tfpexc, %12-15r"},
301 {FPU_VFP_EXT_V1xD, 0x0ee90a10, 0x0fff0fff, "vmsr%c\tfpinst, %12-15r\t@ Impl def"},
302 {FPU_VFP_EXT_V1xD, 0x0eea0a10, 0x0fff0fff, "vmsr%c\tfpinst2, %12-15r\t@ Impl def"},
303 {FPU_VFP_EXT_V1xD, 0x0ef00a10, 0x0fff0fff, "vmrs%c\t%12-15r, fpsid"},
304 {FPU_VFP_EXT_V1xD, 0x0ef1fa10, 0x0fffffff, "vmrs%c\tAPSR_nzcv, fpscr"},
305 {FPU_VFP_EXT_V1xD, 0x0ef10a10, 0x0fff0fff, "vmrs%c\t%12-15r, fpscr"},
306 {FPU_VFP_EXT_V1xD, 0x0ef60a10, 0x0fff0fff, "vmrs%c\t%12-15r, mvfr1"},
307 {FPU_VFP_EXT_V1xD, 0x0ef70a10, 0x0fff0fff, "vmrs%c\t%12-15r, mvfr0"},
308 {FPU_VFP_EXT_V1xD, 0x0ef80a10, 0x0fff0fff, "vmrs%c\t%12-15r, fpexc"},
309 {FPU_VFP_EXT_V1xD, 0x0ef90a10, 0x0fff0fff, "vmrs%c\t%12-15r, fpinst\t@ Impl def"},
310 {FPU_VFP_EXT_V1xD, 0x0efa0a10, 0x0fff0fff, "vmrs%c\t%12-15r, fpinst2\t@ Impl def"},
311 {FPU_VFP_EXT_V1, 0x0e000b10, 0x0fd00fff, "vmov%c.32\t%z2[%21d], %12-15r"},
312 {FPU_VFP_EXT_V1, 0x0e100b10, 0x0fd00fff, "vmov%c.32\t%12-15r, %z2[%21d]"},
313 {FPU_VFP_EXT_V1xD, 0x0ee00a10, 0x0ff00fff, "vmsr%c\t<impl def %16-19x>, %12-15r"},
314 {FPU_VFP_EXT_V1xD, 0x0ef00a10, 0x0ff00fff, "vmrs%c\t%12-15r, <impl def %16-19x>"},
315 {FPU_VFP_EXT_V1xD, 0x0e000a10, 0x0ff00f7f, "vmov%c\t%y2, %12-15r"},
316 {FPU_VFP_EXT_V1xD, 0x0e100a10, 0x0ff00f7f, "vmov%c\t%12-15r, %y2"},
317 {FPU_VFP_EXT_V1xD, 0x0eb50a40, 0x0fbf0f70, "vcmp%7'e%c.f32\t%y1, #0.0"},
318 {FPU_VFP_EXT_V1, 0x0eb50b40, 0x0fbf0f70, "vcmp%7'e%c.f64\t%z1, #0.0"},
319 {FPU_VFP_EXT_V1xD, 0x0eb00a40, 0x0fbf0fd0, "vmov%c.f32\t%y1, %y0"},
320 {FPU_VFP_EXT_V1xD, 0x0eb00ac0, 0x0fbf0fd0, "vabs%c.f32\t%y1, %y0"},
321 {FPU_VFP_EXT_V1, 0x0eb00b40, 0x0fbf0fd0, "vmov%c.f64\t%z1, %z0"},
322 {FPU_VFP_EXT_V1, 0x0eb00bc0, 0x0fbf0fd0, "vabs%c.f64\t%z1, %z0"},
323 {FPU_VFP_EXT_V1xD, 0x0eb10a40, 0x0fbf0fd0, "vneg%c.f32\t%y1, %y0"},
324 {FPU_VFP_EXT_V1xD, 0x0eb10ac0, 0x0fbf0fd0, "vsqrt%c.f32\t%y1, %y0"},
325 {FPU_VFP_EXT_V1, 0x0eb10b40, 0x0fbf0fd0, "vneg%c.f64\t%z1, %z0"},
326 {FPU_VFP_EXT_V1, 0x0eb10bc0, 0x0fbf0fd0, "vsqrt%c.f64\t%z1, %z0"},
327 {FPU_VFP_EXT_V1, 0x0eb70ac0, 0x0fbf0fd0, "vcvt%c.f64.f32\t%z1, %y0"},
328 {FPU_VFP_EXT_V1, 0x0eb70bc0, 0x0fbf0fd0, "vcvt%c.f32.f64\t%y1, %z0"},
329 {FPU_VFP_EXT_V1xD, 0x0eb80a40, 0x0fbf0f50, "vcvt%c.f32.%7?su32\t%y1, %y0"},
330 {FPU_VFP_EXT_V1, 0x0eb80b40, 0x0fbf0f50, "vcvt%c.f64.%7?su32\t%z1, %y0"},
331 {FPU_VFP_EXT_V1xD, 0x0eb40a40, 0x0fbf0f50, "vcmp%7'e%c.f32\t%y1, %y0"},
332 {FPU_VFP_EXT_V1, 0x0eb40b40, 0x0fbf0f50, "vcmp%7'e%c.f64\t%z1, %z0"},
333 {FPU_VFP_EXT_V3, 0x0eba0a40, 0x0fbe0f50, "vcvt%c.f32.%16?us%7?31%7?26\t%y1, %y1, #%5,0-3k"},
334 {FPU_VFP_EXT_V3, 0x0eba0b40, 0x0fbe0f50, "vcvt%c.f64.%16?us%7?31%7?26\t%z1, %z1, #%5,0-3k"},
335 {FPU_VFP_EXT_V1xD, 0x0ebc0a40, 0x0fbe0f50, "vcvt%7`r%c.%16?su32.f32\t%y1, %y0"},
336 {FPU_VFP_EXT_V1, 0x0ebc0b40, 0x0fbe0f50, "vcvt%7`r%c.%16?su32.f64\t%y1, %z0"},
337 {FPU_VFP_EXT_V3, 0x0ebe0a40, 0x0fbe0f50, "vcvt%c.%16?us%7?31%7?26.f32\t%y1, %y1, #%5,0-3k"},
338 {FPU_VFP_EXT_V3, 0x0ebe0b40, 0x0fbe0f50, "vcvt%c.%16?us%7?31%7?26.f64\t%z1, %z1, #%5,0-3k"},
339 {FPU_VFP_EXT_V1, 0x0c500b10, 0x0fb00ff0, "vmov%c\t%12-15r, %16-19r, %z0"},
340 {FPU_VFP_EXT_V3, 0x0eb00a00, 0x0fb00ff0, "vmov%c.f32\t%y1, #%0-3,16-19d"},
341 {FPU_VFP_EXT_V3, 0x0eb00b00, 0x0fb00ff0, "vmov%c.f64\t%z1, #%0-3,16-19d"},
342 {FPU_VFP_EXT_V2, 0x0c400a10, 0x0ff00fd0, "vmov%c\t%y4, %12-15r, %16-19r"},
343 {FPU_VFP_EXT_V2, 0x0c400b10, 0x0ff00fd0, "vmov%c\t%z0, %12-15r, %16-19r"},
344 {FPU_VFP_EXT_V2, 0x0c500a10, 0x0ff00fd0, "vmov%c\t%12-15r, %16-19r, %y4"},
345 {FPU_VFP_EXT_V1xD, 0x0e000a00, 0x0fb00f50, "vmla%c.f32\t%y1, %y2, %y0"},
346 {FPU_VFP_EXT_V1xD, 0x0e000a40, 0x0fb00f50, "vmls%c.f32\t%y1, %y2, %y0"},
347 {FPU_VFP_EXT_V1, 0x0e000b00, 0x0fb00f50, "vmla%c.f64\t%z1, %z2, %z0"},
348 {FPU_VFP_EXT_V1, 0x0e000b40, 0x0fb00f50, "vmls%c.f64\t%z1, %z2, %z0"},
349 {FPU_VFP_EXT_V1xD, 0x0e100a00, 0x0fb00f50, "vnmls%c.f32\t%y1, %y2, %y0"},
350 {FPU_VFP_EXT_V1xD, 0x0e100a40, 0x0fb00f50, "vnmla%c.f32\t%y1, %y2, %y0"},
351 {FPU_VFP_EXT_V1, 0x0e100b00, 0x0fb00f50, "vnmls%c.f64\t%z1, %z2, %z0"},
352 {FPU_VFP_EXT_V1, 0x0e100b40, 0x0fb00f50, "vnmla%c.f64\t%z1, %z2, %z0"},
353 {FPU_VFP_EXT_V1xD, 0x0e200a00, 0x0fb00f50, "vmul%c.f32\t%y1, %y2, %y0"},
354 {FPU_VFP_EXT_V1xD, 0x0e200a40, 0x0fb00f50, "vnmul%c.f32\t%y1, %y2, %y0"},
355 {FPU_VFP_EXT_V1, 0x0e200b00, 0x0fb00f50, "vmul%c.f64\t%z1, %z2, %z0"},
356 {FPU_VFP_EXT_V1, 0x0e200b40, 0x0fb00f50, "vnmul%c.f64\t%z1, %z2, %z0"},
357 {FPU_VFP_EXT_V1xD, 0x0e300a00, 0x0fb00f50, "vadd%c.f32\t%y1, %y2, %y0"},
358 {FPU_VFP_EXT_V1xD, 0x0e300a40, 0x0fb00f50, "vsub%c.f32\t%y1, %y2, %y0"},
359 {FPU_VFP_EXT_V1, 0x0e300b00, 0x0fb00f50, "vadd%c.f64\t%z1, %z2, %z0"},
360 {FPU_VFP_EXT_V1, 0x0e300b40, 0x0fb00f50, "vsub%c.f64\t%z1, %z2, %z0"},
361 {FPU_VFP_EXT_V1xD, 0x0e800a00, 0x0fb00f50, "vdiv%c.f32\t%y1, %y2, %y0"},
362 {FPU_VFP_EXT_V1, 0x0e800b00, 0x0fb00f50, "vdiv%c.f64\t%z1, %z2, %z0"},
364 /* Cirrus coprocessor instructions. */
365 {ARM_CEXT_MAVERICK, 0x0d100400, 0x0f500f00, "cfldrs%c\tmvf%12-15d, %A"},
366 {ARM_CEXT_MAVERICK, 0x0c100400, 0x0f500f00, "cfldrs%c\tmvf%12-15d, %A"},
367 {ARM_CEXT_MAVERICK, 0x0d500400, 0x0f500f00, "cfldrd%c\tmvd%12-15d, %A"},
368 {ARM_CEXT_MAVERICK, 0x0c500400, 0x0f500f00, "cfldrd%c\tmvd%12-15d, %A"},
369 {ARM_CEXT_MAVERICK, 0x0d100500, 0x0f500f00, "cfldr32%c\tmvfx%12-15d, %A"},
370 {ARM_CEXT_MAVERICK, 0x0c100500, 0x0f500f00, "cfldr32%c\tmvfx%12-15d, %A"},
371 {ARM_CEXT_MAVERICK, 0x0d500500, 0x0f500f00, "cfldr64%c\tmvdx%12-15d, %A"},
372 {ARM_CEXT_MAVERICK, 0x0c500500, 0x0f500f00, "cfldr64%c\tmvdx%12-15d, %A"},
373 {ARM_CEXT_MAVERICK, 0x0d000400, 0x0f500f00, "cfstrs%c\tmvf%12-15d, %A"},
374 {ARM_CEXT_MAVERICK, 0x0c000400, 0x0f500f00, "cfstrs%c\tmvf%12-15d, %A"},
375 {ARM_CEXT_MAVERICK, 0x0d400400, 0x0f500f00, "cfstrd%c\tmvd%12-15d, %A"},
376 {ARM_CEXT_MAVERICK, 0x0c400400, 0x0f500f00, "cfstrd%c\tmvd%12-15d, %A"},
377 {ARM_CEXT_MAVERICK, 0x0d000500, 0x0f500f00, "cfstr32%c\tmvfx%12-15d, %A"},
378 {ARM_CEXT_MAVERICK, 0x0c000500, 0x0f500f00, "cfstr32%c\tmvfx%12-15d, %A"},
379 {ARM_CEXT_MAVERICK, 0x0d400500, 0x0f500f00, "cfstr64%c\tmvdx%12-15d, %A"},
380 {ARM_CEXT_MAVERICK, 0x0c400500, 0x0f500f00, "cfstr64%c\tmvdx%12-15d, %A"},
381 {ARM_CEXT_MAVERICK, 0x0e000450, 0x0ff00ff0, "cfmvsr%c\tmvf%16-19d, %12-15r"},
382 {ARM_CEXT_MAVERICK, 0x0e100450, 0x0ff00ff0, "cfmvrs%c\t%12-15r, mvf%16-19d"},
383 {ARM_CEXT_MAVERICK, 0x0e000410, 0x0ff00ff0, "cfmvdlr%c\tmvd%16-19d, %12-15r"},
384 {ARM_CEXT_MAVERICK, 0x0e100410, 0x0ff00ff0, "cfmvrdl%c\t%12-15r, mvd%16-19d"},
385 {ARM_CEXT_MAVERICK, 0x0e000430, 0x0ff00ff0, "cfmvdhr%c\tmvd%16-19d, %12-15r"},
386 {ARM_CEXT_MAVERICK, 0x0e100430, 0x0ff00fff, "cfmvrdh%c\t%12-15r, mvd%16-19d"},
387 {ARM_CEXT_MAVERICK, 0x0e000510, 0x0ff00fff, "cfmv64lr%c\tmvdx%16-19d, %12-15r"},
388 {ARM_CEXT_MAVERICK, 0x0e100510, 0x0ff00fff, "cfmvr64l%c\t%12-15r, mvdx%16-19d"},
389 {ARM_CEXT_MAVERICK, 0x0e000530, 0x0ff00fff, "cfmv64hr%c\tmvdx%16-19d, %12-15r"},
390 {ARM_CEXT_MAVERICK, 0x0e100530, 0x0ff00fff, "cfmvr64h%c\t%12-15r, mvdx%16-19d"},
391 {ARM_CEXT_MAVERICK, 0x0e200440, 0x0ff00fff, "cfmval32%c\tmvax%12-15d, mvfx%16-19d"},
392 {ARM_CEXT_MAVERICK, 0x0e100440, 0x0ff00fff, "cfmv32al%c\tmvfx%12-15d, mvax%16-19d"},
393 {ARM_CEXT_MAVERICK, 0x0e200460, 0x0ff00fff, "cfmvam32%c\tmvax%12-15d, mvfx%16-19d"},
394 {ARM_CEXT_MAVERICK, 0x0e100460, 0x0ff00fff, "cfmv32am%c\tmvfx%12-15d, mvax%16-19d"},
395 {ARM_CEXT_MAVERICK, 0x0e200480, 0x0ff00fff, "cfmvah32%c\tmvax%12-15d, mvfx%16-19d"},
396 {ARM_CEXT_MAVERICK, 0x0e100480, 0x0ff00fff, "cfmv32ah%c\tmvfx%12-15d, mvax%16-19d"},
397 {ARM_CEXT_MAVERICK, 0x0e2004a0, 0x0ff00fff, "cfmva32%c\tmvax%12-15d, mvfx%16-19d"},
398 {ARM_CEXT_MAVERICK, 0x0e1004a0, 0x0ff00fff, "cfmv32a%c\tmvfx%12-15d, mvax%16-19d"},
399 {ARM_CEXT_MAVERICK, 0x0e2004c0, 0x0ff00fff, "cfmva64%c\tmvax%12-15d, mvdx%16-19d"},
400 {ARM_CEXT_MAVERICK, 0x0e1004c0, 0x0ff00fff, "cfmv64a%c\tmvdx%12-15d, mvax%16-19d"},
401 {ARM_CEXT_MAVERICK, 0x0e2004e0, 0x0fff0fff, "cfmvsc32%c\tdspsc, mvdx%12-15d"},
402 {ARM_CEXT_MAVERICK, 0x0e1004e0, 0x0fff0fff, "cfmv32sc%c\tmvdx%12-15d, dspsc"},
403 {ARM_CEXT_MAVERICK, 0x0e000400, 0x0ff00fff, "cfcpys%c\tmvf%12-15d, mvf%16-19d"},
404 {ARM_CEXT_MAVERICK, 0x0e000420, 0x0ff00fff, "cfcpyd%c\tmvd%12-15d, mvd%16-19d"},
405 {ARM_CEXT_MAVERICK, 0x0e000460, 0x0ff00fff, "cfcvtsd%c\tmvd%12-15d, mvf%16-19d"},
406 {ARM_CEXT_MAVERICK, 0x0e000440, 0x0ff00fff, "cfcvtds%c\tmvf%12-15d, mvd%16-19d"},
407 {ARM_CEXT_MAVERICK, 0x0e000480, 0x0ff00fff, "cfcvt32s%c\tmvf%12-15d, mvfx%16-19d"},
408 {ARM_CEXT_MAVERICK, 0x0e0004a0, 0x0ff00fff, "cfcvt32d%c\tmvd%12-15d, mvfx%16-19d"},
409 {ARM_CEXT_MAVERICK, 0x0e0004c0, 0x0ff00fff, "cfcvt64s%c\tmvf%12-15d, mvdx%16-19d"},
410 {ARM_CEXT_MAVERICK, 0x0e0004e0, 0x0ff00fff, "cfcvt64d%c\tmvd%12-15d, mvdx%16-19d"},
411 {ARM_CEXT_MAVERICK, 0x0e100580, 0x0ff00fff, "cfcvts32%c\tmvfx%12-15d, mvf%16-19d"},
412 {ARM_CEXT_MAVERICK, 0x0e1005a0, 0x0ff00fff, "cfcvtd32%c\tmvfx%12-15d, mvd%16-19d"},
413 {ARM_CEXT_MAVERICK, 0x0e1005c0, 0x0ff00fff, "cftruncs32%c\tmvfx%12-15d, mvf%16-19d"},
414 {ARM_CEXT_MAVERICK, 0x0e1005e0, 0x0ff00fff, "cftruncd32%c\tmvfx%12-15d, mvd%16-19d"},
415 {ARM_CEXT_MAVERICK, 0x0e000550, 0x0ff00ff0, "cfrshl32%c\tmvfx%16-19d, mvfx%0-3d, %12-15r"},
416 {ARM_CEXT_MAVERICK, 0x0e000570, 0x0ff00ff0, "cfrshl64%c\tmvdx%16-19d, mvdx%0-3d, %12-15r"},
417 {ARM_CEXT_MAVERICK, 0x0e000500, 0x0ff00f10, "cfsh32%c\tmvfx%12-15d, mvfx%16-19d, #%I"},
418 {ARM_CEXT_MAVERICK, 0x0e200500, 0x0ff00f10, "cfsh64%c\tmvdx%12-15d, mvdx%16-19d, #%I"},
419 {ARM_CEXT_MAVERICK, 0x0e100490, 0x0ff00ff0, "cfcmps%c\t%12-15r, mvf%16-19d, mvf%0-3d"},
420 {ARM_CEXT_MAVERICK, 0x0e1004b0, 0x0ff00ff0, "cfcmpd%c\t%12-15r, mvd%16-19d, mvd%0-3d"},
421 {ARM_CEXT_MAVERICK, 0x0e100590, 0x0ff00ff0, "cfcmp32%c\t%12-15r, mvfx%16-19d, mvfx%0-3d"},
422 {ARM_CEXT_MAVERICK, 0x0e1005b0, 0x0ff00ff0, "cfcmp64%c\t%12-15r, mvdx%16-19d, mvdx%0-3d"},
423 {ARM_CEXT_MAVERICK, 0x0e300400, 0x0ff00fff, "cfabss%c\tmvf%12-15d, mvf%16-19d"},
424 {ARM_CEXT_MAVERICK, 0x0e300420, 0x0ff00fff, "cfabsd%c\tmvd%12-15d, mvd%16-19d"},
425 {ARM_CEXT_MAVERICK, 0x0e300440, 0x0ff00fff, "cfnegs%c\tmvf%12-15d, mvf%16-19d"},
426 {ARM_CEXT_MAVERICK, 0x0e300460, 0x0ff00fff, "cfnegd%c\tmvd%12-15d, mvd%16-19d"},
427 {ARM_CEXT_MAVERICK, 0x0e300480, 0x0ff00ff0, "cfadds%c\tmvf%12-15d, mvf%16-19d, mvf%0-3d"},
428 {ARM_CEXT_MAVERICK, 0x0e3004a0, 0x0ff00ff0, "cfaddd%c\tmvd%12-15d, mvd%16-19d, mvd%0-3d"},
429 {ARM_CEXT_MAVERICK, 0x0e3004c0, 0x0ff00ff0, "cfsubs%c\tmvf%12-15d, mvf%16-19d, mvf%0-3d"},
430 {ARM_CEXT_MAVERICK, 0x0e3004e0, 0x0ff00ff0, "cfsubd%c\tmvd%12-15d, mvd%16-19d, mvd%0-3d"},
431 {ARM_CEXT_MAVERICK, 0x0e100400, 0x0ff00ff0, "cfmuls%c\tmvf%12-15d, mvf%16-19d, mvf%0-3d"},
432 {ARM_CEXT_MAVERICK, 0x0e100420, 0x0ff00ff0, "cfmuld%c\tmvd%12-15d, mvd%16-19d, mvd%0-3d"},
433 {ARM_CEXT_MAVERICK, 0x0e300500, 0x0ff00fff, "cfabs32%c\tmvfx%12-15d, mvfx%16-19d"},
434 {ARM_CEXT_MAVERICK, 0x0e300520, 0x0ff00fff, "cfabs64%c\tmvdx%12-15d, mvdx%16-19d"},
435 {ARM_CEXT_MAVERICK, 0x0e300540, 0x0ff00fff, "cfneg32%c\tmvfx%12-15d, mvfx%16-19d"},
436 {ARM_CEXT_MAVERICK, 0x0e300560, 0x0ff00fff, "cfneg64%c\tmvdx%12-15d, mvdx%16-19d"},
437 {ARM_CEXT_MAVERICK, 0x0e300580, 0x0ff00ff0, "cfadd32%c\tmvfx%12-15d, mvfx%16-19d, mvfx%0-3d"},
438 {ARM_CEXT_MAVERICK, 0x0e3005a0, 0x0ff00ff0, "cfadd64%c\tmvdx%12-15d, mvdx%16-19d, mvdx%0-3d"},
439 {ARM_CEXT_MAVERICK, 0x0e3005c0, 0x0ff00ff0, "cfsub32%c\tmvfx%12-15d, mvfx%16-19d, mvfx%0-3d"},
440 {ARM_CEXT_MAVERICK, 0x0e3005e0, 0x0ff00ff0, "cfsub64%c\tmvdx%12-15d, mvdx%16-19d, mvdx%0-3d"},
441 {ARM_CEXT_MAVERICK, 0x0e100500, 0x0ff00ff0, "cfmul32%c\tmvfx%12-15d, mvfx%16-19d, mvfx%0-3d"},
442 {ARM_CEXT_MAVERICK, 0x0e100520, 0x0ff00ff0, "cfmul64%c\tmvdx%12-15d, mvdx%16-19d, mvdx%0-3d"},
443 {ARM_CEXT_MAVERICK, 0x0e100540, 0x0ff00ff0, "cfmac32%c\tmvfx%12-15d, mvfx%16-19d, mvfx%0-3d"},
444 {ARM_CEXT_MAVERICK, 0x0e100560, 0x0ff00ff0, "cfmsc32%c\tmvfx%12-15d, mvfx%16-19d, mvfx%0-3d"},
445 {ARM_CEXT_MAVERICK, 0x0e000600, 0x0ff00f10, "cfmadd32%c\tmvax%5-7d, mvfx%12-15d, mvfx%16-19d, mvfx%0-3d"},
446 {ARM_CEXT_MAVERICK, 0x0e100600, 0x0ff00f10, "cfmsub32%c\tmvax%5-7d, mvfx%12-15d, mvfx%16-19d, mvfx%0-3d"},
447 {ARM_CEXT_MAVERICK, 0x0e200600, 0x0ff00f10, "cfmadda32%c\tmvax%5-7d, mvax%12-15d, mvfx%16-19d, mvfx%0-3d"},
448 {ARM_CEXT_MAVERICK, 0x0e300600, 0x0ff00f10, "cfmsuba32%c\tmvax%5-7d, mvax%12-15d, mvfx%16-19d, mvfx%0-3d"},
450 /* Generic coprocessor instructions. */
451 { 0, SENTINEL_GENERIC_START, 0, "" },
452 {ARM_EXT_V5E, 0x0c400000, 0x0ff00000, "mcrr%c\t%8-11d, %4-7d, %12-15r, %16-19r, cr%0-3d"},
453 {ARM_EXT_V5E, 0x0c500000, 0x0ff00000, "mrrc%c\t%8-11d, %4-7d, %12-15r, %16-19r, cr%0-3d"},
454 {ARM_EXT_V2, 0x0e000000, 0x0f000010, "cdp%c\t%8-11d, %20-23d, cr%12-15d, cr%16-19d, cr%0-3d, {%5-7d}"},
455 {ARM_EXT_V2, 0x0e100010, 0x0f100010, "mrc%c\t%8-11d, %21-23d, %12-15r, cr%16-19d, cr%0-3d, {%5-7d}"},
456 {ARM_EXT_V2, 0x0e000010, 0x0f100010, "mcr%c\t%8-11d, %21-23d, %12-15r, cr%16-19d, cr%0-3d, {%5-7d}"},
457 {ARM_EXT_V2, 0x0c000000, 0x0e100000, "stc%22'l%c\t%8-11d, cr%12-15d, %A"},
458 {ARM_EXT_V2, 0x0c100000, 0x0e100000, "ldc%22'l%c\t%8-11d, cr%12-15d, %A"},
460 /* V6 coprocessor instructions. */
461 {ARM_EXT_V6, 0xfc500000, 0xfff00000, "mrrc2%c\t%8-11d, %4-7d, %12-15r, %16-19r, cr%0-3d"},
462 {ARM_EXT_V6, 0xfc400000, 0xfff00000, "mcrr2%c\t%8-11d, %4-7d, %12-15r, %16-19r, cr%0-3d"},
464 /* V5 coprocessor instructions. */
465 {ARM_EXT_V5, 0xfc100000, 0xfe100000, "ldc2%22'l%c\t%8-11d, cr%12-15d, %A"},
466 {ARM_EXT_V5, 0xfc000000, 0xfe100000, "stc2%22'l%c\t%8-11d, cr%12-15d, %A"},
467 {ARM_EXT_V5, 0xfe000000, 0xff000010, "cdp2%c\t%8-11d, %20-23d, cr%12-15d, cr%16-19d, cr%0-3d, {%5-7d}"},
468 {ARM_EXT_V5, 0xfe000010, 0xff100010, "mcr2%c\t%8-11d, %21-23d, %12-15r, cr%16-19d, cr%0-3d, {%5-7d}"},
469 {ARM_EXT_V5, 0xfe100010, 0xff100010, "mrc2%c\t%8-11d, %21-23d, %12-15r, cr%16-19d, cr%0-3d, {%5-7d}"},
471 {0, 0, 0, 0}
474 /* Neon opcode table: This does not encode the top byte -- that is
475 checked by the print_insn_neon routine, as it depends on whether we are
476 doing thumb32 or arm32 disassembly. */
478 /* print_insn_neon recognizes the following format control codes:
480 %% %
482 %c print condition code
483 %A print v{st,ld}[1234] operands
484 %B print v{st,ld}[1234] any one operands
485 %C print v{st,ld}[1234] single->all operands
486 %D print scalar
487 %E print vmov, vmvn, vorr, vbic encoded constant
488 %F print vtbl,vtbx register list
490 %<bitfield>r print as an ARM register
491 %<bitfield>d print the bitfield in decimal
492 %<bitfield>e print the 2^N - bitfield in decimal
493 %<bitfield>D print as a NEON D register
494 %<bitfield>Q print as a NEON Q register
495 %<bitfield>R print as a NEON D or Q register
496 %<bitfield>Sn print byte scaled width limited by n
497 %<bitfield>Tn print short scaled width limited by n
498 %<bitfield>Un print long scaled width limited by n
500 %<bitfield>'c print specified char iff bitfield is all ones
501 %<bitfield>`c print specified char iff bitfield is all zeroes
502 %<bitfield>?ab... select from array of values in big endian order. */
504 static const struct opcode32 neon_opcodes[] =
506 /* Extract. */
507 {FPU_NEON_EXT_V1, 0xf2b00840, 0xffb00850, "vext%c.8\t%12-15,22R, %16-19,7R, %0-3,5R, #%8-11d"},
508 {FPU_NEON_EXT_V1, 0xf2b00000, 0xffb00810, "vext%c.8\t%12-15,22R, %16-19,7R, %0-3,5R, #%8-11d"},
510 /* Move data element to all lanes. */
511 {FPU_NEON_EXT_V1, 0xf3b40c00, 0xffb70f90, "vdup%c.32\t%12-15,22R, %0-3,5D[%19d]"},
512 {FPU_NEON_EXT_V1, 0xf3b20c00, 0xffb30f90, "vdup%c.16\t%12-15,22R, %0-3,5D[%18-19d]"},
513 {FPU_NEON_EXT_V1, 0xf3b10c00, 0xffb10f90, "vdup%c.8\t%12-15,22R, %0-3,5D[%17-19d]"},
515 /* Table lookup. */
516 {FPU_NEON_EXT_V1, 0xf3b00800, 0xffb00c50, "vtbl%c.8\t%12-15,22D, %F, %0-3,5D"},
517 {FPU_NEON_EXT_V1, 0xf3b00840, 0xffb00c50, "vtbx%c.8\t%12-15,22D, %F, %0-3,5D"},
519 /* Half-precision conversions. */
520 {FPU_NEON_FP16, 0xf3b60600, 0xffbf0fd0, "vcvt%c.f16.f32\t%12-15,22D, %0-3,5Q"},
521 {FPU_NEON_FP16, 0xf3b60700, 0xffbf0fd0, "vcvt%c.f32.f16\t%12-15,22Q, %0-3,5D"},
523 /* Two registers, miscellaneous. */
524 {FPU_NEON_EXT_V1, 0xf2880a10, 0xfebf0fd0, "vmovl%c.%24?us8\t%12-15,22Q, %0-3,5D"},
525 {FPU_NEON_EXT_V1, 0xf2900a10, 0xfebf0fd0, "vmovl%c.%24?us16\t%12-15,22Q, %0-3,5D"},
526 {FPU_NEON_EXT_V1, 0xf2a00a10, 0xfebf0fd0, "vmovl%c.%24?us32\t%12-15,22Q, %0-3,5D"},
527 {FPU_NEON_EXT_V1, 0xf3b00500, 0xffbf0f90, "vcnt%c.8\t%12-15,22R, %0-3,5R"},
528 {FPU_NEON_EXT_V1, 0xf3b00580, 0xffbf0f90, "vmvn%c\t%12-15,22R, %0-3,5R"},
529 {FPU_NEON_EXT_V1, 0xf3b20000, 0xffbf0f90, "vswp%c\t%12-15,22R, %0-3,5R"},
530 {FPU_NEON_EXT_V1, 0xf3b20200, 0xffb30fd0, "vmovn%c.i%18-19T2\t%12-15,22D, %0-3,5Q"},
531 {FPU_NEON_EXT_V1, 0xf3b20240, 0xffb30fd0, "vqmovun%c.s%18-19T2\t%12-15,22D, %0-3,5Q"},
532 {FPU_NEON_EXT_V1, 0xf3b20280, 0xffb30fd0, "vqmovn%c.s%18-19T2\t%12-15,22D, %0-3,5Q"},
533 {FPU_NEON_EXT_V1, 0xf3b202c0, 0xffb30fd0, "vqmovn%c.u%18-19T2\t%12-15,22D, %0-3,5Q"},
534 {FPU_NEON_EXT_V1, 0xf3b20300, 0xffb30fd0, "vshll%c.i%18-19S2\t%12-15,22Q, %0-3,5D, #%18-19S2"},
535 {FPU_NEON_EXT_V1, 0xf3bb0400, 0xffbf0e90, "vrecpe%c.%8?fu%18-19S2\t%12-15,22R, %0-3,5R"},
536 {FPU_NEON_EXT_V1, 0xf3bb0480, 0xffbf0e90, "vrsqrte%c.%8?fu%18-19S2\t%12-15,22R, %0-3,5R"},
537 {FPU_NEON_EXT_V1, 0xf3b00000, 0xffb30f90, "vrev64%c.%18-19S2\t%12-15,22R, %0-3,5R"},
538 {FPU_NEON_EXT_V1, 0xf3b00080, 0xffb30f90, "vrev32%c.%18-19S2\t%12-15,22R, %0-3,5R"},
539 {FPU_NEON_EXT_V1, 0xf3b00100, 0xffb30f90, "vrev16%c.%18-19S2\t%12-15,22R, %0-3,5R"},
540 {FPU_NEON_EXT_V1, 0xf3b00400, 0xffb30f90, "vcls%c.s%18-19S2\t%12-15,22R, %0-3,5R"},
541 {FPU_NEON_EXT_V1, 0xf3b00480, 0xffb30f90, "vclz%c.i%18-19S2\t%12-15,22R, %0-3,5R"},
542 {FPU_NEON_EXT_V1, 0xf3b00700, 0xffb30f90, "vqabs%c.s%18-19S2\t%12-15,22R, %0-3,5R"},
543 {FPU_NEON_EXT_V1, 0xf3b00780, 0xffb30f90, "vqneg%c.s%18-19S2\t%12-15,22R, %0-3,5R"},
544 {FPU_NEON_EXT_V1, 0xf3b20080, 0xffb30f90, "vtrn%c.%18-19S2\t%12-15,22R, %0-3,5R"},
545 {FPU_NEON_EXT_V1, 0xf3b20100, 0xffb30f90, "vuzp%c.%18-19S2\t%12-15,22R, %0-3,5R"},
546 {FPU_NEON_EXT_V1, 0xf3b20180, 0xffb30f90, "vzip%c.%18-19S2\t%12-15,22R, %0-3,5R"},
547 {FPU_NEON_EXT_V1, 0xf3b10000, 0xffb30b90, "vcgt%c.%10?fs%18-19S2\t%12-15,22R, %0-3,5R, #0"},
548 {FPU_NEON_EXT_V1, 0xf3b10080, 0xffb30b90, "vcge%c.%10?fs%18-19S2\t%12-15,22R, %0-3,5R, #0"},
549 {FPU_NEON_EXT_V1, 0xf3b10100, 0xffb30b90, "vceq%c.%10?fi%18-19S2\t%12-15,22R, %0-3,5R, #0"},
550 {FPU_NEON_EXT_V1, 0xf3b10180, 0xffb30b90, "vcle%c.%10?fs%18-19S2\t%12-15,22R, %0-3,5R, #0"},
551 {FPU_NEON_EXT_V1, 0xf3b10200, 0xffb30b90, "vclt%c.%10?fs%18-19S2\t%12-15,22R, %0-3,5R, #0"},
552 {FPU_NEON_EXT_V1, 0xf3b10300, 0xffb30b90, "vabs%c.%10?fs%18-19S2\t%12-15,22R, %0-3,5R"},
553 {FPU_NEON_EXT_V1, 0xf3b10380, 0xffb30b90, "vneg%c.%10?fs%18-19S2\t%12-15,22R, %0-3,5R"},
554 {FPU_NEON_EXT_V1, 0xf3b00200, 0xffb30f10, "vpaddl%c.%7?us%18-19S2\t%12-15,22R, %0-3,5R"},
555 {FPU_NEON_EXT_V1, 0xf3b00600, 0xffb30f10, "vpadal%c.%7?us%18-19S2\t%12-15,22R, %0-3,5R"},
556 {FPU_NEON_EXT_V1, 0xf3b30600, 0xffb30e10, "vcvt%c.%7-8?usff%18-19Sa.%7-8?ffus%18-19Sa\t%12-15,22R, %0-3,5R"},
558 /* Three registers of the same length. */
559 {FPU_NEON_EXT_V1, 0xf2000110, 0xffb00f10, "vand%c\t%12-15,22R, %16-19,7R, %0-3,5R"},
560 {FPU_NEON_EXT_V1, 0xf2100110, 0xffb00f10, "vbic%c\t%12-15,22R, %16-19,7R, %0-3,5R"},
561 {FPU_NEON_EXT_V1, 0xf2200110, 0xffb00f10, "vorr%c\t%12-15,22R, %16-19,7R, %0-3,5R"},
562 {FPU_NEON_EXT_V1, 0xf2300110, 0xffb00f10, "vorn%c\t%12-15,22R, %16-19,7R, %0-3,5R"},
563 {FPU_NEON_EXT_V1, 0xf3000110, 0xffb00f10, "veor%c\t%12-15,22R, %16-19,7R, %0-3,5R"},
564 {FPU_NEON_EXT_V1, 0xf3100110, 0xffb00f10, "vbsl%c\t%12-15,22R, %16-19,7R, %0-3,5R"},
565 {FPU_NEON_EXT_V1, 0xf3200110, 0xffb00f10, "vbit%c\t%12-15,22R, %16-19,7R, %0-3,5R"},
566 {FPU_NEON_EXT_V1, 0xf3300110, 0xffb00f10, "vbif%c\t%12-15,22R, %16-19,7R, %0-3,5R"},
567 {FPU_NEON_EXT_V1, 0xf2000d00, 0xffa00f10, "vadd%c.f%20U0\t%12-15,22R, %16-19,7R, %0-3,5R"},
568 {FPU_NEON_EXT_V1, 0xf2000d10, 0xffa00f10, "vmla%c.f%20U0\t%12-15,22R, %16-19,7R, %0-3,5R"},
569 {FPU_NEON_EXT_V1, 0xf2000e00, 0xffa00f10, "vceq%c.f%20U0\t%12-15,22R, %16-19,7R, %0-3,5R"},
570 {FPU_NEON_EXT_V1, 0xf2000f00, 0xffa00f10, "vmax%c.f%20U0\t%12-15,22R, %16-19,7R, %0-3,5R"},
571 {FPU_NEON_EXT_V1, 0xf2000f10, 0xffa00f10, "vrecps%c.f%20U0\t%12-15,22R, %16-19,7R, %0-3,5R"},
572 {FPU_NEON_EXT_V1, 0xf2200d00, 0xffa00f10, "vsub%c.f%20U0\t%12-15,22R, %16-19,7R, %0-3,5R"},
573 {FPU_NEON_EXT_V1, 0xf2200d10, 0xffa00f10, "vmls%c.f%20U0\t%12-15,22R, %16-19,7R, %0-3,5R"},
574 {FPU_NEON_EXT_V1, 0xf2200f00, 0xffa00f10, "vmin%c.f%20U0\t%12-15,22R, %16-19,7R, %0-3,5R"},
575 {FPU_NEON_EXT_V1, 0xf2200f10, 0xffa00f10, "vrsqrts%c.f%20U0\t%12-15,22R, %16-19,7R, %0-3,5R"},
576 {FPU_NEON_EXT_V1, 0xf3000d00, 0xffa00f10, "vpadd%c.f%20U0\t%12-15,22R, %16-19,7R, %0-3,5R"},
577 {FPU_NEON_EXT_V1, 0xf3000d10, 0xffa00f10, "vmul%c.f%20U0\t%12-15,22R, %16-19,7R, %0-3,5R"},
578 {FPU_NEON_EXT_V1, 0xf3000e00, 0xffa00f10, "vcge%c.f%20U0\t%12-15,22R, %16-19,7R, %0-3,5R"},
579 {FPU_NEON_EXT_V1, 0xf3000e10, 0xffa00f10, "vacge%c.f%20U0\t%12-15,22R, %16-19,7R, %0-3,5R"},
580 {FPU_NEON_EXT_V1, 0xf3000f00, 0xffa00f10, "vpmax%c.f%20U0\t%12-15,22R, %16-19,7R, %0-3,5R"},
581 {FPU_NEON_EXT_V1, 0xf3200d00, 0xffa00f10, "vabd%c.f%20U0\t%12-15,22R, %16-19,7R, %0-3,5R"},
582 {FPU_NEON_EXT_V1, 0xf3200e00, 0xffa00f10, "vcgt%c.f%20U0\t%12-15,22R, %16-19,7R, %0-3,5R"},
583 {FPU_NEON_EXT_V1, 0xf3200e10, 0xffa00f10, "vacgt%c.f%20U0\t%12-15,22R, %16-19,7R, %0-3,5R"},
584 {FPU_NEON_EXT_V1, 0xf3200f00, 0xffa00f10, "vpmin%c.f%20U0\t%12-15,22R, %16-19,7R, %0-3,5R"},
585 {FPU_NEON_EXT_V1, 0xf2000800, 0xff800f10, "vadd%c.i%20-21S3\t%12-15,22R, %16-19,7R, %0-3,5R"},
586 {FPU_NEON_EXT_V1, 0xf2000810, 0xff800f10, "vtst%c.%20-21S2\t%12-15,22R, %16-19,7R, %0-3,5R"},
587 {FPU_NEON_EXT_V1, 0xf2000900, 0xff800f10, "vmla%c.i%20-21S2\t%12-15,22R, %16-19,7R, %0-3,5R"},
588 {FPU_NEON_EXT_V1, 0xf2000b00, 0xff800f10, "vqdmulh%c.s%20-21S6\t%12-15,22R, %16-19,7R, %0-3,5R"},
589 {FPU_NEON_EXT_V1, 0xf2000b10, 0xff800f10, "vpadd%c.i%20-21S2\t%12-15,22R, %16-19,7R, %0-3,5R"},
590 {FPU_NEON_EXT_V1, 0xf3000800, 0xff800f10, "vsub%c.i%20-21S3\t%12-15,22R, %16-19,7R, %0-3,5R"},
591 {FPU_NEON_EXT_V1, 0xf3000810, 0xff800f10, "vceq%c.i%20-21S2\t%12-15,22R, %16-19,7R, %0-3,5R"},
592 {FPU_NEON_EXT_V1, 0xf3000900, 0xff800f10, "vmls%c.i%20-21S2\t%12-15,22R, %16-19,7R, %0-3,5R"},
593 {FPU_NEON_EXT_V1, 0xf3000b00, 0xff800f10, "vqrdmulh%c.s%20-21S6\t%12-15,22R, %16-19,7R, %0-3,5R"},
594 {FPU_NEON_EXT_V1, 0xf2000000, 0xfe800f10, "vhadd%c.%24?us%20-21S2\t%12-15,22R, %16-19,7R, %0-3,5R"},
595 {FPU_NEON_EXT_V1, 0xf2000010, 0xfe800f10, "vqadd%c.%24?us%20-21S3\t%12-15,22R, %16-19,7R, %0-3,5R"},
596 {FPU_NEON_EXT_V1, 0xf2000100, 0xfe800f10, "vrhadd%c.%24?us%20-21S2\t%12-15,22R, %16-19,7R, %0-3,5R"},
597 {FPU_NEON_EXT_V1, 0xf2000200, 0xfe800f10, "vhsub%c.%24?us%20-21S2\t%12-15,22R, %16-19,7R, %0-3,5R"},
598 {FPU_NEON_EXT_V1, 0xf2000210, 0xfe800f10, "vqsub%c.%24?us%20-21S3\t%12-15,22R, %16-19,7R, %0-3,5R"},
599 {FPU_NEON_EXT_V1, 0xf2000300, 0xfe800f10, "vcgt%c.%24?us%20-21S2\t%12-15,22R, %16-19,7R, %0-3,5R"},
600 {FPU_NEON_EXT_V1, 0xf2000310, 0xfe800f10, "vcge%c.%24?us%20-21S2\t%12-15,22R, %16-19,7R, %0-3,5R"},
601 {FPU_NEON_EXT_V1, 0xf2000400, 0xfe800f10, "vshl%c.%24?us%20-21S3\t%12-15,22R, %0-3,5R, %16-19,7R"},
602 {FPU_NEON_EXT_V1, 0xf2000410, 0xfe800f10, "vqshl%c.%24?us%20-21S3\t%12-15,22R, %0-3,5R, %16-19,7R"},
603 {FPU_NEON_EXT_V1, 0xf2000500, 0xfe800f10, "vrshl%c.%24?us%20-21S3\t%12-15,22R, %0-3,5R, %16-19,7R"},
604 {FPU_NEON_EXT_V1, 0xf2000510, 0xfe800f10, "vqrshl%c.%24?us%20-21S3\t%12-15,22R, %0-3,5R, %16-19,7R"},
605 {FPU_NEON_EXT_V1, 0xf2000600, 0xfe800f10, "vmax%c.%24?us%20-21S2\t%12-15,22R, %16-19,7R, %0-3,5R"},
606 {FPU_NEON_EXT_V1, 0xf2000610, 0xfe800f10, "vmin%c.%24?us%20-21S2\t%12-15,22R, %16-19,7R, %0-3,5R"},
607 {FPU_NEON_EXT_V1, 0xf2000700, 0xfe800f10, "vabd%c.%24?us%20-21S2\t%12-15,22R, %16-19,7R, %0-3,5R"},
608 {FPU_NEON_EXT_V1, 0xf2000710, 0xfe800f10, "vaba%c.%24?us%20-21S2\t%12-15,22R, %16-19,7R, %0-3,5R"},
609 {FPU_NEON_EXT_V1, 0xf2000910, 0xfe800f10, "vmul%c.%24?pi%20-21S2\t%12-15,22R, %16-19,7R, %0-3,5R"},
610 {FPU_NEON_EXT_V1, 0xf2000a00, 0xfe800f10, "vpmax%c.%24?us%20-21S2\t%12-15,22R, %16-19,7R, %0-3,5R"},
611 {FPU_NEON_EXT_V1, 0xf2000a10, 0xfe800f10, "vpmin%c.%24?us%20-21S2\t%12-15,22R, %16-19,7R, %0-3,5R"},
613 /* One register and an immediate value. */
614 {FPU_NEON_EXT_V1, 0xf2800e10, 0xfeb80fb0, "vmov%c.i8\t%12-15,22R, %E"},
615 {FPU_NEON_EXT_V1, 0xf2800e30, 0xfeb80fb0, "vmov%c.i64\t%12-15,22R, %E"},
616 {FPU_NEON_EXT_V1, 0xf2800f10, 0xfeb80fb0, "vmov%c.f32\t%12-15,22R, %E"},
617 {FPU_NEON_EXT_V1, 0xf2800810, 0xfeb80db0, "vmov%c.i16\t%12-15,22R, %E"},
618 {FPU_NEON_EXT_V1, 0xf2800830, 0xfeb80db0, "vmvn%c.i16\t%12-15,22R, %E"},
619 {FPU_NEON_EXT_V1, 0xf2800910, 0xfeb80db0, "vorr%c.i16\t%12-15,22R, %E"},
620 {FPU_NEON_EXT_V1, 0xf2800930, 0xfeb80db0, "vbic%c.i16\t%12-15,22R, %E"},
621 {FPU_NEON_EXT_V1, 0xf2800c10, 0xfeb80eb0, "vmov%c.i32\t%12-15,22R, %E"},
622 {FPU_NEON_EXT_V1, 0xf2800c30, 0xfeb80eb0, "vmvn%c.i32\t%12-15,22R, %E"},
623 {FPU_NEON_EXT_V1, 0xf2800110, 0xfeb809b0, "vorr%c.i32\t%12-15,22R, %E"},
624 {FPU_NEON_EXT_V1, 0xf2800130, 0xfeb809b0, "vbic%c.i32\t%12-15,22R, %E"},
625 {FPU_NEON_EXT_V1, 0xf2800010, 0xfeb808b0, "vmov%c.i32\t%12-15,22R, %E"},
626 {FPU_NEON_EXT_V1, 0xf2800030, 0xfeb808b0, "vmvn%c.i32\t%12-15,22R, %E"},
628 /* Two registers and a shift amount. */
629 {FPU_NEON_EXT_V1, 0xf2880810, 0xffb80fd0, "vshrn%c.i16\t%12-15,22D, %0-3,5Q, #%16-18e"},
630 {FPU_NEON_EXT_V1, 0xf2880850, 0xffb80fd0, "vrshrn%c.i16\t%12-15,22D, %0-3,5Q, #%16-18e"},
631 {FPU_NEON_EXT_V1, 0xf2880810, 0xfeb80fd0, "vqshrun%c.s16\t%12-15,22D, %0-3,5Q, #%16-18e"},
632 {FPU_NEON_EXT_V1, 0xf2880850, 0xfeb80fd0, "vqrshrun%c.s16\t%12-15,22D, %0-3,5Q, #%16-18e"},
633 {FPU_NEON_EXT_V1, 0xf2880910, 0xfeb80fd0, "vqshrn%c.%24?us16\t%12-15,22D, %0-3,5Q, #%16-18e"},
634 {FPU_NEON_EXT_V1, 0xf2880950, 0xfeb80fd0, "vqrshrn%c.%24?us16\t%12-15,22D, %0-3,5Q, #%16-18e"},
635 {FPU_NEON_EXT_V1, 0xf2880a10, 0xfeb80fd0, "vshll%c.%24?us8\t%12-15,22D, %0-3,5Q, #%16-18d"},
636 {FPU_NEON_EXT_V1, 0xf2900810, 0xffb00fd0, "vshrn%c.i32\t%12-15,22D, %0-3,5Q, #%16-19e"},
637 {FPU_NEON_EXT_V1, 0xf2900850, 0xffb00fd0, "vrshrn%c.i32\t%12-15,22D, %0-3,5Q, #%16-19e"},
638 {FPU_NEON_EXT_V1, 0xf2880510, 0xffb80f90, "vshl%c.%24?us8\t%12-15,22R, %0-3,5R, #%16-18d"},
639 {FPU_NEON_EXT_V1, 0xf3880410, 0xffb80f90, "vsri%c.8\t%12-15,22R, %0-3,5R, #%16-18e"},
640 {FPU_NEON_EXT_V1, 0xf3880510, 0xffb80f90, "vsli%c.8\t%12-15,22R, %0-3,5R, #%16-18d"},
641 {FPU_NEON_EXT_V1, 0xf3880610, 0xffb80f90, "vqshlu%c.s8\t%12-15,22R, %0-3,5R, #%16-18d"},
642 {FPU_NEON_EXT_V1, 0xf2900810, 0xfeb00fd0, "vqshrun%c.s32\t%12-15,22D, %0-3,5Q, #%16-19e"},
643 {FPU_NEON_EXT_V1, 0xf2900850, 0xfeb00fd0, "vqrshrun%c.s32\t%12-15,22D, %0-3,5Q, #%16-19e"},
644 {FPU_NEON_EXT_V1, 0xf2900910, 0xfeb00fd0, "vqshrn%c.%24?us32\t%12-15,22D, %0-3,5Q, #%16-19e"},
645 {FPU_NEON_EXT_V1, 0xf2900950, 0xfeb00fd0, "vqrshrn%c.%24?us32\t%12-15,22D, %0-3,5Q, #%16-19e"},
646 {FPU_NEON_EXT_V1, 0xf2900a10, 0xfeb00fd0, "vshll%c.%24?us16\t%12-15,22D, %0-3,5Q, #%16-19d"},
647 {FPU_NEON_EXT_V1, 0xf2880010, 0xfeb80f90, "vshr%c.%24?us8\t%12-15,22R, %0-3,5R, #%16-18e"},
648 {FPU_NEON_EXT_V1, 0xf2880110, 0xfeb80f90, "vsra%c.%24?us8\t%12-15,22R, %0-3,5R, #%16-18e"},
649 {FPU_NEON_EXT_V1, 0xf2880210, 0xfeb80f90, "vrshr%c.%24?us8\t%12-15,22R, %0-3,5R, #%16-18e"},
650 {FPU_NEON_EXT_V1, 0xf2880310, 0xfeb80f90, "vrsra%c.%24?us8\t%12-15,22R, %0-3,5R, #%16-18e"},
651 {FPU_NEON_EXT_V1, 0xf2880710, 0xfeb80f90, "vqshl%c.%24?us8\t%12-15,22R, %0-3,5R, #%16-18d"},
652 {FPU_NEON_EXT_V1, 0xf2a00810, 0xffa00fd0, "vshrn%c.i64\t%12-15,22D, %0-3,5Q, #%16-20e"},
653 {FPU_NEON_EXT_V1, 0xf2a00850, 0xffa00fd0, "vrshrn%c.i64\t%12-15,22D, %0-3,5Q, #%16-20e"},
654 {FPU_NEON_EXT_V1, 0xf2900510, 0xffb00f90, "vshl%c.%24?us16\t%12-15,22R, %0-3,5R, #%16-19d"},
655 {FPU_NEON_EXT_V1, 0xf3900410, 0xffb00f90, "vsri%c.16\t%12-15,22R, %0-3,5R, #%16-19e"},
656 {FPU_NEON_EXT_V1, 0xf3900510, 0xffb00f90, "vsli%c.16\t%12-15,22R, %0-3,5R, #%16-19d"},
657 {FPU_NEON_EXT_V1, 0xf3900610, 0xffb00f90, "vqshlu%c.s16\t%12-15,22R, %0-3,5R, #%16-19d"},
658 {FPU_NEON_EXT_V1, 0xf2a00a10, 0xfea00fd0, "vshll%c.%24?us32\t%12-15,22D, %0-3,5Q, #%16-20d"},
659 {FPU_NEON_EXT_V1, 0xf2900010, 0xfeb00f90, "vshr%c.%24?us16\t%12-15,22R, %0-3,5R, #%16-19e"},
660 {FPU_NEON_EXT_V1, 0xf2900110, 0xfeb00f90, "vsra%c.%24?us16\t%12-15,22R, %0-3,5R, #%16-19e"},
661 {FPU_NEON_EXT_V1, 0xf2900210, 0xfeb00f90, "vrshr%c.%24?us16\t%12-15,22R, %0-3,5R, #%16-19e"},
662 {FPU_NEON_EXT_V1, 0xf2900310, 0xfeb00f90, "vrsra%c.%24?us16\t%12-15,22R, %0-3,5R, #%16-19e"},
663 {FPU_NEON_EXT_V1, 0xf2900710, 0xfeb00f90, "vqshl%c.%24?us16\t%12-15,22R, %0-3,5R, #%16-19d"},
664 {FPU_NEON_EXT_V1, 0xf2a00810, 0xfea00fd0, "vqshrun%c.s64\t%12-15,22D, %0-3,5Q, #%16-20e"},
665 {FPU_NEON_EXT_V1, 0xf2a00850, 0xfea00fd0, "vqrshrun%c.s64\t%12-15,22D, %0-3,5Q, #%16-20e"},
666 {FPU_NEON_EXT_V1, 0xf2a00910, 0xfea00fd0, "vqshrn%c.%24?us64\t%12-15,22D, %0-3,5Q, #%16-20e"},
667 {FPU_NEON_EXT_V1, 0xf2a00950, 0xfea00fd0, "vqrshrn%c.%24?us64\t%12-15,22D, %0-3,5Q, #%16-20e"},
668 {FPU_NEON_EXT_V1, 0xf2a00510, 0xffa00f90, "vshl%c.%24?us32\t%12-15,22R, %0-3,5R, #%16-20d"},
669 {FPU_NEON_EXT_V1, 0xf3a00410, 0xffa00f90, "vsri%c.32\t%12-15,22R, %0-3,5R, #%16-20e"},
670 {FPU_NEON_EXT_V1, 0xf3a00510, 0xffa00f90, "vsli%c.32\t%12-15,22R, %0-3,5R, #%16-20d"},
671 {FPU_NEON_EXT_V1, 0xf3a00610, 0xffa00f90, "vqshlu%c.s32\t%12-15,22R, %0-3,5R, #%16-20d"},
672 {FPU_NEON_EXT_V1, 0xf2a00010, 0xfea00f90, "vshr%c.%24?us32\t%12-15,22R, %0-3,5R, #%16-20e"},
673 {FPU_NEON_EXT_V1, 0xf2a00110, 0xfea00f90, "vsra%c.%24?us32\t%12-15,22R, %0-3,5R, #%16-20e"},
674 {FPU_NEON_EXT_V1, 0xf2a00210, 0xfea00f90, "vrshr%c.%24?us32\t%12-15,22R, %0-3,5R, #%16-20e"},
675 {FPU_NEON_EXT_V1, 0xf2a00310, 0xfea00f90, "vrsra%c.%24?us32\t%12-15,22R, %0-3,5R, #%16-20e"},
676 {FPU_NEON_EXT_V1, 0xf2a00710, 0xfea00f90, "vqshl%c.%24?us32\t%12-15,22R, %0-3,5R, #%16-20d"},
677 {FPU_NEON_EXT_V1, 0xf2800590, 0xff800f90, "vshl%c.%24?us64\t%12-15,22R, %0-3,5R, #%16-21d"},
678 {FPU_NEON_EXT_V1, 0xf3800490, 0xff800f90, "vsri%c.64\t%12-15,22R, %0-3,5R, #%16-21e"},
679 {FPU_NEON_EXT_V1, 0xf3800590, 0xff800f90, "vsli%c.64\t%12-15,22R, %0-3,5R, #%16-21d"},
680 {FPU_NEON_EXT_V1, 0xf3800690, 0xff800f90, "vqshlu%c.s64\t%12-15,22R, %0-3,5R, #%16-21d"},
681 {FPU_NEON_EXT_V1, 0xf2800090, 0xfe800f90, "vshr%c.%24?us64\t%12-15,22R, %0-3,5R, #%16-21e"},
682 {FPU_NEON_EXT_V1, 0xf2800190, 0xfe800f90, "vsra%c.%24?us64\t%12-15,22R, %0-3,5R, #%16-21e"},
683 {FPU_NEON_EXT_V1, 0xf2800290, 0xfe800f90, "vrshr%c.%24?us64\t%12-15,22R, %0-3,5R, #%16-21e"},
684 {FPU_NEON_EXT_V1, 0xf2800390, 0xfe800f90, "vrsra%c.%24?us64\t%12-15,22R, %0-3,5R, #%16-21e"},
685 {FPU_NEON_EXT_V1, 0xf2800790, 0xfe800f90, "vqshl%c.%24?us64\t%12-15,22R, %0-3,5R, #%16-21d"},
686 {FPU_NEON_EXT_V1, 0xf2a00e10, 0xfea00e90, "vcvt%c.%24,8?usff32.%24,8?ffus32\t%12-15,22R, %0-3,5R, #%16-20e"},
688 /* Three registers of different lengths. */
689 {FPU_NEON_EXT_V1, 0xf2800e00, 0xfea00f50, "vmull%c.p%20S0\t%12-15,22Q, %16-19,7D, %0-3,5D"},
690 {FPU_NEON_EXT_V1, 0xf2800400, 0xff800f50, "vaddhn%c.i%20-21T2\t%12-15,22D, %16-19,7Q, %0-3,5Q"},
691 {FPU_NEON_EXT_V1, 0xf2800600, 0xff800f50, "vsubhn%c.i%20-21T2\t%12-15,22D, %16-19,7Q, %0-3,5Q"},
692 {FPU_NEON_EXT_V1, 0xf2800900, 0xff800f50, "vqdmlal%c.s%20-21S6\t%12-15,22Q, %16-19,7D, %0-3,5D"},
693 {FPU_NEON_EXT_V1, 0xf2800b00, 0xff800f50, "vqdmlsl%c.s%20-21S6\t%12-15,22Q, %16-19,7D, %0-3,5D"},
694 {FPU_NEON_EXT_V1, 0xf2800d00, 0xff800f50, "vqdmull%c.s%20-21S6\t%12-15,22Q, %16-19,7D, %0-3,5D"},
695 {FPU_NEON_EXT_V1, 0xf3800400, 0xff800f50, "vraddhn%c.i%20-21T2\t%12-15,22D, %16-19,7Q, %0-3,5Q"},
696 {FPU_NEON_EXT_V1, 0xf3800600, 0xff800f50, "vrsubhn%c.i%20-21T2\t%12-15,22D, %16-19,7Q, %0-3,5Q"},
697 {FPU_NEON_EXT_V1, 0xf2800000, 0xfe800f50, "vaddl%c.%24?us%20-21S2\t%12-15,22Q, %16-19,7D, %0-3,5D"},
698 {FPU_NEON_EXT_V1, 0xf2800100, 0xfe800f50, "vaddw%c.%24?us%20-21S2\t%12-15,22Q, %16-19,7Q, %0-3,5D"},
699 {FPU_NEON_EXT_V1, 0xf2800200, 0xfe800f50, "vsubl%c.%24?us%20-21S2\t%12-15,22Q, %16-19,7D, %0-3,5D"},
700 {FPU_NEON_EXT_V1, 0xf2800300, 0xfe800f50, "vsubw%c.%24?us%20-21S2\t%12-15,22Q, %16-19,7Q, %0-3,5D"},
701 {FPU_NEON_EXT_V1, 0xf2800500, 0xfe800f50, "vabal%c.%24?us%20-21S2\t%12-15,22Q, %16-19,7D, %0-3,5D"},
702 {FPU_NEON_EXT_V1, 0xf2800700, 0xfe800f50, "vabdl%c.%24?us%20-21S2\t%12-15,22Q, %16-19,7D, %0-3,5D"},
703 {FPU_NEON_EXT_V1, 0xf2800800, 0xfe800f50, "vmlal%c.%24?us%20-21S2\t%12-15,22Q, %16-19,7D, %0-3,5D"},
704 {FPU_NEON_EXT_V1, 0xf2800a00, 0xfe800f50, "vmlsl%c.%24?us%20-21S2\t%12-15,22Q, %16-19,7D, %0-3,5D"},
705 {FPU_NEON_EXT_V1, 0xf2800c00, 0xfe800f50, "vmull%c.%24?us%20-21S2\t%12-15,22Q, %16-19,7D, %0-3,5D"},
707 /* Two registers and a scalar. */
708 {FPU_NEON_EXT_V1, 0xf2800040, 0xff800f50, "vmla%c.i%20-21S6\t%12-15,22D, %16-19,7D, %D"},
709 {FPU_NEON_EXT_V1, 0xf2800140, 0xff800f50, "vmla%c.f%20-21Sa\t%12-15,22D, %16-19,7D, %D"},
710 {FPU_NEON_EXT_V1, 0xf2800340, 0xff800f50, "vqdmlal%c.s%20-21S6\t%12-15,22Q, %16-19,7D, %D"},
711 {FPU_NEON_EXT_V1, 0xf2800440, 0xff800f50, "vmls%c.i%20-21S6\t%12-15,22D, %16-19,7D, %D"},
712 {FPU_NEON_EXT_V1, 0xf2800540, 0xff800f50, "vmls%c.f%20-21S6\t%12-15,22D, %16-19,7D, %D"},
713 {FPU_NEON_EXT_V1, 0xf2800740, 0xff800f50, "vqdmlsl%c.s%20-21S6\t%12-15,22Q, %16-19,7D, %D"},
714 {FPU_NEON_EXT_V1, 0xf2800840, 0xff800f50, "vmul%c.i%20-21S6\t%12-15,22D, %16-19,7D, %D"},
715 {FPU_NEON_EXT_V1, 0xf2800940, 0xff800f50, "vmul%c.f%20-21Sa\t%12-15,22D, %16-19,7D, %D"},
716 {FPU_NEON_EXT_V1, 0xf2800b40, 0xff800f50, "vqdmull%c.s%20-21S6\t%12-15,22Q, %16-19,7D, %D"},
717 {FPU_NEON_EXT_V1, 0xf2800c40, 0xff800f50, "vqdmulh%c.s%20-21S6\t%12-15,22D, %16-19,7D, %D"},
718 {FPU_NEON_EXT_V1, 0xf2800d40, 0xff800f50, "vqrdmulh%c.s%20-21S6\t%12-15,22D, %16-19,7D, %D"},
719 {FPU_NEON_EXT_V1, 0xf3800040, 0xff800f50, "vmla%c.i%20-21S6\t%12-15,22Q, %16-19,7Q, %D"},
720 {FPU_NEON_EXT_V1, 0xf3800140, 0xff800f50, "vmla%c.f%20-21Sa\t%12-15,22Q, %16-19,7Q, %D"},
721 {FPU_NEON_EXT_V1, 0xf3800440, 0xff800f50, "vmls%c.i%20-21S6\t%12-15,22Q, %16-19,7Q, %D"},
722 {FPU_NEON_EXT_V1, 0xf3800540, 0xff800f50, "vmls%c.f%20-21Sa\t%12-15,22Q, %16-19,7Q, %D"},
723 {FPU_NEON_EXT_V1, 0xf3800840, 0xff800f50, "vmul%c.i%20-21S6\t%12-15,22Q, %16-19,7Q, %D"},
724 {FPU_NEON_EXT_V1, 0xf3800940, 0xff800f50, "vmul%c.f%20-21Sa\t%12-15,22Q, %16-19,7Q, %D"},
725 {FPU_NEON_EXT_V1, 0xf3800c40, 0xff800f50, "vqdmulh%c.s%20-21S6\t%12-15,22Q, %16-19,7Q, %D"},
726 {FPU_NEON_EXT_V1, 0xf3800d40, 0xff800f50, "vqrdmulh%c.s%20-21S6\t%12-15,22Q, %16-19,7Q, %D"},
727 {FPU_NEON_EXT_V1, 0xf2800240, 0xfe800f50, "vmlal%c.%24?us%20-21S6\t%12-15,22Q, %16-19,7D, %D"},
728 {FPU_NEON_EXT_V1, 0xf2800640, 0xfe800f50, "vmlsl%c.%24?us%20-21S6\t%12-15,22Q, %16-19,7D, %D"},
729 {FPU_NEON_EXT_V1, 0xf2800a40, 0xfe800f50, "vmull%c.%24?us%20-21S6\t%12-15,22Q, %16-19,7D, %D"},
731 /* Element and structure load/store. */
732 {FPU_NEON_EXT_V1, 0xf4a00fc0, 0xffb00fc0, "vld4%c.32\t%C"},
733 {FPU_NEON_EXT_V1, 0xf4a00c00, 0xffb00f00, "vld1%c.%6-7S2\t%C"},
734 {FPU_NEON_EXT_V1, 0xf4a00d00, 0xffb00f00, "vld2%c.%6-7S2\t%C"},
735 {FPU_NEON_EXT_V1, 0xf4a00e00, 0xffb00f00, "vld3%c.%6-7S2\t%C"},
736 {FPU_NEON_EXT_V1, 0xf4a00f00, 0xffb00f00, "vld4%c.%6-7S2\t%C"},
737 {FPU_NEON_EXT_V1, 0xf4000200, 0xff900f00, "v%21?ls%21?dt1%c.%6-7S3\t%A"},
738 {FPU_NEON_EXT_V1, 0xf4000300, 0xff900f00, "v%21?ls%21?dt2%c.%6-7S2\t%A"},
739 {FPU_NEON_EXT_V1, 0xf4000400, 0xff900f00, "v%21?ls%21?dt3%c.%6-7S2\t%A"},
740 {FPU_NEON_EXT_V1, 0xf4000500, 0xff900f00, "v%21?ls%21?dt3%c.%6-7S2\t%A"},
741 {FPU_NEON_EXT_V1, 0xf4000600, 0xff900f00, "v%21?ls%21?dt1%c.%6-7S3\t%A"},
742 {FPU_NEON_EXT_V1, 0xf4000700, 0xff900f00, "v%21?ls%21?dt1%c.%6-7S3\t%A"},
743 {FPU_NEON_EXT_V1, 0xf4000800, 0xff900f00, "v%21?ls%21?dt2%c.%6-7S2\t%A"},
744 {FPU_NEON_EXT_V1, 0xf4000900, 0xff900f00, "v%21?ls%21?dt2%c.%6-7S2\t%A"},
745 {FPU_NEON_EXT_V1, 0xf4000a00, 0xff900f00, "v%21?ls%21?dt1%c.%6-7S3\t%A"},
746 {FPU_NEON_EXT_V1, 0xf4000000, 0xff900e00, "v%21?ls%21?dt4%c.%6-7S2\t%A"},
747 {FPU_NEON_EXT_V1, 0xf4800000, 0xff900300, "v%21?ls%21?dt1%c.%10-11S2\t%B"},
748 {FPU_NEON_EXT_V1, 0xf4800100, 0xff900300, "v%21?ls%21?dt2%c.%10-11S2\t%B"},
749 {FPU_NEON_EXT_V1, 0xf4800200, 0xff900300, "v%21?ls%21?dt3%c.%10-11S2\t%B"},
750 {FPU_NEON_EXT_V1, 0xf4800300, 0xff900300, "v%21?ls%21?dt4%c.%10-11S2\t%B"},
752 {0,0 ,0, 0}
755 /* Opcode tables: ARM, 16-bit Thumb, 32-bit Thumb. All three are partially
756 ordered: they must be searched linearly from the top to obtain a correct
757 match. */
759 /* print_insn_arm recognizes the following format control codes:
761 %% %
763 %a print address for ldr/str instruction
764 %s print address for ldr/str halfword/signextend instruction
765 %b print branch destination
766 %c print condition code (always bits 28-31)
767 %m print register mask for ldm/stm instruction
768 %o print operand2 (immediate or register + shift)
769 %p print 'p' iff bits 12-15 are 15
770 %t print 't' iff bit 21 set and bit 24 clear
771 %B print arm BLX(1) destination
772 %C print the PSR sub type.
773 %U print barrier type.
774 %P print address for pli instruction.
776 %<bitfield>r print as an ARM register
777 %<bitfield>d print the bitfield in decimal
778 %<bitfield>W print the bitfield plus one in decimal
779 %<bitfield>x print the bitfield in hex
780 %<bitfield>X print the bitfield as 1 hex digit without leading "0x"
782 %<bitfield>'c print specified char iff bitfield is all ones
783 %<bitfield>`c print specified char iff bitfield is all zeroes
784 %<bitfield>?ab... select from array of values in big endian order
786 %e print arm SMI operand (bits 0..7,8..19).
787 %E print the LSB and WIDTH fields of a BFI or BFC instruction.
788 %V print the 16-bit immediate field of a MOVT or MOVW instruction. */
790 static const struct opcode32 arm_opcodes[] =
792 /* ARM instructions. */
793 {ARM_EXT_V1, 0xe1a00000, 0xffffffff, "nop\t\t\t; (mov r0, r0)"},
794 {ARM_EXT_V4T | ARM_EXT_V5, 0x012FFF10, 0x0ffffff0, "bx%c\t%0-3r"},
795 {ARM_EXT_V2, 0x00000090, 0x0fe000f0, "mul%20's%c\t%16-19r, %0-3r, %8-11r"},
796 {ARM_EXT_V2, 0x00200090, 0x0fe000f0, "mla%20's%c\t%16-19r, %0-3r, %8-11r, %12-15r"},
797 {ARM_EXT_V2S, 0x01000090, 0x0fb00ff0, "swp%22'b%c\t%12-15r, %0-3r, [%16-19r]"},
798 {ARM_EXT_V3M, 0x00800090, 0x0fa000f0, "%22?sumull%20's%c\t%12-15r, %16-19r, %0-3r, %8-11r"},
799 {ARM_EXT_V3M, 0x00a00090, 0x0fa000f0, "%22?sumlal%20's%c\t%12-15r, %16-19r, %0-3r, %8-11r"},
801 /* V7 instructions. */
802 {ARM_EXT_V7, 0xf450f000, 0xfd70f000, "pli\t%P"},
803 {ARM_EXT_V7, 0x0320f0f0, 0x0ffffff0, "dbg%c\t#%0-3d"},
804 {ARM_EXT_V7, 0xf57ff050, 0xfffffff0, "dmb\t%U"},
805 {ARM_EXT_V7, 0xf57ff040, 0xfffffff0, "dsb\t%U"},
806 {ARM_EXT_V7, 0xf57ff060, 0xfffffff0, "isb\t%U"},
808 /* ARM V6T2 instructions. */
809 {ARM_EXT_V6T2, 0x07c0001f, 0x0fe0007f, "bfc%c\t%12-15r, %E"},
810 {ARM_EXT_V6T2, 0x07c00010, 0x0fe00070, "bfi%c\t%12-15r, %0-3r, %E"},
811 {ARM_EXT_V6T2, 0x00600090, 0x0ff000f0, "mls%c\t%16-19r, %0-3r, %8-11r, %12-15r"},
812 {ARM_EXT_V6T2, 0x006000b0, 0x0f7000f0, "strht%c\t%12-15r, %s"},
813 {ARM_EXT_V6T2, 0x00300090, 0x0f300090, "ldr%6's%5?hbt%c\t%12-15r, %s"},
814 {ARM_EXT_V6T2, 0x03000000, 0x0ff00000, "movw%c\t%12-15r, %V"},
815 {ARM_EXT_V6T2, 0x03400000, 0x0ff00000, "movt%c\t%12-15r, %V"},
816 {ARM_EXT_V6T2, 0x06ff0f30, 0x0fff0ff0, "rbit%c\t%12-15r, %0-3r"},
817 {ARM_EXT_V6T2, 0x07a00050, 0x0fa00070, "%22?usbfx%c\t%12-15r, %0-3r, #%7-11d, #%16-20W"},
819 /* ARM V6Z instructions. */
820 {ARM_EXT_V6Z, 0x01600070, 0x0ff000f0, "smc%c\t%e"},
822 /* ARM V6K instructions. */
823 {ARM_EXT_V6K, 0xf57ff01f, 0xffffffff, "clrex"},
824 {ARM_EXT_V6K, 0x01d00f9f, 0x0ff00fff, "ldrexb%c\t%12-15r, [%16-19r]"},
825 {ARM_EXT_V6K, 0x01b00f9f, 0x0ff00fff, "ldrexd%c\t%12-15r, [%16-19r]"},
826 {ARM_EXT_V6K, 0x01f00f9f, 0x0ff00fff, "ldrexh%c\t%12-15r, [%16-19r]"},
827 {ARM_EXT_V6K, 0x01c00f90, 0x0ff00ff0, "strexb%c\t%12-15r, %0-3r, [%16-19r]"},
828 {ARM_EXT_V6K, 0x01a00f90, 0x0ff00ff0, "strexd%c\t%12-15r, %0-3r, [%16-19r]"},
829 {ARM_EXT_V6K, 0x01e00f90, 0x0ff00ff0, "strexh%c\t%12-15r, %0-3r, [%16-19r]"},
831 /* ARM V6K NOP hints. */
832 {ARM_EXT_V6K, 0x0320f001, 0x0fffffff, "yield%c"},
833 {ARM_EXT_V6K, 0x0320f002, 0x0fffffff, "wfe%c"},
834 {ARM_EXT_V6K, 0x0320f003, 0x0fffffff, "wfi%c"},
835 {ARM_EXT_V6K, 0x0320f004, 0x0fffffff, "sev%c"},
836 {ARM_EXT_V6K, 0x0320f000, 0x0fffff00, "nop%c\t{%0-7d}"},
838 /* ARM V6 instructions. */
839 {ARM_EXT_V6, 0xf1080000, 0xfffffe3f, "cpsie\t%8'a%7'i%6'f"},
840 {ARM_EXT_V6, 0xf10a0000, 0xfffffe20, "cpsie\t%8'a%7'i%6'f,#%0-4d"},
841 {ARM_EXT_V6, 0xf10C0000, 0xfffffe3f, "cpsid\t%8'a%7'i%6'f"},
842 {ARM_EXT_V6, 0xf10e0000, 0xfffffe20, "cpsid\t%8'a%7'i%6'f,#%0-4d"},
843 {ARM_EXT_V6, 0xf1000000, 0xfff1fe20, "cps\t#%0-4d"},
844 {ARM_EXT_V6, 0x06800010, 0x0ff00ff0, "pkhbt%c\t%12-15r, %16-19r, %0-3r"},
845 {ARM_EXT_V6, 0x06800010, 0x0ff00070, "pkhbt%c\t%12-15r, %16-19r, %0-3r, lsl #%7-11d"},
846 {ARM_EXT_V6, 0x06800050, 0x0ff00ff0, "pkhtb%c\t%12-15r, %16-19r, %0-3r, asr #32"},
847 {ARM_EXT_V6, 0x06800050, 0x0ff00070, "pkhtb%c\t%12-15r, %16-19r, %0-3r, asr #%7-11d"},
848 {ARM_EXT_V6, 0x01900f9f, 0x0ff00fff, "ldrex%c\tr%12-15d, [%16-19r]"},
849 {ARM_EXT_V6, 0x06200f10, 0x0ff00ff0, "qadd16%c\t%12-15r, %16-19r, %0-3r"},
850 {ARM_EXT_V6, 0x06200f90, 0x0ff00ff0, "qadd8%c\t%12-15r, %16-19r, %0-3r"},
851 {ARM_EXT_V6, 0x06200f30, 0x0ff00ff0, "qaddsubx%c\t%12-15r, %16-19r, %0-3r"},
852 {ARM_EXT_V6, 0x06200f70, 0x0ff00ff0, "qsub16%c\t%12-15r, %16-19r, %0-3r"},
853 {ARM_EXT_V6, 0x06200ff0, 0x0ff00ff0, "qsub8%c\t%12-15r, %16-19r, %0-3r"},
854 {ARM_EXT_V6, 0x06200f50, 0x0ff00ff0, "qsubaddx%c\t%12-15r, %16-19r, %0-3r"},
855 {ARM_EXT_V6, 0x06100f10, 0x0ff00ff0, "sadd16%c\t%12-15r, %16-19r, %0-3r"},
856 {ARM_EXT_V6, 0x06100f90, 0x0ff00ff0, "sadd8%c\t%12-15r, %16-19r, %0-3r"},
857 {ARM_EXT_V6, 0x06100f30, 0x0ff00ff0, "saddaddx%c\t%12-15r, %16-19r, %0-3r"},
858 {ARM_EXT_V6, 0x06300f10, 0x0ff00ff0, "shadd16%c\t%12-15r, %16-19r, %0-3r"},
859 {ARM_EXT_V6, 0x06300f90, 0x0ff00ff0, "shadd8%c\t%12-15r, %16-19r, %0-3r"},
860 {ARM_EXT_V6, 0x06300f30, 0x0ff00ff0, "shaddsubx%c\t%12-15r, %16-19r, %0-3r"},
861 {ARM_EXT_V6, 0x06300f70, 0x0ff00ff0, "shsub16%c\t%12-15r, %16-19r, %0-3r"},
862 {ARM_EXT_V6, 0x06300ff0, 0x0ff00ff0, "shsub8%c\t%12-15r, %16-19r, %0-3r"},
863 {ARM_EXT_V6, 0x06300f50, 0x0ff00ff0, "shsubaddx%c\t%12-15r, %16-19r, %0-3r"},
864 {ARM_EXT_V6, 0x06100f70, 0x0ff00ff0, "ssub16%c\t%12-15r, %16-19r, %0-3r"},
865 {ARM_EXT_V6, 0x06100ff0, 0x0ff00ff0, "ssub8%c\t%12-15r, %16-19r, %0-3r"},
866 {ARM_EXT_V6, 0x06100f50, 0x0ff00ff0, "ssubaddx%c\t%12-15r, %16-19r, %0-3r"},
867 {ARM_EXT_V6, 0x06500f10, 0x0ff00ff0, "uadd16%c\t%12-15r, %16-19r, %0-3r"},
868 {ARM_EXT_V6, 0x06500f90, 0x0ff00ff0, "uadd8%c\t%12-15r, %16-19r, %0-3r"},
869 {ARM_EXT_V6, 0x06500f30, 0x0ff00ff0, "uaddsubx%c\t%12-15r, %16-19r, %0-3r"},
870 {ARM_EXT_V6, 0x06700f10, 0x0ff00ff0, "uhadd16%c\t%12-15r, %16-19r, %0-3r"},
871 {ARM_EXT_V6, 0x06700f90, 0x0ff00ff0, "uhadd8%c\t%12-15r, %16-19r, %0-3r"},
872 {ARM_EXT_V6, 0x06700f30, 0x0ff00ff0, "uhaddsubx%c\t%12-15r, %16-19r, %0-3r"},
873 {ARM_EXT_V6, 0x06700f70, 0x0ff00ff0, "uhsub16%c\t%12-15r, %16-19r, %0-3r"},
874 {ARM_EXT_V6, 0x06700ff0, 0x0ff00ff0, "uhsub8%c\t%12-15r, %16-19r, %0-3r"},
875 {ARM_EXT_V6, 0x06700f50, 0x0ff00ff0, "uhsubaddx%c\t%12-15r, %16-19r, %0-3r"},
876 {ARM_EXT_V6, 0x06600f10, 0x0ff00ff0, "uqadd16%c\t%12-15r, %16-19r, %0-3r"},
877 {ARM_EXT_V6, 0x06600f90, 0x0ff00ff0, "uqadd8%c\t%12-15r, %16-19r, %0-3r"},
878 {ARM_EXT_V6, 0x06600f30, 0x0ff00ff0, "uqaddsubx%c\t%12-15r, %16-19r, %0-3r"},
879 {ARM_EXT_V6, 0x06600f70, 0x0ff00ff0, "uqsub16%c\t%12-15r, %16-19r, %0-3r"},
880 {ARM_EXT_V6, 0x06600ff0, 0x0ff00ff0, "uqsub8%c\t%12-15r, %16-19r, %0-3r"},
881 {ARM_EXT_V6, 0x06600f50, 0x0ff00ff0, "uqsubaddx%c\t%12-15r, %16-19r, %0-3r"},
882 {ARM_EXT_V6, 0x06500f70, 0x0ff00ff0, "usub16%c\t%12-15r, %16-19r, %0-3r"},
883 {ARM_EXT_V6, 0x06500ff0, 0x0ff00ff0, "usub8%c\t%12-15r, %16-19r, %0-3r"},
884 {ARM_EXT_V6, 0x06500f50, 0x0ff00ff0, "usubaddx%c\t%12-15r, %16-19r, %0-3r"},
885 {ARM_EXT_V6, 0x06bf0f30, 0x0fff0ff0, "rev%c\t%12-15r, %0-3r"},
886 {ARM_EXT_V6, 0x06bf0fb0, 0x0fff0ff0, "rev16%c\t%12-15r, %0-3r"},
887 {ARM_EXT_V6, 0x06ff0fb0, 0x0fff0ff0, "revsh%c\t%12-15r, %0-3r"},
888 {ARM_EXT_V6, 0xf8100a00, 0xfe50ffff, "rfe%23?id%24?ba\t%16-19r%21'!"},
889 {ARM_EXT_V6, 0x06bf0070, 0x0fff0ff0, "sxth%c\t%12-15r, %0-3r"},
890 {ARM_EXT_V6, 0x06bf0470, 0x0fff0ff0, "sxth%c\t%12-15r, %0-3r, ror #8"},
891 {ARM_EXT_V6, 0x06bf0870, 0x0fff0ff0, "sxth%c\t%12-15r, %0-3r, ror #16"},
892 {ARM_EXT_V6, 0x06bf0c70, 0x0fff0ff0, "sxth%c\t%12-15r, %0-3r, ror #24"},
893 {ARM_EXT_V6, 0x068f0070, 0x0fff0ff0, "sxtb16%c\t%12-15r, %0-3r"},
894 {ARM_EXT_V6, 0x068f0470, 0x0fff0ff0, "sxtb16%c\t%12-15r, %0-3r, ror #8"},
895 {ARM_EXT_V6, 0x068f0870, 0x0fff0ff0, "sxtb16%c\t%12-15r, %0-3r, ror #16"},
896 {ARM_EXT_V6, 0x068f0c70, 0x0fff0ff0, "sxtb16%c\t%12-15r, %0-3r, ror #24"},
897 {ARM_EXT_V6, 0x06af0070, 0x0fff0ff0, "sxtb%c\t%12-15r, %0-3r"},
898 {ARM_EXT_V6, 0x06af0470, 0x0fff0ff0, "sxtb%c\t%12-15r, %0-3r, ror #8"},
899 {ARM_EXT_V6, 0x06af0870, 0x0fff0ff0, "sxtb%c\t%12-15r, %0-3r, ror #16"},
900 {ARM_EXT_V6, 0x06af0c70, 0x0fff0ff0, "sxtb%c\t%12-15r, %0-3r, ror #24"},
901 {ARM_EXT_V6, 0x06ff0070, 0x0fff0ff0, "uxth%c\t%12-15r, %0-3r"},
902 {ARM_EXT_V6, 0x06ff0470, 0x0fff0ff0, "uxth%c\t%12-15r, %0-3r, ror #8"},
903 {ARM_EXT_V6, 0x06ff0870, 0x0fff0ff0, "uxth%c\t%12-15r, %0-3r, ror #16"},
904 {ARM_EXT_V6, 0x06ff0c70, 0x0fff0ff0, "uxth%c\t%12-15r, %0-3r, ror #24"},
905 {ARM_EXT_V6, 0x06cf0070, 0x0fff0ff0, "uxtb16%c\t%12-15r, %0-3r"},
906 {ARM_EXT_V6, 0x06cf0470, 0x0fff0ff0, "uxtb16%c\t%12-15r, %0-3r, ror #8"},
907 {ARM_EXT_V6, 0x06cf0870, 0x0fff0ff0, "uxtb16%c\t%12-15r, %0-3r, ror #16"},
908 {ARM_EXT_V6, 0x06cf0c70, 0x0fff0ff0, "uxtb16%c\t%12-15r, %0-3r, ror #24"},
909 {ARM_EXT_V6, 0x06ef0070, 0x0fff0ff0, "uxtb%c\t%12-15r, %0-3r"},
910 {ARM_EXT_V6, 0x06ef0470, 0x0fff0ff0, "uxtb%c\t%12-15r, %0-3r, ror #8"},
911 {ARM_EXT_V6, 0x06ef0870, 0x0fff0ff0, "uxtb%c\t%12-15r, %0-3r, ror #16"},
912 {ARM_EXT_V6, 0x06ef0c70, 0x0fff0ff0, "uxtb%c\t%12-15r, %0-3r, ror #24"},
913 {ARM_EXT_V6, 0x06b00070, 0x0ff00ff0, "sxtah%c\t%12-15r, %16-19r, %0-3r"},
914 {ARM_EXT_V6, 0x06b00470, 0x0ff00ff0, "sxtah%c\t%12-15r, %16-19r, %0-3r, ror #8"},
915 {ARM_EXT_V6, 0x06b00870, 0x0ff00ff0, "sxtah%c\t%12-15r, %16-19r, %0-3r, ror #16"},
916 {ARM_EXT_V6, 0x06b00c70, 0x0ff00ff0, "sxtah%c\t%12-15r, %16-19r, %0-3r, ror #24"},
917 {ARM_EXT_V6, 0x06800070, 0x0ff00ff0, "sxtab16%c\t%12-15r, %16-19r, %0-3r"},
918 {ARM_EXT_V6, 0x06800470, 0x0ff00ff0, "sxtab16%c\t%12-15r, %16-19r, %0-3r, ror #8"},
919 {ARM_EXT_V6, 0x06800870, 0x0ff00ff0, "sxtab16%c\t%12-15r, %16-19r, %0-3r, ror #16"},
920 {ARM_EXT_V6, 0x06800c70, 0x0ff00ff0, "sxtab16%c\t%12-15r, %16-19r, %0-3r, ror #24"},
921 {ARM_EXT_V6, 0x06a00070, 0x0ff00ff0, "sxtab%c\t%12-15r, %16-19r, %0-3r"},
922 {ARM_EXT_V6, 0x06a00470, 0x0ff00ff0, "sxtab%c\t%12-15r, %16-19r, %0-3r, ror #8"},
923 {ARM_EXT_V6, 0x06a00870, 0x0ff00ff0, "sxtab%c\t%12-15r, %16-19r, %0-3r, ror #16"},
924 {ARM_EXT_V6, 0x06a00c70, 0x0ff00ff0, "sxtab%c\t%12-15r, %16-19r, %0-3r, ror #24"},
925 {ARM_EXT_V6, 0x06f00070, 0x0ff00ff0, "uxtah%c\t%12-15r, %16-19r, %0-3r"},
926 {ARM_EXT_V6, 0x06f00470, 0x0ff00ff0, "uxtah%c\t%12-15r, %16-19r, %0-3r, ror #8"},
927 {ARM_EXT_V6, 0x06f00870, 0x0ff00ff0, "uxtah%c\t%12-15r, %16-19r, %0-3r, ror #16"},
928 {ARM_EXT_V6, 0x06f00c70, 0x0ff00ff0, "uxtah%c\t%12-15r, %16-19r, %0-3r, ror #24"},
929 {ARM_EXT_V6, 0x06c00070, 0x0ff00ff0, "uxtab16%c\t%12-15r, %16-19r, %0-3r"},
930 {ARM_EXT_V6, 0x06c00470, 0x0ff00ff0, "uxtab16%c\t%12-15r, %16-19r, %0-3r, ror #8"},
931 {ARM_EXT_V6, 0x06c00870, 0x0ff00ff0, "uxtab16%c\t%12-15r, %16-19r, %0-3r, ror #16"},
932 {ARM_EXT_V6, 0x06c00c70, 0x0ff00ff0, "uxtab16%c\t%12-15r, %16-19r, %0-3r, ROR #24"},
933 {ARM_EXT_V6, 0x06e00070, 0x0ff00ff0, "uxtab%c\t%12-15r, %16-19r, %0-3r"},
934 {ARM_EXT_V6, 0x06e00470, 0x0ff00ff0, "uxtab%c\t%12-15r, %16-19r, %0-3r, ror #8"},
935 {ARM_EXT_V6, 0x06e00870, 0x0ff00ff0, "uxtab%c\t%12-15r, %16-19r, %0-3r, ror #16"},
936 {ARM_EXT_V6, 0x06e00c70, 0x0ff00ff0, "uxtab%c\t%12-15r, %16-19r, %0-3r, ror #24"},
937 {ARM_EXT_V6, 0x06800fb0, 0x0ff00ff0, "sel%c\t%12-15r, %16-19r, %0-3r"},
938 {ARM_EXT_V6, 0xf1010000, 0xfffffc00, "setend\t%9?ble"},
939 {ARM_EXT_V6, 0x0700f010, 0x0ff0f0d0, "smuad%5'x%c\t%16-19r, %0-3r, %8-11r"},
940 {ARM_EXT_V6, 0x0700f050, 0x0ff0f0d0, "smusd%5'x%c\t%16-19r, %0-3r, %8-11r"},
941 {ARM_EXT_V6, 0x07000010, 0x0ff000d0, "smlad%5'x%c\t%16-19r, %0-3r, %8-11r, %12-15r"},
942 {ARM_EXT_V6, 0x07400010, 0x0ff000d0, "smlald%5'x%c\t%12-15r, %16-19r, %0-3r, %8-11r"},
943 {ARM_EXT_V6, 0x07000050, 0x0ff000d0, "smlsd%5'x%c\t%16-19r, %0-3r, %8-11r, %12-15r"},
944 {ARM_EXT_V6, 0x07400050, 0x0ff000d0, "smlsld%5'x%c\t%12-15r, %16-19r, %0-3r, %8-11r"},
945 {ARM_EXT_V6, 0x0750f010, 0x0ff0f0d0, "smmul%5'r%c\t%16-19r, %0-3r, %8-11r"},
946 {ARM_EXT_V6, 0x07500010, 0x0ff000d0, "smmla%5'r%c\t%16-19r, %0-3r, %8-11r, %12-15r"},
947 {ARM_EXT_V6, 0x075000d0, 0x0ff000d0, "smmls%5'r%c\t%16-19r, %0-3r, %8-11r, %12-15r"},
948 {ARM_EXT_V6, 0xf84d0500, 0xfe5fffe0, "srs%23?id%24?ba\t%16-19r%21'!, #%0-4d"},
949 {ARM_EXT_V6, 0x06a00010, 0x0fe00ff0, "ssat%c\t%12-15r, #%16-20W, %0-3r"},
950 {ARM_EXT_V6, 0x06a00010, 0x0fe00070, "ssat%c\t%12-15r, #%16-20W, %0-3r, lsl #%7-11d"},
951 {ARM_EXT_V6, 0x06a00050, 0x0fe00070, "ssat%c\t%12-15r, #%16-20W, %0-3r, asr #%7-11d"},
952 {ARM_EXT_V6, 0x06a00f30, 0x0ff00ff0, "ssat16%c\t%12-15r, #%16-19W, %0-3r"},
953 {ARM_EXT_V6, 0x01800f90, 0x0ff00ff0, "strex%c\t%12-15r, %0-3r, [%16-19r]"},
954 {ARM_EXT_V6, 0x00400090, 0x0ff000f0, "umaal%c\t%12-15r, %16-19r, %0-3r, %8-11r"},
955 {ARM_EXT_V6, 0x0780f010, 0x0ff0f0f0, "usad8%c\t%16-19r, %0-3r, %8-11r"},
956 {ARM_EXT_V6, 0x07800010, 0x0ff000f0, "usada8%c\t%16-19r, %0-3r, %8-11r, %12-15r"},
957 {ARM_EXT_V6, 0x06e00010, 0x0fe00ff0, "usat%c\t%12-15r, #%16-20d, %0-3r"},
958 {ARM_EXT_V6, 0x06e00010, 0x0fe00070, "usat%c\t%12-15r, #%16-20d, %0-3r, lsl #%7-11d"},
959 {ARM_EXT_V6, 0x06e00050, 0x0fe00070, "usat%c\t%12-15r, #%16-20d, %0-3r, asr #%7-11d"},
960 {ARM_EXT_V6, 0x06e00f30, 0x0ff00ff0, "usat16%c\t%12-15r, #%16-19d, %0-3r"},
962 /* V5J instruction. */
963 {ARM_EXT_V5J, 0x012fff20, 0x0ffffff0, "bxj%c\t%0-3r"},
965 /* V5 Instructions. */
966 {ARM_EXT_V5, 0xe1200070, 0xfff000f0, "bkpt\t0x%16-19X%12-15X%8-11X%0-3X"},
967 {ARM_EXT_V5, 0xfa000000, 0xfe000000, "blx\t%B"},
968 {ARM_EXT_V5, 0x012fff30, 0x0ffffff0, "blx%c\t%0-3r"},
969 {ARM_EXT_V5, 0x016f0f10, 0x0fff0ff0, "clz%c\t%12-15r, %0-3r"},
971 /* V5E "El Segundo" Instructions. */
972 {ARM_EXT_V5E, 0x000000d0, 0x0e1000f0, "ldrd%c\t%12-15r, %s"},
973 {ARM_EXT_V5E, 0x000000f0, 0x0e1000f0, "strd%c\t%12-15r, %s"},
974 {ARM_EXT_V5E, 0xf450f000, 0xfc70f000, "pld\t%a"},
975 {ARM_EXT_V5ExP, 0x01000080, 0x0ff000f0, "smlabb%c\t%16-19r, %0-3r, %8-11r, %12-15r"},
976 {ARM_EXT_V5ExP, 0x010000a0, 0x0ff000f0, "smlatb%c\t%16-19r, %0-3r, %8-11r, %12-15r"},
977 {ARM_EXT_V5ExP, 0x010000c0, 0x0ff000f0, "smlabt%c\t%16-19r, %0-3r, %8-11r, %12-15r"},
978 {ARM_EXT_V5ExP, 0x010000e0, 0x0ff000f0, "smlatt%c\t%16-19r, %0-3r, %8-11r, %12-15r"},
980 {ARM_EXT_V5ExP, 0x01200080, 0x0ff000f0, "smlawb%c\t%16-19r, %0-3r, %8-11r, %12-15r"},
981 {ARM_EXT_V5ExP, 0x012000c0, 0x0ff000f0, "smlawt%c\t%16-19r, %0-3r, %8-11r, %12-15r"},
983 {ARM_EXT_V5ExP, 0x01400080, 0x0ff000f0, "smlalbb%c\t%12-15r, %16-19r, %0-3r, %8-11r"},
984 {ARM_EXT_V5ExP, 0x014000a0, 0x0ff000f0, "smlaltb%c\t%12-15r, %16-19r, %0-3r, %8-11r"},
985 {ARM_EXT_V5ExP, 0x014000c0, 0x0ff000f0, "smlalbt%c\t%12-15r, %16-19r, %0-3r, %8-11r"},
986 {ARM_EXT_V5ExP, 0x014000e0, 0x0ff000f0, "smlaltt%c\t%12-15r, %16-19r, %0-3r, %8-11r"},
988 {ARM_EXT_V5ExP, 0x01600080, 0x0ff0f0f0, "smulbb%c\t%16-19r, %0-3r, %8-11r"},
989 {ARM_EXT_V5ExP, 0x016000a0, 0x0ff0f0f0, "smultb%c\t%16-19r, %0-3r, %8-11r"},
990 {ARM_EXT_V5ExP, 0x016000c0, 0x0ff0f0f0, "smulbt%c\t%16-19r, %0-3r, %8-11r"},
991 {ARM_EXT_V5ExP, 0x016000e0, 0x0ff0f0f0, "smultt%c\t%16-19r, %0-3r, %8-11r"},
993 {ARM_EXT_V5ExP, 0x012000a0, 0x0ff0f0f0, "smulwb%c\t%16-19r, %0-3r, %8-11r"},
994 {ARM_EXT_V5ExP, 0x012000e0, 0x0ff0f0f0, "smulwt%c\t%16-19r, %0-3r, %8-11r"},
996 {ARM_EXT_V5ExP, 0x01000050, 0x0ff00ff0, "qadd%c\t%12-15r, %0-3r, %16-19r"},
997 {ARM_EXT_V5ExP, 0x01400050, 0x0ff00ff0, "qdadd%c\t%12-15r, %0-3r, %16-19r"},
998 {ARM_EXT_V5ExP, 0x01200050, 0x0ff00ff0, "qsub%c\t%12-15r, %0-3r, %16-19r"},
999 {ARM_EXT_V5ExP, 0x01600050, 0x0ff00ff0, "qdsub%c\t%12-15r, %0-3r, %16-19r"},
1001 /* ARM Instructions. */
1002 {ARM_EXT_V1, 0x052d0004, 0x0fff0fff, "push%c\t{%12-15r}\t\t; (str%c %12-15r, %a)"},
1003 {ARM_EXT_V1, 0x04000000, 0x0e100000, "str%22'b%t%c\t%12-15r, %a"},
1004 {ARM_EXT_V1, 0x06000000, 0x0e100ff0, "str%22'b%t%c\t%12-15r, %a"},
1005 {ARM_EXT_V1, 0x04000000, 0x0c100010, "str%22'b%t%c\t%12-15r, %a"},
1006 {ARM_EXT_V1, 0x04400000, 0x0e500000, "strb%c\t%12-15r, %a"},
1007 {ARM_EXT_V1, 0x06400000, 0x0e500010, "strb%c\t%12-15r, %a"},
1008 {ARM_EXT_V1, 0x004000b0, 0x0e5000f0, "strh%c\t%12-15r, %s"},
1009 {ARM_EXT_V1, 0x000000b0, 0x0e500ff0, "strh%c\t%12-15r, %s"},
1010 {ARM_EXT_V1, 0x00500090, 0x0e500090, "ldr%6's%5?hb%c\t%12-15r, %s"},
1011 {ARM_EXT_V1, 0x00100090, 0x0e500f90, "ldr%6's%5?hb%c\t%12-15r, %s"},
1013 {ARM_EXT_V1, 0x02000000, 0x0fe00000, "and%20's%c\t%12-15r, %16-19r, %o"},
1014 {ARM_EXT_V1, 0x00000000, 0x0fe00010, "and%20's%c\t%12-15r, %16-19r, %o"},
1015 {ARM_EXT_V1, 0x00000010, 0x0fe00090, "and%20's%c\t%12-15r, %16-19r, %o"},
1017 {ARM_EXT_V1, 0x02200000, 0x0fe00000, "eor%20's%c\t%12-15r, %16-19r, %o"},
1018 {ARM_EXT_V1, 0x00200000, 0x0fe00010, "eor%20's%c\t%12-15r, %16-19r, %o"},
1019 {ARM_EXT_V1, 0x00200010, 0x0fe00090, "eor%20's%c\t%12-15r, %16-19r, %o"},
1021 {ARM_EXT_V1, 0x02400000, 0x0fe00000, "sub%20's%c\t%12-15r, %16-19r, %o"},
1022 {ARM_EXT_V1, 0x00400000, 0x0fe00010, "sub%20's%c\t%12-15r, %16-19r, %o"},
1023 {ARM_EXT_V1, 0x00400010, 0x0fe00090, "sub%20's%c\t%12-15r, %16-19r, %o"},
1025 {ARM_EXT_V1, 0x02600000, 0x0fe00000, "rsb%20's%c\t%12-15r, %16-19r, %o"},
1026 {ARM_EXT_V1, 0x00600000, 0x0fe00010, "rsb%20's%c\t%12-15r, %16-19r, %o"},
1027 {ARM_EXT_V1, 0x00600010, 0x0fe00090, "rsb%20's%c\t%12-15r, %16-19r, %o"},
1029 {ARM_EXT_V1, 0x02800000, 0x0fe00000, "add%20's%c\t%12-15r, %16-19r, %o"},
1030 {ARM_EXT_V1, 0x00800000, 0x0fe00010, "add%20's%c\t%12-15r, %16-19r, %o"},
1031 {ARM_EXT_V1, 0x00800010, 0x0fe00090, "add%20's%c\t%12-15r, %16-19r, %o"},
1033 {ARM_EXT_V1, 0x02a00000, 0x0fe00000, "adc%20's%c\t%12-15r, %16-19r, %o"},
1034 {ARM_EXT_V1, 0x00a00000, 0x0fe00010, "adc%20's%c\t%12-15r, %16-19r, %o"},
1035 {ARM_EXT_V1, 0x00a00010, 0x0fe00090, "adc%20's%c\t%12-15r, %16-19r, %o"},
1037 {ARM_EXT_V1, 0x02c00000, 0x0fe00000, "sbc%20's%c\t%12-15r, %16-19r, %o"},
1038 {ARM_EXT_V1, 0x00c00000, 0x0fe00010, "sbc%20's%c\t%12-15r, %16-19r, %o"},
1039 {ARM_EXT_V1, 0x00c00010, 0x0fe00090, "sbc%20's%c\t%12-15r, %16-19r, %o"},
1041 {ARM_EXT_V1, 0x02e00000, 0x0fe00000, "rsc%20's%c\t%12-15r, %16-19r, %o"},
1042 {ARM_EXT_V1, 0x00e00000, 0x0fe00010, "rsc%20's%c\t%12-15r, %16-19r, %o"},
1043 {ARM_EXT_V1, 0x00e00010, 0x0fe00090, "rsc%20's%c\t%12-15r, %16-19r, %o"},
1045 {ARM_EXT_V3, 0x0120f000, 0x0db0f000, "msr%c\t%22?SCPSR%C, %o"},
1046 {ARM_EXT_V3, 0x010f0000, 0x0fbf0fff, "mrs%c\t%12-15r, %22?SCPSR"},
1048 {ARM_EXT_V1, 0x03000000, 0x0fe00000, "tst%p%c\t%16-19r, %o"},
1049 {ARM_EXT_V1, 0x01000000, 0x0fe00010, "tst%p%c\t%16-19r, %o"},
1050 {ARM_EXT_V1, 0x01000010, 0x0fe00090, "tst%p%c\t%16-19r, %o"},
1052 {ARM_EXT_V1, 0x03200000, 0x0fe00000, "teq%p%c\t%16-19r, %o"},
1053 {ARM_EXT_V1, 0x01200000, 0x0fe00010, "teq%p%c\t%16-19r, %o"},
1054 {ARM_EXT_V1, 0x01200010, 0x0fe00090, "teq%p%c\t%16-19r, %o"},
1056 {ARM_EXT_V1, 0x03400000, 0x0fe00000, "cmp%p%c\t%16-19r, %o"},
1057 {ARM_EXT_V1, 0x01400000, 0x0fe00010, "cmp%p%c\t%16-19r, %o"},
1058 {ARM_EXT_V1, 0x01400010, 0x0fe00090, "cmp%p%c\t%16-19r, %o"},
1060 {ARM_EXT_V1, 0x03600000, 0x0fe00000, "cmn%p%c\t%16-19r, %o"},
1061 {ARM_EXT_V1, 0x01600000, 0x0fe00010, "cmn%p%c\t%16-19r, %o"},
1062 {ARM_EXT_V1, 0x01600010, 0x0fe00090, "cmn%p%c\t%16-19r, %o"},
1064 {ARM_EXT_V1, 0x03800000, 0x0fe00000, "orr%20's%c\t%12-15r, %16-19r, %o"},
1065 {ARM_EXT_V1, 0x01800000, 0x0fe00010, "orr%20's%c\t%12-15r, %16-19r, %o"},
1066 {ARM_EXT_V1, 0x01800010, 0x0fe00090, "orr%20's%c\t%12-15r, %16-19r, %o"},
1068 {ARM_EXT_V1, 0x03a00000, 0x0fef0000, "mov%20's%c\t%12-15r, %o"},
1069 {ARM_EXT_V1, 0x01a00000, 0x0def0ff0, "mov%20's%c\t%12-15r, %0-3r"},
1070 {ARM_EXT_V1, 0x01a00000, 0x0def0060, "lsl%20's%c\t%12-15r, %q"},
1071 {ARM_EXT_V1, 0x01a00020, 0x0def0060, "lsr%20's%c\t%12-15r, %q"},
1072 {ARM_EXT_V1, 0x01a00040, 0x0def0060, "asr%20's%c\t%12-15r, %q"},
1073 {ARM_EXT_V1, 0x01a00060, 0x0def0ff0, "rrx%20's%c\t%12-15r, %0-3r"},
1074 {ARM_EXT_V1, 0x01a00060, 0x0def0060, "ror%20's%c\t%12-15r, %q"},
1076 {ARM_EXT_V1, 0x03c00000, 0x0fe00000, "bic%20's%c\t%12-15r, %16-19r, %o"},
1077 {ARM_EXT_V1, 0x01c00000, 0x0fe00010, "bic%20's%c\t%12-15r, %16-19r, %o"},
1078 {ARM_EXT_V1, 0x01c00010, 0x0fe00090, "bic%20's%c\t%12-15r, %16-19r, %o"},
1080 {ARM_EXT_V1, 0x03e00000, 0x0fe00000, "mvn%20's%c\t%12-15r, %o"},
1081 {ARM_EXT_V1, 0x01e00000, 0x0fe00010, "mvn%20's%c\t%12-15r, %o"},
1082 {ARM_EXT_V1, 0x01e00010, 0x0fe00090, "mvn%20's%c\t%12-15r, %o"},
1084 {ARM_EXT_V1, 0x06000010, 0x0e000010, UNDEFINED_INSTRUCTION},
1085 {ARM_EXT_V1, 0x049d0004, 0x0fff0fff, "pop%c\t{%12-15r}\t\t; (ldr%c %12-15r, %a)"},
1086 {ARM_EXT_V1, 0x04100000, 0x0c100000, "ldr%22'b%t%c\t%12-15r, %a"},
1087 {ARM_EXT_V1, 0x092d0000, 0x0fff0000, "push%c\t%m"},
1088 {ARM_EXT_V1, 0x08800000, 0x0ff00000, "stm%c\t%16-19r%21'!, %m%22'^"},
1089 {ARM_EXT_V1, 0x08000000, 0x0e100000, "stm%23?id%24?ba%c\t%16-19r%21'!, %m%22'^"},
1090 {ARM_EXT_V1, 0x08bd0000, 0x0fff0000, "pop%c\t%m"},
1091 {ARM_EXT_V1, 0x08900000, 0x0f900000, "ldm%c\t%16-19r%21'!, %m%22'^"},
1092 {ARM_EXT_V1, 0x08100000, 0x0e100000, "ldm%23?id%24?ba%c\t%16-19r%21'!, %m%22'^"},
1093 {ARM_EXT_V1, 0x0a000000, 0x0e000000, "b%24'l%c\t%b"},
1094 {ARM_EXT_V1, 0x0f000000, 0x0f000000, "svc%c\t%0-23x"},
1096 /* The rest. */
1097 {ARM_EXT_V1, 0x00000000, 0x00000000, UNDEFINED_INSTRUCTION},
1098 {0, 0x00000000, 0x00000000, 0}
1101 /* print_insn_thumb16 recognizes the following format control codes:
1103 %S print Thumb register (bits 3..5 as high number if bit 6 set)
1104 %D print Thumb register (bits 0..2 as high number if bit 7 set)
1105 %<bitfield>I print bitfield as a signed decimal
1106 (top bit of range being the sign bit)
1107 %N print Thumb register mask (with LR)
1108 %O print Thumb register mask (with PC)
1109 %M print Thumb register mask
1110 %b print CZB's 6-bit unsigned branch destination
1111 %s print Thumb right-shift immediate (6..10; 0 == 32).
1112 %c print the condition code
1113 %C print the condition code, or "s" if not conditional
1114 %x print warning if conditional an not at end of IT block"
1115 %X print "\t; unpredictable <IT:code>" if conditional
1116 %I print IT instruction suffix and operands
1117 %<bitfield>r print bitfield as an ARM register
1118 %<bitfield>d print bitfield as a decimal
1119 %<bitfield>H print (bitfield * 2) as a decimal
1120 %<bitfield>W print (bitfield * 4) as a decimal
1121 %<bitfield>a print (bitfield * 4) as a pc-rel offset + decoded symbol
1122 %<bitfield>B print Thumb branch destination (signed displacement)
1123 %<bitfield>c print bitfield as a condition code
1124 %<bitnum>'c print specified char iff bit is one
1125 %<bitnum>?ab print a if bit is one else print b. */
1127 static const struct opcode16 thumb_opcodes[] =
1129 /* Thumb instructions. */
1131 /* ARM V6K no-argument instructions. */
1132 {ARM_EXT_V6K, 0xbf00, 0xffff, "nop%c"},
1133 {ARM_EXT_V6K, 0xbf10, 0xffff, "yield%c"},
1134 {ARM_EXT_V6K, 0xbf20, 0xffff, "wfe%c"},
1135 {ARM_EXT_V6K, 0xbf30, 0xffff, "wfi%c"},
1136 {ARM_EXT_V6K, 0xbf40, 0xffff, "sev%c"},
1137 {ARM_EXT_V6K, 0xbf00, 0xff0f, "nop%c\t{%4-7d}"},
1139 /* ARM V6T2 instructions. */
1140 {ARM_EXT_V6T2, 0xb900, 0xfd00, "cbnz\t%0-2r, %b%X"},
1141 {ARM_EXT_V6T2, 0xb100, 0xfd00, "cbz\t%0-2r, %b%X"},
1142 {ARM_EXT_V6T2, 0xbf00, 0xff00, "it%I%X"},
1144 /* ARM V6. */
1145 {ARM_EXT_V6, 0xb660, 0xfff8, "cpsie\t%2'a%1'i%0'f%X"},
1146 {ARM_EXT_V6, 0xb670, 0xfff8, "cpsid\t%2'a%1'i%0'f%X"},
1147 {ARM_EXT_V6, 0x4600, 0xffc0, "mov%c\t%0-2r, %3-5r"},
1148 {ARM_EXT_V6, 0xba00, 0xffc0, "rev%c\t%0-2r, %3-5r"},
1149 {ARM_EXT_V6, 0xba40, 0xffc0, "rev16%c\t%0-2r, %3-5r"},
1150 {ARM_EXT_V6, 0xbac0, 0xffc0, "revsh%c\t%0-2r, %3-5r"},
1151 {ARM_EXT_V6, 0xb650, 0xfff7, "setend\t%3?ble%X"},
1152 {ARM_EXT_V6, 0xb200, 0xffc0, "sxth%c\t%0-2r, %3-5r"},
1153 {ARM_EXT_V6, 0xb240, 0xffc0, "sxtb%c\t%0-2r, %3-5r"},
1154 {ARM_EXT_V6, 0xb280, 0xffc0, "uxth%c\t%0-2r, %3-5r"},
1155 {ARM_EXT_V6, 0xb2c0, 0xffc0, "uxtb%c\t%0-2r, %3-5r"},
1157 /* ARM V5 ISA extends Thumb. */
1158 {ARM_EXT_V5T, 0xbe00, 0xff00, "bkpt\t%0-7x"}, /* Is always unconditional. */
1159 /* This is BLX(2). BLX(1) is a 32-bit instruction. */
1160 {ARM_EXT_V5T, 0x4780, 0xff87, "blx%c\t%3-6r%x"}, /* note: 4 bit register number. */
1161 /* ARM V4T ISA (Thumb v1). */
1162 {ARM_EXT_V4T, 0x46C0, 0xFFFF, "nop%c\t\t\t; (mov r8, r8)"},
1163 /* Format 4. */
1164 {ARM_EXT_V4T, 0x4000, 0xFFC0, "and%C\t%0-2r, %3-5r"},
1165 {ARM_EXT_V4T, 0x4040, 0xFFC0, "eor%C\t%0-2r, %3-5r"},
1166 {ARM_EXT_V4T, 0x4080, 0xFFC0, "lsl%C\t%0-2r, %3-5r"},
1167 {ARM_EXT_V4T, 0x40C0, 0xFFC0, "lsr%C\t%0-2r, %3-5r"},
1168 {ARM_EXT_V4T, 0x4100, 0xFFC0, "asr%C\t%0-2r, %3-5r"},
1169 {ARM_EXT_V4T, 0x4140, 0xFFC0, "adc%C\t%0-2r, %3-5r"},
1170 {ARM_EXT_V4T, 0x4180, 0xFFC0, "sbc%C\t%0-2r, %3-5r"},
1171 {ARM_EXT_V4T, 0x41C0, 0xFFC0, "ror%C\t%0-2r, %3-5r"},
1172 {ARM_EXT_V4T, 0x4200, 0xFFC0, "tst%c\t%0-2r, %3-5r"},
1173 {ARM_EXT_V4T, 0x4240, 0xFFC0, "neg%C\t%0-2r, %3-5r"},
1174 {ARM_EXT_V4T, 0x4280, 0xFFC0, "cmp%c\t%0-2r, %3-5r"},
1175 {ARM_EXT_V4T, 0x42C0, 0xFFC0, "cmn%c\t%0-2r, %3-5r"},
1176 {ARM_EXT_V4T, 0x4300, 0xFFC0, "orr%C\t%0-2r, %3-5r"},
1177 {ARM_EXT_V4T, 0x4340, 0xFFC0, "mul%C\t%0-2r, %3-5r"},
1178 {ARM_EXT_V4T, 0x4380, 0xFFC0, "bic%C\t%0-2r, %3-5r"},
1179 {ARM_EXT_V4T, 0x43C0, 0xFFC0, "mvn%C\t%0-2r, %3-5r"},
1180 /* format 13 */
1181 {ARM_EXT_V4T, 0xB000, 0xFF80, "add%c\tsp, #%0-6W"},
1182 {ARM_EXT_V4T, 0xB080, 0xFF80, "sub%c\tsp, #%0-6W"},
1183 /* format 5 */
1184 {ARM_EXT_V4T, 0x4700, 0xFF80, "bx%c\t%S%x"},
1185 {ARM_EXT_V4T, 0x4400, 0xFF00, "add%c\t%D, %S"},
1186 {ARM_EXT_V4T, 0x4500, 0xFF00, "cmp%c\t%D, %S"},
1187 {ARM_EXT_V4T, 0x4600, 0xFF00, "mov%c\t%D, %S"},
1188 /* format 14 */
1189 {ARM_EXT_V4T, 0xB400, 0xFE00, "push%c\t%N"},
1190 {ARM_EXT_V4T, 0xBC00, 0xFE00, "pop%c\t%O"},
1191 /* format 2 */
1192 {ARM_EXT_V4T, 0x1800, 0xFE00, "add%C\t%0-2r, %3-5r, %6-8r"},
1193 {ARM_EXT_V4T, 0x1A00, 0xFE00, "sub%C\t%0-2r, %3-5r, %6-8r"},
1194 {ARM_EXT_V4T, 0x1C00, 0xFE00, "add%C\t%0-2r, %3-5r, #%6-8d"},
1195 {ARM_EXT_V4T, 0x1E00, 0xFE00, "sub%C\t%0-2r, %3-5r, #%6-8d"},
1196 /* format 8 */
1197 {ARM_EXT_V4T, 0x5200, 0xFE00, "strh%c\t%0-2r, [%3-5r, %6-8r]"},
1198 {ARM_EXT_V4T, 0x5A00, 0xFE00, "ldrh%c\t%0-2r, [%3-5r, %6-8r]"},
1199 {ARM_EXT_V4T, 0x5600, 0xF600, "ldrs%11?hb%c\t%0-2r, [%3-5r, %6-8r]"},
1200 /* format 7 */
1201 {ARM_EXT_V4T, 0x5000, 0xFA00, "str%10'b%c\t%0-2r, [%3-5r, %6-8r]"},
1202 {ARM_EXT_V4T, 0x5800, 0xFA00, "ldr%10'b%c\t%0-2r, [%3-5r, %6-8r]"},
1203 /* format 1 */
1204 {ARM_EXT_V4T, 0x0000, 0xF800, "lsl%C\t%0-2r, %3-5r, #%6-10d"},
1205 {ARM_EXT_V4T, 0x0800, 0xF800, "lsr%C\t%0-2r, %3-5r, %s"},
1206 {ARM_EXT_V4T, 0x1000, 0xF800, "asr%C\t%0-2r, %3-5r, %s"},
1207 /* format 3 */
1208 {ARM_EXT_V4T, 0x2000, 0xF800, "mov%C\t%8-10r, #%0-7d"},
1209 {ARM_EXT_V4T, 0x2800, 0xF800, "cmp%c\t%8-10r, #%0-7d"},
1210 {ARM_EXT_V4T, 0x3000, 0xF800, "add%C\t%8-10r, #%0-7d"},
1211 {ARM_EXT_V4T, 0x3800, 0xF800, "sub%C\t%8-10r, #%0-7d"},
1212 /* format 6 */
1213 {ARM_EXT_V4T, 0x4800, 0xF800, "ldr%c\t%8-10r, [pc, #%0-7W]\t; (%0-7a)"}, /* TODO: Disassemble PC relative "LDR rD,=<symbolic>" */
1214 /* format 9 */
1215 {ARM_EXT_V4T, 0x6000, 0xF800, "str%c\t%0-2r, [%3-5r, #%6-10W]"},
1216 {ARM_EXT_V4T, 0x6800, 0xF800, "ldr%c\t%0-2r, [%3-5r, #%6-10W]"},
1217 {ARM_EXT_V4T, 0x7000, 0xF800, "strb%c\t%0-2r, [%3-5r, #%6-10d]"},
1218 {ARM_EXT_V4T, 0x7800, 0xF800, "ldrb%c\t%0-2r, [%3-5r, #%6-10d]"},
1219 /* format 10 */
1220 {ARM_EXT_V4T, 0x8000, 0xF800, "strh%c\t%0-2r, [%3-5r, #%6-10H]"},
1221 {ARM_EXT_V4T, 0x8800, 0xF800, "ldrh%c\t%0-2r, [%3-5r, #%6-10H]"},
1222 /* format 11 */
1223 {ARM_EXT_V4T, 0x9000, 0xF800, "str%c\t%8-10r, [sp, #%0-7W]"},
1224 {ARM_EXT_V4T, 0x9800, 0xF800, "ldr%c\t%8-10r, [sp, #%0-7W]"},
1225 /* format 12 */
1226 {ARM_EXT_V4T, 0xA000, 0xF800, "add%c\t%8-10r, pc, #%0-7W\t; (adr %8-10r, %0-7a)"},
1227 {ARM_EXT_V4T, 0xA800, 0xF800, "add%c\t%8-10r, sp, #%0-7W"},
1228 /* format 15 */
1229 {ARM_EXT_V4T, 0xC000, 0xF800, "stmia%c\t%8-10r!, %M"},
1230 {ARM_EXT_V4T, 0xC800, 0xF800, "ldmia%c\t%8-10r!, %M"},
1231 /* format 17 */
1232 {ARM_EXT_V4T, 0xDF00, 0xFF00, "svc%c\t%0-7d"},
1233 /* format 16 */
1234 {ARM_EXT_V4T, 0xDE00, 0xFE00, UNDEFINED_INSTRUCTION},
1235 {ARM_EXT_V4T, 0xD000, 0xF000, "b%8-11c.n\t%0-7B%X"},
1236 /* format 18 */
1237 {ARM_EXT_V4T, 0xE000, 0xF800, "b%c.n\t%0-10B%x"},
1239 /* The E800 .. FFFF range is unconditionally redirected to the
1240 32-bit table, because even in pre-V6T2 ISAs, BL and BLX(1) pairs
1241 are processed via that table. Thus, we can never encounter a
1242 bare "second half of BL/BLX(1)" instruction here. */
1243 {ARM_EXT_V1, 0x0000, 0x0000, UNDEFINED_INSTRUCTION},
1244 {0, 0, 0, 0}
1247 /* Thumb32 opcodes use the same table structure as the ARM opcodes.
1248 We adopt the convention that hw1 is the high 16 bits of .value and
1249 .mask, hw2 the low 16 bits.
1251 print_insn_thumb32 recognizes the following format control codes:
1253 %% %
1255 %I print a 12-bit immediate from hw1[10],hw2[14:12,7:0]
1256 %M print a modified 12-bit immediate (same location)
1257 %J print a 16-bit immediate from hw1[3:0,10],hw2[14:12,7:0]
1258 %K print a 16-bit immediate from hw2[3:0],hw1[3:0],hw2[11:4]
1259 %S print a possibly-shifted Rm
1261 %a print the address of a plain load/store
1262 %w print the width and signedness of a core load/store
1263 %m print register mask for ldm/stm
1265 %E print the lsb and width fields of a bfc/bfi instruction
1266 %F print the lsb and width fields of a sbfx/ubfx instruction
1267 %b print a conditional branch offset
1268 %B print an unconditional branch offset
1269 %s print the shift field of an SSAT instruction
1270 %R print the rotation field of an SXT instruction
1271 %U print barrier type.
1272 %P print address for pli instruction.
1273 %c print the condition code
1274 %x print warning if conditional an not at end of IT block"
1275 %X print "\t; unpredictable <IT:code>" if conditional
1277 %<bitfield>d print bitfield in decimal
1278 %<bitfield>W print bitfield*4 in decimal
1279 %<bitfield>r print bitfield as an ARM register
1280 %<bitfield>c print bitfield as a condition code
1282 %<bitfield>'c print specified char iff bitfield is all ones
1283 %<bitfield>`c print specified char iff bitfield is all zeroes
1284 %<bitfield>?ab... select from array of values in big endian order
1286 With one exception at the bottom (done because BL and BLX(1) need
1287 to come dead last), this table was machine-sorted first in
1288 decreasing order of number of bits set in the mask, then in
1289 increasing numeric order of mask, then in increasing numeric order
1290 of opcode. This order is not the clearest for a human reader, but
1291 is guaranteed never to catch a special-case bit pattern with a more
1292 general mask, which is important, because this instruction encoding
1293 makes heavy use of special-case bit patterns. */
1294 static const struct opcode32 thumb32_opcodes[] =
1296 /* V7 instructions. */
1297 {ARM_EXT_V7, 0xf910f000, 0xff70f000, "pli%c\t%a"},
1298 {ARM_EXT_V7, 0xf3af80f0, 0xfffffff0, "dbg%c\t#%0-3d"},
1299 {ARM_EXT_V7, 0xf3bf8f50, 0xfffffff0, "dmb%c\t%U"},
1300 {ARM_EXT_V7, 0xf3bf8f40, 0xfffffff0, "dsb%c\t%U"},
1301 {ARM_EXT_V7, 0xf3bf8f60, 0xfffffff0, "isb%c\t%U"},
1302 {ARM_EXT_DIV, 0xfb90f0f0, 0xfff0f0f0, "sdiv%c\t%8-11r, %16-19r, %0-3r"},
1303 {ARM_EXT_DIV, 0xfbb0f0f0, 0xfff0f0f0, "udiv%c\t%8-11r, %16-19r, %0-3r"},
1305 /* Instructions defined in the basic V6T2 set. */
1306 {ARM_EXT_V6T2, 0xf3af8000, 0xffffffff, "nop%c.w"},
1307 {ARM_EXT_V6T2, 0xf3af8001, 0xffffffff, "yield%c.w"},
1308 {ARM_EXT_V6T2, 0xf3af8002, 0xffffffff, "wfe%c.w"},
1309 {ARM_EXT_V6T2, 0xf3af8003, 0xffffffff, "wfi%c.w"},
1310 {ARM_EXT_V6T2, 0xf3af8004, 0xffffffff, "sev%c.w"},
1311 {ARM_EXT_V6T2, 0xf3af8000, 0xffffff00, "nop%c.w\t{%0-7d}"},
1313 {ARM_EXT_V6T2, 0xf3bf8f2f, 0xffffffff, "clrex%c"},
1314 {ARM_EXT_V6T2, 0xf3af8400, 0xffffff1f, "cpsie.w\t%7'a%6'i%5'f%X"},
1315 {ARM_EXT_V6T2, 0xf3af8600, 0xffffff1f, "cpsid.w\t%7'a%6'i%5'f%X"},
1316 {ARM_EXT_V6T2, 0xf3c08f00, 0xfff0ffff, "bxj%c\t%16-19r%x"},
1317 {ARM_EXT_V6T2, 0xe810c000, 0xffd0ffff, "rfedb%c\t%16-19r%21'!"},
1318 {ARM_EXT_V6T2, 0xe990c000, 0xffd0ffff, "rfeia%c\t%16-19r%21'!"},
1319 {ARM_EXT_V6T2, 0xf3ef8000, 0xffeff000, "mrs%c\t%8-11r, %D"},
1320 {ARM_EXT_V6T2, 0xf3af8100, 0xffffffe0, "cps\t#%0-4d%X"},
1321 {ARM_EXT_V6T2, 0xe8d0f000, 0xfff0fff0, "tbb%c\t[%16-19r, %0-3r]%x"},
1322 {ARM_EXT_V6T2, 0xe8d0f010, 0xfff0fff0, "tbh%c\t[%16-19r, %0-3r, lsl #1]%x"},
1323 {ARM_EXT_V6T2, 0xf3af8500, 0xffffff00, "cpsie\t%7'a%6'i%5'f, #%0-4d%X"},
1324 {ARM_EXT_V6T2, 0xf3af8700, 0xffffff00, "cpsid\t%7'a%6'i%5'f, #%0-4d%X"},
1325 {ARM_EXT_V6T2, 0xf3de8f00, 0xffffff00, "subs%c\tpc, lr, #%0-7d"},
1326 {ARM_EXT_V6T2, 0xf3808000, 0xffe0f000, "msr%c\t%C, %16-19r"},
1327 {ARM_EXT_V6T2, 0xe8500f00, 0xfff00fff, "ldrex%c\t%12-15r, [%16-19r]"},
1328 {ARM_EXT_V6T2, 0xe8d00f4f, 0xfff00fef, "ldrex%4?hb%c\t%12-15r, [%16-19r]"},
1329 {ARM_EXT_V6T2, 0xe800c000, 0xffd0ffe0, "srsdb%c\t%16-19r%21'!, #%0-4d"},
1330 {ARM_EXT_V6T2, 0xe980c000, 0xffd0ffe0, "srsia%c\t%16-19r%21'!, #%0-4d"},
1331 {ARM_EXT_V6T2, 0xfa0ff080, 0xfffff0c0, "sxth%c.w\t%8-11r, %0-3r%R"},
1332 {ARM_EXT_V6T2, 0xfa1ff080, 0xfffff0c0, "uxth%c.w\t%8-11r, %0-3r%R"},
1333 {ARM_EXT_V6T2, 0xfa2ff080, 0xfffff0c0, "sxtb16%c\t%8-11r, %0-3r%R"},
1334 {ARM_EXT_V6T2, 0xfa3ff080, 0xfffff0c0, "uxtb16%c\t%8-11r, %0-3r%R"},
1335 {ARM_EXT_V6T2, 0xfa4ff080, 0xfffff0c0, "sxtb%c.w\t%8-11r, %0-3r%R"},
1336 {ARM_EXT_V6T2, 0xfa5ff080, 0xfffff0c0, "uxtb%c.w\t%8-11r, %0-3r%R"},
1337 {ARM_EXT_V6T2, 0xe8400000, 0xfff000ff, "strex%c\t%8-11r, %12-15r, [%16-19r]"},
1338 {ARM_EXT_V6T2, 0xe8d0007f, 0xfff000ff, "ldrexd%c\t%12-15r, %8-11r, [%16-19r]"},
1339 {ARM_EXT_V6T2, 0xfa80f000, 0xfff0f0f0, "sadd8%c\t%8-11r, %16-19r, %0-3r"},
1340 {ARM_EXT_V6T2, 0xfa80f010, 0xfff0f0f0, "qadd8%c\t%8-11r, %16-19r, %0-3r"},
1341 {ARM_EXT_V6T2, 0xfa80f020, 0xfff0f0f0, "shadd8%c\t%8-11r, %16-19r, %0-3r"},
1342 {ARM_EXT_V6T2, 0xfa80f040, 0xfff0f0f0, "uadd8%c\t%8-11r, %16-19r, %0-3r"},
1343 {ARM_EXT_V6T2, 0xfa80f050, 0xfff0f0f0, "uqadd8%c\t%8-11r, %16-19r, %0-3r"},
1344 {ARM_EXT_V6T2, 0xfa80f060, 0xfff0f0f0, "uhadd8%c\t%8-11r, %16-19r, %0-3r"},
1345 {ARM_EXT_V6T2, 0xfa80f080, 0xfff0f0f0, "qadd%c\t%8-11r, %16-19r, %0-3r"},
1346 {ARM_EXT_V6T2, 0xfa80f090, 0xfff0f0f0, "qdadd%c\t%8-11r, %16-19r, %0-3r"},
1347 {ARM_EXT_V6T2, 0xfa80f0a0, 0xfff0f0f0, "qsub%c\t%8-11r, %16-19r, %0-3r"},
1348 {ARM_EXT_V6T2, 0xfa80f0b0, 0xfff0f0f0, "qdsub%c\t%8-11r, %16-19r, %0-3r"},
1349 {ARM_EXT_V6T2, 0xfa90f000, 0xfff0f0f0, "sadd16%c\t%8-11r, %16-19r, %0-3r"},
1350 {ARM_EXT_V6T2, 0xfa90f010, 0xfff0f0f0, "qadd16%c\t%8-11r, %16-19r, %0-3r"},
1351 {ARM_EXT_V6T2, 0xfa90f020, 0xfff0f0f0, "shadd16%c\t%8-11r, %16-19r, %0-3r"},
1352 {ARM_EXT_V6T2, 0xfa90f040, 0xfff0f0f0, "uadd16%c\t%8-11r, %16-19r, %0-3r"},
1353 {ARM_EXT_V6T2, 0xfa90f050, 0xfff0f0f0, "uqadd16%c\t%8-11r, %16-19r, %0-3r"},
1354 {ARM_EXT_V6T2, 0xfa90f060, 0xfff0f0f0, "uhadd16%c\t%8-11r, %16-19r, %0-3r"},
1355 {ARM_EXT_V6T2, 0xfa90f080, 0xfff0f0f0, "rev%c.w\t%8-11r, %16-19r"},
1356 {ARM_EXT_V6T2, 0xfa90f090, 0xfff0f0f0, "rev16%c.w\t%8-11r, %16-19r"},
1357 {ARM_EXT_V6T2, 0xfa90f0a0, 0xfff0f0f0, "rbit%c\t%8-11r, %16-19r"},
1358 {ARM_EXT_V6T2, 0xfa90f0b0, 0xfff0f0f0, "revsh%c.w\t%8-11r, %16-19r"},
1359 {ARM_EXT_V6T2, 0xfaa0f000, 0xfff0f0f0, "saddsubx%c\t%8-11r, %16-19r, %0-3r"},
1360 {ARM_EXT_V6T2, 0xfaa0f010, 0xfff0f0f0, "qaddsubx%c\t%8-11r, %16-19r, %0-3r"},
1361 {ARM_EXT_V6T2, 0xfaa0f020, 0xfff0f0f0, "shaddsubx%c\t%8-11r, %16-19r, %0-3r"},
1362 {ARM_EXT_V6T2, 0xfaa0f040, 0xfff0f0f0, "uaddsubx%c\t%8-11r, %16-19r, %0-3r"},
1363 {ARM_EXT_V6T2, 0xfaa0f050, 0xfff0f0f0, "uqaddsubx%c\t%8-11r, %16-19r, %0-3r"},
1364 {ARM_EXT_V6T2, 0xfaa0f060, 0xfff0f0f0, "uhaddsubx%c\t%8-11r, %16-19r, %0-3r"},
1365 {ARM_EXT_V6T2, 0xfaa0f080, 0xfff0f0f0, "sel%c\t%8-11r, %16-19r, %0-3r"},
1366 {ARM_EXT_V6T2, 0xfab0f080, 0xfff0f0f0, "clz%c\t%8-11r, %16-19r"},
1367 {ARM_EXT_V6T2, 0xfac0f000, 0xfff0f0f0, "ssub8%c\t%8-11r, %16-19r, %0-3r"},
1368 {ARM_EXT_V6T2, 0xfac0f010, 0xfff0f0f0, "qsub8%c\t%8-11r, %16-19r, %0-3r"},
1369 {ARM_EXT_V6T2, 0xfac0f020, 0xfff0f0f0, "shsub8%c\t%8-11r, %16-19r, %0-3r"},
1370 {ARM_EXT_V6T2, 0xfac0f040, 0xfff0f0f0, "usub8%c\t%8-11r, %16-19r, %0-3r"},
1371 {ARM_EXT_V6T2, 0xfac0f050, 0xfff0f0f0, "uqsub8%c\t%8-11r, %16-19r, %0-3r"},
1372 {ARM_EXT_V6T2, 0xfac0f060, 0xfff0f0f0, "uhsub8%c\t%8-11r, %16-19r, %0-3r"},
1373 {ARM_EXT_V6T2, 0xfad0f000, 0xfff0f0f0, "ssub16%c\t%8-11r, %16-19r, %0-3r"},
1374 {ARM_EXT_V6T2, 0xfad0f010, 0xfff0f0f0, "qsub16%c\t%8-11r, %16-19r, %0-3r"},
1375 {ARM_EXT_V6T2, 0xfad0f020, 0xfff0f0f0, "shsub16%c\t%8-11r, %16-19r, %0-3r"},
1376 {ARM_EXT_V6T2, 0xfad0f040, 0xfff0f0f0, "usub16%c\t%8-11r, %16-19r, %0-3r"},
1377 {ARM_EXT_V6T2, 0xfad0f050, 0xfff0f0f0, "uqsub16%c\t%8-11r, %16-19r, %0-3r"},
1378 {ARM_EXT_V6T2, 0xfad0f060, 0xfff0f0f0, "uhsub16%c\t%8-11r, %16-19r, %0-3r"},
1379 {ARM_EXT_V6T2, 0xfae0f000, 0xfff0f0f0, "ssubaddx%c\t%8-11r, %16-19r, %0-3r"},
1380 {ARM_EXT_V6T2, 0xfae0f010, 0xfff0f0f0, "qsubaddx%c\t%8-11r, %16-19r, %0-3r"},
1381 {ARM_EXT_V6T2, 0xfae0f020, 0xfff0f0f0, "shsubaddx%c\t%8-11r, %16-19r, %0-3r"},
1382 {ARM_EXT_V6T2, 0xfae0f040, 0xfff0f0f0, "usubaddx%c\t%8-11r, %16-19r, %0-3r"},
1383 {ARM_EXT_V6T2, 0xfae0f050, 0xfff0f0f0, "uqsubaddx%c\t%8-11r, %16-19r, %0-3r"},
1384 {ARM_EXT_V6T2, 0xfae0f060, 0xfff0f0f0, "uhsubaddx%c\t%8-11r, %16-19r, %0-3r"},
1385 {ARM_EXT_V6T2, 0xfb00f000, 0xfff0f0f0, "mul%c.w\t%8-11r, %16-19r, %0-3r"},
1386 {ARM_EXT_V6T2, 0xfb70f000, 0xfff0f0f0, "usad8%c\t%8-11r, %16-19r, %0-3r"},
1387 {ARM_EXT_V6T2, 0xfa00f000, 0xffe0f0f0, "lsl%20's%c.w\t%8-11r, %16-19r, %0-3r"},
1388 {ARM_EXT_V6T2, 0xfa20f000, 0xffe0f0f0, "lsr%20's%c.w\t%8-11r, %16-19r, %0-3r"},
1389 {ARM_EXT_V6T2, 0xfa40f000, 0xffe0f0f0, "asr%20's%c.w\t%8-11r, %16-19r, %0-3r"},
1390 {ARM_EXT_V6T2, 0xfa60f000, 0xffe0f0f0, "ror%20's%c.w\t%8-11r, %16-19r, %0-3r"},
1391 {ARM_EXT_V6T2, 0xe8c00f40, 0xfff00fe0, "strex%4?hb%c\t%0-3r, %12-15r, [%16-19r]"},
1392 {ARM_EXT_V6T2, 0xf3200000, 0xfff0f0e0, "ssat16%c\t%8-11r, #%0-4d, %16-19r"},
1393 {ARM_EXT_V6T2, 0xf3a00000, 0xfff0f0e0, "usat16%c\t%8-11r, #%0-4d, %16-19r"},
1394 {ARM_EXT_V6T2, 0xfb20f000, 0xfff0f0e0, "smuad%4'x%c\t%8-11r, %16-19r, %0-3r"},
1395 {ARM_EXT_V6T2, 0xfb30f000, 0xfff0f0e0, "smulw%4?tb%c\t%8-11r, %16-19r, %0-3r"},
1396 {ARM_EXT_V6T2, 0xfb40f000, 0xfff0f0e0, "smusd%4'x%c\t%8-11r, %16-19r, %0-3r"},
1397 {ARM_EXT_V6T2, 0xfb50f000, 0xfff0f0e0, "smmul%4'r%c\t%8-11r, %16-19r, %0-3r"},
1398 {ARM_EXT_V6T2, 0xfa00f080, 0xfff0f0c0, "sxtah%c\t%8-11r, %16-19r, %0-3r%R"},
1399 {ARM_EXT_V6T2, 0xfa10f080, 0xfff0f0c0, "uxtah%c\t%8-11r, %16-19r, %0-3r%R"},
1400 {ARM_EXT_V6T2, 0xfa20f080, 0xfff0f0c0, "sxtab16%c\t%8-11r, %16-19r, %0-3r%R"},
1401 {ARM_EXT_V6T2, 0xfa30f080, 0xfff0f0c0, "uxtab16%c\t%8-11r, %16-19r, %0-3r%R"},
1402 {ARM_EXT_V6T2, 0xfa40f080, 0xfff0f0c0, "sxtab%c\t%8-11r, %16-19r, %0-3r%R"},
1403 {ARM_EXT_V6T2, 0xfa50f080, 0xfff0f0c0, "uxtab%c\t%8-11r, %16-19r, %0-3r%R"},
1404 {ARM_EXT_V6T2, 0xfb10f000, 0xfff0f0c0, "smul%5?tb%4?tb%c\t%8-11r, %16-19r, %0-3r"},
1405 {ARM_EXT_V6T2, 0xf36f0000, 0xffff8020, "bfc%c\t%8-11r, %E"},
1406 {ARM_EXT_V6T2, 0xea100f00, 0xfff08f00, "tst%c.w\t%16-19r, %S"},
1407 {ARM_EXT_V6T2, 0xea900f00, 0xfff08f00, "teq%c\t%16-19r, %S"},
1408 {ARM_EXT_V6T2, 0xeb100f00, 0xfff08f00, "cmn%c.w\t%16-19r, %S"},
1409 {ARM_EXT_V6T2, 0xebb00f00, 0xfff08f00, "cmp%c.w\t%16-19r, %S"},
1410 {ARM_EXT_V6T2, 0xf0100f00, 0xfbf08f00, "tst%c.w\t%16-19r, %M"},
1411 {ARM_EXT_V6T2, 0xf0900f00, 0xfbf08f00, "teq%c\t%16-19r, %M"},
1412 {ARM_EXT_V6T2, 0xf1100f00, 0xfbf08f00, "cmn%c.w\t%16-19r, %M"},
1413 {ARM_EXT_V6T2, 0xf1b00f00, 0xfbf08f00, "cmp%c.w\t%16-19r, %M"},
1414 {ARM_EXT_V6T2, 0xea4f0000, 0xffef8000, "mov%20's%c.w\t%8-11r, %S"},
1415 {ARM_EXT_V6T2, 0xea6f0000, 0xffef8000, "mvn%20's%c.w\t%8-11r, %S"},
1416 {ARM_EXT_V6T2, 0xe8c00070, 0xfff000f0, "strexd%c\t%0-3r, %12-15r, %8-11r, [%16-19r]"},
1417 {ARM_EXT_V6T2, 0xfb000000, 0xfff000f0, "mla%c\t%8-11r, %16-19r, %0-3r, %12-15r"},
1418 {ARM_EXT_V6T2, 0xfb000010, 0xfff000f0, "mls%c\t%8-11r, %16-19r, %0-3r, %12-15r"},
1419 {ARM_EXT_V6T2, 0xfb700000, 0xfff000f0, "usada8%c\t%8-11r, %16-19r, %0-3r, %12-15r"},
1420 {ARM_EXT_V6T2, 0xfb800000, 0xfff000f0, "smull%c\t%12-15r, %8-11r, %16-19r, %0-3r"},
1421 {ARM_EXT_V6T2, 0xfba00000, 0xfff000f0, "umull%c\t%12-15r, %8-11r, %16-19r, %0-3r"},
1422 {ARM_EXT_V6T2, 0xfbc00000, 0xfff000f0, "smlal%c\t%12-15r, %8-11r, %16-19r, %0-3r"},
1423 {ARM_EXT_V6T2, 0xfbe00000, 0xfff000f0, "umlal%c\t%12-15r, %8-11r, %16-19r, %0-3r"},
1424 {ARM_EXT_V6T2, 0xfbe00060, 0xfff000f0, "umaal%c\t%12-15r, %8-11r, %16-19r, %0-3r"},
1425 {ARM_EXT_V6T2, 0xe8500f00, 0xfff00f00, "ldrex%c\t%12-15r, [%16-19r, #%0-7W]"},
1426 {ARM_EXT_V6T2, 0xf7f08000, 0xfff0f000, "smc%c\t%K"},
1427 {ARM_EXT_V6T2, 0xf04f0000, 0xfbef8000, "mov%20's%c.w\t%8-11r, %M"},
1428 {ARM_EXT_V6T2, 0xf06f0000, 0xfbef8000, "mvn%20's%c.w\t%8-11r, %M"},
1429 {ARM_EXT_V6T2, 0xf810f000, 0xff70f000, "pld%c\t%a"},
1430 {ARM_EXT_V6T2, 0xfb200000, 0xfff000e0, "smlad%4'x%c\t%8-11r, %16-19r, %0-3r, %12-15r"},
1431 {ARM_EXT_V6T2, 0xfb300000, 0xfff000e0, "smlaw%4?tb%c\t%8-11r, %16-19r, %0-3r, %12-15r"},
1432 {ARM_EXT_V6T2, 0xfb400000, 0xfff000e0, "smlsd%4'x%c\t%8-11r, %16-19r, %0-3r, %12-15r"},
1433 {ARM_EXT_V6T2, 0xfb500000, 0xfff000e0, "smmla%4'r%c\t%8-11r, %16-19r, %0-3r, %12-15r"},
1434 {ARM_EXT_V6T2, 0xfb600000, 0xfff000e0, "smmls%4'r%c\t%8-11r, %16-19r, %0-3r, %12-15r"},
1435 {ARM_EXT_V6T2, 0xfbc000c0, 0xfff000e0, "smlald%4'x%c\t%12-15r, %8-11r, %16-19r, %0-3r"},
1436 {ARM_EXT_V6T2, 0xfbd000c0, 0xfff000e0, "smlsld%4'x%c\t%12-15r, %8-11r, %16-19r, %0-3r"},
1437 {ARM_EXT_V6T2, 0xeac00000, 0xfff08030, "pkhbt%c\t%8-11r, %16-19r, %S"},
1438 {ARM_EXT_V6T2, 0xeac00020, 0xfff08030, "pkhtb%c\t%8-11r, %16-19r, %S"},
1439 {ARM_EXT_V6T2, 0xf3400000, 0xfff08020, "sbfx%c\t%8-11r, %16-19r, %F"},
1440 {ARM_EXT_V6T2, 0xf3c00000, 0xfff08020, "ubfx%c\t%8-11r, %16-19r, %F"},
1441 {ARM_EXT_V6T2, 0xf8000e00, 0xff900f00, "str%wt%c\t%12-15r, %a"},
1442 {ARM_EXT_V6T2, 0xfb100000, 0xfff000c0, "smla%5?tb%4?tb%c\t%8-11r, %16-19r, %0-3r, %12-15r"},
1443 {ARM_EXT_V6T2, 0xfbc00080, 0xfff000c0, "smlal%5?tb%4?tb%c\t%12-15r, %8-11r, %16-19r, %0-3r"},
1444 {ARM_EXT_V6T2, 0xf3600000, 0xfff08020, "bfi%c\t%8-11r, %16-19r, %E"},
1445 {ARM_EXT_V6T2, 0xf8100e00, 0xfe900f00, "ldr%wt%c\t%12-15r, %a"},
1446 {ARM_EXT_V6T2, 0xf3000000, 0xffd08020, "ssat%c\t%8-11r, #%0-4d, %16-19r%s"},
1447 {ARM_EXT_V6T2, 0xf3800000, 0xffd08020, "usat%c\t%8-11r, #%0-4d, %16-19r%s"},
1448 {ARM_EXT_V6T2, 0xf2000000, 0xfbf08000, "addw%c\t%8-11r, %16-19r, %I"},
1449 {ARM_EXT_V6T2, 0xf2400000, 0xfbf08000, "movw%c\t%8-11r, %J"},
1450 {ARM_EXT_V6T2, 0xf2a00000, 0xfbf08000, "subw%c\t%8-11r, %16-19r, %I"},
1451 {ARM_EXT_V6T2, 0xf2c00000, 0xfbf08000, "movt%c\t%8-11r, %J"},
1452 {ARM_EXT_V6T2, 0xea000000, 0xffe08000, "and%20's%c.w\t%8-11r, %16-19r, %S"},
1453 {ARM_EXT_V6T2, 0xea200000, 0xffe08000, "bic%20's%c.w\t%8-11r, %16-19r, %S"},
1454 {ARM_EXT_V6T2, 0xea400000, 0xffe08000, "orr%20's%c.w\t%8-11r, %16-19r, %S"},
1455 {ARM_EXT_V6T2, 0xea600000, 0xffe08000, "orn%20's%c\t%8-11r, %16-19r, %S"},
1456 {ARM_EXT_V6T2, 0xea800000, 0xffe08000, "eor%20's%c.w\t%8-11r, %16-19r, %S"},
1457 {ARM_EXT_V6T2, 0xeb000000, 0xffe08000, "add%20's%c.w\t%8-11r, %16-19r, %S"},
1458 {ARM_EXT_V6T2, 0xeb400000, 0xffe08000, "adc%20's%c.w\t%8-11r, %16-19r, %S"},
1459 {ARM_EXT_V6T2, 0xeb600000, 0xffe08000, "sbc%20's%c.w\t%8-11r, %16-19r, %S"},
1460 {ARM_EXT_V6T2, 0xeba00000, 0xffe08000, "sub%20's%c.w\t%8-11r, %16-19r, %S"},
1461 {ARM_EXT_V6T2, 0xebc00000, 0xffe08000, "rsb%20's%c\t%8-11r, %16-19r, %S"},
1462 {ARM_EXT_V6T2, 0xe8400000, 0xfff00000, "strex%c\t%8-11r, %12-15r, [%16-19r, #%0-7W]"},
1463 {ARM_EXT_V6T2, 0xf0000000, 0xfbe08000, "and%20's%c.w\t%8-11r, %16-19r, %M"},
1464 {ARM_EXT_V6T2, 0xf0200000, 0xfbe08000, "bic%20's%c.w\t%8-11r, %16-19r, %M"},
1465 {ARM_EXT_V6T2, 0xf0400000, 0xfbe08000, "orr%20's%c.w\t%8-11r, %16-19r, %M"},
1466 {ARM_EXT_V6T2, 0xf0600000, 0xfbe08000, "orn%20's%c\t%8-11r, %16-19r, %M"},
1467 {ARM_EXT_V6T2, 0xf0800000, 0xfbe08000, "eor%20's%c.w\t%8-11r, %16-19r, %M"},
1468 {ARM_EXT_V6T2, 0xf1000000, 0xfbe08000, "add%20's%c.w\t%8-11r, %16-19r, %M"},
1469 {ARM_EXT_V6T2, 0xf1400000, 0xfbe08000, "adc%20's%c.w\t%8-11r, %16-19r, %M"},
1470 {ARM_EXT_V6T2, 0xf1600000, 0xfbe08000, "sbc%20's%c.w\t%8-11r, %16-19r, %M"},
1471 {ARM_EXT_V6T2, 0xf1a00000, 0xfbe08000, "sub%20's%c.w\t%8-11r, %16-19r, %M"},
1472 {ARM_EXT_V6T2, 0xf1c00000, 0xfbe08000, "rsb%20's%c\t%8-11r, %16-19r, %M"},
1473 {ARM_EXT_V6T2, 0xe8800000, 0xffd00000, "stmia%c.w\t%16-19r%21'!, %m"},
1474 {ARM_EXT_V6T2, 0xe8900000, 0xffd00000, "ldmia%c.w\t%16-19r%21'!, %m"},
1475 {ARM_EXT_V6T2, 0xe9000000, 0xffd00000, "stmdb%c\t%16-19r%21'!, %m"},
1476 {ARM_EXT_V6T2, 0xe9100000, 0xffd00000, "ldmdb%c\t%16-19r%21'!, %m"},
1477 {ARM_EXT_V6T2, 0xe9c00000, 0xffd000ff, "strd%c\t%12-15r, %8-11r, [%16-19r]"},
1478 {ARM_EXT_V6T2, 0xe9d00000, 0xffd000ff, "ldrd%c\t%12-15r, %8-11r, [%16-19r]"},
1479 {ARM_EXT_V6T2, 0xe9400000, 0xff500000, "strd%c\t%12-15r, %8-11r, [%16-19r, #%23`-%0-7W]%21'!"},
1480 {ARM_EXT_V6T2, 0xe9500000, 0xff500000, "ldrd%c\t%12-15r, %8-11r, [%16-19r, #%23`-%0-7W]%21'!"},
1481 {ARM_EXT_V6T2, 0xe8600000, 0xff700000, "strd%c\t%12-15r, %8-11r, [%16-19r], #%23`-%0-7W"},
1482 {ARM_EXT_V6T2, 0xe8700000, 0xff700000, "ldrd%c\t%12-15r, %8-11r, [%16-19r], #%23`-%0-7W"},
1483 {ARM_EXT_V6T2, 0xf8000000, 0xff100000, "str%w%c.w\t%12-15r, %a"},
1484 {ARM_EXT_V6T2, 0xf8100000, 0xfe100000, "ldr%w%c.w\t%12-15r, %a"},
1486 /* Filter out Bcc with cond=E or F, which are used for other instructions. */
1487 {ARM_EXT_V6T2, 0xf3c08000, 0xfbc0d000, "undefined (bcc, cond=0xF)"},
1488 {ARM_EXT_V6T2, 0xf3808000, 0xfbc0d000, "undefined (bcc, cond=0xE)"},
1489 {ARM_EXT_V6T2, 0xf0008000, 0xf800d000, "b%22-25c.w\t%b%X"},
1490 {ARM_EXT_V6T2, 0xf0009000, 0xf800d000, "b%c.w\t%B%x"},
1492 /* These have been 32-bit since the invention of Thumb. */
1493 {ARM_EXT_V4T, 0xf000c000, 0xf800d000, "blx%c\t%B%x"},
1494 {ARM_EXT_V4T, 0xf000d000, 0xf800d000, "bl%c\t%B%x"},
1496 /* Fallback. */
1497 {ARM_EXT_V1, 0x00000000, 0x00000000, UNDEFINED_INSTRUCTION},
1498 {0, 0, 0, 0}
1501 static const char *const arm_conditional[] =
1502 {"eq", "ne", "cs", "cc", "mi", "pl", "vs", "vc",
1503 "hi", "ls", "ge", "lt", "gt", "le", "al", "<und>", ""};
1505 static const char *const arm_fp_const[] =
1506 {"0.0", "1.0", "2.0", "3.0", "4.0", "5.0", "0.5", "10.0"};
1508 static const char *const arm_shift[] =
1509 {"lsl", "lsr", "asr", "ror"};
1511 typedef struct
1513 const char *name;
1514 const char *description;
1515 const char *reg_names[16];
1517 arm_regname;
1519 static const arm_regname regnames[] =
1521 { "raw" , "Select raw register names",
1522 { "r0", "r1", "r2", "r3", "r4", "r5", "r6", "r7", "r8", "r9", "r10", "r11", "r12", "r13", "r14", "r15"}},
1523 { "gcc", "Select register names used by GCC",
1524 { "r0", "r1", "r2", "r3", "r4", "r5", "r6", "r7", "r8", "r9", "sl", "fp", "ip", "sp", "lr", "pc" }},
1525 { "std", "Select register names used in ARM's ISA documentation",
1526 { "r0", "r1", "r2", "r3", "r4", "r5", "r6", "r7", "r8", "r9", "r10", "r11", "r12", "sp", "lr", "pc" }},
1527 { "apcs", "Select register names used in the APCS",
1528 { "a1", "a2", "a3", "a4", "v1", "v2", "v3", "v4", "v5", "v6", "sl", "fp", "ip", "sp", "lr", "pc" }},
1529 { "atpcs", "Select register names used in the ATPCS",
1530 { "a1", "a2", "a3", "a4", "v1", "v2", "v3", "v4", "v5", "v6", "v7", "v8", "IP", "SP", "LR", "PC" }},
1531 { "special-atpcs", "Select special register names used in the ATPCS",
1532 { "a1", "a2", "a3", "a4", "v1", "v2", "v3", "WR", "v5", "SB", "SL", "FP", "IP", "SP", "LR", "PC" }},
1535 static const char *const iwmmxt_wwnames[] =
1536 {"b", "h", "w", "d"};
1538 static const char *const iwmmxt_wwssnames[] =
1539 {"b", "bus", "bc", "bss",
1540 "h", "hus", "hc", "hss",
1541 "w", "wus", "wc", "wss",
1542 "d", "dus", "dc", "dss"
1545 static const char *const iwmmxt_regnames[] =
1546 { "wr0", "wr1", "wr2", "wr3", "wr4", "wr5", "wr6", "wr7",
1547 "wr8", "wr9", "wr10", "wr11", "wr12", "wr13", "wr14", "wr15"
1550 static const char *const iwmmxt_cregnames[] =
1551 { "wcid", "wcon", "wcssf", "wcasf", "reserved", "reserved", "reserved", "reserved",
1552 "wcgr0", "wcgr1", "wcgr2", "wcgr3", "reserved", "reserved", "reserved", "reserved"
1555 /* Default to GCC register name set. */
1556 static unsigned int regname_selected = 1;
1558 #define NUM_ARM_REGNAMES NUM_ELEM (regnames)
1559 #define arm_regnames regnames[regname_selected].reg_names
1561 static bfd_boolean force_thumb = FALSE;
1563 /* Current IT instruction state. This contains the same state as the IT
1564 bits in the CPSR. */
1565 static unsigned int ifthen_state;
1566 /* IT state for the next instruction. */
1567 static unsigned int ifthen_next_state;
1568 /* The address of the insn for which the IT state is valid. */
1569 static bfd_vma ifthen_address;
1570 #define IFTHEN_COND ((ifthen_state >> 4) & 0xf)
1572 /* Cached mapping symbol state. */
1573 enum map_type
1575 MAP_ARM,
1576 MAP_THUMB,
1577 MAP_DATA
1580 enum map_type last_type;
1581 int last_mapping_sym = -1;
1582 bfd_vma last_mapping_addr = 0;
1585 /* Functions. */
1587 get_arm_regname_num_options (void)
1589 return NUM_ARM_REGNAMES;
1593 set_arm_regname_option (int option)
1595 int old = regname_selected;
1596 regname_selected = option;
1597 return old;
1601 get_arm_regnames (int option,
1602 const char **setname,
1603 const char **setdescription,
1604 const char *const **register_names)
1606 *setname = regnames[option].name;
1607 *setdescription = regnames[option].description;
1608 *register_names = regnames[option].reg_names;
1609 return 16;
1612 /* Decode a bitfield of the form matching regexp (N(-N)?,)*N(-N)?.
1613 Returns pointer to following character of the format string and
1614 fills in *VALUEP and *WIDTHP with the extracted value and number of
1615 bits extracted. WIDTHP can be NULL. */
1617 static const char *
1618 arm_decode_bitfield (const char *ptr,
1619 unsigned long insn,
1620 unsigned long *valuep,
1621 int *widthp)
1623 unsigned long value = 0;
1624 int width = 0;
1628 int start, end;
1629 int bits;
1631 for (start = 0; *ptr >= '0' && *ptr <= '9'; ptr++)
1632 start = start * 10 + *ptr - '0';
1633 if (*ptr == '-')
1634 for (end = 0, ptr++; *ptr >= '0' && *ptr <= '9'; ptr++)
1635 end = end * 10 + *ptr - '0';
1636 else
1637 end = start;
1638 bits = end - start;
1639 if (bits < 0)
1640 abort ();
1641 value |= ((insn >> start) & ((2ul << bits) - 1)) << width;
1642 width += bits + 1;
1644 while (*ptr++ == ',');
1645 *valuep = value;
1646 if (widthp)
1647 *widthp = width;
1648 return ptr - 1;
1651 static void
1652 arm_decode_shift (long given, fprintf_ftype func, void *stream,
1653 bfd_boolean print_shift)
1655 func (stream, "%s", arm_regnames[given & 0xf]);
1657 if ((given & 0xff0) != 0)
1659 if ((given & 0x10) == 0)
1661 int amount = (given & 0xf80) >> 7;
1662 int shift = (given & 0x60) >> 5;
1664 if (amount == 0)
1666 if (shift == 3)
1668 func (stream, ", rrx");
1669 return;
1672 amount = 32;
1675 if (print_shift)
1676 func (stream, ", %s #%d", arm_shift[shift], amount);
1677 else
1678 func (stream, ", #%d", amount);
1680 else if ((given & 0x80) == 0x80)
1681 func (stream, ", <illegal shifter operand>");
1682 else if (print_shift)
1683 func (stream, ", %s %s", arm_shift[(given & 0x60) >> 5],
1684 arm_regnames[(given & 0xf00) >> 8]);
1685 else
1686 func (stream, ", %s", arm_regnames[(given & 0xf00) >> 8]);
1690 /* Print one coprocessor instruction on INFO->STREAM.
1691 Return TRUE if the instuction matched, FALSE if this is not a
1692 recognised coprocessor instruction. */
1694 static bfd_boolean
1695 print_insn_coprocessor (bfd_vma pc,
1696 struct disassemble_info *info,
1697 long given,
1698 bfd_boolean thumb)
1700 const struct opcode32 *insn;
1701 void *stream = info->stream;
1702 fprintf_ftype func = info->fprintf_func;
1703 unsigned long mask;
1704 unsigned long value;
1705 unsigned long allowed_arches = ((arm_feature_set *) info->private_data)->coproc;
1706 int cond;
1708 for (insn = coprocessor_opcodes; insn->assembler; insn++)
1710 signed long value_in_comment = 0;
1711 const char *c;
1713 if (insn->arch == 0)
1714 switch (insn->value)
1716 case SENTINEL_IWMMXT_START:
1717 if (info->mach != bfd_mach_arm_XScale
1718 && info->mach != bfd_mach_arm_iWMMXt
1719 && info->mach != bfd_mach_arm_iWMMXt2)
1721 insn++;
1722 while (insn->arch != 0 && insn->value != SENTINEL_IWMMXT_END);
1723 continue;
1725 case SENTINEL_IWMMXT_END:
1726 continue;
1728 case SENTINEL_GENERIC_START:
1729 allowed_arches = ((arm_feature_set *) info->private_data)->core;
1730 continue;
1732 default:
1733 abort ();
1736 mask = insn->mask;
1737 value = insn->value;
1738 if (thumb)
1740 /* The high 4 bits are 0xe for Arm conditional instructions, and
1741 0xe for arm unconditional instructions. The rest of the
1742 encoding is the same. */
1743 mask |= 0xf0000000;
1744 value |= 0xe0000000;
1745 if (ifthen_state)
1746 cond = IFTHEN_COND;
1747 else
1748 cond = 16;
1750 else
1752 /* Only match unconditional instuctions against unconditional
1753 patterns. */
1754 if ((given & 0xf0000000) == 0xf0000000)
1756 mask |= 0xf0000000;
1757 cond = 16;
1759 else
1761 cond = (given >> 28) & 0xf;
1762 if (cond == 0xe)
1763 cond = 16;
1767 if ((given & mask) != value)
1768 continue;
1770 if ((insn->arch & allowed_arches) == 0)
1771 continue;
1773 for (c = insn->assembler; *c; c++)
1775 if (*c == '%')
1777 switch (*++c)
1779 case '%':
1780 func (stream, "%%");
1781 break;
1783 case 'A':
1785 int offset = given & 0xff;
1787 func (stream, "[%s", arm_regnames [(given >> 16) & 0xf]);
1789 value_in_comment = offset * 4;
1790 if ((given & 0x00800000) == 0)
1791 value_in_comment = - value_in_comment;
1793 if ((given & (1 << 24)) != 0)
1795 if (offset)
1796 func (stream, ", #%d]%s",
1797 value_in_comment,
1798 ((given & 0x00200000) != 0 ? "!" : ""));
1799 else
1800 func (stream, "]");
1802 else
1804 func (stream, "]");
1806 if (given & (1 << 21))
1808 if (offset)
1809 func (stream, ", #%d", value_in_comment);
1811 else
1813 func (stream, ", {%d}", offset);
1814 value_in_comment = offset;
1818 break;
1820 case 'B':
1822 int regno = ((given >> 12) & 0xf) | ((given >> (22 - 4)) & 0x10);
1823 int offset = (given >> 1) & 0x3f;
1825 if (offset == 1)
1826 func (stream, "{d%d}", regno);
1827 else if (regno + offset > 32)
1828 func (stream, "{d%d-<overflow reg d%d>}", regno, regno + offset - 1);
1829 else
1830 func (stream, "{d%d-d%d}", regno, regno + offset - 1);
1832 break;
1834 case 'C':
1836 int rn = (given >> 16) & 0xf;
1837 int offset = (given & 0xff) * 4;
1838 int add = (given >> 23) & 1;
1840 func (stream, "[%s", arm_regnames[rn]);
1842 if (offset)
1844 if (!add)
1845 offset = -offset;
1846 func (stream, ", #%d", offset);
1847 if (rn != 15)
1848 value_in_comment = offset;
1850 func (stream, "]");
1851 if (rn == 15)
1853 func (stream, "\t; ");
1854 /* FIXME: Unsure if info->bytes_per_chunk is the
1855 right thing to use here. */
1856 info->print_address_func (offset + pc
1857 + info->bytes_per_chunk * 2, info);
1860 break;
1862 case 'c':
1863 func (stream, "%s", arm_conditional[cond]);
1864 break;
1866 case 'I':
1867 /* Print a Cirrus/DSP shift immediate. */
1868 /* Immediates are 7bit signed ints with bits 0..3 in
1869 bits 0..3 of opcode and bits 4..6 in bits 5..7
1870 of opcode. */
1872 int imm;
1874 imm = (given & 0xf) | ((given & 0xe0) >> 1);
1876 /* Is ``imm'' a negative number? */
1877 if (imm & 0x40)
1878 imm |= (-1 << 7);
1880 func (stream, "%d", imm);
1883 break;
1885 case 'F':
1886 switch (given & 0x00408000)
1888 case 0:
1889 func (stream, "4");
1890 break;
1891 case 0x8000:
1892 func (stream, "1");
1893 break;
1894 case 0x00400000:
1895 func (stream, "2");
1896 break;
1897 default:
1898 func (stream, "3");
1900 break;
1902 case 'P':
1903 switch (given & 0x00080080)
1905 case 0:
1906 func (stream, "s");
1907 break;
1908 case 0x80:
1909 func (stream, "d");
1910 break;
1911 case 0x00080000:
1912 func (stream, "e");
1913 break;
1914 default:
1915 func (stream, _("<illegal precision>"));
1916 break;
1918 break;
1920 case 'Q':
1921 switch (given & 0x00408000)
1923 case 0:
1924 func (stream, "s");
1925 break;
1926 case 0x8000:
1927 func (stream, "d");
1928 break;
1929 case 0x00400000:
1930 func (stream, "e");
1931 break;
1932 default:
1933 func (stream, "p");
1934 break;
1936 break;
1938 case 'R':
1939 switch (given & 0x60)
1941 case 0:
1942 break;
1943 case 0x20:
1944 func (stream, "p");
1945 break;
1946 case 0x40:
1947 func (stream, "m");
1948 break;
1949 default:
1950 func (stream, "z");
1951 break;
1953 break;
1955 case '0': case '1': case '2': case '3': case '4':
1956 case '5': case '6': case '7': case '8': case '9':
1958 int width;
1959 unsigned long value;
1961 c = arm_decode_bitfield (c, given, &value, &width);
1963 switch (*c)
1965 case 'r':
1966 func (stream, "%s", arm_regnames[value]);
1967 break;
1968 case 'D':
1969 func (stream, "d%ld", value);
1970 break;
1971 case 'Q':
1972 if (value & 1)
1973 func (stream, "<illegal reg q%ld.5>", value >> 1);
1974 else
1975 func (stream, "q%ld", value >> 1);
1976 break;
1977 case 'd':
1978 func (stream, "%ld", value);
1979 value_in_comment = value;
1980 break;
1981 case 'k':
1983 int from = (given & (1 << 7)) ? 32 : 16;
1984 func (stream, "%ld", from - value);
1986 break;
1988 case 'f':
1989 if (value > 7)
1990 func (stream, "#%s", arm_fp_const[value & 7]);
1991 else
1992 func (stream, "f%ld", value);
1993 break;
1995 case 'w':
1996 if (width == 2)
1997 func (stream, "%s", iwmmxt_wwnames[value]);
1998 else
1999 func (stream, "%s", iwmmxt_wwssnames[value]);
2000 break;
2002 case 'g':
2003 func (stream, "%s", iwmmxt_regnames[value]);
2004 break;
2005 case 'G':
2006 func (stream, "%s", iwmmxt_cregnames[value]);
2007 break;
2009 case 'x':
2010 func (stream, "0x%lx", (value & 0xffffffffUL));
2011 break;
2013 case '`':
2014 c++;
2015 if (value == 0)
2016 func (stream, "%c", *c);
2017 break;
2018 case '\'':
2019 c++;
2020 if (value == ((1ul << width) - 1))
2021 func (stream, "%c", *c);
2022 break;
2023 case '?':
2024 func (stream, "%c", c[(1 << width) - (int) value]);
2025 c += 1 << width;
2026 break;
2027 default:
2028 abort ();
2030 break;
2032 case 'y':
2033 case 'z':
2035 int single = *c++ == 'y';
2036 int regno;
2038 switch (*c)
2040 case '4': /* Sm pair */
2041 case '0': /* Sm, Dm */
2042 regno = given & 0x0000000f;
2043 if (single)
2045 regno <<= 1;
2046 regno += (given >> 5) & 1;
2048 else
2049 regno += ((given >> 5) & 1) << 4;
2050 break;
2052 case '1': /* Sd, Dd */
2053 regno = (given >> 12) & 0x0000000f;
2054 if (single)
2056 regno <<= 1;
2057 regno += (given >> 22) & 1;
2059 else
2060 regno += ((given >> 22) & 1) << 4;
2061 break;
2063 case '2': /* Sn, Dn */
2064 regno = (given >> 16) & 0x0000000f;
2065 if (single)
2067 regno <<= 1;
2068 regno += (given >> 7) & 1;
2070 else
2071 regno += ((given >> 7) & 1) << 4;
2072 break;
2074 case '3': /* List */
2075 func (stream, "{");
2076 regno = (given >> 12) & 0x0000000f;
2077 if (single)
2079 regno <<= 1;
2080 regno += (given >> 22) & 1;
2082 else
2083 regno += ((given >> 22) & 1) << 4;
2084 break;
2086 default:
2087 abort ();
2090 func (stream, "%c%d", single ? 's' : 'd', regno);
2092 if (*c == '3')
2094 int count = given & 0xff;
2096 if (single == 0)
2097 count >>= 1;
2099 if (--count)
2101 func (stream, "-%c%d",
2102 single ? 's' : 'd',
2103 regno + count);
2106 func (stream, "}");
2108 else if (*c == '4')
2109 func (stream, ", %c%d", single ? 's' : 'd',
2110 regno + 1);
2112 break;
2114 case 'L':
2115 switch (given & 0x00400100)
2117 case 0x00000000: func (stream, "b"); break;
2118 case 0x00400000: func (stream, "h"); break;
2119 case 0x00000100: func (stream, "w"); break;
2120 case 0x00400100: func (stream, "d"); break;
2121 default:
2122 break;
2124 break;
2126 case 'Z':
2128 int value;
2129 /* given (20, 23) | given (0, 3) */
2130 value = ((given >> 16) & 0xf0) | (given & 0xf);
2131 func (stream, "%d", value);
2133 break;
2135 case 'l':
2136 /* This is like the 'A' operator, except that if
2137 the width field "M" is zero, then the offset is
2138 *not* multiplied by four. */
2140 int offset = given & 0xff;
2141 int multiplier = (given & 0x00000100) ? 4 : 1;
2143 func (stream, "[%s", arm_regnames [(given >> 16) & 0xf]);
2145 if (multiplier > 1)
2147 value_in_comment = offset * multiplier;
2148 if ((given & 0x00800000) == 0)
2149 value_in_comment = - value_in_comment;
2152 if (offset)
2154 if ((given & 0x01000000) != 0)
2155 func (stream, ", #%s%d]%s",
2156 ((given & 0x00800000) == 0 ? "-" : ""),
2157 offset * multiplier,
2158 ((given & 0x00200000) != 0 ? "!" : ""));
2159 else
2160 func (stream, "], #%s%d",
2161 ((given & 0x00800000) == 0 ? "-" : ""),
2162 offset * multiplier);
2164 else
2165 func (stream, "]");
2167 break;
2169 case 'r':
2171 int imm4 = (given >> 4) & 0xf;
2172 int puw_bits = ((given >> 22) & 6) | ((given >> 21) & 1);
2173 int ubit = (given >> 23) & 1;
2174 const char *rm = arm_regnames [given & 0xf];
2175 const char *rn = arm_regnames [(given >> 16) & 0xf];
2177 switch (puw_bits)
2179 case 1:
2180 case 3:
2181 func (stream, "[%s], %c%s", rn, ubit ? '+' : '-', rm);
2182 if (imm4)
2183 func (stream, ", lsl #%d", imm4);
2184 break;
2186 case 4:
2187 case 5:
2188 case 6:
2189 case 7:
2190 func (stream, "[%s, %c%s", rn, ubit ? '+' : '-', rm);
2191 if (imm4 > 0)
2192 func (stream, ", lsl #%d", imm4);
2193 func (stream, "]");
2194 if (puw_bits == 5 || puw_bits == 7)
2195 func (stream, "!");
2196 break;
2198 default:
2199 func (stream, "INVALID");
2202 break;
2204 case 'i':
2206 long imm5;
2207 imm5 = ((given & 0x100) >> 4) | (given & 0xf);
2208 func (stream, "%ld", (imm5 == 0) ? 32 : imm5);
2210 break;
2212 default:
2213 abort ();
2217 else
2218 func (stream, "%c", *c);
2221 if (value_in_comment > 32 || value_in_comment < -16)
2222 func (stream, "\t; 0x%lx", (value_in_comment & 0xffffffffUL));
2224 return TRUE;
2226 return FALSE;
2229 /* Decodes and prints ARM addressing modes. Returns the offset
2230 used in the address, if any, if it is worthwhile printing the
2231 offset as a hexadecimal value in a comment at the end of the
2232 line of disassembly. */
2234 static signed long
2235 print_arm_address (bfd_vma pc, struct disassemble_info *info, long given)
2237 void *stream = info->stream;
2238 fprintf_ftype func = info->fprintf_func;
2239 int offset = 0;
2241 if (((given & 0x000f0000) == 0x000f0000)
2242 && ((given & 0x02000000) == 0))
2244 offset = given & 0xfff;
2246 func (stream, "[pc");
2248 if (given & 0x01000000)
2250 if ((given & 0x00800000) == 0)
2251 offset = - offset;
2253 /* Pre-indexed. */
2254 func (stream, ", #%d]", offset);
2256 offset += pc + 8;
2258 /* Cope with the possibility of write-back
2259 being used. Probably a very dangerous thing
2260 for the programmer to do, but who are we to
2261 argue ? */
2262 if (given & 0x00200000)
2263 func (stream, "!");
2265 else
2267 /* Post indexed. */
2268 func (stream, "], #%d", offset);
2270 /* ie ignore the offset. */
2271 offset = pc + 8;
2274 func (stream, "\t; ");
2275 info->print_address_func (offset, info);
2276 offset = 0;
2278 else
2280 func (stream, "[%s",
2281 arm_regnames[(given >> 16) & 0xf]);
2282 if ((given & 0x01000000) != 0)
2284 if ((given & 0x02000000) == 0)
2286 offset = given & 0xfff;
2287 if (offset)
2288 func (stream, ", #%s%d",
2289 (((given & 0x00800000) == 0)
2290 ? "-" : ""), offset);
2292 else
2294 func (stream, ", %s",
2295 (((given & 0x00800000) == 0)
2296 ? "-" : ""));
2297 arm_decode_shift (given, func, stream, TRUE);
2300 func (stream, "]%s",
2301 ((given & 0x00200000) != 0) ? "!" : "");
2303 else
2305 if ((given & 0x02000000) == 0)
2307 offset = given & 0xfff;
2308 if (offset)
2309 func (stream, "], #%s%d",
2310 (((given & 0x00800000) == 0)
2311 ? "-" : ""), offset);
2312 else
2313 func (stream, "]");
2315 else
2317 func (stream, "], %s",
2318 (((given & 0x00800000) == 0)
2319 ? "-" : ""));
2320 arm_decode_shift (given, func, stream, TRUE);
2325 return (signed long) offset;
2328 /* Print one neon instruction on INFO->STREAM.
2329 Return TRUE if the instuction matched, FALSE if this is not a
2330 recognised neon instruction. */
2332 static bfd_boolean
2333 print_insn_neon (struct disassemble_info *info, long given, bfd_boolean thumb)
2335 const struct opcode32 *insn;
2336 void *stream = info->stream;
2337 fprintf_ftype func = info->fprintf_func;
2339 if (thumb)
2341 if ((given & 0xef000000) == 0xef000000)
2343 /* Move bit 28 to bit 24 to translate Thumb2 to ARM encoding. */
2344 unsigned long bit28 = given & (1 << 28);
2346 given &= 0x00ffffff;
2347 if (bit28)
2348 given |= 0xf3000000;
2349 else
2350 given |= 0xf2000000;
2352 else if ((given & 0xff000000) == 0xf9000000)
2353 given ^= 0xf9000000 ^ 0xf4000000;
2354 else
2355 return FALSE;
2358 for (insn = neon_opcodes; insn->assembler; insn++)
2360 if ((given & insn->mask) == insn->value)
2362 signed long value_in_comment = 0;
2363 const char *c;
2365 for (c = insn->assembler; *c; c++)
2367 if (*c == '%')
2369 switch (*++c)
2371 case '%':
2372 func (stream, "%%");
2373 break;
2375 case 'c':
2376 if (thumb && ifthen_state)
2377 func (stream, "%s", arm_conditional[IFTHEN_COND]);
2378 break;
2380 case 'A':
2382 static const unsigned char enc[16] =
2384 0x4, 0x14, /* st4 0,1 */
2385 0x4, /* st1 2 */
2386 0x4, /* st2 3 */
2387 0x3, /* st3 4 */
2388 0x13, /* st3 5 */
2389 0x3, /* st1 6 */
2390 0x1, /* st1 7 */
2391 0x2, /* st2 8 */
2392 0x12, /* st2 9 */
2393 0x2, /* st1 10 */
2394 0, 0, 0, 0, 0
2396 int rd = ((given >> 12) & 0xf) | (((given >> 22) & 1) << 4);
2397 int rn = ((given >> 16) & 0xf);
2398 int rm = ((given >> 0) & 0xf);
2399 int align = ((given >> 4) & 0x3);
2400 int type = ((given >> 8) & 0xf);
2401 int n = enc[type] & 0xf;
2402 int stride = (enc[type] >> 4) + 1;
2403 int ix;
2405 func (stream, "{");
2406 if (stride > 1)
2407 for (ix = 0; ix != n; ix++)
2408 func (stream, "%sd%d", ix ? "," : "", rd + ix * stride);
2409 else if (n == 1)
2410 func (stream, "d%d", rd);
2411 else
2412 func (stream, "d%d-d%d", rd, rd + n - 1);
2413 func (stream, "}, [%s", arm_regnames[rn]);
2414 if (align)
2415 func (stream, ", :%d", 32 << align);
2416 func (stream, "]");
2417 if (rm == 0xd)
2418 func (stream, "!");
2419 else if (rm != 0xf)
2420 func (stream, ", %s", arm_regnames[rm]);
2422 break;
2424 case 'B':
2426 int rd = ((given >> 12) & 0xf) | (((given >> 22) & 1) << 4);
2427 int rn = ((given >> 16) & 0xf);
2428 int rm = ((given >> 0) & 0xf);
2429 int idx_align = ((given >> 4) & 0xf);
2430 int align = 0;
2431 int size = ((given >> 10) & 0x3);
2432 int idx = idx_align >> (size + 1);
2433 int length = ((given >> 8) & 3) + 1;
2434 int stride = 1;
2435 int i;
2437 if (length > 1 && size > 0)
2438 stride = (idx_align & (1 << size)) ? 2 : 1;
2440 switch (length)
2442 case 1:
2444 int amask = (1 << size) - 1;
2445 if ((idx_align & (1 << size)) != 0)
2446 return FALSE;
2447 if (size > 0)
2449 if ((idx_align & amask) == amask)
2450 align = 8 << size;
2451 else if ((idx_align & amask) != 0)
2452 return FALSE;
2455 break;
2457 case 2:
2458 if (size == 2 && (idx_align & 2) != 0)
2459 return FALSE;
2460 align = (idx_align & 1) ? 16 << size : 0;
2461 break;
2463 case 3:
2464 if ((size == 2 && (idx_align & 3) != 0)
2465 || (idx_align & 1) != 0)
2466 return FALSE;
2467 break;
2469 case 4:
2470 if (size == 2)
2472 if ((idx_align & 3) == 3)
2473 return FALSE;
2474 align = (idx_align & 3) * 64;
2476 else
2477 align = (idx_align & 1) ? 32 << size : 0;
2478 break;
2480 default:
2481 abort ();
2484 func (stream, "{");
2485 for (i = 0; i < length; i++)
2486 func (stream, "%sd%d[%d]", (i == 0) ? "" : ",",
2487 rd + i * stride, idx);
2488 func (stream, "}, [%s", arm_regnames[rn]);
2489 if (align)
2490 func (stream, ", :%d", align);
2491 func (stream, "]");
2492 if (rm == 0xd)
2493 func (stream, "!");
2494 else if (rm != 0xf)
2495 func (stream, ", %s", arm_regnames[rm]);
2497 break;
2499 case 'C':
2501 int rd = ((given >> 12) & 0xf) | (((given >> 22) & 1) << 4);
2502 int rn = ((given >> 16) & 0xf);
2503 int rm = ((given >> 0) & 0xf);
2504 int align = ((given >> 4) & 0x1);
2505 int size = ((given >> 6) & 0x3);
2506 int type = ((given >> 8) & 0x3);
2507 int n = type + 1;
2508 int stride = ((given >> 5) & 0x1);
2509 int ix;
2511 if (stride && (n == 1))
2512 n++;
2513 else
2514 stride++;
2516 func (stream, "{");
2517 if (stride > 1)
2518 for (ix = 0; ix != n; ix++)
2519 func (stream, "%sd%d[]", ix ? "," : "", rd + ix * stride);
2520 else if (n == 1)
2521 func (stream, "d%d[]", rd);
2522 else
2523 func (stream, "d%d[]-d%d[]", rd, rd + n - 1);
2524 func (stream, "}, [%s", arm_regnames[rn]);
2525 if (align)
2527 int align = (8 * (type + 1)) << size;
2528 if (type == 3)
2529 align = (size > 1) ? align >> 1 : align;
2530 if (type == 2 || (type == 0 && !size))
2531 func (stream, ", :<bad align %d>", align);
2532 else
2533 func (stream, ", :%d", align);
2535 func (stream, "]");
2536 if (rm == 0xd)
2537 func (stream, "!");
2538 else if (rm != 0xf)
2539 func (stream, ", %s", arm_regnames[rm]);
2541 break;
2543 case 'D':
2545 int raw_reg = (given & 0xf) | ((given >> 1) & 0x10);
2546 int size = (given >> 20) & 3;
2547 int reg = raw_reg & ((4 << size) - 1);
2548 int ix = raw_reg >> size >> 2;
2550 func (stream, "d%d[%d]", reg, ix);
2552 break;
2554 case 'E':
2555 /* Neon encoded constant for mov, mvn, vorr, vbic. */
2557 int bits = 0;
2558 int cmode = (given >> 8) & 0xf;
2559 int op = (given >> 5) & 0x1;
2560 unsigned long value = 0, hival = 0;
2561 unsigned shift;
2562 int size = 0;
2563 int isfloat = 0;
2565 bits |= ((given >> 24) & 1) << 7;
2566 bits |= ((given >> 16) & 7) << 4;
2567 bits |= ((given >> 0) & 15) << 0;
2569 if (cmode < 8)
2571 shift = (cmode >> 1) & 3;
2572 value = (unsigned long) bits << (8 * shift);
2573 size = 32;
2575 else if (cmode < 12)
2577 shift = (cmode >> 1) & 1;
2578 value = (unsigned long) bits << (8 * shift);
2579 size = 16;
2581 else if (cmode < 14)
2583 shift = (cmode & 1) + 1;
2584 value = (unsigned long) bits << (8 * shift);
2585 value |= (1ul << (8 * shift)) - 1;
2586 size = 32;
2588 else if (cmode == 14)
2590 if (op)
2592 /* Bit replication into bytes. */
2593 int ix;
2594 unsigned long mask;
2596 value = 0;
2597 hival = 0;
2598 for (ix = 7; ix >= 0; ix--)
2600 mask = ((bits >> ix) & 1) ? 0xff : 0;
2601 if (ix <= 3)
2602 value = (value << 8) | mask;
2603 else
2604 hival = (hival << 8) | mask;
2606 size = 64;
2608 else
2610 /* Byte replication. */
2611 value = (unsigned long) bits;
2612 size = 8;
2615 else if (!op)
2617 /* Floating point encoding. */
2618 int tmp;
2620 value = (unsigned long) (bits & 0x7f) << 19;
2621 value |= (unsigned long) (bits & 0x80) << 24;
2622 tmp = bits & 0x40 ? 0x3c : 0x40;
2623 value |= (unsigned long) tmp << 24;
2624 size = 32;
2625 isfloat = 1;
2627 else
2629 func (stream, "<illegal constant %.8x:%x:%x>",
2630 bits, cmode, op);
2631 size = 32;
2632 break;
2634 switch (size)
2636 case 8:
2637 func (stream, "#%ld\t; 0x%.2lx", value, value);
2638 break;
2640 case 16:
2641 func (stream, "#%ld\t; 0x%.4lx", value, value);
2642 break;
2644 case 32:
2645 if (isfloat)
2647 unsigned char valbytes[4];
2648 double fvalue;
2650 /* Do this a byte at a time so we don't have to
2651 worry about the host's endianness. */
2652 valbytes[0] = value & 0xff;
2653 valbytes[1] = (value >> 8) & 0xff;
2654 valbytes[2] = (value >> 16) & 0xff;
2655 valbytes[3] = (value >> 24) & 0xff;
2657 floatformat_to_double
2658 (&floatformat_ieee_single_little, valbytes,
2659 &fvalue);
2661 func (stream, "#%.7g\t; 0x%.8lx", fvalue,
2662 value);
2664 else
2665 func (stream, "#%ld\t; 0x%.8lx",
2666 (long) ((value & 0x80000000)
2667 ? value | ~0xffffffffl : value), value);
2668 break;
2670 case 64:
2671 func (stream, "#0x%.8lx%.8lx", hival, value);
2672 break;
2674 default:
2675 abort ();
2678 break;
2680 case 'F':
2682 int regno = ((given >> 16) & 0xf) | ((given >> (7 - 4)) & 0x10);
2683 int num = (given >> 8) & 0x3;
2685 if (!num)
2686 func (stream, "{d%d}", regno);
2687 else if (num + regno >= 32)
2688 func (stream, "{d%d-<overflow reg d%d}", regno, regno + num);
2689 else
2690 func (stream, "{d%d-d%d}", regno, regno + num);
2692 break;
2695 case '0': case '1': case '2': case '3': case '4':
2696 case '5': case '6': case '7': case '8': case '9':
2698 int width;
2699 unsigned long value;
2701 c = arm_decode_bitfield (c, given, &value, &width);
2703 switch (*c)
2705 case 'r':
2706 func (stream, "%s", arm_regnames[value]);
2707 break;
2708 case 'd':
2709 func (stream, "%ld", value);
2710 value_in_comment = value;
2711 break;
2712 case 'e':
2713 func (stream, "%ld", (1ul << width) - value);
2714 break;
2716 case 'S':
2717 case 'T':
2718 case 'U':
2719 /* Various width encodings. */
2721 int base = 8 << (*c - 'S'); /* 8,16 or 32 */
2722 int limit;
2723 unsigned low, high;
2725 c++;
2726 if (*c >= '0' && *c <= '9')
2727 limit = *c - '0';
2728 else if (*c >= 'a' && *c <= 'f')
2729 limit = *c - 'a' + 10;
2730 else
2731 abort ();
2732 low = limit >> 2;
2733 high = limit & 3;
2735 if (value < low || value > high)
2736 func (stream, "<illegal width %d>", base << value);
2737 else
2738 func (stream, "%d", base << value);
2740 break;
2741 case 'R':
2742 if (given & (1 << 6))
2743 goto Q;
2744 /* FALLTHROUGH */
2745 case 'D':
2746 func (stream, "d%ld", value);
2747 break;
2748 case 'Q':
2750 if (value & 1)
2751 func (stream, "<illegal reg q%ld.5>", value >> 1);
2752 else
2753 func (stream, "q%ld", value >> 1);
2754 break;
2756 case '`':
2757 c++;
2758 if (value == 0)
2759 func (stream, "%c", *c);
2760 break;
2761 case '\'':
2762 c++;
2763 if (value == ((1ul << width) - 1))
2764 func (stream, "%c", *c);
2765 break;
2766 case '?':
2767 func (stream, "%c", c[(1 << width) - (int) value]);
2768 c += 1 << width;
2769 break;
2770 default:
2771 abort ();
2773 break;
2775 default:
2776 abort ();
2780 else
2781 func (stream, "%c", *c);
2784 if (value_in_comment > 32 || value_in_comment < -16)
2785 func (stream, "\t; 0x%lx", value_in_comment);
2787 return TRUE;
2790 return FALSE;
2793 /* Print one ARM instruction from PC on INFO->STREAM. */
2795 static void
2796 print_insn_arm (bfd_vma pc, struct disassemble_info *info, long given)
2798 const struct opcode32 *insn;
2799 void *stream = info->stream;
2800 fprintf_ftype func = info->fprintf_func;
2802 if (print_insn_coprocessor (pc, info, given, FALSE))
2803 return;
2805 if (print_insn_neon (info, given, FALSE))
2806 return;
2808 for (insn = arm_opcodes; insn->assembler; insn++)
2810 if ((given & insn->mask) != insn->value)
2811 continue;
2813 if ((insn->arch & ((arm_feature_set *) info->private_data)->core) == 0)
2814 continue;
2816 /* Special case: an instruction with all bits set in the condition field
2817 (0xFnnn_nnnn) is only matched if all those bits are set in insn->mask,
2818 or by the catchall at the end of the table. */
2819 if ((given & 0xF0000000) != 0xF0000000
2820 || (insn->mask & 0xF0000000) == 0xF0000000
2821 || (insn->mask == 0 && insn->value == 0))
2823 signed long value_in_comment = 0;
2824 const char *c;
2826 for (c = insn->assembler; *c; c++)
2828 if (*c == '%')
2830 switch (*++c)
2832 case '%':
2833 func (stream, "%%");
2834 break;
2836 case 'a':
2837 value_in_comment = print_arm_address (pc, info, given);
2838 break;
2840 case 'P':
2841 /* Set P address bit and use normal address
2842 printing routine. */
2843 value_in_comment = print_arm_address (pc, info, given | (1 << 24));
2844 break;
2846 case 's':
2847 if ((given & 0x004f0000) == 0x004f0000)
2849 /* PC relative with immediate offset. */
2850 int offset = ((given & 0xf00) >> 4) | (given & 0xf);
2852 if ((given & 0x00800000) == 0)
2853 offset = -offset;
2855 func (stream, "[pc, #%d]\t; ", offset);
2856 info->print_address_func (offset + pc + 8, info);
2858 else
2860 bfd_boolean negative = (given & 0x00800000) == 0;
2861 int offset = ((given & 0xf00) >> 4) | (given & 0xf);
2863 if (negative)
2864 offset = -offset;
2866 func (stream, "[%s",
2867 arm_regnames[(given >> 16) & 0xf]);
2869 if ((given & 0x01000000) != 0)
2871 /* Pre-indexed. */
2872 if ((given & 0x00400000) == 0x00400000)
2874 /* Immediate. */
2875 if (offset)
2876 func (stream, ", #%d", offset);
2877 value_in_comment = offset;
2879 else
2881 /* Register. */
2882 func (stream, ", %s%s", negative ? "-" : "",
2883 arm_regnames[given & 0xf]);
2886 func (stream, "]%s",
2887 ((given & 0x00200000) != 0) ? "!" : "");
2889 else
2891 /* Post-indexed. */
2892 if ((given & 0x00400000) == 0x00400000)
2894 /* Immediate. */
2895 if (offset)
2896 func (stream, "], #%d", offset);
2897 else
2898 func (stream, "]");
2900 value_in_comment = offset;
2902 else
2904 /* Register. */
2905 func (stream, "], %s%s", negative ? "-" : "",
2906 arm_regnames[given & 0xf]);
2910 break;
2912 case 'b':
2914 int disp = (((given & 0xffffff) ^ 0x800000) - 0x800000);
2915 info->print_address_func (disp * 4 + pc + 8, info);
2917 break;
2919 case 'c':
2920 if (((given >> 28) & 0xf) != 0xe)
2921 func (stream, "%s",
2922 arm_conditional [(given >> 28) & 0xf]);
2923 break;
2925 case 'm':
2927 int started = 0;
2928 int reg;
2930 func (stream, "{");
2931 for (reg = 0; reg < 16; reg++)
2932 if ((given & (1 << reg)) != 0)
2934 if (started)
2935 func (stream, ", ");
2936 started = 1;
2937 func (stream, "%s", arm_regnames[reg]);
2939 func (stream, "}");
2941 break;
2943 case 'q':
2944 arm_decode_shift (given, func, stream, FALSE);
2945 break;
2947 case 'o':
2948 if ((given & 0x02000000) != 0)
2950 int rotate = (given & 0xf00) >> 7;
2951 int immed = (given & 0xff);
2953 immed = (((immed << (32 - rotate))
2954 | (immed >> rotate)) & 0xffffffff);
2955 func (stream, "#%d", immed);
2956 value_in_comment = immed;
2958 else
2959 arm_decode_shift (given, func, stream, TRUE);
2960 break;
2962 case 'p':
2963 if ((given & 0x0000f000) == 0x0000f000)
2964 func (stream, "p");
2965 break;
2967 case 't':
2968 if ((given & 0x01200000) == 0x00200000)
2969 func (stream, "t");
2970 break;
2972 case 'A':
2974 int offset = given & 0xff;
2976 value_in_comment = offset * 4;
2977 if ((given & 0x00800000) == 0)
2978 value_in_comment = - value_in_comment;
2980 func (stream, "[%s", arm_regnames [(given >> 16) & 0xf]);
2982 if ((given & (1 << 24)) != 0)
2984 if (offset)
2985 func (stream, ", #%d]%s",
2986 value_in_comment,
2987 ((given & 0x00200000) != 0 ? "!" : ""));
2988 else
2989 func (stream, "]");
2991 else
2993 func (stream, "]");
2995 if (given & (1 << 21))
2997 if (offset)
2998 func (stream, ", #%d", value_in_comment);
3000 else
3002 func (stream, ", {%d}", offset);
3003 value_in_comment = offset;
3007 break;
3009 case 'B':
3010 /* Print ARM V5 BLX(1) address: pc+25 bits. */
3012 bfd_vma address;
3013 bfd_vma offset = 0;
3015 if (given & 0x00800000)
3016 /* Is signed, hi bits should be ones. */
3017 offset = (-1) ^ 0x00ffffff;
3019 /* Offset is (SignExtend(offset field)<<2). */
3020 offset += given & 0x00ffffff;
3021 offset <<= 2;
3022 address = offset + pc + 8;
3024 if (given & 0x01000000)
3025 /* H bit allows addressing to 2-byte boundaries. */
3026 address += 2;
3028 info->print_address_func (address, info);
3030 break;
3032 case 'C':
3033 func (stream, "_");
3034 if (given & 0x80000)
3035 func (stream, "f");
3036 if (given & 0x40000)
3037 func (stream, "s");
3038 if (given & 0x20000)
3039 func (stream, "x");
3040 if (given & 0x10000)
3041 func (stream, "c");
3042 break;
3044 case 'U':
3045 switch (given & 0xf)
3047 case 0xf: func (stream, "sy"); break;
3048 case 0x7: func (stream, "un"); break;
3049 case 0xe: func (stream, "st"); break;
3050 case 0x6: func (stream, "unst"); break;
3051 default:
3052 func (stream, "#%d", (int) given & 0xf);
3053 break;
3055 break;
3057 case '0': case '1': case '2': case '3': case '4':
3058 case '5': case '6': case '7': case '8': case '9':
3060 int width;
3061 unsigned long value;
3063 c = arm_decode_bitfield (c, given, &value, &width);
3065 switch (*c)
3067 case 'r':
3068 func (stream, "%s", arm_regnames[value]);
3069 break;
3070 case 'd':
3071 func (stream, "%ld", value);
3072 value_in_comment = value;
3073 break;
3074 case 'b':
3075 func (stream, "%ld", value * 8);
3076 value_in_comment = value * 8;
3077 break;
3078 case 'W':
3079 func (stream, "%ld", value + 1);
3080 value_in_comment = value + 1;
3081 break;
3082 case 'x':
3083 func (stream, "0x%08lx", value);
3085 /* Some SWI instructions have special
3086 meanings. */
3087 if ((given & 0x0fffffff) == 0x0FF00000)
3088 func (stream, "\t; IMB");
3089 else if ((given & 0x0fffffff) == 0x0FF00001)
3090 func (stream, "\t; IMBRange");
3091 break;
3092 case 'X':
3093 func (stream, "%01lx", value & 0xf);
3094 value_in_comment = value;
3095 break;
3096 case '`':
3097 c++;
3098 if (value == 0)
3099 func (stream, "%c", *c);
3100 break;
3101 case '\'':
3102 c++;
3103 if (value == ((1ul << width) - 1))
3104 func (stream, "%c", *c);
3105 break;
3106 case '?':
3107 func (stream, "%c", c[(1 << width) - (int) value]);
3108 c += 1 << width;
3109 break;
3110 default:
3111 abort ();
3113 break;
3115 case 'e':
3117 int imm;
3119 imm = (given & 0xf) | ((given & 0xfff00) >> 4);
3120 func (stream, "%d", imm);
3121 value_in_comment = imm;
3123 break;
3125 case 'E':
3126 /* LSB and WIDTH fields of BFI or BFC. The machine-
3127 language instruction encodes LSB and MSB. */
3129 long msb = (given & 0x001f0000) >> 16;
3130 long lsb = (given & 0x00000f80) >> 7;
3131 long width = msb - lsb + 1;
3133 if (width > 0)
3134 func (stream, "#%lu, #%lu", lsb, width);
3135 else
3136 func (stream, "(invalid: %lu:%lu)", lsb, msb);
3138 break;
3140 case 'V':
3141 /* 16-bit unsigned immediate from a MOVT or MOVW
3142 instruction, encoded in bits 0:11 and 15:19. */
3144 long hi = (given & 0x000f0000) >> 4;
3145 long lo = (given & 0x00000fff);
3146 long imm16 = hi | lo;
3148 func (stream, "#%lu", imm16);
3149 value_in_comment = imm16;
3151 break;
3153 default:
3154 abort ();
3158 else
3159 func (stream, "%c", *c);
3162 if (value_in_comment > 32 || value_in_comment < -16)
3163 func (stream, "\t; 0x%lx", (value_in_comment & 0xffffffffUL));
3164 return;
3167 abort ();
3170 /* Print one 16-bit Thumb instruction from PC on INFO->STREAM. */
3172 static void
3173 print_insn_thumb16 (bfd_vma pc, struct disassemble_info *info, long given)
3175 const struct opcode16 *insn;
3176 void *stream = info->stream;
3177 fprintf_ftype func = info->fprintf_func;
3179 for (insn = thumb_opcodes; insn->assembler; insn++)
3180 if ((given & insn->mask) == insn->value)
3182 signed long value_in_comment = 0;
3183 const char *c = insn->assembler;
3185 for (; *c; c++)
3187 int domaskpc = 0;
3188 int domasklr = 0;
3190 if (*c != '%')
3192 func (stream, "%c", *c);
3193 continue;
3196 switch (*++c)
3198 case '%':
3199 func (stream, "%%");
3200 break;
3202 case 'c':
3203 if (ifthen_state)
3204 func (stream, "%s", arm_conditional[IFTHEN_COND]);
3205 break;
3207 case 'C':
3208 if (ifthen_state)
3209 func (stream, "%s", arm_conditional[IFTHEN_COND]);
3210 else
3211 func (stream, "s");
3212 break;
3214 case 'I':
3216 unsigned int tmp;
3218 ifthen_next_state = given & 0xff;
3219 for (tmp = given << 1; tmp & 0xf; tmp <<= 1)
3220 func (stream, ((given ^ tmp) & 0x10) ? "e" : "t");
3221 func (stream, "\t%s", arm_conditional[(given >> 4) & 0xf]);
3223 break;
3225 case 'x':
3226 if (ifthen_next_state)
3227 func (stream, "\t; unpredictable branch in IT block\n");
3228 break;
3230 case 'X':
3231 if (ifthen_state)
3232 func (stream, "\t; unpredictable <IT:%s>",
3233 arm_conditional[IFTHEN_COND]);
3234 break;
3236 case 'S':
3238 long reg;
3240 reg = (given >> 3) & 0x7;
3241 if (given & (1 << 6))
3242 reg += 8;
3244 func (stream, "%s", arm_regnames[reg]);
3246 break;
3248 case 'D':
3250 long reg;
3252 reg = given & 0x7;
3253 if (given & (1 << 7))
3254 reg += 8;
3256 func (stream, "%s", arm_regnames[reg]);
3258 break;
3260 case 'N':
3261 if (given & (1 << 8))
3262 domasklr = 1;
3263 /* Fall through. */
3264 case 'O':
3265 if (*c == 'O' && (given & (1 << 8)))
3266 domaskpc = 1;
3267 /* Fall through. */
3268 case 'M':
3270 int started = 0;
3271 int reg;
3273 func (stream, "{");
3275 /* It would be nice if we could spot
3276 ranges, and generate the rS-rE format: */
3277 for (reg = 0; (reg < 8); reg++)
3278 if ((given & (1 << reg)) != 0)
3280 if (started)
3281 func (stream, ", ");
3282 started = 1;
3283 func (stream, "%s", arm_regnames[reg]);
3286 if (domasklr)
3288 if (started)
3289 func (stream, ", ");
3290 started = 1;
3291 func (stream, arm_regnames[14] /* "lr" */);
3294 if (domaskpc)
3296 if (started)
3297 func (stream, ", ");
3298 func (stream, arm_regnames[15] /* "pc" */);
3301 func (stream, "}");
3303 break;
3305 case 'b':
3306 /* Print ARM V6T2 CZB address: pc+4+6 bits. */
3308 bfd_vma address = (pc + 4
3309 + ((given & 0x00f8) >> 2)
3310 + ((given & 0x0200) >> 3));
3311 info->print_address_func (address, info);
3313 break;
3315 case 's':
3316 /* Right shift immediate -- bits 6..10; 1-31 print
3317 as themselves, 0 prints as 32. */
3319 long imm = (given & 0x07c0) >> 6;
3320 if (imm == 0)
3321 imm = 32;
3322 func (stream, "#%ld", imm);
3324 break;
3326 case '0': case '1': case '2': case '3': case '4':
3327 case '5': case '6': case '7': case '8': case '9':
3329 int bitstart = *c++ - '0';
3330 int bitend = 0;
3332 while (*c >= '0' && *c <= '9')
3333 bitstart = (bitstart * 10) + *c++ - '0';
3335 switch (*c)
3337 case '-':
3339 long reg;
3341 c++;
3342 while (*c >= '0' && *c <= '9')
3343 bitend = (bitend * 10) + *c++ - '0';
3344 if (!bitend)
3345 abort ();
3346 reg = given >> bitstart;
3347 reg &= (2 << (bitend - bitstart)) - 1;
3348 switch (*c)
3350 case 'r':
3351 func (stream, "%s", arm_regnames[reg]);
3352 break;
3354 case 'd':
3355 func (stream, "%ld", reg);
3356 value_in_comment = reg;
3357 break;
3359 case 'H':
3360 func (stream, "%ld", reg << 1);
3361 value_in_comment = reg << 1;
3362 break;
3364 case 'W':
3365 func (stream, "%ld", reg << 2);
3366 value_in_comment = reg << 2;
3367 break;
3369 case 'a':
3370 /* PC-relative address -- the bottom two
3371 bits of the address are dropped
3372 before the calculation. */
3373 info->print_address_func
3374 (((pc + 4) & ~3) + (reg << 2), info);
3375 value_in_comment = 0;
3376 break;
3378 case 'x':
3379 func (stream, "0x%04lx", reg);
3380 break;
3382 case 'B':
3383 reg = ((reg ^ (1 << bitend)) - (1 << bitend));
3384 info->print_address_func (reg * 2 + pc + 4, info);
3385 value_in_comment = 0;
3386 break;
3388 case 'c':
3389 func (stream, "%s", arm_conditional [reg]);
3390 break;
3392 default:
3393 abort ();
3396 break;
3398 case '\'':
3399 c++;
3400 if ((given & (1 << bitstart)) != 0)
3401 func (stream, "%c", *c);
3402 break;
3404 case '?':
3405 ++c;
3406 if ((given & (1 << bitstart)) != 0)
3407 func (stream, "%c", *c++);
3408 else
3409 func (stream, "%c", *++c);
3410 break;
3412 default:
3413 abort ();
3416 break;
3418 default:
3419 abort ();
3423 if (value_in_comment > 32 || value_in_comment < -16)
3424 func (stream, "\t; 0x%lx", value_in_comment);
3425 return;
3428 /* No match. */
3429 abort ();
3432 /* Return the name of an V7M special register. */
3434 static const char *
3435 psr_name (int regno)
3437 switch (regno)
3439 case 0: return "APSR";
3440 case 1: return "IAPSR";
3441 case 2: return "EAPSR";
3442 case 3: return "PSR";
3443 case 5: return "IPSR";
3444 case 6: return "EPSR";
3445 case 7: return "IEPSR";
3446 case 8: return "MSP";
3447 case 9: return "PSP";
3448 case 16: return "PRIMASK";
3449 case 17: return "BASEPRI";
3450 case 18: return "BASEPRI_MASK";
3451 case 19: return "FAULTMASK";
3452 case 20: return "CONTROL";
3453 default: return "<unknown>";
3457 /* Print one 32-bit Thumb instruction from PC on INFO->STREAM. */
3459 static void
3460 print_insn_thumb32 (bfd_vma pc, struct disassemble_info *info, long given)
3462 const struct opcode32 *insn;
3463 void *stream = info->stream;
3464 fprintf_ftype func = info->fprintf_func;
3466 if (print_insn_coprocessor (pc, info, given, TRUE))
3467 return;
3469 if (print_insn_neon (info, given, TRUE))
3470 return;
3472 for (insn = thumb32_opcodes; insn->assembler; insn++)
3473 if ((given & insn->mask) == insn->value)
3475 signed long value_in_comment = 0;
3476 const char *c = insn->assembler;
3478 for (; *c; c++)
3480 if (*c != '%')
3482 func (stream, "%c", *c);
3483 continue;
3486 switch (*++c)
3488 case '%':
3489 func (stream, "%%");
3490 break;
3492 case 'c':
3493 if (ifthen_state)
3494 func (stream, "%s", arm_conditional[IFTHEN_COND]);
3495 break;
3497 case 'x':
3498 if (ifthen_next_state)
3499 func (stream, "\t; unpredictable branch in IT block\n");
3500 break;
3502 case 'X':
3503 if (ifthen_state)
3504 func (stream, "\t; unpredictable <IT:%s>",
3505 arm_conditional[IFTHEN_COND]);
3506 break;
3508 case 'I':
3510 unsigned int imm12 = 0;
3512 imm12 |= (given & 0x000000ffu);
3513 imm12 |= (given & 0x00007000u) >> 4;
3514 imm12 |= (given & 0x04000000u) >> 15;
3515 func (stream, "#%u", imm12);
3516 value_in_comment = imm12;
3518 break;
3520 case 'M':
3522 unsigned int bits = 0, imm, imm8, mod;
3524 bits |= (given & 0x000000ffu);
3525 bits |= (given & 0x00007000u) >> 4;
3526 bits |= (given & 0x04000000u) >> 15;
3527 imm8 = (bits & 0x0ff);
3528 mod = (bits & 0xf00) >> 8;
3529 switch (mod)
3531 case 0: imm = imm8; break;
3532 case 1: imm = ((imm8<<16) | imm8); break;
3533 case 2: imm = ((imm8<<24) | (imm8 << 8)); break;
3534 case 3: imm = ((imm8<<24) | (imm8 << 16) | (imm8 << 8) | imm8); break;
3535 default:
3536 mod = (bits & 0xf80) >> 7;
3537 imm8 = (bits & 0x07f) | 0x80;
3538 imm = (((imm8 << (32 - mod)) | (imm8 >> mod)) & 0xffffffff);
3540 func (stream, "#%u", imm);
3541 value_in_comment = imm;
3543 break;
3545 case 'J':
3547 unsigned int imm = 0;
3549 imm |= (given & 0x000000ffu);
3550 imm |= (given & 0x00007000u) >> 4;
3551 imm |= (given & 0x04000000u) >> 15;
3552 imm |= (given & 0x000f0000u) >> 4;
3553 func (stream, "#%u", imm);
3554 value_in_comment = imm;
3556 break;
3558 case 'K':
3560 unsigned int imm = 0;
3562 imm |= (given & 0x000f0000u) >> 16;
3563 imm |= (given & 0x00000ff0u) >> 0;
3564 imm |= (given & 0x0000000fu) << 12;
3565 func (stream, "#%u", imm);
3566 value_in_comment = imm;
3568 break;
3570 case 'S':
3572 unsigned int reg = (given & 0x0000000fu);
3573 unsigned int stp = (given & 0x00000030u) >> 4;
3574 unsigned int imm = 0;
3575 imm |= (given & 0x000000c0u) >> 6;
3576 imm |= (given & 0x00007000u) >> 10;
3578 func (stream, "%s", arm_regnames[reg]);
3579 switch (stp)
3581 case 0:
3582 if (imm > 0)
3583 func (stream, ", lsl #%u", imm);
3584 break;
3586 case 1:
3587 if (imm == 0)
3588 imm = 32;
3589 func (stream, ", lsr #%u", imm);
3590 break;
3592 case 2:
3593 if (imm == 0)
3594 imm = 32;
3595 func (stream, ", asr #%u", imm);
3596 break;
3598 case 3:
3599 if (imm == 0)
3600 func (stream, ", rrx");
3601 else
3602 func (stream, ", ror #%u", imm);
3605 break;
3607 case 'a':
3609 unsigned int Rn = (given & 0x000f0000) >> 16;
3610 unsigned int U = (given & 0x00800000) >> 23;
3611 unsigned int op = (given & 0x00000f00) >> 8;
3612 unsigned int i12 = (given & 0x00000fff);
3613 unsigned int i8 = (given & 0x000000ff);
3614 bfd_boolean writeback = FALSE, postind = FALSE;
3615 int offset = 0;
3617 func (stream, "[%s", arm_regnames[Rn]);
3618 if (U) /* 12-bit positive immediate offset. */
3620 offset = i12;
3621 if (Rn != 15)
3622 value_in_comment = offset;
3624 else if (Rn == 15) /* 12-bit negative immediate offset. */
3625 offset = - (int) i12;
3626 else if (op == 0x0) /* Shifted register offset. */
3628 unsigned int Rm = (i8 & 0x0f);
3629 unsigned int sh = (i8 & 0x30) >> 4;
3631 func (stream, ", %s", arm_regnames[Rm]);
3632 if (sh)
3633 func (stream, ", lsl #%u", sh);
3634 func (stream, "]");
3635 break;
3637 else switch (op)
3639 case 0xE: /* 8-bit positive immediate offset. */
3640 offset = i8;
3641 break;
3643 case 0xC: /* 8-bit negative immediate offset. */
3644 offset = -i8;
3645 break;
3647 case 0xF: /* 8-bit + preindex with wb. */
3648 offset = i8;
3649 writeback = TRUE;
3650 break;
3652 case 0xD: /* 8-bit - preindex with wb. */
3653 offset = -i8;
3654 writeback = TRUE;
3655 break;
3657 case 0xB: /* 8-bit + postindex. */
3658 offset = i8;
3659 postind = TRUE;
3660 break;
3662 case 0x9: /* 8-bit - postindex. */
3663 offset = -i8;
3664 postind = TRUE;
3665 break;
3667 default:
3668 func (stream, ", <undefined>]");
3669 goto skip;
3672 if (postind)
3673 func (stream, "], #%d", offset);
3674 else
3676 if (offset)
3677 func (stream, ", #%d", offset);
3678 func (stream, writeback ? "]!" : "]");
3681 if (Rn == 15)
3683 func (stream, "\t; ");
3684 info->print_address_func (((pc + 4) & ~3) + offset, info);
3687 skip:
3688 break;
3690 case 'A':
3692 unsigned int P = (given & 0x01000000) >> 24;
3693 unsigned int U = (given & 0x00800000) >> 23;
3694 unsigned int W = (given & 0x00400000) >> 21;
3695 unsigned int Rn = (given & 0x000f0000) >> 16;
3696 unsigned int off = (given & 0x000000ff);
3698 func (stream, "[%s", arm_regnames[Rn]);
3699 if (P)
3701 if (off || !U)
3703 func (stream, ", #%c%u", U ? '+' : '-', off * 4);
3704 value_in_comment = off * 4 * U ? 1 : -1;
3706 func (stream, "]");
3707 if (W)
3708 func (stream, "!");
3710 else
3712 func (stream, "], ");
3713 if (W)
3715 func (stream, "#%c%u", U ? '+' : '-', off * 4);
3716 value_in_comment = off * 4 * U ? 1 : -1;
3718 else
3720 func (stream, "{%u}", off);
3721 value_in_comment = off;
3725 break;
3727 case 'w':
3729 unsigned int Sbit = (given & 0x01000000) >> 24;
3730 unsigned int type = (given & 0x00600000) >> 21;
3732 switch (type)
3734 case 0: func (stream, Sbit ? "sb" : "b"); break;
3735 case 1: func (stream, Sbit ? "sh" : "h"); break;
3736 case 2:
3737 if (Sbit)
3738 func (stream, "??");
3739 break;
3740 case 3:
3741 func (stream, "??");
3742 break;
3745 break;
3747 case 'm':
3749 int started = 0;
3750 int reg;
3752 func (stream, "{");
3753 for (reg = 0; reg < 16; reg++)
3754 if ((given & (1 << reg)) != 0)
3756 if (started)
3757 func (stream, ", ");
3758 started = 1;
3759 func (stream, "%s", arm_regnames[reg]);
3761 func (stream, "}");
3763 break;
3765 case 'E':
3767 unsigned int msb = (given & 0x0000001f);
3768 unsigned int lsb = 0;
3770 lsb |= (given & 0x000000c0u) >> 6;
3771 lsb |= (given & 0x00007000u) >> 10;
3772 func (stream, "#%u, #%u", lsb, msb - lsb + 1);
3774 break;
3776 case 'F':
3778 unsigned int width = (given & 0x0000001f) + 1;
3779 unsigned int lsb = 0;
3781 lsb |= (given & 0x000000c0u) >> 6;
3782 lsb |= (given & 0x00007000u) >> 10;
3783 func (stream, "#%u, #%u", lsb, width);
3785 break;
3787 case 'b':
3789 unsigned int S = (given & 0x04000000u) >> 26;
3790 unsigned int J1 = (given & 0x00002000u) >> 13;
3791 unsigned int J2 = (given & 0x00000800u) >> 11;
3792 int offset = 0;
3794 offset |= !S << 20;
3795 offset |= J2 << 19;
3796 offset |= J1 << 18;
3797 offset |= (given & 0x003f0000) >> 4;
3798 offset |= (given & 0x000007ff) << 1;
3799 offset -= (1 << 20);
3801 info->print_address_func (pc + 4 + offset, info);
3803 break;
3805 case 'B':
3807 unsigned int S = (given & 0x04000000u) >> 26;
3808 unsigned int I1 = (given & 0x00002000u) >> 13;
3809 unsigned int I2 = (given & 0x00000800u) >> 11;
3810 int offset = 0;
3812 offset |= !S << 24;
3813 offset |= !(I1 ^ S) << 23;
3814 offset |= !(I2 ^ S) << 22;
3815 offset |= (given & 0x03ff0000u) >> 4;
3816 offset |= (given & 0x000007ffu) << 1;
3817 offset -= (1 << 24);
3818 offset += pc + 4;
3820 /* BLX target addresses are always word aligned. */
3821 if ((given & 0x00001000u) == 0)
3822 offset &= ~2u;
3824 info->print_address_func (offset, info);
3826 break;
3828 case 's':
3830 unsigned int shift = 0;
3832 shift |= (given & 0x000000c0u) >> 6;
3833 shift |= (given & 0x00007000u) >> 10;
3834 if (given & 0x00200000u)
3835 func (stream, ", asr #%u", shift);
3836 else if (shift)
3837 func (stream, ", lsl #%u", shift);
3838 /* else print nothing - lsl #0 */
3840 break;
3842 case 'R':
3844 unsigned int rot = (given & 0x00000030) >> 4;
3846 if (rot)
3847 func (stream, ", ror #%u", rot * 8);
3849 break;
3851 case 'U':
3852 switch (given & 0xf)
3854 case 0xf: func (stream, "sy"); break;
3855 case 0x7: func (stream, "un"); break;
3856 case 0xe: func (stream, "st"); break;
3857 case 0x6: func (stream, "unst"); break;
3858 default:
3859 func (stream, "#%d", (int) given & 0xf);
3860 break;
3862 break;
3864 case 'C':
3865 if ((given & 0xff) == 0)
3867 func (stream, "%cPSR_", (given & 0x100000) ? 'S' : 'C');
3868 if (given & 0x800)
3869 func (stream, "f");
3870 if (given & 0x400)
3871 func (stream, "s");
3872 if (given & 0x200)
3873 func (stream, "x");
3874 if (given & 0x100)
3875 func (stream, "c");
3877 else
3879 func (stream, psr_name (given & 0xff));
3881 break;
3883 case 'D':
3884 if ((given & 0xff) == 0)
3885 func (stream, "%cPSR", (given & 0x100000) ? 'S' : 'C');
3886 else
3887 func (stream, psr_name (given & 0xff));
3888 break;
3890 case '0': case '1': case '2': case '3': case '4':
3891 case '5': case '6': case '7': case '8': case '9':
3893 int width;
3894 unsigned long val;
3896 c = arm_decode_bitfield (c, given, &val, &width);
3898 switch (*c)
3900 case 'd':
3901 func (stream, "%lu", val);
3902 value_in_comment = val;
3903 break;
3904 case 'W':
3905 func (stream, "%lu", val * 4);
3906 value_in_comment = val * 4;
3907 break;
3908 case 'r': func (stream, "%s", arm_regnames[val]); break;
3910 case 'c':
3911 func (stream, "%s", arm_conditional[val]);
3912 break;
3914 case '\'':
3915 c++;
3916 if (val == ((1ul << width) - 1))
3917 func (stream, "%c", *c);
3918 break;
3920 case '`':
3921 c++;
3922 if (val == 0)
3923 func (stream, "%c", *c);
3924 break;
3926 case '?':
3927 func (stream, "%c", c[(1 << width) - (int) val]);
3928 c += 1 << width;
3929 break;
3931 default:
3932 abort ();
3935 break;
3937 default:
3938 abort ();
3942 if (value_in_comment > 32 || value_in_comment < -16)
3943 func (stream, "\t; 0x%lx", value_in_comment);
3944 return;
3947 /* No match. */
3948 abort ();
3951 /* Print data bytes on INFO->STREAM. */
3953 static void
3954 print_insn_data (bfd_vma pc ATTRIBUTE_UNUSED,
3955 struct disassemble_info *info,
3956 long given)
3958 switch (info->bytes_per_chunk)
3960 case 1:
3961 info->fprintf_func (info->stream, ".byte\t0x%02lx", given);
3962 break;
3963 case 2:
3964 info->fprintf_func (info->stream, ".short\t0x%04lx", given);
3965 break;
3966 case 4:
3967 info->fprintf_func (info->stream, ".word\t0x%08lx", given);
3968 break;
3969 default:
3970 abort ();
3974 /* Disallow mapping symbols ($a, $b, $d, $t etc) from
3975 being displayed in symbol relative addresses. */
3977 bfd_boolean
3978 arm_symbol_is_valid (asymbol * sym,
3979 struct disassemble_info * info ATTRIBUTE_UNUSED)
3981 const char * name;
3983 if (sym == NULL)
3984 return FALSE;
3986 name = bfd_asymbol_name (sym);
3988 return (name && *name != '$');
3991 /* Parse an individual disassembler option. */
3993 void
3994 parse_arm_disassembler_option (char *option)
3996 if (option == NULL)
3997 return;
3999 if (CONST_STRNEQ (option, "reg-names-"))
4001 int i;
4003 option += 10;
4005 for (i = NUM_ARM_REGNAMES; i--;)
4006 if (strneq (option, regnames[i].name, strlen (regnames[i].name)))
4008 regname_selected = i;
4009 break;
4012 if (i < 0)
4013 /* XXX - should break 'option' at following delimiter. */
4014 fprintf (stderr, _("Unrecognised register name set: %s\n"), option);
4016 else if (CONST_STRNEQ (option, "force-thumb"))
4017 force_thumb = 1;
4018 else if (CONST_STRNEQ (option, "no-force-thumb"))
4019 force_thumb = 0;
4020 else
4021 /* XXX - should break 'option' at following delimiter. */
4022 fprintf (stderr, _("Unrecognised disassembler option: %s\n"), option);
4024 return;
4027 /* Parse the string of disassembler options, spliting it at whitespaces
4028 or commas. (Whitespace separators supported for backwards compatibility). */
4030 static void
4031 parse_disassembler_options (char *options)
4033 if (options == NULL)
4034 return;
4036 while (*options)
4038 parse_arm_disassembler_option (options);
4040 /* Skip forward to next seperator. */
4041 while ((*options) && (! ISSPACE (*options)) && (*options != ','))
4042 ++ options;
4043 /* Skip forward past seperators. */
4044 while (ISSPACE (*options) || (*options == ','))
4045 ++ options;
4049 /* Search back through the insn stream to determine if this instruction is
4050 conditionally executed. */
4052 static void
4053 find_ifthen_state (bfd_vma pc,
4054 struct disassemble_info *info,
4055 bfd_boolean little)
4057 unsigned char b[2];
4058 unsigned int insn;
4059 int status;
4060 /* COUNT is twice the number of instructions seen. It will be odd if we
4061 just crossed an instruction boundary. */
4062 int count;
4063 int it_count;
4064 unsigned int seen_it;
4065 bfd_vma addr;
4067 ifthen_address = pc;
4068 ifthen_state = 0;
4070 addr = pc;
4071 count = 1;
4072 it_count = 0;
4073 seen_it = 0;
4074 /* Scan backwards looking for IT instructions, keeping track of where
4075 instruction boundaries are. We don't know if something is actually an
4076 IT instruction until we find a definite instruction boundary. */
4077 for (;;)
4079 if (addr == 0 || info->symbol_at_address_func (addr, info))
4081 /* A symbol must be on an instruction boundary, and will not
4082 be within an IT block. */
4083 if (seen_it && (count & 1))
4084 break;
4086 return;
4088 addr -= 2;
4089 status = info->read_memory_func (addr, (bfd_byte *) b, 2, info);
4090 if (status)
4091 return;
4093 if (little)
4094 insn = (b[0]) | (b[1] << 8);
4095 else
4096 insn = (b[1]) | (b[0] << 8);
4097 if (seen_it)
4099 if ((insn & 0xf800) < 0xe800)
4101 /* Addr + 2 is an instruction boundary. See if this matches
4102 the expected boundary based on the position of the last
4103 IT candidate. */
4104 if (count & 1)
4105 break;
4106 seen_it = 0;
4109 if ((insn & 0xff00) == 0xbf00 && (insn & 0xf) != 0)
4111 /* This could be an IT instruction. */
4112 seen_it = insn;
4113 it_count = count >> 1;
4115 if ((insn & 0xf800) >= 0xe800)
4116 count++;
4117 else
4118 count = (count + 2) | 1;
4119 /* IT blocks contain at most 4 instructions. */
4120 if (count >= 8 && !seen_it)
4121 return;
4123 /* We found an IT instruction. */
4124 ifthen_state = (seen_it & 0xe0) | ((seen_it << it_count) & 0x1f);
4125 if ((ifthen_state & 0xf) == 0)
4126 ifthen_state = 0;
4129 /* Try to infer the code type (Arm or Thumb) from a symbol.
4130 Returns nonzero if *MAP_TYPE was set. */
4132 static int
4133 get_sym_code_type (struct disassemble_info *info,
4134 int n,
4135 enum map_type *map_type)
4137 elf_symbol_type *es;
4138 unsigned int type;
4139 const char *name;
4141 es = *(elf_symbol_type **)(info->symtab + n);
4142 type = ELF_ST_TYPE (es->internal_elf_sym.st_info);
4144 /* If the symbol has function type then use that. */
4145 if (type == STT_FUNC || type == STT_ARM_TFUNC)
4147 *map_type = (type == STT_ARM_TFUNC) ? MAP_THUMB : MAP_ARM;
4148 return TRUE;
4151 /* Check for mapping symbols. */
4152 name = bfd_asymbol_name (info->symtab[n]);
4153 if (name[0] == '$' && (name[1] == 'a' || name[1] == 't' || name[1] == 'd')
4154 && (name[2] == 0 || name[2] == '.'))
4156 *map_type = ((name[1] == 'a') ? MAP_ARM
4157 : (name[1] == 't') ? MAP_THUMB
4158 : MAP_DATA);
4159 return TRUE;
4162 return FALSE;
4165 /* Given a bfd_mach_arm_XXX value, this function fills in the fields
4166 of the supplied arm_feature_set structure with bitmasks indicating
4167 the support base architectures and coprocessor extensions.
4169 FIXME: This could more efficiently implemented as a constant array,
4170 although it would also be less robust. */
4172 static void
4173 select_arm_features (unsigned long mach,
4174 arm_feature_set * features)
4176 #undef ARM_FEATURE
4177 #define ARM_FEATURE(ARCH,CEXT) \
4178 features->core = (ARCH); \
4179 features->coproc = (CEXT) | FPU_FPA; \
4180 return
4182 switch (mach)
4184 case bfd_mach_arm_2: ARM_ARCH_V2;
4185 case bfd_mach_arm_2a: ARM_ARCH_V2S;
4186 case bfd_mach_arm_3: ARM_ARCH_V3;
4187 case bfd_mach_arm_3M: ARM_ARCH_V3M;
4188 case bfd_mach_arm_4: ARM_ARCH_V4;
4189 case bfd_mach_arm_4T: ARM_ARCH_V4T;
4190 case bfd_mach_arm_5: ARM_ARCH_V5;
4191 case bfd_mach_arm_5T: ARM_ARCH_V5T;
4192 case bfd_mach_arm_5TE: ARM_ARCH_V5TE;
4193 case bfd_mach_arm_XScale: ARM_ARCH_XSCALE;
4194 case bfd_mach_arm_ep9312: ARM_FEATURE (ARM_AEXT_V4T, ARM_CEXT_MAVERICK | FPU_MAVERICK);
4195 case bfd_mach_arm_iWMMXt: ARM_ARCH_IWMMXT;
4196 case bfd_mach_arm_iWMMXt2: ARM_ARCH_IWMMXT2;
4197 /* If the machine type is unknown allow all
4198 architecture types and all extensions. */
4199 case bfd_mach_arm_unknown: ARM_FEATURE (-1UL, -1UL);
4200 default:
4201 abort ();
4206 /* NOTE: There are no checks in these routines that
4207 the relevant number of data bytes exist. */
4209 static int
4210 print_insn (bfd_vma pc, struct disassemble_info *info, bfd_boolean little)
4212 unsigned char b[4];
4213 long given;
4214 int status;
4215 int is_thumb = FALSE;
4216 int is_data = FALSE;
4217 int little_code;
4218 unsigned int size = 4;
4219 void (*printer) (bfd_vma, struct disassemble_info *, long);
4220 bfd_boolean found = FALSE;
4222 if (info->disassembler_options)
4224 parse_disassembler_options (info->disassembler_options);
4226 /* To avoid repeated parsing of these options, we remove them here. */
4227 info->disassembler_options = NULL;
4230 /* PR 10288: Control which instructions will be disassembled. */
4231 if (info->private_data == NULL)
4233 static arm_feature_set features;
4235 if ((info->flags & USER_SPECIFIED_MACHINE_TYPE) == 0)
4236 /* If the user did not use the -m command line switch then default to
4237 disassembling all types of ARM instruction.
4239 The info->mach value has to be ignored as this will be based on
4240 the default archictecture for the target and/or hints in the notes
4241 section, but it will never be greater than the current largest arm
4242 machine value (iWMMXt2), which is only equivalent to the V5TE
4243 architecture. ARM architectures have advanced beyond the machine
4244 value encoding, and these newer architectures would be ignored if
4245 the machine value was used.
4247 Ie the -m switch is used to restrict which instructions will be
4248 disassembled. If it is necessary to use the -m switch to tell
4249 objdump that an ARM binary is being disassembled, eg because the
4250 input is a raw binary file, but it is also desired to disassemble
4251 all ARM instructions then use "-marm". This will select the
4252 "unknown" arm architecture which is compatible with any ARM
4253 instruction. */
4254 info->mach = bfd_mach_arm_unknown;
4256 /* Compute the architecture bitmask from the machine number.
4257 Note: This assumes that the machine number will not change
4258 during disassembly.... */
4259 select_arm_features (info->mach, & features);
4261 info->private_data = & features;
4264 /* Decide if our code is going to be little-endian, despite what the
4265 function argument might say. */
4266 little_code = ((info->endian_code == BFD_ENDIAN_LITTLE) || little);
4268 /* First check the full symtab for a mapping symbol, even if there
4269 are no usable non-mapping symbols for this address. */
4270 if (info->symtab_size != 0
4271 && bfd_asymbol_flavour (*info->symtab) == bfd_target_elf_flavour)
4273 bfd_vma addr;
4274 int n;
4275 int last_sym = -1;
4276 enum map_type type = MAP_ARM;
4278 if (pc <= last_mapping_addr)
4279 last_mapping_sym = -1;
4280 is_thumb = (last_type == MAP_THUMB);
4281 found = FALSE;
4282 /* Start scanning at the start of the function, or wherever
4283 we finished last time. */
4284 n = info->symtab_pos + 1;
4285 if (n < last_mapping_sym)
4286 n = last_mapping_sym;
4288 /* Scan up to the location being disassembled. */
4289 for (; n < info->symtab_size; n++)
4291 addr = bfd_asymbol_value (info->symtab[n]);
4292 if (addr > pc)
4293 break;
4294 if ((info->section == NULL
4295 || info->section == info->symtab[n]->section)
4296 && get_sym_code_type (info, n, &type))
4298 last_sym = n;
4299 found = TRUE;
4303 if (!found)
4305 n = info->symtab_pos;
4306 if (n < last_mapping_sym - 1)
4307 n = last_mapping_sym - 1;
4309 /* No mapping symbol found at this address. Look backwards
4310 for a preceeding one. */
4311 for (; n >= 0; n--)
4313 if ((info->section == NULL
4314 || info->section == info->symtab[n]->section)
4315 && get_sym_code_type (info, n, &type))
4317 last_sym = n;
4318 found = TRUE;
4319 break;
4324 last_mapping_sym = last_sym;
4325 last_type = type;
4326 is_thumb = (last_type == MAP_THUMB);
4327 is_data = (last_type == MAP_DATA);
4329 /* Look a little bit ahead to see if we should print out
4330 two or four bytes of data. If there's a symbol,
4331 mapping or otherwise, after two bytes then don't
4332 print more. */
4333 if (is_data)
4335 size = 4 - (pc & 3);
4336 for (n = last_sym + 1; n < info->symtab_size; n++)
4338 addr = bfd_asymbol_value (info->symtab[n]);
4339 if (addr > pc)
4341 if (addr - pc < size)
4342 size = addr - pc;
4343 break;
4346 /* If the next symbol is after three bytes, we need to
4347 print only part of the data, so that we can use either
4348 .byte or .short. */
4349 if (size == 3)
4350 size = (pc & 1) ? 1 : 2;
4354 if (info->symbols != NULL)
4356 if (bfd_asymbol_flavour (*info->symbols) == bfd_target_coff_flavour)
4358 coff_symbol_type * cs;
4360 cs = coffsymbol (*info->symbols);
4361 is_thumb = ( cs->native->u.syment.n_sclass == C_THUMBEXT
4362 || cs->native->u.syment.n_sclass == C_THUMBSTAT
4363 || cs->native->u.syment.n_sclass == C_THUMBLABEL
4364 || cs->native->u.syment.n_sclass == C_THUMBEXTFUNC
4365 || cs->native->u.syment.n_sclass == C_THUMBSTATFUNC);
4367 else if (bfd_asymbol_flavour (*info->symbols) == bfd_target_elf_flavour
4368 && !found)
4370 /* If no mapping symbol has been found then fall back to the type
4371 of the function symbol. */
4372 elf_symbol_type * es;
4373 unsigned int type;
4375 es = *(elf_symbol_type **)(info->symbols);
4376 type = ELF_ST_TYPE (es->internal_elf_sym.st_info);
4378 is_thumb = (type == STT_ARM_TFUNC) || (type == STT_ARM_16BIT);
4382 if (force_thumb)
4383 is_thumb = TRUE;
4385 if (is_data)
4386 info->display_endian = little ? BFD_ENDIAN_LITTLE : BFD_ENDIAN_BIG;
4387 else
4388 info->display_endian = little_code ? BFD_ENDIAN_LITTLE : BFD_ENDIAN_BIG;
4390 info->bytes_per_line = 4;
4392 /* PR 10263: Disassemble data if requested to do so by the user. */
4393 if (is_data && ((info->flags & DISASSEMBLE_DATA) == 0))
4395 int i;
4397 /* Size was already set above. */
4398 info->bytes_per_chunk = size;
4399 printer = print_insn_data;
4401 status = info->read_memory_func (pc, (bfd_byte *) b, size, info);
4402 given = 0;
4403 if (little)
4404 for (i = size - 1; i >= 0; i--)
4405 given = b[i] | (given << 8);
4406 else
4407 for (i = 0; i < (int) size; i++)
4408 given = b[i] | (given << 8);
4410 else if (!is_thumb)
4412 /* In ARM mode endianness is a straightforward issue: the instruction
4413 is four bytes long and is either ordered 0123 or 3210. */
4414 printer = print_insn_arm;
4415 info->bytes_per_chunk = 4;
4416 size = 4;
4418 status = info->read_memory_func (pc, (bfd_byte *) b, 4, info);
4419 if (little_code)
4420 given = (b[0]) | (b[1] << 8) | (b[2] << 16) | (b[3] << 24);
4421 else
4422 given = (b[3]) | (b[2] << 8) | (b[1] << 16) | (b[0] << 24);
4424 else
4426 /* In Thumb mode we have the additional wrinkle of two
4427 instruction lengths. Fortunately, the bits that determine
4428 the length of the current instruction are always to be found
4429 in the first two bytes. */
4430 printer = print_insn_thumb16;
4431 info->bytes_per_chunk = 2;
4432 size = 2;
4434 status = info->read_memory_func (pc, (bfd_byte *) b, 2, info);
4435 if (little_code)
4436 given = (b[0]) | (b[1] << 8);
4437 else
4438 given = (b[1]) | (b[0] << 8);
4440 if (!status)
4442 /* These bit patterns signal a four-byte Thumb
4443 instruction. */
4444 if ((given & 0xF800) == 0xF800
4445 || (given & 0xF800) == 0xF000
4446 || (given & 0xF800) == 0xE800)
4448 status = info->read_memory_func (pc + 2, (bfd_byte *) b, 2, info);
4449 if (little_code)
4450 given = (b[0]) | (b[1] << 8) | (given << 16);
4451 else
4452 given = (b[1]) | (b[0] << 8) | (given << 16);
4454 printer = print_insn_thumb32;
4455 size = 4;
4459 if (ifthen_address != pc)
4460 find_ifthen_state (pc, info, little_code);
4462 if (ifthen_state)
4464 if ((ifthen_state & 0xf) == 0x8)
4465 ifthen_next_state = 0;
4466 else
4467 ifthen_next_state = (ifthen_state & 0xe0)
4468 | ((ifthen_state & 0xf) << 1);
4472 if (status)
4474 info->memory_error_func (status, pc, info);
4475 return -1;
4477 if (info->flags & INSN_HAS_RELOC)
4478 /* If the instruction has a reloc associated with it, then
4479 the offset field in the instruction will actually be the
4480 addend for the reloc. (We are using REL type relocs).
4481 In such cases, we can ignore the pc when computing
4482 addresses, since the addend is not currently pc-relative. */
4483 pc = 0;
4485 printer (pc, info, given);
4487 if (is_thumb)
4489 ifthen_state = ifthen_next_state;
4490 ifthen_address += size;
4492 return size;
4496 print_insn_big_arm (bfd_vma pc, struct disassemble_info *info)
4498 /* Detect BE8-ness and record it in the disassembler info. */
4499 if (info->flavour == bfd_target_elf_flavour
4500 && info->section != NULL
4501 && (elf_elfheader (info->section->owner)->e_flags & EF_ARM_BE8))
4502 info->endian_code = BFD_ENDIAN_LITTLE;
4504 return print_insn (pc, info, FALSE);
4508 print_insn_little_arm (bfd_vma pc, struct disassemble_info *info)
4510 return print_insn (pc, info, TRUE);
4513 void
4514 print_arm_disassembler_options (FILE *stream)
4516 int i;
4518 fprintf (stream, _("\n\
4519 The following ARM specific disassembler options are supported for use with\n\
4520 the -M switch:\n"));
4522 for (i = NUM_ARM_REGNAMES; i--;)
4523 fprintf (stream, " reg-names-%s %*c%s\n",
4524 regnames[i].name,
4525 (int)(14 - strlen (regnames[i].name)), ' ',
4526 regnames[i].description);
4528 fprintf (stream, " force-thumb Assume all insns are Thumb insns\n");
4529 fprintf (stream, " no-force-thumb Examine preceeding label to determine an insn's type\n\n");