]> git.proxmox.com Git - mirror_qemu.git/blob - disas/riscv-xthead.c
99da679d16c8760ad3c117c803db58a2e4d78c1d
[mirror_qemu.git] / disas / riscv-xthead.c
1 /*
2 * QEMU RISC-V Disassembler for xthead.
3 *
4 * SPDX-License-Identifier: GPL-2.0-or-later
5 */
6
7 #include "disas/riscv.h"
8 #include "disas/riscv-xthead.h"
9
10 typedef enum {
11 /* 0 is reserved for rv_op_illegal. */
12 /* XTheadBa */
13 rv_op_th_addsl = 1,
14 /* XTheadBb */
15 rv_op_th_srri,
16 rv_op_th_srriw,
17 rv_op_th_ext,
18 rv_op_th_extu,
19 rv_op_th_ff0,
20 rv_op_th_ff1,
21 rv_op_th_rev,
22 rv_op_th_revw,
23 rv_op_th_tstnbz,
24 /* XTheadBs */
25 rv_op_th_tst,
26 /* XTheadCmo */
27 rv_op_th_dcache_call,
28 rv_op_th_dcache_ciall,
29 rv_op_th_dcache_iall,
30 rv_op_th_dcache_cpa,
31 rv_op_th_dcache_cipa,
32 rv_op_th_dcache_ipa,
33 rv_op_th_dcache_cva,
34 rv_op_th_dcache_civa,
35 rv_op_th_dcache_iva,
36 rv_op_th_dcache_csw,
37 rv_op_th_dcache_cisw,
38 rv_op_th_dcache_isw,
39 rv_op_th_dcache_cpal1,
40 rv_op_th_dcache_cval1,
41 rv_op_th_icache_iall,
42 rv_op_th_icache_ialls,
43 rv_op_th_icache_ipa,
44 rv_op_th_icache_iva,
45 rv_op_th_l2cache_call,
46 rv_op_th_l2cache_ciall,
47 rv_op_th_l2cache_iall,
48 /* XTheadCondMov */
49 rv_op_th_mveqz,
50 rv_op_th_mvnez,
51 /* XTheadFMemIdx */
52 rv_op_th_flrd,
53 rv_op_th_flrw,
54 rv_op_th_flurd,
55 rv_op_th_flurw,
56 rv_op_th_fsrd,
57 rv_op_th_fsrw,
58 rv_op_th_fsurd,
59 rv_op_th_fsurw,
60 /* XTheadFmv */
61 rv_op_th_fmv_hw_x,
62 rv_op_th_fmv_x_hw,
63 /* XTheadMac */
64 rv_op_th_mula,
65 rv_op_th_mulah,
66 rv_op_th_mulaw,
67 rv_op_th_muls,
68 rv_op_th_mulsw,
69 rv_op_th_mulsh,
70 /* XTheadMemIdx */
71 rv_op_th_lbia,
72 rv_op_th_lbib,
73 rv_op_th_lbuia,
74 rv_op_th_lbuib,
75 rv_op_th_lhia,
76 rv_op_th_lhib,
77 rv_op_th_lhuia,
78 rv_op_th_lhuib,
79 rv_op_th_lwia,
80 rv_op_th_lwib,
81 rv_op_th_lwuia,
82 rv_op_th_lwuib,
83 rv_op_th_ldia,
84 rv_op_th_ldib,
85 rv_op_th_sbia,
86 rv_op_th_sbib,
87 rv_op_th_shia,
88 rv_op_th_shib,
89 rv_op_th_swia,
90 rv_op_th_swib,
91 rv_op_th_sdia,
92 rv_op_th_sdib,
93 rv_op_th_lrb,
94 rv_op_th_lrbu,
95 rv_op_th_lrh,
96 rv_op_th_lrhu,
97 rv_op_th_lrw,
98 rv_op_th_lrwu,
99 rv_op_th_lrd,
100 rv_op_th_srb,
101 rv_op_th_srh,
102 rv_op_th_srw,
103 rv_op_th_srd,
104 rv_op_th_lurb,
105 rv_op_th_lurbu,
106 rv_op_th_lurh,
107 rv_op_th_lurhu,
108 rv_op_th_lurw,
109 rv_op_th_lurwu,
110 rv_op_th_lurd,
111 rv_op_th_surb,
112 rv_op_th_surh,
113 rv_op_th_surw,
114 rv_op_th_surd,
115 /* XTheadMemPair */
116 rv_op_th_ldd,
117 rv_op_th_lwd,
118 rv_op_th_lwud,
119 rv_op_th_sdd,
120 rv_op_th_swd,
121 /* XTheadSync */
122 rv_op_th_sfence_vmas,
123 rv_op_th_sync,
124 rv_op_th_sync_i,
125 rv_op_th_sync_is,
126 rv_op_th_sync_s,
127 } rv_xthead_op;
128
129 const rv_opcode_data xthead_opcode_data[] = {
130 { "th.illegal", rv_codec_illegal, rv_fmt_none, NULL, 0, 0, 0 },
131 /* XTheadBa */
132 { "th.addsl", rv_codec_r_imm2, rv_fmt_rd_rs1_rs2_imm, NULL, 0, 0, 0 },
133 /* XTheadBb */
134 { "th.srri", rv_codec_r2_imm6, rv_fmt_rd_rs1_imm, NULL, 0, 0, 0 },
135 { "th.srriw", rv_codec_r2_imm5, rv_fmt_rd_rs1_imm, NULL, 0, 0, 0 },
136 { "th.ext", rv_codec_r2_immhl, rv_fmt_rd_rs1_immh_imml, NULL, 0, 0, 0 },
137 { "th.extu", rv_codec_r2_immhl, rv_fmt_rd_rs1_immh_imml, NULL, 0, 0, 0 },
138 { "th.ff0", rv_codec_r2, rv_fmt_rd_rs1, NULL, 0, 0, 0 },
139 { "th.ff1", rv_codec_r2, rv_fmt_rd_rs1, NULL, 0, 0, 0 },
140 { "th.rev", rv_codec_r2, rv_fmt_rd_rs1, NULL, 0, 0, 0 },
141 { "th.revw", rv_codec_r2, rv_fmt_rd_rs1, NULL, 0, 0, 0 },
142 { "th.tstnbz", rv_codec_r2, rv_fmt_rd_rs1, NULL, 0, 0, 0 },
143 /* XTheadBs */
144 { "th.tst", rv_codec_r2_imm6, rv_fmt_rd_rs1_imm, NULL, 0, 0, 0 },
145 /* XTheadCmo */
146 { "th.dcache.call", rv_codec_none, rv_fmt_none, NULL, 0, 0, 0 },
147 { "th.dcache.ciall", rv_codec_none, rv_fmt_none, NULL, 0, 0, 0 },
148 { "th.dcache.iall", rv_codec_none, rv_fmt_none, NULL, 0, 0, 0 },
149 { "th.dcache.cpa", rv_codec_r, rv_fmt_rs1, NULL, 0, 0, 0 },
150 { "th.dcache.cipa", rv_codec_r, rv_fmt_rs1, NULL, 0, 0, 0 },
151 { "th.dcache.ipa", rv_codec_r, rv_fmt_rs1, NULL, 0, 0, 0 },
152 { "th.dcache.cva", rv_codec_r, rv_fmt_rs1, NULL, 0, 0, 0 },
153 { "th.dcache.civa", rv_codec_r, rv_fmt_rs1, NULL, 0, 0, 0 },
154 { "th.dcache.iva", rv_codec_r, rv_fmt_rs1, NULL, 0, 0, 0 },
155 { "th.dcache.csw", rv_codec_r, rv_fmt_rs1, NULL, 0, 0, 0 },
156 { "th.dcache.cisw", rv_codec_r, rv_fmt_rs1, NULL, 0, 0, 0 },
157 { "th.dcache.isw", rv_codec_r, rv_fmt_rs1, NULL, 0, 0, 0 },
158 { "th.dcache.cpal1", rv_codec_r, rv_fmt_rs1, NULL, 0, 0, 0 },
159 { "th.dcache.cval1", rv_codec_r, rv_fmt_rs1, NULL, 0, 0, 0 },
160 { "th.icache.iall", rv_codec_none, rv_fmt_none, NULL, 0, 0, 0 },
161 { "th.icache.ialls", rv_codec_none, rv_fmt_none, NULL, 0, 0, 0 },
162 { "th.icache.ipa", rv_codec_r, rv_fmt_rs1, NULL, 0, 0, 0 },
163 { "th.icache.iva", rv_codec_r, rv_fmt_rs1, NULL, 0, 0, 0 },
164 { "th.l2cache.call", rv_codec_none, rv_fmt_none, NULL, 0, 0, 0 },
165 { "th.l2cache.ciall", rv_codec_none, rv_fmt_none, NULL, 0, 0, 0 },
166 { "th.l2cache.iall", rv_codec_none, rv_fmt_none, NULL, 0, 0, 0 },
167 /* XTheadCondMov */
168 { "th.mveqz", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },
169 { "th.mvnez", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },
170 /* XTheadFMemIdx */
171 { "th.flrd", rv_codec_r_imm2, rv_fmt_frd_rs1_rs2_imm, NULL, 0, 0, 0 },
172 { "th.flrw", rv_codec_r_imm2, rv_fmt_frd_rs1_rs2_imm, NULL, 0, 0, 0 },
173 { "th.flurd", rv_codec_r_imm2, rv_fmt_frd_rs1_rs2_imm, NULL, 0, 0, 0 },
174 { "th.flurw", rv_codec_r_imm2, rv_fmt_frd_rs1_rs2_imm, NULL, 0, 0, 0 },
175 { "th.fsrd", rv_codec_r_imm2, rv_fmt_frd_rs1_rs2_imm, NULL, 0, 0, 0 },
176 { "th.fsrw", rv_codec_r_imm2, rv_fmt_frd_rs1_rs2_imm, NULL, 0, 0, 0 },
177 { "th.fsurd", rv_codec_r_imm2, rv_fmt_frd_rs1_rs2_imm, NULL, 0, 0, 0 },
178 { "th.fsurw", rv_codec_r_imm2, rv_fmt_frd_rs1_rs2_imm, NULL, 0, 0, 0 },
179 /* XTheadFmv */
180 { "th.fmv.hw.x", rv_codec_r, rv_fmt_rd_frs1, NULL, 0, 0, 0 },
181 { "th.fmv.x.hw", rv_codec_r, rv_fmt_rd_frs1, NULL, 0, 0, 0 },
182 /* XTheadMac */
183 { "th.mula", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },
184 { "th.mulaw", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },
185 { "th.mulah", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },
186 { "th.muls", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },
187 { "th.mulsw", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },
188 { "th.mulsh", rv_codec_r, rv_fmt_rd_rs1_rs2, NULL, 0, 0, 0 },
189 /* XTheadMemIdx */
190 { "th.lbia", rv_codec_r2_imm2_imm5, rv_fmt_rd_rs1_immh_imml_addr, NULL, 0, 0, 0 },
191 { "th.lbib", rv_codec_r2_imm2_imm5, rv_fmt_rd_rs1_immh_imml, NULL, 0, 0, 0 },
192 { "th.lbuia", rv_codec_r2_imm2_imm5, rv_fmt_rd_rs1_immh_imml_addr, NULL, 0, 0, 0 },
193 { "th.lbuib", rv_codec_r2_imm2_imm5, rv_fmt_rd_rs1_immh_imml_addr, NULL, 0, 0, 0 },
194 { "th.lhia", rv_codec_r2_imm2_imm5, rv_fmt_rd_rs1_immh_imml_addr, NULL, 0, 0, 0 },
195 { "th.lhib", rv_codec_r2_imm2_imm5, rv_fmt_rd_rs1_immh_imml_addr, NULL, 0, 0, 0 },
196 { "th.lhuia", rv_codec_r2_imm2_imm5, rv_fmt_rd_rs1_immh_imml_addr, NULL, 0, 0, 0 },
197 { "th.lhuib", rv_codec_r2_imm2_imm5, rv_fmt_rd_rs1_immh_imml_addr, NULL, 0, 0, 0 },
198 { "th.lwia", rv_codec_r2_imm2_imm5, rv_fmt_rd_rs1_immh_imml_addr, NULL, 0, 0, 0 },
199 { "th.lwib", rv_codec_r2_imm2_imm5, rv_fmt_rd_rs1_immh_imml_addr, NULL, 0, 0, 0 },
200 { "th.lwuia", rv_codec_r2_imm2_imm5, rv_fmt_rd_rs1_immh_imml_addr, NULL, 0, 0, 0 },
201 { "th.lwuib", rv_codec_r2_imm2_imm5, rv_fmt_rd_rs1_immh_imml_addr, NULL, 0, 0, 0 },
202 { "th.ldia", rv_codec_r2_imm2_imm5, rv_fmt_rd_rs1_immh_imml_addr, NULL, 0, 0, 0 },
203 { "th.ldib", rv_codec_r2_imm2_imm5, rv_fmt_rd_rs1_immh_imml_addr, NULL, 0, 0, 0 },
204 { "th.sbia", rv_codec_r2_imm2_imm5, rv_fmt_rd_rs1_immh_imml_addr, NULL, 0, 0, 0 },
205 { "th.sbib", rv_codec_r2_imm2_imm5, rv_fmt_rd_rs1_immh_imml_addr, NULL, 0, 0, 0 },
206 { "th.shia", rv_codec_r2_imm2_imm5, rv_fmt_rd_rs1_immh_imml_addr, NULL, 0, 0, 0 },
207 { "th.shib", rv_codec_r2_imm2_imm5, rv_fmt_rd_rs1_immh_imml_addr, NULL, 0, 0, 0 },
208 { "th.swia", rv_codec_r2_imm2_imm5, rv_fmt_rd_rs1_immh_imml_addr, NULL, 0, 0, 0 },
209 { "th.swib", rv_codec_r2_imm2_imm5, rv_fmt_rd_rs1_immh_imml_addr, NULL, 0, 0, 0 },
210 { "th.sdia", rv_codec_r2_imm2_imm5, rv_fmt_rd_rs1_immh_imml_addr, NULL, 0, 0, 0 },
211 { "th.sdib", rv_codec_r2_imm2_imm5, rv_fmt_rd_rs1_immh_imml_addr, NULL, 0, 0, 0 },
212 { "th.lrb", rv_codec_r_imm2, rv_fmt_rd_rs1_rs2_imm, NULL, 0, 0, 0 },
213 { "th.lrbu", rv_codec_r_imm2, rv_fmt_rd_rs1_rs2_imm, NULL, 0, 0, 0 },
214 { "th.lrh", rv_codec_r_imm2, rv_fmt_rd_rs1_rs2_imm, NULL, 0, 0, 0 },
215 { "th.lrhu", rv_codec_r_imm2, rv_fmt_rd_rs1_rs2_imm, NULL, 0, 0, 0 },
216 { "th.lrw", rv_codec_r_imm2, rv_fmt_rd_rs1_rs2_imm, NULL, 0, 0, 0 },
217 { "th.lrwu", rv_codec_r_imm2, rv_fmt_rd_rs1_rs2_imm, NULL, 0, 0, 0 },
218 { "th.lrd", rv_codec_r_imm2, rv_fmt_rd_rs1_rs2_imm, NULL, 0, 0, 0 },
219 { "th.srb", rv_codec_r_imm2, rv_fmt_rd_rs1_rs2_imm, NULL, 0, 0, 0 },
220 { "th.srh", rv_codec_r_imm2, rv_fmt_rd_rs1_rs2_imm, NULL, 0, 0, 0 },
221 { "th.srw", rv_codec_r_imm2, rv_fmt_rd_rs1_rs2_imm, NULL, 0, 0, 0 },
222 { "th.srd", rv_codec_r_imm2, rv_fmt_rd_rs1_rs2_imm, NULL, 0, 0, 0 },
223 { "th.lurb", rv_codec_r_imm2, rv_fmt_rd_rs1_rs2_imm, NULL, 0, 0, 0 },
224 { "th.lurbu", rv_codec_r_imm2, rv_fmt_rd_rs1_rs2_imm, NULL, 0, 0, 0 },
225 { "th.lurh", rv_codec_r_imm2, rv_fmt_rd_rs1_rs2_imm, NULL, 0, 0, 0 },
226 { "th.lurhu", rv_codec_r_imm2, rv_fmt_rd_rs1_rs2_imm, NULL, 0, 0, 0 },
227 { "th.lurw", rv_codec_r_imm2, rv_fmt_rd_rs1_rs2_imm, NULL, 0, 0, 0 },
228 { "th.lurwu", rv_codec_r_imm2, rv_fmt_rd_rs1_rs2_imm, NULL, 0, 0, 0 },
229 { "th.lurd", rv_codec_r_imm2, rv_fmt_rd_rs1_rs2_imm, NULL, 0, 0, 0 },
230 { "th.surb", rv_codec_r_imm2, rv_fmt_rd_rs1_rs2_imm, NULL, 0, 0, 0 },
231 { "th.surh", rv_codec_r_imm2, rv_fmt_rd_rs1_rs2_imm, NULL, 0, 0, 0 },
232 { "th.surw", rv_codec_r_imm2, rv_fmt_rd_rs1_rs2_imm, NULL, 0, 0, 0 },
233 { "th.surd", rv_codec_r_imm2, rv_fmt_rd_rs1_rs2_imm, NULL, 0, 0, 0 },
234 /* XTheadMemPair */
235 { "th.ldd", rv_codec_r_imm2, rv_fmt_rd2_imm, NULL, 0, 0, 0 },
236 { "th.lwd", rv_codec_r_imm2, rv_fmt_rd2_imm, NULL, 0, 0, 0 },
237 { "th.lwud", rv_codec_r_imm2, rv_fmt_rd2_imm, NULL, 0, 0, 0 },
238 { "th.sdd", rv_codec_r_imm2, rv_fmt_rd2_imm, NULL, 0, 0, 0 },
239 { "th.swd", rv_codec_r_imm2, rv_fmt_rd2_imm, NULL, 0, 0, 0 },
240 /* XTheadSync */
241 { "th.sfence.vmas", rv_codec_r, rv_fmt_rs1_rs2, NULL, 0, 0, 0 },
242 { "th.sync", rv_codec_none, rv_fmt_none, NULL, 0, 0, 0 },
243 { "th.sync.i", rv_codec_none, rv_fmt_none, NULL, 0, 0, 0 },
244 { "th.sync.is", rv_codec_none, rv_fmt_none, NULL, 0, 0, 0 },
245 { "th.sync.s", rv_codec_none, rv_fmt_none, NULL, 0, 0, 0 },
246 };
247
248 void decode_xtheadba(rv_decode *dec, rv_isa isa)
249 {
250 rv_inst inst = dec->inst;
251 rv_opcode op = rv_op_illegal;
252
253 switch (((inst >> 0) & 0b11)) {
254 case 3:
255 switch (((inst >> 2) & 0b11111)) {
256 case 2:
257 /* custom-0 */
258 switch ((inst >> 12) & 0b111) {
259 case 1:
260 switch ((inst >> 25) & 0b1111111) {
261 case 0b0000000:
262 case 0b0000001:
263 case 0b0000010:
264 case 0b0000011: op = rv_op_th_addsl; break;
265 }
266 break;
267 }
268 break;
269 /* custom-0 */
270 }
271 break;
272 }
273
274 dec->op = op;
275 }
276
277 void decode_xtheadbb(rv_decode *dec, rv_isa isa)
278 {
279 rv_inst inst = dec->inst;
280 rv_opcode op = rv_op_illegal;
281
282 switch (((inst >> 0) & 0b11)) {
283 case 3:
284 switch (((inst >> 2) & 0b11111)) {
285 case 2:
286 /* custom-0 */
287 switch ((inst >> 12) & 0b111) {
288 case 1:
289 switch ((inst >> 25) & 0b1111111) {
290 case 0b0001010: op = rv_op_th_srriw; break;
291 case 0b1000000:
292 if (((inst >> 20) & 0b11111) == 0) {
293 op = rv_op_th_tstnbz;
294 }
295 break;
296 case 0b1000001:
297 if (((inst >> 20) & 0b11111) == 0) {
298 op = rv_op_th_rev;
299 }
300 break;
301 case 0b1000010:
302 if (((inst >> 20) & 0b11111) == 0) {
303 op = rv_op_th_ff0;
304 }
305 break;
306 case 0b1000011:
307 if (((inst >> 20) & 0b11111) == 0) {
308 op = rv_op_th_ff1;
309 }
310 break;
311 case 0b1000100:
312 case 0b1001000:
313 if (((inst >> 20) & 0b11111) == 0) {
314 op = rv_op_th_revw;
315 }
316 break;
317 case 0b0000100:
318 case 0b0000101: op = rv_op_th_srri; break;
319 }
320 break;
321 case 2: op = rv_op_th_ext; break;
322 case 3: op = rv_op_th_extu; break;
323 }
324 break;
325 /* custom-0 */
326 }
327 break;
328 }
329
330 dec->op = op;
331 }
332
333 void decode_xtheadbs(rv_decode *dec, rv_isa isa)
334 {
335 rv_inst inst = dec->inst;
336 rv_opcode op = rv_op_illegal;
337
338 switch (((inst >> 0) & 0b11)) {
339 case 3:
340 switch (((inst >> 2) & 0b11111)) {
341 case 2:
342 /* custom-0 */
343 switch ((inst >> 12) & 0b111) {
344 case 1:
345 switch ((inst >> 26) & 0b111111) {
346 case 0b100010: op = rv_op_th_tst; break;
347 }
348 break;
349 }
350 break;
351 /* custom-0 */
352 }
353 break;
354 }
355
356 dec->op = op;
357 }
358
359 void decode_xtheadcmo(rv_decode *dec, rv_isa isa)
360 {
361 rv_inst inst = dec->inst;
362 rv_opcode op = rv_op_illegal;
363
364 switch (((inst >> 0) & 0b11)) {
365 case 3:
366 switch (((inst >> 2) & 0b11111)) {
367 case 2:
368 /* custom-0 */
369 switch ((inst >> 12) & 0b111) {
370 case 0:
371 switch ((inst >> 20 & 0b111111111111)) {
372 case 0b000000000001:
373 if (((inst >> 20) & 0b11111) == 0) {
374 op = rv_op_th_dcache_call;
375 }
376 break;
377 case 0b000000000011:
378 if (((inst >> 20) & 0b11111) == 0) {
379 op = rv_op_th_dcache_ciall;
380 }
381 break;
382 case 0b000000000010:
383 if (((inst >> 20) & 0b11111) == 0) {
384 op = rv_op_th_dcache_iall;
385 }
386 break;
387 case 0b000000101001: op = rv_op_th_dcache_cpa; break;
388 case 0b000000101011: op = rv_op_th_dcache_cipa; break;
389 case 0b000000101010: op = rv_op_th_dcache_ipa; break;
390 case 0b000000100101: op = rv_op_th_dcache_cva; break;
391 case 0b000000100111: op = rv_op_th_dcache_civa; break;
392 case 0b000000100110: op = rv_op_th_dcache_iva; break;
393 case 0b000000100001: op = rv_op_th_dcache_csw; break;
394 case 0b000000100011: op = rv_op_th_dcache_cisw; break;
395 case 0b000000100010: op = rv_op_th_dcache_isw; break;
396 case 0b000000101000: op = rv_op_th_dcache_cpal1; break;
397 case 0b000000100100: op = rv_op_th_dcache_cval1; break;
398 case 0b000000010000:
399 if (((inst >> 20) & 0b11111) == 0) {
400 op = rv_op_th_icache_iall;
401 }
402 break;
403 case 0b000000010001:
404 if (((inst >> 20) & 0b11111) == 0) {
405 op = rv_op_th_icache_ialls;
406 }
407 break;
408 case 0b000000111000: op = rv_op_th_icache_ipa; break;
409 case 0b000000110000: op = rv_op_th_icache_iva; break;
410 case 0b000000010101:
411 if (((inst >> 20) & 0b11111) == 0) {
412 op = rv_op_th_l2cache_call;
413 }
414 break;
415 case 0b000000010111:
416 if (((inst >> 20) & 0b11111) == 0) {
417 op = rv_op_th_l2cache_ciall;
418 }
419 break;
420 case 0b000000010110:
421 if (((inst >> 20) & 0b11111) == 0) {
422 op = rv_op_th_l2cache_iall;
423 }
424 break;
425 }
426 break;
427 }
428 break;
429 /* custom-0 */
430 }
431 break;
432 }
433
434 dec->op = op;
435 }
436
437 void decode_xtheadcondmov(rv_decode *dec, rv_isa isa)
438 {
439 rv_inst inst = dec->inst;
440 rv_opcode op = rv_op_illegal;
441
442 switch (((inst >> 0) & 0b11)) {
443 case 3:
444 switch (((inst >> 2) & 0b11111)) {
445 case 2:
446 /* custom-0 */
447 switch ((inst >> 12) & 0b111) {
448 case 1:
449 switch ((inst >> 25) & 0b1111111) {
450 case 0b0100000: op = rv_op_th_mveqz; break;
451 case 0b0100001: op = rv_op_th_mvnez; break;
452 }
453 break;
454 }
455 break;
456 /* custom-0 */
457 }
458 break;
459 }
460
461 dec->op = op;
462 }
463
464 void decode_xtheadfmemidx(rv_decode *dec, rv_isa isa)
465 {
466 rv_inst inst = dec->inst;
467 rv_opcode op = rv_op_illegal;
468
469 switch (((inst >> 0) & 0b11)) {
470 case 3:
471 switch (((inst >> 2) & 0b11111)) {
472 case 2:
473 /* custom-0 */
474 switch ((inst >> 12) & 0b111) {
475 case 6:
476 switch ((inst >> 27) & 0b11111) {
477 case 8: op = rv_op_th_flrw; break;
478 case 10: op = rv_op_th_flurw; break;
479 case 12: op = rv_op_th_flrd; break;
480 case 14: op = rv_op_th_flurd; break;
481 }
482 break;
483 case 7:
484 switch ((inst >> 27) & 0b11111) {
485 case 8: op = rv_op_th_fsrw; break;
486 case 10: op = rv_op_th_fsurw; break;
487 case 12: op = rv_op_th_fsrd; break;
488 case 14: op = rv_op_th_fsurd; break;
489 }
490 break;
491 }
492 break;
493 /* custom-0 */
494 }
495 break;
496 }
497
498 dec->op = op;
499 }
500
501 void decode_xtheadfmv(rv_decode *dec, rv_isa isa)
502 {
503 rv_inst inst = dec->inst;
504 rv_opcode op = rv_op_illegal;
505
506 switch (((inst >> 0) & 0b11)) {
507 case 3:
508 switch (((inst >> 2) & 0b11111)) {
509 case 2:
510 /* custom-0 */
511 switch ((inst >> 12) & 0b111) {
512 case 1:
513 switch ((inst >> 25) & 0b1111111) {
514 case 0b1010000:
515 if (((inst >> 20) & 0b11111) == 0) {
516 op = rv_op_th_fmv_hw_x;
517 }
518 break;
519 case 0b1100000:
520 if (((inst >> 20) & 0b11111) == 0) {
521 op = rv_op_th_fmv_x_hw;
522 }
523 break;
524 }
525 break;
526 }
527 break;
528 /* custom-0 */
529 }
530 break;
531 }
532
533 dec->op = op;
534 }
535
536 void decode_xtheadmac(rv_decode *dec, rv_isa isa)
537 {
538 rv_inst inst = dec->inst;
539 rv_opcode op = rv_op_illegal;
540
541 switch (((inst >> 0) & 0b11)) {
542 case 3:
543 switch (((inst >> 2) & 0b11111)) {
544 case 2:
545 /* custom-0 */
546 switch ((inst >> 12) & 0b111) {
547 case 1:
548 switch ((inst >> 25) & 0b1111111) {
549 case 0b0010000: op = rv_op_th_mula; break;
550 case 0b0010001: op = rv_op_th_muls; break;
551 case 0b0010010: op = rv_op_th_mulaw; break;
552 case 0b0010011: op = rv_op_th_mulsw; break;
553 case 0b0010100: op = rv_op_th_mulah; break;
554 case 0b0010101: op = rv_op_th_mulsh; break;
555 }
556 break;
557 }
558 break;
559 /* custom-0 */
560 }
561 break;
562 }
563
564 dec->op = op;
565 }
566
567 void decode_xtheadmemidx(rv_decode *dec, rv_isa isa)
568 {
569 rv_inst inst = dec->inst;
570 rv_opcode op = rv_op_illegal;
571
572 switch (((inst >> 0) & 0b11)) {
573 case 3:
574 switch (((inst >> 2) & 0b11111)) {
575 case 2:
576 /* custom-0 */
577 switch ((inst >> 12) & 0b111) {
578 case 4:
579 switch ((inst >> 27) & 0b11111) {
580 case 0: op = rv_op_th_lrb; break;
581 case 1: op = rv_op_th_lbib; break;
582 case 2: op = rv_op_th_lurb; break;
583 case 3: op = rv_op_th_lbia; break;
584 case 4: op = rv_op_th_lrh; break;
585 case 5: op = rv_op_th_lhib; break;
586 case 6: op = rv_op_th_lurh; break;
587 case 7: op = rv_op_th_lhia; break;
588 case 8: op = rv_op_th_lrw; break;
589 case 9: op = rv_op_th_lwib; break;
590 case 10: op = rv_op_th_lurw; break;
591 case 11: op = rv_op_th_lwia; break;
592 case 12: op = rv_op_th_lrd; break;
593 case 13: op = rv_op_th_ldib; break;
594 case 14: op = rv_op_th_lurd; break;
595 case 15: op = rv_op_th_ldia; break;
596 case 16: op = rv_op_th_lrbu; break;
597 case 17: op = rv_op_th_lbuib; break;
598 case 18: op = rv_op_th_lurbu; break;
599 case 19: op = rv_op_th_lbuia; break;
600 case 20: op = rv_op_th_lrhu; break;
601 case 21: op = rv_op_th_lhuib; break;
602 case 22: op = rv_op_th_lurhu; break;
603 case 23: op = rv_op_th_lhuia; break;
604 case 24: op = rv_op_th_lrwu; break;
605 case 25: op = rv_op_th_lwuib; break;
606 case 26: op = rv_op_th_lurwu; break;
607 case 27: op = rv_op_th_lwuia; break;
608 }
609 break;
610 case 5:
611 switch ((inst >> 27) & 0b11111) {
612 case 0: op = rv_op_th_srb; break;
613 case 1: op = rv_op_th_sbib; break;
614 case 2: op = rv_op_th_surb; break;
615 case 3: op = rv_op_th_sbia; break;
616 case 4: op = rv_op_th_srh; break;
617 case 5: op = rv_op_th_shib; break;
618 case 6: op = rv_op_th_surh; break;
619 case 7: op = rv_op_th_shia; break;
620 case 8: op = rv_op_th_srw; break;
621 case 9: op = rv_op_th_swib; break;
622 case 10: op = rv_op_th_surw; break;
623 case 11: op = rv_op_th_swia; break;
624 case 12: op = rv_op_th_srd; break;
625 case 13: op = rv_op_th_sdib; break;
626 case 14: op = rv_op_th_surd; break;
627 case 15: op = rv_op_th_sdia; break;
628 }
629 break;
630 break;
631 }
632 break;
633 /* custom-0 */
634 }
635 break;
636 }
637
638 dec->op = op;
639 }
640
641 void decode_xtheadmempair(rv_decode *dec, rv_isa isa)
642 {
643 rv_inst inst = dec->inst;
644 rv_opcode op = rv_op_illegal;
645
646 switch (((inst >> 0) & 0b11)) {
647 case 3:
648 switch (((inst >> 2) & 0b11111)) {
649 case 2:
650 /* custom-0 */
651 switch ((inst >> 12) & 0b111) {
652 case 4:
653 switch ((inst >> 27) & 0b11111) {
654 case 28: op = rv_op_th_lwd; break;
655 case 30: op = rv_op_th_lwud; break;
656 case 31: op = rv_op_th_ldd; break;
657 }
658 break;
659 case 5:
660 switch ((inst >> 27) & 0b11111) {
661 case 28: op = rv_op_th_swd; break;
662 case 31: op = rv_op_th_sdd; break;
663 }
664 break;
665 }
666 break;
667 /* custom-0 */
668 }
669 break;
670 }
671
672 dec->op = op;
673 }
674
675 void decode_xtheadsync(rv_decode *dec, rv_isa isa)
676 {
677 rv_inst inst = dec->inst;
678 rv_opcode op = rv_op_illegal;
679
680 switch (((inst >> 0) & 0b11)) {
681 case 3:
682 switch (((inst >> 2) & 0b11111)) {
683 case 2:
684 /* custom-0 */
685 switch ((inst >> 12) & 0b111) {
686 case 0:
687 switch ((inst >> 25) & 0b1111111) {
688 case 0b0000010: op = rv_op_th_sfence_vmas; break;
689 case 0b0000000:
690 switch ((inst >> 20) & 0b11111) {
691 case 0b11000: op = rv_op_th_sync; break;
692 case 0b11010: op = rv_op_th_sync_i; break;
693 case 0b11011: op = rv_op_th_sync_is; break;
694 case 0b11001: op = rv_op_th_sync_s; break;
695 }
696 break;
697 }
698 break;
699 }
700 break;
701 /* custom-0 */
702 }
703 break;
704 }
705
706 dec->op = op;
707 }