1 /* 2 * Xtensa ISA: 3 * http://www.tensilica.com/products/literature-docs/documentation/xtensa-isa-databook.htm 4 * 5 * Copyright (c) 2011, Max Filippov, Open Source and Linux Lab. 6 * All rights reserved. 7 * 8 * Redistribution and use in source and binary forms, with or without 9 * modification, are permitted provided that the following conditions are met: 10 * * Redistributions of source code must retain the above copyright 11 * notice, this list of conditions and the following disclaimer. 12 * * Redistributions in binary form must reproduce the above copyright 13 * notice, this list of conditions and the following disclaimer in the 14 * documentation and/or other materials provided with the distribution. 15 * * Neither the name of the Open Source and Linux Lab nor the 16 * names of its contributors may be used to endorse or promote products 17 * derived from this software without specific prior written permission. 18 * 19 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" 20 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 21 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 22 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY 23 * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES 24 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; 25 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND 26 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 27 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS 28 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 29 */ 30 31 #include "qemu/osdep.h" 32 33 #include "cpu.h" 34 #include "exec/exec-all.h" 35 #include "disas/disas.h" 36 #include "tcg/tcg-op.h" 37 #include "qemu/log.h" 38 #include "qemu/qemu-print.h" 39 #include "exec/cpu_ldst.h" 40 #include "semihosting/semihost.h" 41 #include "exec/translator.h" 42 43 #include "exec/helper-proto.h" 44 #include "exec/helper-gen.h" 45 46 #include "exec/log.h" 47 48 #define HELPER_H "helper.h" 49 #include "exec/helper-info.c.inc" 50 #undef HELPER_H 51 52 53 struct DisasContext { 54 DisasContextBase base; 55 const XtensaConfig *config; 56 uint32_t pc; 57 int cring; 58 int ring; 59 uint32_t lbeg_off; 60 uint32_t lend; 61 62 bool sar_5bit; 63 bool sar_m32_5bit; 64 TCGv_i32 sar_m32; 65 66 unsigned window; 67 unsigned callinc; 68 bool cwoe; 69 70 bool debug; 71 bool icount; 72 TCGv_i32 next_icount; 73 74 unsigned cpenable; 75 76 uint32_t op_flags; 77 xtensa_insnbuf_word insnbuf[MAX_INSNBUF_LENGTH]; 78 xtensa_insnbuf_word slotbuf[MAX_INSNBUF_LENGTH]; 79 }; 80 81 static TCGv_i32 cpu_pc; 82 static TCGv_i32 cpu_R[16]; 83 static TCGv_i32 cpu_FR[16]; 84 static TCGv_i64 cpu_FRD[16]; 85 static TCGv_i32 cpu_MR[4]; 86 static TCGv_i32 cpu_BR[16]; 87 static TCGv_i32 cpu_BR4[4]; 88 static TCGv_i32 cpu_BR8[2]; 89 static TCGv_i32 cpu_SR[256]; 90 static TCGv_i32 cpu_UR[256]; 91 static TCGv_i32 cpu_windowbase_next; 92 static TCGv_i32 cpu_exclusive_addr; 93 static TCGv_i32 cpu_exclusive_val; 94 95 static GHashTable *xtensa_regfile_table; 96 97 static char *sr_name[256]; 98 static char *ur_name[256]; 99 100 void xtensa_collect_sr_names(const XtensaConfig *config) 101 { 102 xtensa_isa isa = config->isa; 103 int n = xtensa_isa_num_sysregs(isa); 104 int i; 105 106 for (i = 0; i < n; ++i) { 107 int sr = xtensa_sysreg_number(isa, i); 108 109 if (sr >= 0 && sr < 256) { 110 const char *name = xtensa_sysreg_name(isa, i); 111 char **pname = 112 (xtensa_sysreg_is_user(isa, i) ? ur_name : sr_name) + sr; 113 114 if (*pname) { 115 if (strstr(*pname, name) == NULL) { 116 char *new_name = 117 malloc(strlen(*pname) + strlen(name) + 2); 118 119 strcpy(new_name, *pname); 120 strcat(new_name, "/"); 121 strcat(new_name, name); 122 free(*pname); 123 *pname = new_name; 124 } 125 } else { 126 *pname = strdup(name); 127 } 128 } 129 } 130 } 131 132 void xtensa_translate_init(void) 133 { 134 static const char * const regnames[] = { 135 "ar0", "ar1", "ar2", "ar3", 136 "ar4", "ar5", "ar6", "ar7", 137 "ar8", "ar9", "ar10", "ar11", 138 "ar12", "ar13", "ar14", "ar15", 139 }; 140 static const char * const fregnames[] = { 141 "f0", "f1", "f2", "f3", 142 "f4", "f5", "f6", "f7", 143 "f8", "f9", "f10", "f11", 144 "f12", "f13", "f14", "f15", 145 }; 146 static const char * const mregnames[] = { 147 "m0", "m1", "m2", "m3", 148 }; 149 static const char * const bregnames[] = { 150 "b0", "b1", "b2", "b3", 151 "b4", "b5", "b6", "b7", 152 "b8", "b9", "b10", "b11", 153 "b12", "b13", "b14", "b15", 154 }; 155 int i; 156 157 cpu_pc = tcg_global_mem_new_i32(tcg_env, 158 offsetof(CPUXtensaState, pc), "pc"); 159 160 for (i = 0; i < 16; i++) { 161 cpu_R[i] = tcg_global_mem_new_i32(tcg_env, 162 offsetof(CPUXtensaState, regs[i]), 163 regnames[i]); 164 } 165 166 for (i = 0; i < 16; i++) { 167 cpu_FR[i] = tcg_global_mem_new_i32(tcg_env, 168 offsetof(CPUXtensaState, 169 fregs[i].f32[FP_F32_LOW]), 170 fregnames[i]); 171 } 172 173 for (i = 0; i < 16; i++) { 174 cpu_FRD[i] = tcg_global_mem_new_i64(tcg_env, 175 offsetof(CPUXtensaState, 176 fregs[i].f64), 177 fregnames[i]); 178 } 179 180 for (i = 0; i < 4; i++) { 181 cpu_MR[i] = tcg_global_mem_new_i32(tcg_env, 182 offsetof(CPUXtensaState, 183 sregs[MR + i]), 184 mregnames[i]); 185 } 186 187 for (i = 0; i < 16; i++) { 188 cpu_BR[i] = tcg_global_mem_new_i32(tcg_env, 189 offsetof(CPUXtensaState, 190 sregs[BR]), 191 bregnames[i]); 192 if (i % 4 == 0) { 193 cpu_BR4[i / 4] = tcg_global_mem_new_i32(tcg_env, 194 offsetof(CPUXtensaState, 195 sregs[BR]), 196 bregnames[i]); 197 } 198 if (i % 8 == 0) { 199 cpu_BR8[i / 8] = tcg_global_mem_new_i32(tcg_env, 200 offsetof(CPUXtensaState, 201 sregs[BR]), 202 bregnames[i]); 203 } 204 } 205 206 for (i = 0; i < 256; ++i) { 207 if (sr_name[i]) { 208 cpu_SR[i] = tcg_global_mem_new_i32(tcg_env, 209 offsetof(CPUXtensaState, 210 sregs[i]), 211 sr_name[i]); 212 } 213 } 214 215 for (i = 0; i < 256; ++i) { 216 if (ur_name[i]) { 217 cpu_UR[i] = tcg_global_mem_new_i32(tcg_env, 218 offsetof(CPUXtensaState, 219 uregs[i]), 220 ur_name[i]); 221 } 222 } 223 224 cpu_windowbase_next = 225 tcg_global_mem_new_i32(tcg_env, 226 offsetof(CPUXtensaState, windowbase_next), 227 "windowbase_next"); 228 cpu_exclusive_addr = 229 tcg_global_mem_new_i32(tcg_env, 230 offsetof(CPUXtensaState, exclusive_addr), 231 "exclusive_addr"); 232 cpu_exclusive_val = 233 tcg_global_mem_new_i32(tcg_env, 234 offsetof(CPUXtensaState, exclusive_val), 235 "exclusive_val"); 236 } 237 238 void **xtensa_get_regfile_by_name(const char *name, int entries, int bits) 239 { 240 char *geometry_name; 241 void **res; 242 243 if (xtensa_regfile_table == NULL) { 244 xtensa_regfile_table = g_hash_table_new(g_str_hash, g_str_equal); 245 /* 246 * AR is special. Xtensa translator uses it as a current register 247 * window, but configuration overlays represent it as a complete 248 * physical register file. 249 */ 250 g_hash_table_insert(xtensa_regfile_table, 251 (void *)"AR 16x32", (void *)cpu_R); 252 g_hash_table_insert(xtensa_regfile_table, 253 (void *)"AR 32x32", (void *)cpu_R); 254 g_hash_table_insert(xtensa_regfile_table, 255 (void *)"AR 64x32", (void *)cpu_R); 256 257 g_hash_table_insert(xtensa_regfile_table, 258 (void *)"MR 4x32", (void *)cpu_MR); 259 260 g_hash_table_insert(xtensa_regfile_table, 261 (void *)"FR 16x32", (void *)cpu_FR); 262 g_hash_table_insert(xtensa_regfile_table, 263 (void *)"FR 16x64", (void *)cpu_FRD); 264 265 g_hash_table_insert(xtensa_regfile_table, 266 (void *)"BR 16x1", (void *)cpu_BR); 267 g_hash_table_insert(xtensa_regfile_table, 268 (void *)"BR4 4x4", (void *)cpu_BR4); 269 g_hash_table_insert(xtensa_regfile_table, 270 (void *)"BR8 2x8", (void *)cpu_BR8); 271 } 272 273 geometry_name = g_strdup_printf("%s %dx%d", name, entries, bits); 274 res = (void **)g_hash_table_lookup(xtensa_regfile_table, geometry_name); 275 g_free(geometry_name); 276 return res; 277 } 278 279 static inline bool option_enabled(DisasContext *dc, int opt) 280 { 281 return xtensa_option_enabled(dc->config, opt); 282 } 283 284 static void init_sar_tracker(DisasContext *dc) 285 { 286 dc->sar_5bit = false; 287 dc->sar_m32_5bit = false; 288 dc->sar_m32 = NULL; 289 } 290 291 static void gen_right_shift_sar(DisasContext *dc, TCGv_i32 sa) 292 { 293 tcg_gen_andi_i32(cpu_SR[SAR], sa, 0x1f); 294 if (dc->sar_m32_5bit) { 295 tcg_gen_discard_i32(dc->sar_m32); 296 } 297 dc->sar_5bit = true; 298 dc->sar_m32_5bit = false; 299 } 300 301 static void gen_left_shift_sar(DisasContext *dc, TCGv_i32 sa) 302 { 303 if (!dc->sar_m32) { 304 dc->sar_m32 = tcg_temp_new_i32(); 305 } 306 tcg_gen_andi_i32(dc->sar_m32, sa, 0x1f); 307 tcg_gen_sub_i32(cpu_SR[SAR], tcg_constant_i32(32), dc->sar_m32); 308 dc->sar_5bit = false; 309 dc->sar_m32_5bit = true; 310 } 311 312 static void gen_exception(DisasContext *dc, int excp) 313 { 314 gen_helper_exception(tcg_env, tcg_constant_i32(excp)); 315 } 316 317 static void gen_exception_cause(DisasContext *dc, uint32_t cause) 318 { 319 TCGv_i32 pc = tcg_constant_i32(dc->pc); 320 gen_helper_exception_cause(tcg_env, pc, tcg_constant_i32(cause)); 321 if (cause == ILLEGAL_INSTRUCTION_CAUSE || 322 cause == SYSCALL_CAUSE) { 323 dc->base.is_jmp = DISAS_NORETURN; 324 } 325 } 326 327 static void gen_debug_exception(DisasContext *dc, uint32_t cause) 328 { 329 TCGv_i32 pc = tcg_constant_i32(dc->pc); 330 gen_helper_debug_exception(tcg_env, pc, tcg_constant_i32(cause)); 331 if (cause & (DEBUGCAUSE_IB | DEBUGCAUSE_BI | DEBUGCAUSE_BN)) { 332 dc->base.is_jmp = DISAS_NORETURN; 333 } 334 } 335 336 static bool gen_check_privilege(DisasContext *dc) 337 { 338 #ifndef CONFIG_USER_ONLY 339 if (!dc->cring) { 340 return true; 341 } 342 #endif 343 gen_exception_cause(dc, PRIVILEGED_CAUSE); 344 dc->base.is_jmp = DISAS_NORETURN; 345 return false; 346 } 347 348 static bool gen_check_cpenable(DisasContext *dc, uint32_t cp_mask) 349 { 350 cp_mask &= ~dc->cpenable; 351 352 if (option_enabled(dc, XTENSA_OPTION_COPROCESSOR) && cp_mask) { 353 gen_exception_cause(dc, COPROCESSOR0_DISABLED + ctz32(cp_mask)); 354 dc->base.is_jmp = DISAS_NORETURN; 355 return false; 356 } 357 return true; 358 } 359 360 static int gen_postprocess(DisasContext *dc, int slot); 361 362 static void gen_jump_slot(DisasContext *dc, TCGv dest, int slot) 363 { 364 tcg_gen_mov_i32(cpu_pc, dest); 365 if (dc->icount) { 366 tcg_gen_mov_i32(cpu_SR[ICOUNT], dc->next_icount); 367 } 368 if (dc->op_flags & XTENSA_OP_POSTPROCESS) { 369 slot = gen_postprocess(dc, slot); 370 } 371 if (slot >= 0) { 372 tcg_gen_goto_tb(slot); 373 tcg_gen_exit_tb(dc->base.tb, slot); 374 } else { 375 tcg_gen_exit_tb(NULL, 0); 376 } 377 dc->base.is_jmp = DISAS_NORETURN; 378 } 379 380 static void gen_jump(DisasContext *dc, TCGv dest) 381 { 382 gen_jump_slot(dc, dest, -1); 383 } 384 385 static int adjust_jump_slot(DisasContext *dc, uint32_t dest, int slot) 386 { 387 return translator_use_goto_tb(&dc->base, dest) ? slot : -1; 388 } 389 390 static void gen_jumpi(DisasContext *dc, uint32_t dest, int slot) 391 { 392 gen_jump_slot(dc, tcg_constant_i32(dest), 393 adjust_jump_slot(dc, dest, slot)); 394 } 395 396 static void gen_callw_slot(DisasContext *dc, int callinc, TCGv_i32 dest, 397 int slot) 398 { 399 tcg_gen_deposit_i32(cpu_SR[PS], cpu_SR[PS], 400 tcg_constant_i32(callinc), PS_CALLINC_SHIFT, PS_CALLINC_LEN); 401 tcg_gen_movi_i32(cpu_R[callinc << 2], 402 (callinc << 30) | (dc->base.pc_next & 0x3fffffff)); 403 gen_jump_slot(dc, dest, slot); 404 } 405 406 static bool gen_check_loop_end(DisasContext *dc, int slot) 407 { 408 if (dc->base.pc_next == dc->lend) { 409 TCGLabel *label = gen_new_label(); 410 411 tcg_gen_brcondi_i32(TCG_COND_EQ, cpu_SR[LCOUNT], 0, label); 412 tcg_gen_subi_i32(cpu_SR[LCOUNT], cpu_SR[LCOUNT], 1); 413 if (dc->lbeg_off) { 414 gen_jumpi(dc, dc->base.pc_next - dc->lbeg_off, slot); 415 } else { 416 gen_jump(dc, cpu_SR[LBEG]); 417 } 418 gen_set_label(label); 419 gen_jumpi(dc, dc->base.pc_next, -1); 420 return true; 421 } 422 return false; 423 } 424 425 static void gen_jumpi_check_loop_end(DisasContext *dc, int slot) 426 { 427 if (!gen_check_loop_end(dc, slot)) { 428 gen_jumpi(dc, dc->base.pc_next, slot); 429 } 430 } 431 432 static void gen_brcond(DisasContext *dc, TCGCond cond, 433 TCGv_i32 t0, TCGv_i32 t1, uint32_t addr) 434 { 435 TCGLabel *label = gen_new_label(); 436 437 tcg_gen_brcond_i32(cond, t0, t1, label); 438 gen_jumpi_check_loop_end(dc, 0); 439 gen_set_label(label); 440 gen_jumpi(dc, addr, 1); 441 } 442 443 static void gen_brcondi(DisasContext *dc, TCGCond cond, 444 TCGv_i32 t0, uint32_t t1, uint32_t addr) 445 { 446 gen_brcond(dc, cond, t0, tcg_constant_i32(t1), addr); 447 } 448 449 static uint32_t test_exceptions_sr(DisasContext *dc, const OpcodeArg arg[], 450 const uint32_t par[]) 451 { 452 return xtensa_option_enabled(dc->config, par[1]) ? 0 : XTENSA_OP_ILL; 453 } 454 455 static uint32_t test_exceptions_ccompare(DisasContext *dc, 456 const OpcodeArg arg[], 457 const uint32_t par[]) 458 { 459 unsigned n = par[0] - CCOMPARE; 460 461 if (n >= dc->config->nccompare) { 462 return XTENSA_OP_ILL; 463 } 464 return test_exceptions_sr(dc, arg, par); 465 } 466 467 static uint32_t test_exceptions_dbreak(DisasContext *dc, const OpcodeArg arg[], 468 const uint32_t par[]) 469 { 470 unsigned n = MAX_NDBREAK; 471 472 if (par[0] >= DBREAKA && par[0] < DBREAKA + MAX_NDBREAK) { 473 n = par[0] - DBREAKA; 474 } 475 if (par[0] >= DBREAKC && par[0] < DBREAKC + MAX_NDBREAK) { 476 n = par[0] - DBREAKC; 477 } 478 if (n >= dc->config->ndbreak) { 479 return XTENSA_OP_ILL; 480 } 481 return test_exceptions_sr(dc, arg, par); 482 } 483 484 static uint32_t test_exceptions_ibreak(DisasContext *dc, const OpcodeArg arg[], 485 const uint32_t par[]) 486 { 487 unsigned n = par[0] - IBREAKA; 488 489 if (n >= dc->config->nibreak) { 490 return XTENSA_OP_ILL; 491 } 492 return test_exceptions_sr(dc, arg, par); 493 } 494 495 static uint32_t test_exceptions_hpi(DisasContext *dc, const OpcodeArg arg[], 496 const uint32_t par[]) 497 { 498 unsigned n = MAX_NLEVEL + 1; 499 500 if (par[0] >= EXCSAVE1 && par[0] < EXCSAVE1 + MAX_NLEVEL) { 501 n = par[0] - EXCSAVE1 + 1; 502 } 503 if (par[0] >= EPC1 && par[0] < EPC1 + MAX_NLEVEL) { 504 n = par[0] - EPC1 + 1; 505 } 506 if (par[0] >= EPS2 && par[0] < EPS2 + MAX_NLEVEL - 1) { 507 n = par[0] - EPS2 + 2; 508 } 509 if (n > dc->config->nlevel) { 510 return XTENSA_OP_ILL; 511 } 512 return test_exceptions_sr(dc, arg, par); 513 } 514 515 static MemOp gen_load_store_alignment(DisasContext *dc, MemOp mop, 516 TCGv_i32 addr) 517 { 518 if ((mop & MO_SIZE) == MO_8) { 519 return mop; 520 } 521 if ((mop & MO_AMASK) == MO_UNALN && 522 !option_enabled(dc, XTENSA_OPTION_HW_ALIGNMENT)) { 523 mop |= MO_ALIGN; 524 } 525 if (!option_enabled(dc, XTENSA_OPTION_UNALIGNED_EXCEPTION)) { 526 tcg_gen_andi_i32(addr, addr, ~0 << get_alignment_bits(mop)); 527 } 528 return mop; 529 } 530 531 static bool gen_window_check(DisasContext *dc, uint32_t mask) 532 { 533 unsigned r = 31 - clz32(mask); 534 535 if (r / 4 > dc->window) { 536 TCGv_i32 pc = tcg_constant_i32(dc->pc); 537 TCGv_i32 w = tcg_constant_i32(r / 4); 538 539 gen_helper_window_check(tcg_env, pc, w); 540 dc->base.is_jmp = DISAS_NORETURN; 541 return false; 542 } 543 return true; 544 } 545 546 static TCGv_i32 gen_mac16_m(TCGv_i32 v, bool hi, bool is_unsigned) 547 { 548 TCGv_i32 m = tcg_temp_new_i32(); 549 550 if (hi) { 551 (is_unsigned ? tcg_gen_shri_i32 : tcg_gen_sari_i32)(m, v, 16); 552 } else { 553 (is_unsigned ? tcg_gen_ext16u_i32 : tcg_gen_ext16s_i32)(m, v); 554 } 555 return m; 556 } 557 558 static void gen_zero_check(DisasContext *dc, const OpcodeArg arg[]) 559 { 560 TCGLabel *label = gen_new_label(); 561 562 tcg_gen_brcondi_i32(TCG_COND_NE, arg[2].in, 0, label); 563 gen_exception_cause(dc, INTEGER_DIVIDE_BY_ZERO_CAUSE); 564 gen_set_label(label); 565 } 566 567 static inline unsigned xtensa_op0_insn_len(DisasContext *dc, uint8_t op0) 568 { 569 return xtensa_isa_length_from_chars(dc->config->isa, &op0); 570 } 571 572 static int gen_postprocess(DisasContext *dc, int slot) 573 { 574 uint32_t op_flags = dc->op_flags; 575 576 #ifndef CONFIG_USER_ONLY 577 if (op_flags & XTENSA_OP_CHECK_INTERRUPTS) { 578 translator_io_start(&dc->base); 579 gen_helper_check_interrupts(tcg_env); 580 } 581 #endif 582 if (op_flags & XTENSA_OP_SYNC_REGISTER_WINDOW) { 583 gen_helper_sync_windowbase(tcg_env); 584 } 585 if (op_flags & XTENSA_OP_EXIT_TB_M1) { 586 slot = -1; 587 } 588 return slot; 589 } 590 591 struct opcode_arg_copy { 592 uint32_t resource; 593 void *temp; 594 OpcodeArg *arg; 595 }; 596 597 struct opcode_arg_info { 598 uint32_t resource; 599 int index; 600 }; 601 602 struct slot_prop { 603 XtensaOpcodeOps *ops; 604 OpcodeArg arg[MAX_OPCODE_ARGS]; 605 struct opcode_arg_info in[MAX_OPCODE_ARGS]; 606 struct opcode_arg_info out[MAX_OPCODE_ARGS]; 607 unsigned n_in; 608 unsigned n_out; 609 uint32_t op_flags; 610 }; 611 612 enum resource_type { 613 RES_REGFILE, 614 RES_STATE, 615 RES_MAX, 616 }; 617 618 static uint32_t encode_resource(enum resource_type r, unsigned g, unsigned n) 619 { 620 assert(r < RES_MAX && g < 256 && n < 65536); 621 return (r << 24) | (g << 16) | n; 622 } 623 624 static enum resource_type get_resource_type(uint32_t resource) 625 { 626 return resource >> 24; 627 } 628 629 /* 630 * a depends on b if b must be executed before a, 631 * because a's side effects will destroy b's inputs. 632 */ 633 static bool op_depends_on(const struct slot_prop *a, 634 const struct slot_prop *b) 635 { 636 unsigned i = 0; 637 unsigned j = 0; 638 639 if (a->op_flags & XTENSA_OP_CONTROL_FLOW) { 640 return true; 641 } 642 if ((a->op_flags & XTENSA_OP_LOAD_STORE) < 643 (b->op_flags & XTENSA_OP_LOAD_STORE)) { 644 return true; 645 } 646 while (i < a->n_out && j < b->n_in) { 647 if (a->out[i].resource < b->in[j].resource) { 648 ++i; 649 } else if (a->out[i].resource > b->in[j].resource) { 650 ++j; 651 } else { 652 return true; 653 } 654 } 655 return false; 656 } 657 658 /* 659 * Try to break a dependency on b, append temporary register copy records 660 * to the end of copy and update n_copy in case of success. 661 * This is not always possible: e.g. control flow must always be the last, 662 * load/store must be first and state dependencies are not supported yet. 663 */ 664 static bool break_dependency(struct slot_prop *a, 665 struct slot_prop *b, 666 struct opcode_arg_copy *copy, 667 unsigned *n_copy) 668 { 669 unsigned i = 0; 670 unsigned j = 0; 671 unsigned n = *n_copy; 672 bool rv = false; 673 674 if (a->op_flags & XTENSA_OP_CONTROL_FLOW) { 675 return false; 676 } 677 if ((a->op_flags & XTENSA_OP_LOAD_STORE) < 678 (b->op_flags & XTENSA_OP_LOAD_STORE)) { 679 return false; 680 } 681 while (i < a->n_out && j < b->n_in) { 682 if (a->out[i].resource < b->in[j].resource) { 683 ++i; 684 } else if (a->out[i].resource > b->in[j].resource) { 685 ++j; 686 } else { 687 int index = b->in[j].index; 688 689 if (get_resource_type(a->out[i].resource) != RES_REGFILE || 690 index < 0) { 691 return false; 692 } 693 copy[n].resource = b->in[j].resource; 694 copy[n].arg = b->arg + index; 695 ++n; 696 ++j; 697 rv = true; 698 } 699 } 700 *n_copy = n; 701 return rv; 702 } 703 704 /* 705 * Calculate evaluation order for slot opcodes. 706 * Build opcode order graph and output its nodes in topological sort order. 707 * An edge a -> b in the graph means that opcode a must be followed by 708 * opcode b. 709 */ 710 static bool tsort(struct slot_prop *slot, 711 struct slot_prop *sorted[], 712 unsigned n, 713 struct opcode_arg_copy *copy, 714 unsigned *n_copy) 715 { 716 struct tsnode { 717 unsigned n_in_edge; 718 unsigned n_out_edge; 719 unsigned out_edge[MAX_INSN_SLOTS]; 720 } node[MAX_INSN_SLOTS]; 721 722 unsigned in[MAX_INSN_SLOTS]; 723 unsigned i, j; 724 unsigned n_in = 0; 725 unsigned n_out = 0; 726 unsigned n_edge = 0; 727 unsigned in_idx = 0; 728 unsigned node_idx = 0; 729 730 for (i = 0; i < n; ++i) { 731 node[i].n_in_edge = 0; 732 node[i].n_out_edge = 0; 733 } 734 735 for (i = 0; i < n; ++i) { 736 unsigned n_out_edge = 0; 737 738 for (j = 0; j < n; ++j) { 739 if (i != j && op_depends_on(slot + j, slot + i)) { 740 node[i].out_edge[n_out_edge] = j; 741 ++node[j].n_in_edge; 742 ++n_out_edge; 743 ++n_edge; 744 } 745 } 746 node[i].n_out_edge = n_out_edge; 747 } 748 749 for (i = 0; i < n; ++i) { 750 if (!node[i].n_in_edge) { 751 in[n_in] = i; 752 ++n_in; 753 } 754 } 755 756 again: 757 for (; in_idx < n_in; ++in_idx) { 758 i = in[in_idx]; 759 sorted[n_out] = slot + i; 760 ++n_out; 761 for (j = 0; j < node[i].n_out_edge; ++j) { 762 --n_edge; 763 if (--node[node[i].out_edge[j]].n_in_edge == 0) { 764 in[n_in] = node[i].out_edge[j]; 765 ++n_in; 766 } 767 } 768 } 769 if (n_edge) { 770 for (; node_idx < n; ++node_idx) { 771 struct tsnode *cnode = node + node_idx; 772 773 if (cnode->n_in_edge) { 774 for (j = 0; j < cnode->n_out_edge; ++j) { 775 unsigned k = cnode->out_edge[j]; 776 777 if (break_dependency(slot + k, slot + node_idx, 778 copy, n_copy) && 779 --node[k].n_in_edge == 0) { 780 in[n_in] = k; 781 ++n_in; 782 --n_edge; 783 cnode->out_edge[j] = 784 cnode->out_edge[cnode->n_out_edge - 1]; 785 --cnode->n_out_edge; 786 goto again; 787 } 788 } 789 } 790 } 791 } 792 return n_edge == 0; 793 } 794 795 static void opcode_add_resource(struct slot_prop *op, 796 uint32_t resource, char direction, 797 int index) 798 { 799 switch (direction) { 800 case 'm': 801 case 'i': 802 assert(op->n_in < ARRAY_SIZE(op->in)); 803 op->in[op->n_in].resource = resource; 804 op->in[op->n_in].index = index; 805 ++op->n_in; 806 /* fall through */ 807 case 'o': 808 if (direction == 'm' || direction == 'o') { 809 assert(op->n_out < ARRAY_SIZE(op->out)); 810 op->out[op->n_out].resource = resource; 811 op->out[op->n_out].index = index; 812 ++op->n_out; 813 } 814 break; 815 default: 816 g_assert_not_reached(); 817 } 818 } 819 820 static int resource_compare(const void *a, const void *b) 821 { 822 const struct opcode_arg_info *pa = a; 823 const struct opcode_arg_info *pb = b; 824 825 return pa->resource < pb->resource ? 826 -1 : (pa->resource > pb->resource ? 1 : 0); 827 } 828 829 static int arg_copy_compare(const void *a, const void *b) 830 { 831 const struct opcode_arg_copy *pa = a; 832 const struct opcode_arg_copy *pb = b; 833 834 return pa->resource < pb->resource ? 835 -1 : (pa->resource > pb->resource ? 1 : 0); 836 } 837 838 static void disas_xtensa_insn(CPUXtensaState *env, DisasContext *dc) 839 { 840 xtensa_isa isa = dc->config->isa; 841 unsigned char b[MAX_INSN_LENGTH] = {translator_ldub(env, &dc->base, 842 dc->pc)}; 843 unsigned len = xtensa_op0_insn_len(dc, b[0]); 844 xtensa_format fmt; 845 int slot, slots; 846 unsigned i; 847 uint32_t op_flags = 0; 848 struct slot_prop slot_prop[MAX_INSN_SLOTS]; 849 struct slot_prop *ordered[MAX_INSN_SLOTS]; 850 struct opcode_arg_copy arg_copy[MAX_INSN_SLOTS * MAX_OPCODE_ARGS]; 851 unsigned n_arg_copy = 0; 852 uint32_t debug_cause = 0; 853 uint32_t windowed_register = 0; 854 uint32_t coprocessor = 0; 855 856 if (len == XTENSA_UNDEFINED) { 857 qemu_log_mask(LOG_GUEST_ERROR, 858 "unknown instruction length (pc = %08x)\n", 859 dc->pc); 860 gen_exception_cause(dc, ILLEGAL_INSTRUCTION_CAUSE); 861 dc->base.pc_next = dc->pc + 1; 862 return; 863 } 864 865 dc->base.pc_next = dc->pc + len; 866 for (i = 1; i < len; ++i) { 867 b[i] = translator_ldub(env, &dc->base, dc->pc + i); 868 } 869 xtensa_insnbuf_from_chars(isa, dc->insnbuf, b, len); 870 fmt = xtensa_format_decode(isa, dc->insnbuf); 871 if (fmt == XTENSA_UNDEFINED) { 872 qemu_log_mask(LOG_GUEST_ERROR, 873 "unrecognized instruction format (pc = %08x)\n", 874 dc->pc); 875 gen_exception_cause(dc, ILLEGAL_INSTRUCTION_CAUSE); 876 return; 877 } 878 slots = xtensa_format_num_slots(isa, fmt); 879 for (slot = 0; slot < slots; ++slot) { 880 xtensa_opcode opc; 881 int opnd, vopnd, opnds; 882 OpcodeArg *arg = slot_prop[slot].arg; 883 XtensaOpcodeOps *ops; 884 885 xtensa_format_get_slot(isa, fmt, slot, dc->insnbuf, dc->slotbuf); 886 opc = xtensa_opcode_decode(isa, fmt, slot, dc->slotbuf); 887 if (opc == XTENSA_UNDEFINED) { 888 qemu_log_mask(LOG_GUEST_ERROR, 889 "unrecognized opcode in slot %d (pc = %08x)\n", 890 slot, dc->pc); 891 gen_exception_cause(dc, ILLEGAL_INSTRUCTION_CAUSE); 892 return; 893 } 894 opnds = xtensa_opcode_num_operands(isa, opc); 895 896 for (opnd = vopnd = 0; opnd < opnds; ++opnd) { 897 void **register_file = NULL; 898 xtensa_regfile rf; 899 900 if (xtensa_operand_is_register(isa, opc, opnd)) { 901 rf = xtensa_operand_regfile(isa, opc, opnd); 902 register_file = dc->config->regfile[rf]; 903 904 if (rf == dc->config->a_regfile) { 905 uint32_t v; 906 907 xtensa_operand_get_field(isa, opc, opnd, fmt, slot, 908 dc->slotbuf, &v); 909 xtensa_operand_decode(isa, opc, opnd, &v); 910 windowed_register |= 1u << v; 911 } 912 } 913 if (xtensa_operand_is_visible(isa, opc, opnd)) { 914 uint32_t v; 915 916 xtensa_operand_get_field(isa, opc, opnd, fmt, slot, 917 dc->slotbuf, &v); 918 xtensa_operand_decode(isa, opc, opnd, &v); 919 arg[vopnd].raw_imm = v; 920 if (xtensa_operand_is_PCrelative(isa, opc, opnd)) { 921 xtensa_operand_undo_reloc(isa, opc, opnd, &v, dc->pc); 922 } 923 arg[vopnd].imm = v; 924 if (register_file) { 925 arg[vopnd].in = register_file[v]; 926 arg[vopnd].out = register_file[v]; 927 arg[vopnd].num_bits = xtensa_regfile_num_bits(isa, rf); 928 } else { 929 arg[vopnd].num_bits = 32; 930 } 931 ++vopnd; 932 } 933 } 934 ops = dc->config->opcode_ops[opc]; 935 slot_prop[slot].ops = ops; 936 937 if (ops) { 938 op_flags |= ops->op_flags; 939 if (ops->test_exceptions) { 940 op_flags |= ops->test_exceptions(dc, arg, ops->par); 941 } 942 } else { 943 qemu_log_mask(LOG_UNIMP, 944 "unimplemented opcode '%s' in slot %d (pc = %08x)\n", 945 xtensa_opcode_name(isa, opc), slot, dc->pc); 946 op_flags |= XTENSA_OP_ILL; 947 } 948 if (op_flags & XTENSA_OP_ILL) { 949 gen_exception_cause(dc, ILLEGAL_INSTRUCTION_CAUSE); 950 return; 951 } 952 if (op_flags & XTENSA_OP_DEBUG_BREAK) { 953 debug_cause |= ops->par[0]; 954 } 955 if (ops->test_overflow) { 956 windowed_register |= ops->test_overflow(dc, arg, ops->par); 957 } 958 coprocessor |= ops->coprocessor; 959 960 if (slots > 1) { 961 slot_prop[slot].n_in = 0; 962 slot_prop[slot].n_out = 0; 963 slot_prop[slot].op_flags = ops->op_flags & XTENSA_OP_LOAD_STORE; 964 965 opnds = xtensa_opcode_num_operands(isa, opc); 966 967 for (opnd = vopnd = 0; opnd < opnds; ++opnd) { 968 bool visible = xtensa_operand_is_visible(isa, opc, opnd); 969 970 if (xtensa_operand_is_register(isa, opc, opnd)) { 971 xtensa_regfile rf = xtensa_operand_regfile(isa, opc, opnd); 972 uint32_t v = 0; 973 974 xtensa_operand_get_field(isa, opc, opnd, fmt, slot, 975 dc->slotbuf, &v); 976 xtensa_operand_decode(isa, opc, opnd, &v); 977 opcode_add_resource(slot_prop + slot, 978 encode_resource(RES_REGFILE, rf, v), 979 xtensa_operand_inout(isa, opc, opnd), 980 visible ? vopnd : -1); 981 } 982 if (visible) { 983 ++vopnd; 984 } 985 } 986 987 opnds = xtensa_opcode_num_stateOperands(isa, opc); 988 989 for (opnd = 0; opnd < opnds; ++opnd) { 990 xtensa_state state = xtensa_stateOperand_state(isa, opc, opnd); 991 992 opcode_add_resource(slot_prop + slot, 993 encode_resource(RES_STATE, 0, state), 994 xtensa_stateOperand_inout(isa, opc, opnd), 995 -1); 996 } 997 if (xtensa_opcode_is_branch(isa, opc) || 998 xtensa_opcode_is_jump(isa, opc) || 999 xtensa_opcode_is_loop(isa, opc) || 1000 xtensa_opcode_is_call(isa, opc)) { 1001 slot_prop[slot].op_flags |= XTENSA_OP_CONTROL_FLOW; 1002 } 1003 1004 qsort(slot_prop[slot].in, slot_prop[slot].n_in, 1005 sizeof(slot_prop[slot].in[0]), resource_compare); 1006 qsort(slot_prop[slot].out, slot_prop[slot].n_out, 1007 sizeof(slot_prop[slot].out[0]), resource_compare); 1008 } 1009 } 1010 1011 if (slots > 1) { 1012 if (!tsort(slot_prop, ordered, slots, arg_copy, &n_arg_copy)) { 1013 qemu_log_mask(LOG_UNIMP, 1014 "Circular resource dependencies (pc = %08x)\n", 1015 dc->pc); 1016 gen_exception_cause(dc, ILLEGAL_INSTRUCTION_CAUSE); 1017 return; 1018 } 1019 } else { 1020 ordered[0] = slot_prop + 0; 1021 } 1022 1023 if ((op_flags & XTENSA_OP_PRIVILEGED) && 1024 !gen_check_privilege(dc)) { 1025 return; 1026 } 1027 1028 if (op_flags & XTENSA_OP_SYSCALL) { 1029 gen_exception_cause(dc, SYSCALL_CAUSE); 1030 return; 1031 } 1032 1033 if ((op_flags & XTENSA_OP_DEBUG_BREAK) && dc->debug) { 1034 gen_debug_exception(dc, debug_cause); 1035 return; 1036 } 1037 1038 if (windowed_register && !gen_window_check(dc, windowed_register)) { 1039 return; 1040 } 1041 1042 if (op_flags & XTENSA_OP_UNDERFLOW) { 1043 TCGv_i32 pc = tcg_constant_i32(dc->pc); 1044 1045 gen_helper_test_underflow_retw(tcg_env, pc); 1046 } 1047 1048 if (op_flags & XTENSA_OP_ALLOCA) { 1049 TCGv_i32 pc = tcg_constant_i32(dc->pc); 1050 1051 gen_helper_movsp(tcg_env, pc); 1052 } 1053 1054 if (coprocessor && !gen_check_cpenable(dc, coprocessor)) { 1055 return; 1056 } 1057 1058 if (n_arg_copy) { 1059 uint32_t resource; 1060 void *temp; 1061 unsigned j; 1062 1063 qsort(arg_copy, n_arg_copy, sizeof(*arg_copy), arg_copy_compare); 1064 for (i = j = 0; i < n_arg_copy; ++i) { 1065 if (i == 0 || arg_copy[i].resource != resource) { 1066 resource = arg_copy[i].resource; 1067 if (arg_copy[i].arg->num_bits <= 32) { 1068 temp = tcg_temp_new_i32(); 1069 tcg_gen_mov_i32(temp, arg_copy[i].arg->in); 1070 } else if (arg_copy[i].arg->num_bits <= 64) { 1071 temp = tcg_temp_new_i64(); 1072 tcg_gen_mov_i64(temp, arg_copy[i].arg->in); 1073 } else { 1074 g_assert_not_reached(); 1075 } 1076 arg_copy[i].temp = temp; 1077 1078 if (i != j) { 1079 arg_copy[j] = arg_copy[i]; 1080 } 1081 ++j; 1082 } 1083 arg_copy[i].arg->in = temp; 1084 } 1085 n_arg_copy = j; 1086 } 1087 1088 if (op_flags & XTENSA_OP_DIVIDE_BY_ZERO) { 1089 for (slot = 0; slot < slots; ++slot) { 1090 if (slot_prop[slot].ops->op_flags & XTENSA_OP_DIVIDE_BY_ZERO) { 1091 gen_zero_check(dc, slot_prop[slot].arg); 1092 } 1093 } 1094 } 1095 1096 dc->op_flags = op_flags; 1097 1098 for (slot = 0; slot < slots; ++slot) { 1099 struct slot_prop *pslot = ordered[slot]; 1100 XtensaOpcodeOps *ops = pslot->ops; 1101 1102 ops->translate(dc, pslot->arg, ops->par); 1103 } 1104 1105 if (dc->base.is_jmp == DISAS_NEXT) { 1106 gen_postprocess(dc, 0); 1107 dc->op_flags = 0; 1108 if (op_flags & XTENSA_OP_EXIT_TB_M1) { 1109 /* Change in mmu index, memory mapping or tb->flags; exit tb */ 1110 gen_jumpi_check_loop_end(dc, -1); 1111 } else if (op_flags & XTENSA_OP_EXIT_TB_0) { 1112 gen_jumpi_check_loop_end(dc, 0); 1113 } else { 1114 gen_check_loop_end(dc, 0); 1115 } 1116 } 1117 dc->pc = dc->base.pc_next; 1118 } 1119 1120 static inline unsigned xtensa_insn_len(CPUXtensaState *env, DisasContext *dc) 1121 { 1122 uint8_t b0 = cpu_ldub_code(env, dc->pc); 1123 return xtensa_op0_insn_len(dc, b0); 1124 } 1125 1126 static void gen_ibreak_check(CPUXtensaState *env, DisasContext *dc) 1127 { 1128 unsigned i; 1129 1130 for (i = 0; i < dc->config->nibreak; ++i) { 1131 if ((env->sregs[IBREAKENABLE] & (1 << i)) && 1132 env->sregs[IBREAKA + i] == dc->pc) { 1133 gen_debug_exception(dc, DEBUGCAUSE_IB); 1134 break; 1135 } 1136 } 1137 } 1138 1139 static void xtensa_tr_init_disas_context(DisasContextBase *dcbase, 1140 CPUState *cpu) 1141 { 1142 DisasContext *dc = container_of(dcbase, DisasContext, base); 1143 CPUXtensaState *env = cpu_env(cpu); 1144 uint32_t tb_flags = dc->base.tb->flags; 1145 1146 dc->config = env->config; 1147 dc->pc = dc->base.pc_first; 1148 dc->ring = tb_flags & XTENSA_TBFLAG_RING_MASK; 1149 dc->cring = (tb_flags & XTENSA_TBFLAG_EXCM) ? 0 : dc->ring; 1150 dc->lbeg_off = (dc->base.tb->cs_base & XTENSA_CSBASE_LBEG_OFF_MASK) >> 1151 XTENSA_CSBASE_LBEG_OFF_SHIFT; 1152 dc->lend = (dc->base.tb->cs_base & XTENSA_CSBASE_LEND_MASK) + 1153 (dc->base.pc_first & TARGET_PAGE_MASK); 1154 dc->debug = tb_flags & XTENSA_TBFLAG_DEBUG; 1155 dc->icount = tb_flags & XTENSA_TBFLAG_ICOUNT; 1156 dc->cpenable = (tb_flags & XTENSA_TBFLAG_CPENABLE_MASK) >> 1157 XTENSA_TBFLAG_CPENABLE_SHIFT; 1158 dc->window = ((tb_flags & XTENSA_TBFLAG_WINDOW_MASK) >> 1159 XTENSA_TBFLAG_WINDOW_SHIFT); 1160 dc->cwoe = tb_flags & XTENSA_TBFLAG_CWOE; 1161 dc->callinc = ((tb_flags & XTENSA_TBFLAG_CALLINC_MASK) >> 1162 XTENSA_TBFLAG_CALLINC_SHIFT); 1163 init_sar_tracker(dc); 1164 } 1165 1166 static void xtensa_tr_tb_start(DisasContextBase *dcbase, CPUState *cpu) 1167 { 1168 DisasContext *dc = container_of(dcbase, DisasContext, base); 1169 1170 if (dc->icount) { 1171 dc->next_icount = tcg_temp_new_i32(); 1172 } 1173 } 1174 1175 static void xtensa_tr_insn_start(DisasContextBase *dcbase, CPUState *cpu) 1176 { 1177 tcg_gen_insn_start(dcbase->pc_next); 1178 } 1179 1180 static void xtensa_tr_translate_insn(DisasContextBase *dcbase, CPUState *cpu) 1181 { 1182 DisasContext *dc = container_of(dcbase, DisasContext, base); 1183 CPUXtensaState *env = cpu_env(cpu); 1184 target_ulong page_start; 1185 1186 /* These two conditions only apply to the first insn in the TB, 1187 but this is the first TranslateOps hook that allows exiting. */ 1188 if ((tb_cflags(dc->base.tb) & CF_USE_ICOUNT) 1189 && (dc->base.tb->flags & XTENSA_TBFLAG_YIELD)) { 1190 gen_exception(dc, EXCP_YIELD); 1191 dc->base.pc_next = dc->pc + 1; 1192 dc->base.is_jmp = DISAS_NORETURN; 1193 return; 1194 } 1195 1196 if (dc->icount) { 1197 TCGLabel *label = gen_new_label(); 1198 1199 tcg_gen_addi_i32(dc->next_icount, cpu_SR[ICOUNT], 1); 1200 tcg_gen_brcondi_i32(TCG_COND_NE, dc->next_icount, 0, label); 1201 tcg_gen_mov_i32(dc->next_icount, cpu_SR[ICOUNT]); 1202 if (dc->debug) { 1203 gen_debug_exception(dc, DEBUGCAUSE_IC); 1204 } 1205 gen_set_label(label); 1206 } 1207 1208 if (dc->debug) { 1209 gen_ibreak_check(env, dc); 1210 } 1211 1212 disas_xtensa_insn(env, dc); 1213 1214 if (dc->icount) { 1215 tcg_gen_mov_i32(cpu_SR[ICOUNT], dc->next_icount); 1216 } 1217 1218 /* End the TB if the next insn will cross into the next page. */ 1219 page_start = dc->base.pc_first & TARGET_PAGE_MASK; 1220 if (dc->base.is_jmp == DISAS_NEXT && 1221 (dc->pc - page_start >= TARGET_PAGE_SIZE || 1222 dc->pc - page_start + xtensa_insn_len(env, dc) > TARGET_PAGE_SIZE)) { 1223 dc->base.is_jmp = DISAS_TOO_MANY; 1224 } 1225 } 1226 1227 static void xtensa_tr_tb_stop(DisasContextBase *dcbase, CPUState *cpu) 1228 { 1229 DisasContext *dc = container_of(dcbase, DisasContext, base); 1230 1231 switch (dc->base.is_jmp) { 1232 case DISAS_NORETURN: 1233 break; 1234 case DISAS_TOO_MANY: 1235 gen_jumpi(dc, dc->pc, 0); 1236 break; 1237 default: 1238 g_assert_not_reached(); 1239 } 1240 } 1241 1242 static void xtensa_tr_disas_log(const DisasContextBase *dcbase, 1243 CPUState *cpu, FILE *logfile) 1244 { 1245 fprintf(logfile, "IN: %s\n", lookup_symbol(dcbase->pc_first)); 1246 target_disas(logfile, cpu, dcbase->pc_first, dcbase->tb->size); 1247 } 1248 1249 static const TranslatorOps xtensa_translator_ops = { 1250 .init_disas_context = xtensa_tr_init_disas_context, 1251 .tb_start = xtensa_tr_tb_start, 1252 .insn_start = xtensa_tr_insn_start, 1253 .translate_insn = xtensa_tr_translate_insn, 1254 .tb_stop = xtensa_tr_tb_stop, 1255 .disas_log = xtensa_tr_disas_log, 1256 }; 1257 1258 void gen_intermediate_code(CPUState *cpu, TranslationBlock *tb, int *max_insns, 1259 target_ulong pc, void *host_pc) 1260 { 1261 DisasContext dc = {}; 1262 translator_loop(cpu, tb, max_insns, pc, host_pc, 1263 &xtensa_translator_ops, &dc.base); 1264 } 1265 1266 void xtensa_cpu_dump_state(CPUState *cs, FILE *f, int flags) 1267 { 1268 XtensaCPU *cpu = XTENSA_CPU(cs); 1269 CPUXtensaState *env = &cpu->env; 1270 xtensa_isa isa = env->config->isa; 1271 int i, j; 1272 1273 qemu_fprintf(f, "PC=%08x\n\n", env->pc); 1274 1275 for (i = j = 0; i < xtensa_isa_num_sysregs(isa); ++i) { 1276 const uint32_t *reg = 1277 xtensa_sysreg_is_user(isa, i) ? env->uregs : env->sregs; 1278 int regno = xtensa_sysreg_number(isa, i); 1279 1280 if (regno >= 0) { 1281 qemu_fprintf(f, "%12s=%08x%c", 1282 xtensa_sysreg_name(isa, i), 1283 reg[regno], 1284 (j++ % 4) == 3 ? '\n' : ' '); 1285 } 1286 } 1287 1288 qemu_fprintf(f, (j % 4) == 0 ? "\n" : "\n\n"); 1289 1290 for (i = 0; i < 16; ++i) { 1291 qemu_fprintf(f, " A%02d=%08x%c", 1292 i, env->regs[i], (i % 4) == 3 ? '\n' : ' '); 1293 } 1294 1295 xtensa_sync_phys_from_window(env); 1296 qemu_fprintf(f, "\n"); 1297 1298 for (i = 0; i < env->config->nareg; ++i) { 1299 qemu_fprintf(f, "AR%02d=%08x ", i, env->phys_regs[i]); 1300 if (i % 4 == 3) { 1301 bool ws = (env->sregs[WINDOW_START] & (1 << (i / 4))) != 0; 1302 bool cw = env->sregs[WINDOW_BASE] == i / 4; 1303 1304 qemu_fprintf(f, "%c%c\n", ws ? '<' : ' ', cw ? '=' : ' '); 1305 } 1306 } 1307 1308 if ((flags & CPU_DUMP_FPU) && 1309 xtensa_option_enabled(env->config, XTENSA_OPTION_FP_COPROCESSOR)) { 1310 qemu_fprintf(f, "\n"); 1311 1312 for (i = 0; i < 16; ++i) { 1313 qemu_fprintf(f, "F%02d=%08x (%-+15.8e)%c", i, 1314 float32_val(env->fregs[i].f32[FP_F32_LOW]), 1315 *(float *)(env->fregs[i].f32 + FP_F32_LOW), 1316 (i % 2) == 1 ? '\n' : ' '); 1317 } 1318 } 1319 1320 if ((flags & CPU_DUMP_FPU) && 1321 xtensa_option_enabled(env->config, XTENSA_OPTION_DFP_COPROCESSOR) && 1322 !xtensa_option_enabled(env->config, XTENSA_OPTION_DFPU_SINGLE_ONLY)) { 1323 qemu_fprintf(f, "\n"); 1324 1325 for (i = 0; i < 16; ++i) { 1326 qemu_fprintf(f, "F%02d=%016"PRIx64" (%-+24.16le)%c", i, 1327 float64_val(env->fregs[i].f64), 1328 *(double *)(&env->fregs[i].f64), 1329 (i % 2) == 1 ? '\n' : ' '); 1330 } 1331 } 1332 } 1333 1334 static void translate_abs(DisasContext *dc, const OpcodeArg arg[], 1335 const uint32_t par[]) 1336 { 1337 tcg_gen_abs_i32(arg[0].out, arg[1].in); 1338 } 1339 1340 static void translate_add(DisasContext *dc, const OpcodeArg arg[], 1341 const uint32_t par[]) 1342 { 1343 tcg_gen_add_i32(arg[0].out, arg[1].in, arg[2].in); 1344 } 1345 1346 static void translate_addi(DisasContext *dc, const OpcodeArg arg[], 1347 const uint32_t par[]) 1348 { 1349 tcg_gen_addi_i32(arg[0].out, arg[1].in, arg[2].imm); 1350 } 1351 1352 static void translate_addx(DisasContext *dc, const OpcodeArg arg[], 1353 const uint32_t par[]) 1354 { 1355 TCGv_i32 tmp = tcg_temp_new_i32(); 1356 tcg_gen_shli_i32(tmp, arg[1].in, par[0]); 1357 tcg_gen_add_i32(arg[0].out, tmp, arg[2].in); 1358 } 1359 1360 static void translate_all(DisasContext *dc, const OpcodeArg arg[], 1361 const uint32_t par[]) 1362 { 1363 uint32_t shift = par[1]; 1364 TCGv_i32 mask = tcg_constant_i32(((1 << shift) - 1) << arg[1].imm); 1365 TCGv_i32 tmp = tcg_temp_new_i32(); 1366 1367 tcg_gen_and_i32(tmp, arg[1].in, mask); 1368 if (par[0]) { 1369 tcg_gen_addi_i32(tmp, tmp, 1 << arg[1].imm); 1370 } else { 1371 tcg_gen_add_i32(tmp, tmp, mask); 1372 } 1373 tcg_gen_shri_i32(tmp, tmp, arg[1].imm + shift); 1374 tcg_gen_deposit_i32(arg[0].out, arg[0].out, 1375 tmp, arg[0].imm, 1); 1376 } 1377 1378 static void translate_and(DisasContext *dc, const OpcodeArg arg[], 1379 const uint32_t par[]) 1380 { 1381 tcg_gen_and_i32(arg[0].out, arg[1].in, arg[2].in); 1382 } 1383 1384 static void translate_ball(DisasContext *dc, const OpcodeArg arg[], 1385 const uint32_t par[]) 1386 { 1387 TCGv_i32 tmp = tcg_temp_new_i32(); 1388 tcg_gen_and_i32(tmp, arg[0].in, arg[1].in); 1389 gen_brcond(dc, par[0], tmp, arg[1].in, arg[2].imm); 1390 } 1391 1392 static void translate_bany(DisasContext *dc, const OpcodeArg arg[], 1393 const uint32_t par[]) 1394 { 1395 TCGv_i32 tmp = tcg_temp_new_i32(); 1396 tcg_gen_and_i32(tmp, arg[0].in, arg[1].in); 1397 gen_brcondi(dc, par[0], tmp, 0, arg[2].imm); 1398 } 1399 1400 static void translate_b(DisasContext *dc, const OpcodeArg arg[], 1401 const uint32_t par[]) 1402 { 1403 gen_brcond(dc, par[0], arg[0].in, arg[1].in, arg[2].imm); 1404 } 1405 1406 static void translate_bb(DisasContext *dc, const OpcodeArg arg[], 1407 const uint32_t par[]) 1408 { 1409 TCGv_i32 tmp = tcg_temp_new_i32(); 1410 1411 tcg_gen_andi_i32(tmp, arg[1].in, 0x1f); 1412 if (TARGET_BIG_ENDIAN) { 1413 tcg_gen_shr_i32(tmp, tcg_constant_i32(0x80000000u), tmp); 1414 } else { 1415 tcg_gen_shl_i32(tmp, tcg_constant_i32(0x00000001u), tmp); 1416 } 1417 tcg_gen_and_i32(tmp, arg[0].in, tmp); 1418 gen_brcondi(dc, par[0], tmp, 0, arg[2].imm); 1419 } 1420 1421 static void translate_bbi(DisasContext *dc, const OpcodeArg arg[], 1422 const uint32_t par[]) 1423 { 1424 TCGv_i32 tmp = tcg_temp_new_i32(); 1425 #if TARGET_BIG_ENDIAN 1426 tcg_gen_andi_i32(tmp, arg[0].in, 0x80000000u >> arg[1].imm); 1427 #else 1428 tcg_gen_andi_i32(tmp, arg[0].in, 0x00000001u << arg[1].imm); 1429 #endif 1430 gen_brcondi(dc, par[0], tmp, 0, arg[2].imm); 1431 } 1432 1433 static void translate_bi(DisasContext *dc, const OpcodeArg arg[], 1434 const uint32_t par[]) 1435 { 1436 gen_brcondi(dc, par[0], arg[0].in, arg[1].imm, arg[2].imm); 1437 } 1438 1439 static void translate_bz(DisasContext *dc, const OpcodeArg arg[], 1440 const uint32_t par[]) 1441 { 1442 gen_brcondi(dc, par[0], arg[0].in, 0, arg[1].imm); 1443 } 1444 1445 enum { 1446 BOOLEAN_AND, 1447 BOOLEAN_ANDC, 1448 BOOLEAN_OR, 1449 BOOLEAN_ORC, 1450 BOOLEAN_XOR, 1451 }; 1452 1453 static void translate_boolean(DisasContext *dc, const OpcodeArg arg[], 1454 const uint32_t par[]) 1455 { 1456 static void (* const op[])(TCGv_i32, TCGv_i32, TCGv_i32) = { 1457 [BOOLEAN_AND] = tcg_gen_and_i32, 1458 [BOOLEAN_ANDC] = tcg_gen_andc_i32, 1459 [BOOLEAN_OR] = tcg_gen_or_i32, 1460 [BOOLEAN_ORC] = tcg_gen_orc_i32, 1461 [BOOLEAN_XOR] = tcg_gen_xor_i32, 1462 }; 1463 1464 TCGv_i32 tmp1 = tcg_temp_new_i32(); 1465 TCGv_i32 tmp2 = tcg_temp_new_i32(); 1466 1467 tcg_gen_shri_i32(tmp1, arg[1].in, arg[1].imm); 1468 tcg_gen_shri_i32(tmp2, arg[2].in, arg[2].imm); 1469 op[par[0]](tmp1, tmp1, tmp2); 1470 tcg_gen_deposit_i32(arg[0].out, arg[0].out, tmp1, arg[0].imm, 1); 1471 } 1472 1473 static void translate_bp(DisasContext *dc, const OpcodeArg arg[], 1474 const uint32_t par[]) 1475 { 1476 TCGv_i32 tmp = tcg_temp_new_i32(); 1477 1478 tcg_gen_andi_i32(tmp, arg[0].in, 1 << arg[0].imm); 1479 gen_brcondi(dc, par[0], tmp, 0, arg[1].imm); 1480 } 1481 1482 static void translate_call0(DisasContext *dc, const OpcodeArg arg[], 1483 const uint32_t par[]) 1484 { 1485 tcg_gen_movi_i32(cpu_R[0], dc->base.pc_next); 1486 gen_jumpi(dc, arg[0].imm, 0); 1487 } 1488 1489 static void translate_callw(DisasContext *dc, const OpcodeArg arg[], 1490 const uint32_t par[]) 1491 { 1492 TCGv_i32 tmp = tcg_constant_i32(arg[0].imm); 1493 gen_callw_slot(dc, par[0], tmp, adjust_jump_slot(dc, arg[0].imm, 0)); 1494 } 1495 1496 static void translate_callx0(DisasContext *dc, const OpcodeArg arg[], 1497 const uint32_t par[]) 1498 { 1499 TCGv_i32 tmp = tcg_temp_new_i32(); 1500 tcg_gen_mov_i32(tmp, arg[0].in); 1501 tcg_gen_movi_i32(cpu_R[0], dc->base.pc_next); 1502 gen_jump(dc, tmp); 1503 } 1504 1505 static void translate_callxw(DisasContext *dc, const OpcodeArg arg[], 1506 const uint32_t par[]) 1507 { 1508 TCGv_i32 tmp = tcg_temp_new_i32(); 1509 1510 tcg_gen_mov_i32(tmp, arg[0].in); 1511 gen_callw_slot(dc, par[0], tmp, -1); 1512 } 1513 1514 static void translate_clamps(DisasContext *dc, const OpcodeArg arg[], 1515 const uint32_t par[]) 1516 { 1517 TCGv_i32 tmp1 = tcg_constant_i32(-1u << arg[2].imm); 1518 TCGv_i32 tmp2 = tcg_constant_i32((1 << arg[2].imm) - 1); 1519 1520 tcg_gen_smax_i32(arg[0].out, tmp1, arg[1].in); 1521 tcg_gen_smin_i32(arg[0].out, arg[0].out, tmp2); 1522 } 1523 1524 static void translate_clrb_expstate(DisasContext *dc, const OpcodeArg arg[], 1525 const uint32_t par[]) 1526 { 1527 /* TODO: GPIO32 may be a part of coprocessor */ 1528 tcg_gen_andi_i32(cpu_UR[EXPSTATE], cpu_UR[EXPSTATE], ~(1u << arg[0].imm)); 1529 } 1530 1531 static void translate_clrex(DisasContext *dc, const OpcodeArg arg[], 1532 const uint32_t par[]) 1533 { 1534 tcg_gen_movi_i32(cpu_exclusive_addr, -1); 1535 } 1536 1537 static void translate_const16(DisasContext *dc, const OpcodeArg arg[], 1538 const uint32_t par[]) 1539 { 1540 TCGv_i32 c = tcg_constant_i32(arg[1].imm); 1541 1542 tcg_gen_deposit_i32(arg[0].out, c, arg[0].in, 16, 16); 1543 } 1544 1545 static void translate_dcache(DisasContext *dc, const OpcodeArg arg[], 1546 const uint32_t par[]) 1547 { 1548 TCGv_i32 addr = tcg_temp_new_i32(); 1549 TCGv_i32 res = tcg_temp_new_i32(); 1550 1551 tcg_gen_addi_i32(addr, arg[0].in, arg[1].imm); 1552 tcg_gen_qemu_ld_i32(res, addr, dc->cring, MO_UB); 1553 } 1554 1555 static void translate_depbits(DisasContext *dc, const OpcodeArg arg[], 1556 const uint32_t par[]) 1557 { 1558 tcg_gen_deposit_i32(arg[1].out, arg[1].in, arg[0].in, 1559 arg[2].imm, arg[3].imm); 1560 } 1561 1562 static void translate_diwbuip(DisasContext *dc, const OpcodeArg arg[], 1563 const uint32_t par[]) 1564 { 1565 tcg_gen_addi_i32(arg[0].out, arg[0].in, dc->config->dcache_line_bytes); 1566 } 1567 1568 static uint32_t test_exceptions_entry(DisasContext *dc, const OpcodeArg arg[], 1569 const uint32_t par[]) 1570 { 1571 if (arg[0].imm > 3 || !dc->cwoe) { 1572 qemu_log_mask(LOG_GUEST_ERROR, 1573 "Illegal entry instruction(pc = %08x)\n", dc->pc); 1574 return XTENSA_OP_ILL; 1575 } else { 1576 return 0; 1577 } 1578 } 1579 1580 static uint32_t test_overflow_entry(DisasContext *dc, const OpcodeArg arg[], 1581 const uint32_t par[]) 1582 { 1583 return 1 << (dc->callinc * 4); 1584 } 1585 1586 static void translate_entry(DisasContext *dc, const OpcodeArg arg[], 1587 const uint32_t par[]) 1588 { 1589 TCGv_i32 pc = tcg_constant_i32(dc->pc); 1590 TCGv_i32 s = tcg_constant_i32(arg[0].imm); 1591 TCGv_i32 imm = tcg_constant_i32(arg[1].imm); 1592 gen_helper_entry(tcg_env, pc, s, imm); 1593 } 1594 1595 static void translate_extui(DisasContext *dc, const OpcodeArg arg[], 1596 const uint32_t par[]) 1597 { 1598 int maskimm = (1 << arg[3].imm) - 1; 1599 1600 TCGv_i32 tmp = tcg_temp_new_i32(); 1601 tcg_gen_shri_i32(tmp, arg[1].in, arg[2].imm); 1602 tcg_gen_andi_i32(arg[0].out, tmp, maskimm); 1603 } 1604 1605 static void translate_getex(DisasContext *dc, const OpcodeArg arg[], 1606 const uint32_t par[]) 1607 { 1608 TCGv_i32 tmp = tcg_temp_new_i32(); 1609 1610 tcg_gen_extract_i32(tmp, cpu_SR[ATOMCTL], 8, 1); 1611 tcg_gen_deposit_i32(cpu_SR[ATOMCTL], cpu_SR[ATOMCTL], arg[0].in, 8, 1); 1612 tcg_gen_mov_i32(arg[0].out, tmp); 1613 } 1614 1615 static void translate_icache(DisasContext *dc, const OpcodeArg arg[], 1616 const uint32_t par[]) 1617 { 1618 #ifndef CONFIG_USER_ONLY 1619 TCGv_i32 addr = tcg_temp_new_i32(); 1620 1621 tcg_gen_movi_i32(cpu_pc, dc->pc); 1622 tcg_gen_addi_i32(addr, arg[0].in, arg[1].imm); 1623 gen_helper_itlb_hit_test(tcg_env, addr); 1624 #endif 1625 } 1626 1627 static void translate_itlb(DisasContext *dc, const OpcodeArg arg[], 1628 const uint32_t par[]) 1629 { 1630 #ifndef CONFIG_USER_ONLY 1631 TCGv_i32 dtlb = tcg_constant_i32(par[0]); 1632 1633 gen_helper_itlb(tcg_env, arg[0].in, dtlb); 1634 #endif 1635 } 1636 1637 static void translate_j(DisasContext *dc, const OpcodeArg arg[], 1638 const uint32_t par[]) 1639 { 1640 gen_jumpi(dc, arg[0].imm, 0); 1641 } 1642 1643 static void translate_jx(DisasContext *dc, const OpcodeArg arg[], 1644 const uint32_t par[]) 1645 { 1646 gen_jump(dc, arg[0].in); 1647 } 1648 1649 static void translate_l32e(DisasContext *dc, const OpcodeArg arg[], 1650 const uint32_t par[]) 1651 { 1652 TCGv_i32 addr = tcg_temp_new_i32(); 1653 MemOp mop; 1654 1655 tcg_gen_addi_i32(addr, arg[1].in, arg[2].imm); 1656 mop = gen_load_store_alignment(dc, MO_TEUL, addr); 1657 tcg_gen_qemu_ld_tl(arg[0].out, addr, dc->ring, mop); 1658 } 1659 1660 #ifdef CONFIG_USER_ONLY 1661 static void gen_check_exclusive(DisasContext *dc, TCGv_i32 addr, bool is_write) 1662 { 1663 } 1664 #else 1665 static void gen_check_exclusive(DisasContext *dc, TCGv_i32 addr, bool is_write) 1666 { 1667 if (!option_enabled(dc, XTENSA_OPTION_MPU)) { 1668 TCGv_i32 pc = tcg_constant_i32(dc->pc); 1669 1670 gen_helper_check_exclusive(tcg_env, pc, addr, 1671 tcg_constant_i32(is_write)); 1672 } 1673 } 1674 #endif 1675 1676 static void translate_l32ex(DisasContext *dc, const OpcodeArg arg[], 1677 const uint32_t par[]) 1678 { 1679 TCGv_i32 addr = tcg_temp_new_i32(); 1680 MemOp mop; 1681 1682 tcg_gen_mov_i32(addr, arg[1].in); 1683 mop = gen_load_store_alignment(dc, MO_TEUL | MO_ALIGN, addr); 1684 gen_check_exclusive(dc, addr, false); 1685 tcg_gen_qemu_ld_i32(arg[0].out, addr, dc->cring, mop); 1686 tcg_gen_mov_i32(cpu_exclusive_addr, addr); 1687 tcg_gen_mov_i32(cpu_exclusive_val, arg[0].out); 1688 } 1689 1690 static void translate_ldst(DisasContext *dc, const OpcodeArg arg[], 1691 const uint32_t par[]) 1692 { 1693 TCGv_i32 addr = tcg_temp_new_i32(); 1694 MemOp mop; 1695 1696 tcg_gen_addi_i32(addr, arg[1].in, arg[2].imm); 1697 mop = gen_load_store_alignment(dc, par[0], addr); 1698 1699 if (par[2]) { 1700 if (par[1]) { 1701 tcg_gen_mb(TCG_BAR_STRL | TCG_MO_ALL); 1702 } 1703 tcg_gen_qemu_st_tl(arg[0].in, addr, dc->cring, mop); 1704 } else { 1705 tcg_gen_qemu_ld_tl(arg[0].out, addr, dc->cring, mop); 1706 if (par[1]) { 1707 tcg_gen_mb(TCG_BAR_LDAQ | TCG_MO_ALL); 1708 } 1709 } 1710 } 1711 1712 static void translate_lct(DisasContext *dc, const OpcodeArg arg[], 1713 const uint32_t par[]) 1714 { 1715 tcg_gen_movi_i32(arg[0].out, 0); 1716 } 1717 1718 static void translate_l32r(DisasContext *dc, const OpcodeArg arg[], 1719 const uint32_t par[]) 1720 { 1721 TCGv_i32 tmp; 1722 1723 if (dc->base.tb->flags & XTENSA_TBFLAG_LITBASE) { 1724 tmp = tcg_temp_new(); 1725 tcg_gen_addi_i32(tmp, cpu_SR[LITBASE], arg[1].raw_imm - 1); 1726 } else { 1727 tmp = tcg_constant_i32(arg[1].imm); 1728 } 1729 tcg_gen_qemu_ld_i32(arg[0].out, tmp, dc->cring, MO_TEUL); 1730 } 1731 1732 static void translate_loop(DisasContext *dc, const OpcodeArg arg[], 1733 const uint32_t par[]) 1734 { 1735 uint32_t lend = arg[1].imm; 1736 1737 tcg_gen_subi_i32(cpu_SR[LCOUNT], arg[0].in, 1); 1738 tcg_gen_movi_i32(cpu_SR[LBEG], dc->base.pc_next); 1739 tcg_gen_movi_i32(cpu_SR[LEND], lend); 1740 1741 if (par[0] != TCG_COND_NEVER) { 1742 TCGLabel *label = gen_new_label(); 1743 tcg_gen_brcondi_i32(par[0], arg[0].in, 0, label); 1744 gen_jumpi(dc, lend, 1); 1745 gen_set_label(label); 1746 } 1747 1748 gen_jumpi(dc, dc->base.pc_next, 0); 1749 } 1750 1751 enum { 1752 MAC16_UMUL, 1753 MAC16_MUL, 1754 MAC16_MULA, 1755 MAC16_MULS, 1756 MAC16_NONE, 1757 }; 1758 1759 enum { 1760 MAC16_LL, 1761 MAC16_HL, 1762 MAC16_LH, 1763 MAC16_HH, 1764 1765 MAC16_HX = 0x1, 1766 MAC16_XH = 0x2, 1767 }; 1768 1769 static void translate_mac16(DisasContext *dc, const OpcodeArg arg[], 1770 const uint32_t par[]) 1771 { 1772 int op = par[0]; 1773 unsigned half = par[1]; 1774 uint32_t ld_offset = par[2]; 1775 unsigned off = ld_offset ? 2 : 0; 1776 TCGv_i32 vaddr = tcg_temp_new_i32(); 1777 TCGv_i32 mem32 = tcg_temp_new_i32(); 1778 1779 if (ld_offset) { 1780 MemOp mop; 1781 1782 tcg_gen_addi_i32(vaddr, arg[1].in, ld_offset); 1783 mop = gen_load_store_alignment(dc, MO_TEUL, vaddr); 1784 tcg_gen_qemu_ld_tl(mem32, vaddr, dc->cring, mop); 1785 } 1786 if (op != MAC16_NONE) { 1787 TCGv_i32 m1 = gen_mac16_m(arg[off].in, 1788 half & MAC16_HX, op == MAC16_UMUL); 1789 TCGv_i32 m2 = gen_mac16_m(arg[off + 1].in, 1790 half & MAC16_XH, op == MAC16_UMUL); 1791 1792 if (op == MAC16_MUL || op == MAC16_UMUL) { 1793 tcg_gen_mul_i32(cpu_SR[ACCLO], m1, m2); 1794 if (op == MAC16_UMUL) { 1795 tcg_gen_movi_i32(cpu_SR[ACCHI], 0); 1796 } else { 1797 tcg_gen_sari_i32(cpu_SR[ACCHI], cpu_SR[ACCLO], 31); 1798 } 1799 } else { 1800 TCGv_i32 lo = tcg_temp_new_i32(); 1801 TCGv_i32 hi = tcg_temp_new_i32(); 1802 1803 tcg_gen_mul_i32(lo, m1, m2); 1804 tcg_gen_sari_i32(hi, lo, 31); 1805 if (op == MAC16_MULA) { 1806 tcg_gen_add2_i32(cpu_SR[ACCLO], cpu_SR[ACCHI], 1807 cpu_SR[ACCLO], cpu_SR[ACCHI], 1808 lo, hi); 1809 } else { 1810 tcg_gen_sub2_i32(cpu_SR[ACCLO], cpu_SR[ACCHI], 1811 cpu_SR[ACCLO], cpu_SR[ACCHI], 1812 lo, hi); 1813 } 1814 tcg_gen_ext8s_i32(cpu_SR[ACCHI], cpu_SR[ACCHI]); 1815 } 1816 } 1817 if (ld_offset) { 1818 tcg_gen_mov_i32(arg[1].out, vaddr); 1819 tcg_gen_mov_i32(cpu_SR[MR + arg[0].imm], mem32); 1820 } 1821 } 1822 1823 static void translate_memw(DisasContext *dc, const OpcodeArg arg[], 1824 const uint32_t par[]) 1825 { 1826 tcg_gen_mb(TCG_BAR_SC | TCG_MO_ALL); 1827 } 1828 1829 static void translate_smin(DisasContext *dc, const OpcodeArg arg[], 1830 const uint32_t par[]) 1831 { 1832 tcg_gen_smin_i32(arg[0].out, arg[1].in, arg[2].in); 1833 } 1834 1835 static void translate_umin(DisasContext *dc, const OpcodeArg arg[], 1836 const uint32_t par[]) 1837 { 1838 tcg_gen_umin_i32(arg[0].out, arg[1].in, arg[2].in); 1839 } 1840 1841 static void translate_smax(DisasContext *dc, const OpcodeArg arg[], 1842 const uint32_t par[]) 1843 { 1844 tcg_gen_smax_i32(arg[0].out, arg[1].in, arg[2].in); 1845 } 1846 1847 static void translate_umax(DisasContext *dc, const OpcodeArg arg[], 1848 const uint32_t par[]) 1849 { 1850 tcg_gen_umax_i32(arg[0].out, arg[1].in, arg[2].in); 1851 } 1852 1853 static void translate_mov(DisasContext *dc, const OpcodeArg arg[], 1854 const uint32_t par[]) 1855 { 1856 tcg_gen_mov_i32(arg[0].out, arg[1].in); 1857 } 1858 1859 static void translate_movcond(DisasContext *dc, const OpcodeArg arg[], 1860 const uint32_t par[]) 1861 { 1862 TCGv_i32 zero = tcg_constant_i32(0); 1863 1864 tcg_gen_movcond_i32(par[0], arg[0].out, 1865 arg[2].in, zero, arg[1].in, arg[0].in); 1866 } 1867 1868 static void translate_movi(DisasContext *dc, const OpcodeArg arg[], 1869 const uint32_t par[]) 1870 { 1871 tcg_gen_movi_i32(arg[0].out, arg[1].imm); 1872 } 1873 1874 static void translate_movp(DisasContext *dc, const OpcodeArg arg[], 1875 const uint32_t par[]) 1876 { 1877 TCGv_i32 zero = tcg_constant_i32(0); 1878 TCGv_i32 tmp = tcg_temp_new_i32(); 1879 1880 tcg_gen_andi_i32(tmp, arg[2].in, 1 << arg[2].imm); 1881 tcg_gen_movcond_i32(par[0], 1882 arg[0].out, tmp, zero, 1883 arg[1].in, arg[0].in); 1884 } 1885 1886 static void translate_movsp(DisasContext *dc, const OpcodeArg arg[], 1887 const uint32_t par[]) 1888 { 1889 tcg_gen_mov_i32(arg[0].out, arg[1].in); 1890 } 1891 1892 static void translate_mul16(DisasContext *dc, const OpcodeArg arg[], 1893 const uint32_t par[]) 1894 { 1895 TCGv_i32 v1 = tcg_temp_new_i32(); 1896 TCGv_i32 v2 = tcg_temp_new_i32(); 1897 1898 if (par[0]) { 1899 tcg_gen_ext16s_i32(v1, arg[1].in); 1900 tcg_gen_ext16s_i32(v2, arg[2].in); 1901 } else { 1902 tcg_gen_ext16u_i32(v1, arg[1].in); 1903 tcg_gen_ext16u_i32(v2, arg[2].in); 1904 } 1905 tcg_gen_mul_i32(arg[0].out, v1, v2); 1906 } 1907 1908 static void translate_mull(DisasContext *dc, const OpcodeArg arg[], 1909 const uint32_t par[]) 1910 { 1911 tcg_gen_mul_i32(arg[0].out, arg[1].in, arg[2].in); 1912 } 1913 1914 static void translate_mulh(DisasContext *dc, const OpcodeArg arg[], 1915 const uint32_t par[]) 1916 { 1917 TCGv_i32 lo = tcg_temp_new(); 1918 1919 if (par[0]) { 1920 tcg_gen_muls2_i32(lo, arg[0].out, arg[1].in, arg[2].in); 1921 } else { 1922 tcg_gen_mulu2_i32(lo, arg[0].out, arg[1].in, arg[2].in); 1923 } 1924 } 1925 1926 static void translate_neg(DisasContext *dc, const OpcodeArg arg[], 1927 const uint32_t par[]) 1928 { 1929 tcg_gen_neg_i32(arg[0].out, arg[1].in); 1930 } 1931 1932 static void translate_nop(DisasContext *dc, const OpcodeArg arg[], 1933 const uint32_t par[]) 1934 { 1935 } 1936 1937 static void translate_nsa(DisasContext *dc, const OpcodeArg arg[], 1938 const uint32_t par[]) 1939 { 1940 tcg_gen_clrsb_i32(arg[0].out, arg[1].in); 1941 } 1942 1943 static void translate_nsau(DisasContext *dc, const OpcodeArg arg[], 1944 const uint32_t par[]) 1945 { 1946 tcg_gen_clzi_i32(arg[0].out, arg[1].in, 32); 1947 } 1948 1949 static void translate_or(DisasContext *dc, const OpcodeArg arg[], 1950 const uint32_t par[]) 1951 { 1952 tcg_gen_or_i32(arg[0].out, arg[1].in, arg[2].in); 1953 } 1954 1955 static void translate_ptlb(DisasContext *dc, const OpcodeArg arg[], 1956 const uint32_t par[]) 1957 { 1958 #ifndef CONFIG_USER_ONLY 1959 TCGv_i32 dtlb = tcg_constant_i32(par[0]); 1960 1961 tcg_gen_movi_i32(cpu_pc, dc->pc); 1962 gen_helper_ptlb(arg[0].out, tcg_env, arg[1].in, dtlb); 1963 #endif 1964 } 1965 1966 static void translate_pptlb(DisasContext *dc, const OpcodeArg arg[], 1967 const uint32_t par[]) 1968 { 1969 #ifndef CONFIG_USER_ONLY 1970 tcg_gen_movi_i32(cpu_pc, dc->pc); 1971 gen_helper_pptlb(arg[0].out, tcg_env, arg[1].in); 1972 #endif 1973 } 1974 1975 static void translate_quos(DisasContext *dc, const OpcodeArg arg[], 1976 const uint32_t par[]) 1977 { 1978 TCGLabel *label1 = gen_new_label(); 1979 TCGLabel *label2 = gen_new_label(); 1980 1981 tcg_gen_brcondi_i32(TCG_COND_NE, arg[1].in, 0x80000000, 1982 label1); 1983 tcg_gen_brcondi_i32(TCG_COND_NE, arg[2].in, 0xffffffff, 1984 label1); 1985 tcg_gen_movi_i32(arg[0].out, 1986 par[0] ? 0x80000000 : 0); 1987 tcg_gen_br(label2); 1988 gen_set_label(label1); 1989 if (par[0]) { 1990 tcg_gen_div_i32(arg[0].out, 1991 arg[1].in, arg[2].in); 1992 } else { 1993 tcg_gen_rem_i32(arg[0].out, 1994 arg[1].in, arg[2].in); 1995 } 1996 gen_set_label(label2); 1997 } 1998 1999 static void translate_quou(DisasContext *dc, const OpcodeArg arg[], 2000 const uint32_t par[]) 2001 { 2002 tcg_gen_divu_i32(arg[0].out, 2003 arg[1].in, arg[2].in); 2004 } 2005 2006 static void translate_read_impwire(DisasContext *dc, const OpcodeArg arg[], 2007 const uint32_t par[]) 2008 { 2009 /* TODO: GPIO32 may be a part of coprocessor */ 2010 tcg_gen_movi_i32(arg[0].out, 0); 2011 } 2012 2013 static void translate_remu(DisasContext *dc, const OpcodeArg arg[], 2014 const uint32_t par[]) 2015 { 2016 tcg_gen_remu_i32(arg[0].out, 2017 arg[1].in, arg[2].in); 2018 } 2019 2020 static void translate_rer(DisasContext *dc, const OpcodeArg arg[], 2021 const uint32_t par[]) 2022 { 2023 gen_helper_rer(arg[0].out, tcg_env, arg[1].in); 2024 } 2025 2026 static void translate_ret(DisasContext *dc, const OpcodeArg arg[], 2027 const uint32_t par[]) 2028 { 2029 gen_jump(dc, cpu_R[0]); 2030 } 2031 2032 static uint32_t test_exceptions_retw(DisasContext *dc, const OpcodeArg arg[], 2033 const uint32_t par[]) 2034 { 2035 if (!dc->cwoe) { 2036 qemu_log_mask(LOG_GUEST_ERROR, 2037 "Illegal retw instruction(pc = %08x)\n", dc->pc); 2038 return XTENSA_OP_ILL; 2039 } else { 2040 TCGv_i32 pc = tcg_constant_i32(dc->pc); 2041 2042 gen_helper_test_ill_retw(tcg_env, pc); 2043 return 0; 2044 } 2045 } 2046 2047 static void translate_retw(DisasContext *dc, const OpcodeArg arg[], 2048 const uint32_t par[]) 2049 { 2050 TCGv_i32 tmp = tcg_temp_new(); 2051 tcg_gen_shl_i32(tmp, tcg_constant_i32(1), cpu_SR[WINDOW_BASE]); 2052 tcg_gen_andc_i32(cpu_SR[WINDOW_START], 2053 cpu_SR[WINDOW_START], tmp); 2054 tcg_gen_movi_i32(tmp, dc->pc); 2055 tcg_gen_deposit_i32(tmp, tmp, cpu_R[0], 0, 30); 2056 gen_helper_retw(tcg_env, cpu_R[0]); 2057 gen_jump(dc, tmp); 2058 } 2059 2060 static void translate_rfde(DisasContext *dc, const OpcodeArg arg[], 2061 const uint32_t par[]) 2062 { 2063 gen_jump(dc, cpu_SR[dc->config->ndepc ? DEPC : EPC1]); 2064 } 2065 2066 static void translate_rfe(DisasContext *dc, const OpcodeArg arg[], 2067 const uint32_t par[]) 2068 { 2069 tcg_gen_andi_i32(cpu_SR[PS], cpu_SR[PS], ~PS_EXCM); 2070 gen_jump(dc, cpu_SR[EPC1]); 2071 } 2072 2073 static void translate_rfi(DisasContext *dc, const OpcodeArg arg[], 2074 const uint32_t par[]) 2075 { 2076 tcg_gen_mov_i32(cpu_SR[PS], cpu_SR[EPS2 + arg[0].imm - 2]); 2077 gen_jump(dc, cpu_SR[EPC1 + arg[0].imm - 1]); 2078 } 2079 2080 static void translate_rfw(DisasContext *dc, const OpcodeArg arg[], 2081 const uint32_t par[]) 2082 { 2083 TCGv_i32 tmp = tcg_temp_new(); 2084 2085 tcg_gen_andi_i32(cpu_SR[PS], cpu_SR[PS], ~PS_EXCM); 2086 tcg_gen_shl_i32(tmp, tcg_constant_i32(1), cpu_SR[WINDOW_BASE]); 2087 2088 if (par[0]) { 2089 tcg_gen_andc_i32(cpu_SR[WINDOW_START], 2090 cpu_SR[WINDOW_START], tmp); 2091 } else { 2092 tcg_gen_or_i32(cpu_SR[WINDOW_START], 2093 cpu_SR[WINDOW_START], tmp); 2094 } 2095 2096 gen_helper_restore_owb(tcg_env); 2097 gen_jump(dc, cpu_SR[EPC1]); 2098 } 2099 2100 static void translate_rotw(DisasContext *dc, const OpcodeArg arg[], 2101 const uint32_t par[]) 2102 { 2103 tcg_gen_addi_i32(cpu_windowbase_next, cpu_SR[WINDOW_BASE], arg[0].imm); 2104 } 2105 2106 static void translate_rsil(DisasContext *dc, const OpcodeArg arg[], 2107 const uint32_t par[]) 2108 { 2109 tcg_gen_mov_i32(arg[0].out, cpu_SR[PS]); 2110 tcg_gen_andi_i32(cpu_SR[PS], cpu_SR[PS], ~PS_INTLEVEL); 2111 tcg_gen_ori_i32(cpu_SR[PS], cpu_SR[PS], arg[1].imm); 2112 } 2113 2114 static void translate_rsr(DisasContext *dc, const OpcodeArg arg[], 2115 const uint32_t par[]) 2116 { 2117 if (sr_name[par[0]]) { 2118 tcg_gen_mov_i32(arg[0].out, cpu_SR[par[0]]); 2119 } else { 2120 tcg_gen_movi_i32(arg[0].out, 0); 2121 } 2122 } 2123 2124 static void translate_rsr_ccount(DisasContext *dc, const OpcodeArg arg[], 2125 const uint32_t par[]) 2126 { 2127 #ifndef CONFIG_USER_ONLY 2128 translator_io_start(&dc->base); 2129 gen_helper_update_ccount(tcg_env); 2130 tcg_gen_mov_i32(arg[0].out, cpu_SR[par[0]]); 2131 #endif 2132 } 2133 2134 static void translate_rsr_ptevaddr(DisasContext *dc, const OpcodeArg arg[], 2135 const uint32_t par[]) 2136 { 2137 #ifndef CONFIG_USER_ONLY 2138 TCGv_i32 tmp = tcg_temp_new_i32(); 2139 2140 tcg_gen_shri_i32(tmp, cpu_SR[EXCVADDR], 10); 2141 tcg_gen_or_i32(tmp, tmp, cpu_SR[PTEVADDR]); 2142 tcg_gen_andi_i32(arg[0].out, tmp, 0xfffffffc); 2143 #endif 2144 } 2145 2146 static void translate_rtlb(DisasContext *dc, const OpcodeArg arg[], 2147 const uint32_t par[]) 2148 { 2149 #ifndef CONFIG_USER_ONLY 2150 static void (* const helper[])(TCGv_i32 r, TCGv_env env, TCGv_i32 a1, 2151 TCGv_i32 a2) = { 2152 gen_helper_rtlb0, 2153 gen_helper_rtlb1, 2154 }; 2155 TCGv_i32 dtlb = tcg_constant_i32(par[0]); 2156 2157 helper[par[1]](arg[0].out, tcg_env, arg[1].in, dtlb); 2158 #endif 2159 } 2160 2161 static void translate_rptlb0(DisasContext *dc, const OpcodeArg arg[], 2162 const uint32_t par[]) 2163 { 2164 #ifndef CONFIG_USER_ONLY 2165 gen_helper_rptlb0(arg[0].out, tcg_env, arg[1].in); 2166 #endif 2167 } 2168 2169 static void translate_rptlb1(DisasContext *dc, const OpcodeArg arg[], 2170 const uint32_t par[]) 2171 { 2172 #ifndef CONFIG_USER_ONLY 2173 gen_helper_rptlb1(arg[0].out, tcg_env, arg[1].in); 2174 #endif 2175 } 2176 2177 static void translate_rur(DisasContext *dc, const OpcodeArg arg[], 2178 const uint32_t par[]) 2179 { 2180 tcg_gen_mov_i32(arg[0].out, cpu_UR[par[0]]); 2181 } 2182 2183 static void translate_setb_expstate(DisasContext *dc, const OpcodeArg arg[], 2184 const uint32_t par[]) 2185 { 2186 /* TODO: GPIO32 may be a part of coprocessor */ 2187 tcg_gen_ori_i32(cpu_UR[EXPSTATE], cpu_UR[EXPSTATE], 1u << arg[0].imm); 2188 } 2189 2190 #ifdef CONFIG_USER_ONLY 2191 static void gen_check_atomctl(DisasContext *dc, TCGv_i32 addr) 2192 { 2193 } 2194 #else 2195 static void gen_check_atomctl(DisasContext *dc, TCGv_i32 addr) 2196 { 2197 TCGv_i32 pc = tcg_constant_i32(dc->pc); 2198 2199 gen_helper_check_atomctl(tcg_env, pc, addr); 2200 } 2201 #endif 2202 2203 static void translate_s32c1i(DisasContext *dc, const OpcodeArg arg[], 2204 const uint32_t par[]) 2205 { 2206 TCGv_i32 tmp = tcg_temp_new_i32(); 2207 TCGv_i32 addr = tcg_temp_new_i32(); 2208 MemOp mop; 2209 2210 tcg_gen_mov_i32(tmp, arg[0].in); 2211 tcg_gen_addi_i32(addr, arg[1].in, arg[2].imm); 2212 mop = gen_load_store_alignment(dc, MO_TEUL | MO_ALIGN, addr); 2213 gen_check_atomctl(dc, addr); 2214 tcg_gen_atomic_cmpxchg_i32(arg[0].out, addr, cpu_SR[SCOMPARE1], 2215 tmp, dc->cring, mop); 2216 } 2217 2218 static void translate_s32e(DisasContext *dc, const OpcodeArg arg[], 2219 const uint32_t par[]) 2220 { 2221 TCGv_i32 addr = tcg_temp_new_i32(); 2222 MemOp mop; 2223 2224 tcg_gen_addi_i32(addr, arg[1].in, arg[2].imm); 2225 mop = gen_load_store_alignment(dc, MO_TEUL, addr); 2226 tcg_gen_qemu_st_tl(arg[0].in, addr, dc->ring, mop); 2227 } 2228 2229 static void translate_s32ex(DisasContext *dc, const OpcodeArg arg[], 2230 const uint32_t par[]) 2231 { 2232 TCGv_i32 prev = tcg_temp_new_i32(); 2233 TCGv_i32 addr = tcg_temp_new_i32(); 2234 TCGv_i32 res = tcg_temp_new_i32(); 2235 TCGLabel *label = gen_new_label(); 2236 MemOp mop; 2237 2238 tcg_gen_movi_i32(res, 0); 2239 tcg_gen_mov_i32(addr, arg[1].in); 2240 mop = gen_load_store_alignment(dc, MO_TEUL | MO_ALIGN, addr); 2241 tcg_gen_brcond_i32(TCG_COND_NE, addr, cpu_exclusive_addr, label); 2242 gen_check_exclusive(dc, addr, true); 2243 tcg_gen_atomic_cmpxchg_i32(prev, cpu_exclusive_addr, cpu_exclusive_val, 2244 arg[0].in, dc->cring, mop); 2245 tcg_gen_setcond_i32(TCG_COND_EQ, res, prev, cpu_exclusive_val); 2246 tcg_gen_movcond_i32(TCG_COND_EQ, cpu_exclusive_val, 2247 prev, cpu_exclusive_val, prev, cpu_exclusive_val); 2248 tcg_gen_movi_i32(cpu_exclusive_addr, -1); 2249 gen_set_label(label); 2250 tcg_gen_extract_i32(arg[0].out, cpu_SR[ATOMCTL], 8, 1); 2251 tcg_gen_deposit_i32(cpu_SR[ATOMCTL], cpu_SR[ATOMCTL], res, 8, 1); 2252 } 2253 2254 static void translate_salt(DisasContext *dc, const OpcodeArg arg[], 2255 const uint32_t par[]) 2256 { 2257 tcg_gen_setcond_i32(par[0], 2258 arg[0].out, 2259 arg[1].in, arg[2].in); 2260 } 2261 2262 static void translate_sext(DisasContext *dc, const OpcodeArg arg[], 2263 const uint32_t par[]) 2264 { 2265 int shift = 31 - arg[2].imm; 2266 2267 if (shift == 24) { 2268 tcg_gen_ext8s_i32(arg[0].out, arg[1].in); 2269 } else if (shift == 16) { 2270 tcg_gen_ext16s_i32(arg[0].out, arg[1].in); 2271 } else { 2272 TCGv_i32 tmp = tcg_temp_new_i32(); 2273 tcg_gen_shli_i32(tmp, arg[1].in, shift); 2274 tcg_gen_sari_i32(arg[0].out, tmp, shift); 2275 } 2276 } 2277 2278 static uint32_t test_exceptions_simcall(DisasContext *dc, 2279 const OpcodeArg arg[], 2280 const uint32_t par[]) 2281 { 2282 bool is_semi = semihosting_enabled(dc->cring != 0); 2283 #ifdef CONFIG_USER_ONLY 2284 bool ill = true; 2285 #else 2286 /* Between RE.2 and RE.3 simcall opcode's become nop for the hardware. */ 2287 bool ill = dc->config->hw_version <= 250002 && !is_semi; 2288 #endif 2289 if (ill || !is_semi) { 2290 qemu_log_mask(LOG_GUEST_ERROR, "SIMCALL but semihosting is disabled\n"); 2291 } 2292 return ill ? XTENSA_OP_ILL : 0; 2293 } 2294 2295 static void translate_simcall(DisasContext *dc, const OpcodeArg arg[], 2296 const uint32_t par[]) 2297 { 2298 #ifndef CONFIG_USER_ONLY 2299 if (semihosting_enabled(dc->cring != 0)) { 2300 gen_helper_simcall(tcg_env); 2301 } 2302 #endif 2303 } 2304 2305 /* 2306 * Note: 64 bit ops are used here solely because SAR values 2307 * have range 0..63 2308 */ 2309 #define gen_shift_reg(cmd, reg) do { \ 2310 TCGv_i64 tmp = tcg_temp_new_i64(); \ 2311 tcg_gen_extu_i32_i64(tmp, reg); \ 2312 tcg_gen_##cmd##_i64(v, v, tmp); \ 2313 tcg_gen_extrl_i64_i32(arg[0].out, v); \ 2314 } while (0) 2315 2316 #define gen_shift(cmd) gen_shift_reg(cmd, cpu_SR[SAR]) 2317 2318 static void translate_sll(DisasContext *dc, const OpcodeArg arg[], 2319 const uint32_t par[]) 2320 { 2321 if (dc->sar_m32_5bit) { 2322 tcg_gen_shl_i32(arg[0].out, arg[1].in, dc->sar_m32); 2323 } else { 2324 TCGv_i64 v = tcg_temp_new_i64(); 2325 TCGv_i32 s = tcg_temp_new(); 2326 tcg_gen_subfi_i32(s, 32, cpu_SR[SAR]); 2327 tcg_gen_andi_i32(s, s, 0x3f); 2328 tcg_gen_extu_i32_i64(v, arg[1].in); 2329 gen_shift_reg(shl, s); 2330 } 2331 } 2332 2333 static void translate_slli(DisasContext *dc, const OpcodeArg arg[], 2334 const uint32_t par[]) 2335 { 2336 if (arg[2].imm == 32) { 2337 qemu_log_mask(LOG_GUEST_ERROR, "slli a%d, a%d, 32 is undefined\n", 2338 arg[0].imm, arg[1].imm); 2339 } 2340 tcg_gen_shli_i32(arg[0].out, arg[1].in, arg[2].imm & 0x1f); 2341 } 2342 2343 static void translate_sra(DisasContext *dc, const OpcodeArg arg[], 2344 const uint32_t par[]) 2345 { 2346 if (dc->sar_m32_5bit) { 2347 tcg_gen_sar_i32(arg[0].out, arg[1].in, cpu_SR[SAR]); 2348 } else { 2349 TCGv_i64 v = tcg_temp_new_i64(); 2350 tcg_gen_ext_i32_i64(v, arg[1].in); 2351 gen_shift(sar); 2352 } 2353 } 2354 2355 static void translate_srai(DisasContext *dc, const OpcodeArg arg[], 2356 const uint32_t par[]) 2357 { 2358 tcg_gen_sari_i32(arg[0].out, arg[1].in, arg[2].imm); 2359 } 2360 2361 static void translate_src(DisasContext *dc, const OpcodeArg arg[], 2362 const uint32_t par[]) 2363 { 2364 TCGv_i64 v = tcg_temp_new_i64(); 2365 tcg_gen_concat_i32_i64(v, arg[2].in, arg[1].in); 2366 gen_shift(shr); 2367 } 2368 2369 static void translate_srl(DisasContext *dc, const OpcodeArg arg[], 2370 const uint32_t par[]) 2371 { 2372 if (dc->sar_m32_5bit) { 2373 tcg_gen_shr_i32(arg[0].out, arg[1].in, cpu_SR[SAR]); 2374 } else { 2375 TCGv_i64 v = tcg_temp_new_i64(); 2376 tcg_gen_extu_i32_i64(v, arg[1].in); 2377 gen_shift(shr); 2378 } 2379 } 2380 2381 #undef gen_shift 2382 #undef gen_shift_reg 2383 2384 static void translate_srli(DisasContext *dc, const OpcodeArg arg[], 2385 const uint32_t par[]) 2386 { 2387 tcg_gen_shri_i32(arg[0].out, arg[1].in, arg[2].imm); 2388 } 2389 2390 static void translate_ssa8b(DisasContext *dc, const OpcodeArg arg[], 2391 const uint32_t par[]) 2392 { 2393 TCGv_i32 tmp = tcg_temp_new_i32(); 2394 tcg_gen_shli_i32(tmp, arg[0].in, 3); 2395 gen_left_shift_sar(dc, tmp); 2396 } 2397 2398 static void translate_ssa8l(DisasContext *dc, const OpcodeArg arg[], 2399 const uint32_t par[]) 2400 { 2401 TCGv_i32 tmp = tcg_temp_new_i32(); 2402 tcg_gen_shli_i32(tmp, arg[0].in, 3); 2403 gen_right_shift_sar(dc, tmp); 2404 } 2405 2406 static void translate_ssai(DisasContext *dc, const OpcodeArg arg[], 2407 const uint32_t par[]) 2408 { 2409 gen_right_shift_sar(dc, tcg_constant_i32(arg[0].imm)); 2410 } 2411 2412 static void translate_ssl(DisasContext *dc, const OpcodeArg arg[], 2413 const uint32_t par[]) 2414 { 2415 gen_left_shift_sar(dc, arg[0].in); 2416 } 2417 2418 static void translate_ssr(DisasContext *dc, const OpcodeArg arg[], 2419 const uint32_t par[]) 2420 { 2421 gen_right_shift_sar(dc, arg[0].in); 2422 } 2423 2424 static void translate_sub(DisasContext *dc, const OpcodeArg arg[], 2425 const uint32_t par[]) 2426 { 2427 tcg_gen_sub_i32(arg[0].out, arg[1].in, arg[2].in); 2428 } 2429 2430 static void translate_subx(DisasContext *dc, const OpcodeArg arg[], 2431 const uint32_t par[]) 2432 { 2433 TCGv_i32 tmp = tcg_temp_new_i32(); 2434 tcg_gen_shli_i32(tmp, arg[1].in, par[0]); 2435 tcg_gen_sub_i32(arg[0].out, tmp, arg[2].in); 2436 } 2437 2438 static void translate_waiti(DisasContext *dc, const OpcodeArg arg[], 2439 const uint32_t par[]) 2440 { 2441 #ifndef CONFIG_USER_ONLY 2442 TCGv_i32 pc = tcg_constant_i32(dc->base.pc_next); 2443 2444 translator_io_start(&dc->base); 2445 gen_helper_waiti(tcg_env, pc, tcg_constant_i32(arg[0].imm)); 2446 #endif 2447 } 2448 2449 static void translate_wtlb(DisasContext *dc, const OpcodeArg arg[], 2450 const uint32_t par[]) 2451 { 2452 #ifndef CONFIG_USER_ONLY 2453 TCGv_i32 dtlb = tcg_constant_i32(par[0]); 2454 2455 gen_helper_wtlb(tcg_env, arg[0].in, arg[1].in, dtlb); 2456 #endif 2457 } 2458 2459 static void translate_wptlb(DisasContext *dc, const OpcodeArg arg[], 2460 const uint32_t par[]) 2461 { 2462 #ifndef CONFIG_USER_ONLY 2463 gen_helper_wptlb(tcg_env, arg[0].in, arg[1].in); 2464 #endif 2465 } 2466 2467 static void translate_wer(DisasContext *dc, const OpcodeArg arg[], 2468 const uint32_t par[]) 2469 { 2470 gen_helper_wer(tcg_env, arg[0].in, arg[1].in); 2471 } 2472 2473 static void translate_wrmsk_expstate(DisasContext *dc, const OpcodeArg arg[], 2474 const uint32_t par[]) 2475 { 2476 /* TODO: GPIO32 may be a part of coprocessor */ 2477 tcg_gen_and_i32(cpu_UR[EXPSTATE], arg[0].in, arg[1].in); 2478 } 2479 2480 static void translate_wsr(DisasContext *dc, const OpcodeArg arg[], 2481 const uint32_t par[]) 2482 { 2483 if (sr_name[par[0]]) { 2484 tcg_gen_mov_i32(cpu_SR[par[0]], arg[0].in); 2485 } 2486 } 2487 2488 static void translate_wsr_mask(DisasContext *dc, const OpcodeArg arg[], 2489 const uint32_t par[]) 2490 { 2491 if (sr_name[par[0]]) { 2492 tcg_gen_andi_i32(cpu_SR[par[0]], arg[0].in, par[2]); 2493 } 2494 } 2495 2496 static void translate_wsr_acchi(DisasContext *dc, const OpcodeArg arg[], 2497 const uint32_t par[]) 2498 { 2499 tcg_gen_ext8s_i32(cpu_SR[par[0]], arg[0].in); 2500 } 2501 2502 static void translate_wsr_ccompare(DisasContext *dc, const OpcodeArg arg[], 2503 const uint32_t par[]) 2504 { 2505 #ifndef CONFIG_USER_ONLY 2506 uint32_t id = par[0] - CCOMPARE; 2507 2508 assert(id < dc->config->nccompare); 2509 translator_io_start(&dc->base); 2510 tcg_gen_mov_i32(cpu_SR[par[0]], arg[0].in); 2511 gen_helper_update_ccompare(tcg_env, tcg_constant_i32(id)); 2512 #endif 2513 } 2514 2515 static void translate_wsr_ccount(DisasContext *dc, const OpcodeArg arg[], 2516 const uint32_t par[]) 2517 { 2518 #ifndef CONFIG_USER_ONLY 2519 translator_io_start(&dc->base); 2520 gen_helper_wsr_ccount(tcg_env, arg[0].in); 2521 #endif 2522 } 2523 2524 static void translate_wsr_dbreaka(DisasContext *dc, const OpcodeArg arg[], 2525 const uint32_t par[]) 2526 { 2527 #ifndef CONFIG_USER_ONLY 2528 unsigned id = par[0] - DBREAKA; 2529 2530 assert(id < dc->config->ndbreak); 2531 gen_helper_wsr_dbreaka(tcg_env, tcg_constant_i32(id), arg[0].in); 2532 #endif 2533 } 2534 2535 static void translate_wsr_dbreakc(DisasContext *dc, const OpcodeArg arg[], 2536 const uint32_t par[]) 2537 { 2538 #ifndef CONFIG_USER_ONLY 2539 unsigned id = par[0] - DBREAKC; 2540 2541 assert(id < dc->config->ndbreak); 2542 gen_helper_wsr_dbreakc(tcg_env, tcg_constant_i32(id), arg[0].in); 2543 #endif 2544 } 2545 2546 static void translate_wsr_ibreaka(DisasContext *dc, const OpcodeArg arg[], 2547 const uint32_t par[]) 2548 { 2549 #ifndef CONFIG_USER_ONLY 2550 unsigned id = par[0] - IBREAKA; 2551 2552 assert(id < dc->config->nibreak); 2553 gen_helper_wsr_ibreaka(tcg_env, tcg_constant_i32(id), arg[0].in); 2554 #endif 2555 } 2556 2557 static void translate_wsr_ibreakenable(DisasContext *dc, const OpcodeArg arg[], 2558 const uint32_t par[]) 2559 { 2560 #ifndef CONFIG_USER_ONLY 2561 gen_helper_wsr_ibreakenable(tcg_env, arg[0].in); 2562 #endif 2563 } 2564 2565 static void translate_wsr_icount(DisasContext *dc, const OpcodeArg arg[], 2566 const uint32_t par[]) 2567 { 2568 #ifndef CONFIG_USER_ONLY 2569 if (dc->icount) { 2570 tcg_gen_mov_i32(dc->next_icount, arg[0].in); 2571 } else { 2572 tcg_gen_mov_i32(cpu_SR[par[0]], arg[0].in); 2573 } 2574 #endif 2575 } 2576 2577 static void translate_wsr_intclear(DisasContext *dc, const OpcodeArg arg[], 2578 const uint32_t par[]) 2579 { 2580 #ifndef CONFIG_USER_ONLY 2581 gen_helper_intclear(tcg_env, arg[0].in); 2582 #endif 2583 } 2584 2585 static void translate_wsr_intset(DisasContext *dc, const OpcodeArg arg[], 2586 const uint32_t par[]) 2587 { 2588 #ifndef CONFIG_USER_ONLY 2589 gen_helper_intset(tcg_env, arg[0].in); 2590 #endif 2591 } 2592 2593 static void translate_wsr_memctl(DisasContext *dc, const OpcodeArg arg[], 2594 const uint32_t par[]) 2595 { 2596 #ifndef CONFIG_USER_ONLY 2597 gen_helper_wsr_memctl(tcg_env, arg[0].in); 2598 #endif 2599 } 2600 2601 static void translate_wsr_mpuenb(DisasContext *dc, const OpcodeArg arg[], 2602 const uint32_t par[]) 2603 { 2604 #ifndef CONFIG_USER_ONLY 2605 gen_helper_wsr_mpuenb(tcg_env, arg[0].in); 2606 #endif 2607 } 2608 2609 static void translate_wsr_ps(DisasContext *dc, const OpcodeArg arg[], 2610 const uint32_t par[]) 2611 { 2612 #ifndef CONFIG_USER_ONLY 2613 uint32_t mask = PS_WOE | PS_CALLINC | PS_OWB | 2614 PS_UM | PS_EXCM | PS_INTLEVEL; 2615 2616 if (option_enabled(dc, XTENSA_OPTION_MMU) || 2617 option_enabled(dc, XTENSA_OPTION_MPU)) { 2618 mask |= PS_RING; 2619 } 2620 tcg_gen_andi_i32(cpu_SR[par[0]], arg[0].in, mask); 2621 #endif 2622 } 2623 2624 static void translate_wsr_rasid(DisasContext *dc, const OpcodeArg arg[], 2625 const uint32_t par[]) 2626 { 2627 #ifndef CONFIG_USER_ONLY 2628 gen_helper_wsr_rasid(tcg_env, arg[0].in); 2629 #endif 2630 } 2631 2632 static void translate_wsr_sar(DisasContext *dc, const OpcodeArg arg[], 2633 const uint32_t par[]) 2634 { 2635 tcg_gen_andi_i32(cpu_SR[par[0]], arg[0].in, 0x3f); 2636 if (dc->sar_m32_5bit) { 2637 tcg_gen_discard_i32(dc->sar_m32); 2638 } 2639 dc->sar_5bit = false; 2640 dc->sar_m32_5bit = false; 2641 } 2642 2643 static void translate_wsr_windowbase(DisasContext *dc, const OpcodeArg arg[], 2644 const uint32_t par[]) 2645 { 2646 #ifndef CONFIG_USER_ONLY 2647 tcg_gen_mov_i32(cpu_windowbase_next, arg[0].in); 2648 #endif 2649 } 2650 2651 static void translate_wsr_windowstart(DisasContext *dc, const OpcodeArg arg[], 2652 const uint32_t par[]) 2653 { 2654 #ifndef CONFIG_USER_ONLY 2655 tcg_gen_andi_i32(cpu_SR[par[0]], arg[0].in, 2656 (1 << dc->config->nareg / 4) - 1); 2657 #endif 2658 } 2659 2660 static void translate_wur(DisasContext *dc, const OpcodeArg arg[], 2661 const uint32_t par[]) 2662 { 2663 tcg_gen_mov_i32(cpu_UR[par[0]], arg[0].in); 2664 } 2665 2666 static void translate_xor(DisasContext *dc, const OpcodeArg arg[], 2667 const uint32_t par[]) 2668 { 2669 tcg_gen_xor_i32(arg[0].out, arg[1].in, arg[2].in); 2670 } 2671 2672 static void translate_xsr(DisasContext *dc, const OpcodeArg arg[], 2673 const uint32_t par[]) 2674 { 2675 if (sr_name[par[0]]) { 2676 TCGv_i32 tmp = tcg_temp_new_i32(); 2677 2678 tcg_gen_mov_i32(tmp, arg[0].in); 2679 tcg_gen_mov_i32(arg[0].out, cpu_SR[par[0]]); 2680 tcg_gen_mov_i32(cpu_SR[par[0]], tmp); 2681 } else { 2682 tcg_gen_movi_i32(arg[0].out, 0); 2683 } 2684 } 2685 2686 static void translate_xsr_mask(DisasContext *dc, const OpcodeArg arg[], 2687 const uint32_t par[]) 2688 { 2689 if (sr_name[par[0]]) { 2690 TCGv_i32 tmp = tcg_temp_new_i32(); 2691 2692 tcg_gen_mov_i32(tmp, arg[0].in); 2693 tcg_gen_mov_i32(arg[0].out, cpu_SR[par[0]]); 2694 tcg_gen_andi_i32(cpu_SR[par[0]], tmp, par[2]); 2695 } else { 2696 tcg_gen_movi_i32(arg[0].out, 0); 2697 } 2698 } 2699 2700 static void translate_xsr_ccount(DisasContext *dc, const OpcodeArg arg[], 2701 const uint32_t par[]) 2702 { 2703 #ifndef CONFIG_USER_ONLY 2704 TCGv_i32 tmp = tcg_temp_new_i32(); 2705 2706 translator_io_start(&dc->base); 2707 gen_helper_update_ccount(tcg_env); 2708 tcg_gen_mov_i32(tmp, cpu_SR[par[0]]); 2709 gen_helper_wsr_ccount(tcg_env, arg[0].in); 2710 tcg_gen_mov_i32(arg[0].out, tmp); 2711 2712 #endif 2713 } 2714 2715 #define gen_translate_xsr(name) \ 2716 static void translate_xsr_##name(DisasContext *dc, const OpcodeArg arg[], \ 2717 const uint32_t par[]) \ 2718 { \ 2719 TCGv_i32 tmp = tcg_temp_new_i32(); \ 2720 \ 2721 if (sr_name[par[0]]) { \ 2722 tcg_gen_mov_i32(tmp, cpu_SR[par[0]]); \ 2723 } else { \ 2724 tcg_gen_movi_i32(tmp, 0); \ 2725 } \ 2726 translate_wsr_##name(dc, arg, par); \ 2727 tcg_gen_mov_i32(arg[0].out, tmp); \ 2728 } 2729 2730 gen_translate_xsr(acchi) 2731 gen_translate_xsr(ccompare) 2732 gen_translate_xsr(dbreaka) 2733 gen_translate_xsr(dbreakc) 2734 gen_translate_xsr(ibreaka) 2735 gen_translate_xsr(ibreakenable) 2736 gen_translate_xsr(icount) 2737 gen_translate_xsr(memctl) 2738 gen_translate_xsr(mpuenb) 2739 gen_translate_xsr(ps) 2740 gen_translate_xsr(rasid) 2741 gen_translate_xsr(sar) 2742 gen_translate_xsr(windowbase) 2743 gen_translate_xsr(windowstart) 2744 2745 #undef gen_translate_xsr 2746 2747 static const XtensaOpcodeOps core_ops[] = { 2748 { 2749 .name = "abs", 2750 .translate = translate_abs, 2751 }, { 2752 .name = (const char * const[]) { 2753 "add", "add.n", NULL, 2754 }, 2755 .translate = translate_add, 2756 .op_flags = XTENSA_OP_NAME_ARRAY, 2757 }, { 2758 .name = (const char * const[]) { 2759 "addi", "addi.n", NULL, 2760 }, 2761 .translate = translate_addi, 2762 .op_flags = XTENSA_OP_NAME_ARRAY, 2763 }, { 2764 .name = "addmi", 2765 .translate = translate_addi, 2766 }, { 2767 .name = "addx2", 2768 .translate = translate_addx, 2769 .par = (const uint32_t[]){1}, 2770 }, { 2771 .name = "addx4", 2772 .translate = translate_addx, 2773 .par = (const uint32_t[]){2}, 2774 }, { 2775 .name = "addx8", 2776 .translate = translate_addx, 2777 .par = (const uint32_t[]){3}, 2778 }, { 2779 .name = "all4", 2780 .translate = translate_all, 2781 .par = (const uint32_t[]){true, 4}, 2782 }, { 2783 .name = "all8", 2784 .translate = translate_all, 2785 .par = (const uint32_t[]){true, 8}, 2786 }, { 2787 .name = "and", 2788 .translate = translate_and, 2789 }, { 2790 .name = "andb", 2791 .translate = translate_boolean, 2792 .par = (const uint32_t[]){BOOLEAN_AND}, 2793 }, { 2794 .name = "andbc", 2795 .translate = translate_boolean, 2796 .par = (const uint32_t[]){BOOLEAN_ANDC}, 2797 }, { 2798 .name = "any4", 2799 .translate = translate_all, 2800 .par = (const uint32_t[]){false, 4}, 2801 }, { 2802 .name = "any8", 2803 .translate = translate_all, 2804 .par = (const uint32_t[]){false, 8}, 2805 }, { 2806 .name = (const char * const[]) { 2807 "ball", "ball.w15", "ball.w18", NULL, 2808 }, 2809 .translate = translate_ball, 2810 .par = (const uint32_t[]){TCG_COND_EQ}, 2811 .op_flags = XTENSA_OP_NAME_ARRAY, 2812 }, { 2813 .name = (const char * const[]) { 2814 "bany", "bany.w15", "bany.w18", NULL, 2815 }, 2816 .translate = translate_bany, 2817 .par = (const uint32_t[]){TCG_COND_NE}, 2818 .op_flags = XTENSA_OP_NAME_ARRAY, 2819 }, { 2820 .name = (const char * const[]) { 2821 "bbc", "bbc.w15", "bbc.w18", NULL, 2822 }, 2823 .translate = translate_bb, 2824 .par = (const uint32_t[]){TCG_COND_EQ}, 2825 .op_flags = XTENSA_OP_NAME_ARRAY, 2826 }, { 2827 .name = (const char * const[]) { 2828 "bbci", "bbci.w15", "bbci.w18", NULL, 2829 }, 2830 .translate = translate_bbi, 2831 .par = (const uint32_t[]){TCG_COND_EQ}, 2832 .op_flags = XTENSA_OP_NAME_ARRAY, 2833 }, { 2834 .name = (const char * const[]) { 2835 "bbs", "bbs.w15", "bbs.w18", NULL, 2836 }, 2837 .translate = translate_bb, 2838 .par = (const uint32_t[]){TCG_COND_NE}, 2839 .op_flags = XTENSA_OP_NAME_ARRAY, 2840 }, { 2841 .name = (const char * const[]) { 2842 "bbsi", "bbsi.w15", "bbsi.w18", NULL, 2843 }, 2844 .translate = translate_bbi, 2845 .par = (const uint32_t[]){TCG_COND_NE}, 2846 .op_flags = XTENSA_OP_NAME_ARRAY, 2847 }, { 2848 .name = (const char * const[]) { 2849 "beq", "beq.w15", "beq.w18", NULL, 2850 }, 2851 .translate = translate_b, 2852 .par = (const uint32_t[]){TCG_COND_EQ}, 2853 .op_flags = XTENSA_OP_NAME_ARRAY, 2854 }, { 2855 .name = (const char * const[]) { 2856 "beqi", "beqi.w15", "beqi.w18", NULL, 2857 }, 2858 .translate = translate_bi, 2859 .par = (const uint32_t[]){TCG_COND_EQ}, 2860 .op_flags = XTENSA_OP_NAME_ARRAY, 2861 }, { 2862 .name = (const char * const[]) { 2863 "beqz", "beqz.n", "beqz.w15", "beqz.w18", NULL, 2864 }, 2865 .translate = translate_bz, 2866 .par = (const uint32_t[]){TCG_COND_EQ}, 2867 .op_flags = XTENSA_OP_NAME_ARRAY, 2868 }, { 2869 .name = "bf", 2870 .translate = translate_bp, 2871 .par = (const uint32_t[]){TCG_COND_EQ}, 2872 }, { 2873 .name = (const char * const[]) { 2874 "bge", "bge.w15", "bge.w18", NULL, 2875 }, 2876 .translate = translate_b, 2877 .par = (const uint32_t[]){TCG_COND_GE}, 2878 .op_flags = XTENSA_OP_NAME_ARRAY, 2879 }, { 2880 .name = (const char * const[]) { 2881 "bgei", "bgei.w15", "bgei.w18", NULL, 2882 }, 2883 .translate = translate_bi, 2884 .par = (const uint32_t[]){TCG_COND_GE}, 2885 .op_flags = XTENSA_OP_NAME_ARRAY, 2886 }, { 2887 .name = (const char * const[]) { 2888 "bgeu", "bgeu.w15", "bgeu.w18", NULL, 2889 }, 2890 .translate = translate_b, 2891 .par = (const uint32_t[]){TCG_COND_GEU}, 2892 .op_flags = XTENSA_OP_NAME_ARRAY, 2893 }, { 2894 .name = (const char * const[]) { 2895 "bgeui", "bgeui.w15", "bgeui.w18", NULL, 2896 }, 2897 .translate = translate_bi, 2898 .par = (const uint32_t[]){TCG_COND_GEU}, 2899 .op_flags = XTENSA_OP_NAME_ARRAY, 2900 }, { 2901 .name = (const char * const[]) { 2902 "bgez", "bgez.w15", "bgez.w18", NULL, 2903 }, 2904 .translate = translate_bz, 2905 .par = (const uint32_t[]){TCG_COND_GE}, 2906 .op_flags = XTENSA_OP_NAME_ARRAY, 2907 }, { 2908 .name = (const char * const[]) { 2909 "blt", "blt.w15", "blt.w18", NULL, 2910 }, 2911 .translate = translate_b, 2912 .par = (const uint32_t[]){TCG_COND_LT}, 2913 .op_flags = XTENSA_OP_NAME_ARRAY, 2914 }, { 2915 .name = (const char * const[]) { 2916 "blti", "blti.w15", "blti.w18", NULL, 2917 }, 2918 .translate = translate_bi, 2919 .par = (const uint32_t[]){TCG_COND_LT}, 2920 .op_flags = XTENSA_OP_NAME_ARRAY, 2921 }, { 2922 .name = (const char * const[]) { 2923 "bltu", "bltu.w15", "bltu.w18", NULL, 2924 }, 2925 .translate = translate_b, 2926 .par = (const uint32_t[]){TCG_COND_LTU}, 2927 .op_flags = XTENSA_OP_NAME_ARRAY, 2928 }, { 2929 .name = (const char * const[]) { 2930 "bltui", "bltui.w15", "bltui.w18", NULL, 2931 }, 2932 .translate = translate_bi, 2933 .par = (const uint32_t[]){TCG_COND_LTU}, 2934 .op_flags = XTENSA_OP_NAME_ARRAY, 2935 }, { 2936 .name = (const char * const[]) { 2937 "bltz", "bltz.w15", "bltz.w18", NULL, 2938 }, 2939 .translate = translate_bz, 2940 .par = (const uint32_t[]){TCG_COND_LT}, 2941 .op_flags = XTENSA_OP_NAME_ARRAY, 2942 }, { 2943 .name = (const char * const[]) { 2944 "bnall", "bnall.w15", "bnall.w18", NULL, 2945 }, 2946 .translate = translate_ball, 2947 .par = (const uint32_t[]){TCG_COND_NE}, 2948 .op_flags = XTENSA_OP_NAME_ARRAY, 2949 }, { 2950 .name = (const char * const[]) { 2951 "bne", "bne.w15", "bne.w18", NULL, 2952 }, 2953 .translate = translate_b, 2954 .par = (const uint32_t[]){TCG_COND_NE}, 2955 .op_flags = XTENSA_OP_NAME_ARRAY, 2956 }, { 2957 .name = (const char * const[]) { 2958 "bnei", "bnei.w15", "bnei.w18", NULL, 2959 }, 2960 .translate = translate_bi, 2961 .par = (const uint32_t[]){TCG_COND_NE}, 2962 .op_flags = XTENSA_OP_NAME_ARRAY, 2963 }, { 2964 .name = (const char * const[]) { 2965 "bnez", "bnez.n", "bnez.w15", "bnez.w18", NULL, 2966 }, 2967 .translate = translate_bz, 2968 .par = (const uint32_t[]){TCG_COND_NE}, 2969 .op_flags = XTENSA_OP_NAME_ARRAY, 2970 }, { 2971 .name = (const char * const[]) { 2972 "bnone", "bnone.w15", "bnone.w18", NULL, 2973 }, 2974 .translate = translate_bany, 2975 .par = (const uint32_t[]){TCG_COND_EQ}, 2976 .op_flags = XTENSA_OP_NAME_ARRAY, 2977 }, { 2978 .name = "break", 2979 .translate = translate_nop, 2980 .par = (const uint32_t[]){DEBUGCAUSE_BI}, 2981 .op_flags = XTENSA_OP_DEBUG_BREAK, 2982 }, { 2983 .name = "break.n", 2984 .translate = translate_nop, 2985 .par = (const uint32_t[]){DEBUGCAUSE_BN}, 2986 .op_flags = XTENSA_OP_DEBUG_BREAK, 2987 }, { 2988 .name = "bt", 2989 .translate = translate_bp, 2990 .par = (const uint32_t[]){TCG_COND_NE}, 2991 }, { 2992 .name = "call0", 2993 .translate = translate_call0, 2994 }, { 2995 .name = "call12", 2996 .translate = translate_callw, 2997 .par = (const uint32_t[]){3}, 2998 }, { 2999 .name = "call4", 3000 .translate = translate_callw, 3001 .par = (const uint32_t[]){1}, 3002 }, { 3003 .name = "call8", 3004 .translate = translate_callw, 3005 .par = (const uint32_t[]){2}, 3006 }, { 3007 .name = "callx0", 3008 .translate = translate_callx0, 3009 }, { 3010 .name = "callx12", 3011 .translate = translate_callxw, 3012 .par = (const uint32_t[]){3}, 3013 }, { 3014 .name = "callx4", 3015 .translate = translate_callxw, 3016 .par = (const uint32_t[]){1}, 3017 }, { 3018 .name = "callx8", 3019 .translate = translate_callxw, 3020 .par = (const uint32_t[]){2}, 3021 }, { 3022 .name = "clamps", 3023 .translate = translate_clamps, 3024 }, { 3025 .name = "clrb_expstate", 3026 .translate = translate_clrb_expstate, 3027 }, { 3028 .name = "clrex", 3029 .translate = translate_clrex, 3030 }, { 3031 .name = "const16", 3032 .translate = translate_const16, 3033 }, { 3034 .name = "depbits", 3035 .translate = translate_depbits, 3036 }, { 3037 .name = "dhi", 3038 .translate = translate_dcache, 3039 .op_flags = XTENSA_OP_PRIVILEGED, 3040 }, { 3041 .name = "dhi.b", 3042 .translate = translate_nop, 3043 }, { 3044 .name = "dhu", 3045 .translate = translate_dcache, 3046 .op_flags = XTENSA_OP_PRIVILEGED, 3047 }, { 3048 .name = "dhwb", 3049 .translate = translate_dcache, 3050 }, { 3051 .name = "dhwb.b", 3052 .translate = translate_nop, 3053 }, { 3054 .name = "dhwbi", 3055 .translate = translate_dcache, 3056 }, { 3057 .name = "dhwbi.b", 3058 .translate = translate_nop, 3059 }, { 3060 .name = "dii", 3061 .translate = translate_nop, 3062 .op_flags = XTENSA_OP_PRIVILEGED, 3063 }, { 3064 .name = "diu", 3065 .translate = translate_nop, 3066 .op_flags = XTENSA_OP_PRIVILEGED, 3067 }, { 3068 .name = "diwb", 3069 .translate = translate_nop, 3070 .op_flags = XTENSA_OP_PRIVILEGED, 3071 }, { 3072 .name = "diwbi", 3073 .translate = translate_nop, 3074 .op_flags = XTENSA_OP_PRIVILEGED, 3075 }, { 3076 .name = "diwbui.p", 3077 .translate = translate_diwbuip, 3078 .op_flags = XTENSA_OP_PRIVILEGED, 3079 }, { 3080 .name = "dpfl", 3081 .translate = translate_dcache, 3082 .op_flags = XTENSA_OP_PRIVILEGED, 3083 }, { 3084 .name = "dpfm.b", 3085 .translate = translate_nop, 3086 }, { 3087 .name = "dpfm.bf", 3088 .translate = translate_nop, 3089 }, { 3090 .name = "dpfr", 3091 .translate = translate_nop, 3092 }, { 3093 .name = "dpfr.b", 3094 .translate = translate_nop, 3095 }, { 3096 .name = "dpfr.bf", 3097 .translate = translate_nop, 3098 }, { 3099 .name = "dpfro", 3100 .translate = translate_nop, 3101 }, { 3102 .name = "dpfw", 3103 .translate = translate_nop, 3104 }, { 3105 .name = "dpfw.b", 3106 .translate = translate_nop, 3107 }, { 3108 .name = "dpfw.bf", 3109 .translate = translate_nop, 3110 }, { 3111 .name = "dpfwo", 3112 .translate = translate_nop, 3113 }, { 3114 .name = "dsync", 3115 .translate = translate_nop, 3116 }, { 3117 .name = "entry", 3118 .translate = translate_entry, 3119 .test_exceptions = test_exceptions_entry, 3120 .test_overflow = test_overflow_entry, 3121 .op_flags = XTENSA_OP_EXIT_TB_M1 | 3122 XTENSA_OP_SYNC_REGISTER_WINDOW, 3123 }, { 3124 .name = "esync", 3125 .translate = translate_nop, 3126 }, { 3127 .name = "excw", 3128 .translate = translate_nop, 3129 }, { 3130 .name = "extui", 3131 .translate = translate_extui, 3132 }, { 3133 .name = "extw", 3134 .translate = translate_memw, 3135 }, { 3136 .name = "getex", 3137 .translate = translate_getex, 3138 }, { 3139 .name = "hwwdtlba", 3140 .op_flags = XTENSA_OP_ILL, 3141 }, { 3142 .name = "hwwitlba", 3143 .op_flags = XTENSA_OP_ILL, 3144 }, { 3145 .name = "idtlb", 3146 .translate = translate_itlb, 3147 .par = (const uint32_t[]){true}, 3148 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_EXIT_TB_M1, 3149 }, { 3150 .name = "ihi", 3151 .translate = translate_icache, 3152 }, { 3153 .name = "ihu", 3154 .translate = translate_icache, 3155 .op_flags = XTENSA_OP_PRIVILEGED, 3156 }, { 3157 .name = "iii", 3158 .translate = translate_nop, 3159 .op_flags = XTENSA_OP_PRIVILEGED, 3160 }, { 3161 .name = "iitlb", 3162 .translate = translate_itlb, 3163 .par = (const uint32_t[]){false}, 3164 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_EXIT_TB_M1, 3165 }, { 3166 .name = "iiu", 3167 .translate = translate_nop, 3168 .op_flags = XTENSA_OP_PRIVILEGED, 3169 }, { 3170 .name = (const char * const[]) { 3171 "ill", "ill.n", NULL, 3172 }, 3173 .op_flags = XTENSA_OP_ILL | XTENSA_OP_NAME_ARRAY, 3174 }, { 3175 .name = "ipf", 3176 .translate = translate_nop, 3177 }, { 3178 .name = "ipfl", 3179 .translate = translate_icache, 3180 .op_flags = XTENSA_OP_PRIVILEGED, 3181 }, { 3182 .name = "isync", 3183 .translate = translate_nop, 3184 }, { 3185 .name = "j", 3186 .translate = translate_j, 3187 }, { 3188 .name = "jx", 3189 .translate = translate_jx, 3190 }, { 3191 .name = "l16si", 3192 .translate = translate_ldst, 3193 .par = (const uint32_t[]){MO_TESW, false, false}, 3194 .op_flags = XTENSA_OP_LOAD, 3195 }, { 3196 .name = "l16ui", 3197 .translate = translate_ldst, 3198 .par = (const uint32_t[]){MO_TEUW, false, false}, 3199 .op_flags = XTENSA_OP_LOAD, 3200 }, { 3201 .name = "l32ai", 3202 .translate = translate_ldst, 3203 .par = (const uint32_t[]){MO_TEUL | MO_ALIGN, true, false}, 3204 .op_flags = XTENSA_OP_LOAD, 3205 }, { 3206 .name = "l32e", 3207 .translate = translate_l32e, 3208 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_LOAD, 3209 }, { 3210 .name = "l32ex", 3211 .translate = translate_l32ex, 3212 .op_flags = XTENSA_OP_LOAD, 3213 }, { 3214 .name = (const char * const[]) { 3215 "l32i", "l32i.n", NULL, 3216 }, 3217 .translate = translate_ldst, 3218 .par = (const uint32_t[]){MO_TEUL, false, false}, 3219 .op_flags = XTENSA_OP_NAME_ARRAY | XTENSA_OP_LOAD, 3220 }, { 3221 .name = "l32r", 3222 .translate = translate_l32r, 3223 .op_flags = XTENSA_OP_LOAD, 3224 }, { 3225 .name = "l8ui", 3226 .translate = translate_ldst, 3227 .par = (const uint32_t[]){MO_UB, false, false}, 3228 .op_flags = XTENSA_OP_LOAD, 3229 }, { 3230 .name = "ldct", 3231 .translate = translate_lct, 3232 .op_flags = XTENSA_OP_PRIVILEGED, 3233 }, { 3234 .name = "ldcw", 3235 .translate = translate_nop, 3236 .op_flags = XTENSA_OP_PRIVILEGED, 3237 }, { 3238 .name = "lddec", 3239 .translate = translate_mac16, 3240 .par = (const uint32_t[]){MAC16_NONE, 0, -4}, 3241 .op_flags = XTENSA_OP_LOAD, 3242 }, { 3243 .name = "ldinc", 3244 .translate = translate_mac16, 3245 .par = (const uint32_t[]){MAC16_NONE, 0, 4}, 3246 .op_flags = XTENSA_OP_LOAD, 3247 }, { 3248 .name = "ldpte", 3249 .op_flags = XTENSA_OP_ILL, 3250 }, { 3251 .name = "lict", 3252 .translate = translate_lct, 3253 .op_flags = XTENSA_OP_PRIVILEGED, 3254 }, { 3255 .name = "licw", 3256 .translate = translate_nop, 3257 .op_flags = XTENSA_OP_PRIVILEGED, 3258 }, { 3259 .name = (const char * const[]) { 3260 "loop", "loop.w15", NULL, 3261 }, 3262 .translate = translate_loop, 3263 .par = (const uint32_t[]){TCG_COND_NEVER}, 3264 .op_flags = XTENSA_OP_NAME_ARRAY, 3265 }, { 3266 .name = (const char * const[]) { 3267 "loopgtz", "loopgtz.w15", NULL, 3268 }, 3269 .translate = translate_loop, 3270 .par = (const uint32_t[]){TCG_COND_GT}, 3271 .op_flags = XTENSA_OP_NAME_ARRAY, 3272 }, { 3273 .name = (const char * const[]) { 3274 "loopnez", "loopnez.w15", NULL, 3275 }, 3276 .translate = translate_loop, 3277 .par = (const uint32_t[]){TCG_COND_NE}, 3278 .op_flags = XTENSA_OP_NAME_ARRAY, 3279 }, { 3280 .name = "max", 3281 .translate = translate_smax, 3282 }, { 3283 .name = "maxu", 3284 .translate = translate_umax, 3285 }, { 3286 .name = "memw", 3287 .translate = translate_memw, 3288 }, { 3289 .name = "min", 3290 .translate = translate_smin, 3291 }, { 3292 .name = "minu", 3293 .translate = translate_umin, 3294 }, { 3295 .name = (const char * const[]) { 3296 "mov", "mov.n", NULL, 3297 }, 3298 .translate = translate_mov, 3299 .op_flags = XTENSA_OP_NAME_ARRAY, 3300 }, { 3301 .name = "moveqz", 3302 .translate = translate_movcond, 3303 .par = (const uint32_t[]){TCG_COND_EQ}, 3304 }, { 3305 .name = "movf", 3306 .translate = translate_movp, 3307 .par = (const uint32_t[]){TCG_COND_EQ}, 3308 }, { 3309 .name = "movgez", 3310 .translate = translate_movcond, 3311 .par = (const uint32_t[]){TCG_COND_GE}, 3312 }, { 3313 .name = "movi", 3314 .translate = translate_movi, 3315 }, { 3316 .name = "movi.n", 3317 .translate = translate_movi, 3318 }, { 3319 .name = "movltz", 3320 .translate = translate_movcond, 3321 .par = (const uint32_t[]){TCG_COND_LT}, 3322 }, { 3323 .name = "movnez", 3324 .translate = translate_movcond, 3325 .par = (const uint32_t[]){TCG_COND_NE}, 3326 }, { 3327 .name = "movsp", 3328 .translate = translate_movsp, 3329 .op_flags = XTENSA_OP_ALLOCA, 3330 }, { 3331 .name = "movt", 3332 .translate = translate_movp, 3333 .par = (const uint32_t[]){TCG_COND_NE}, 3334 }, { 3335 .name = "mul.aa.hh", 3336 .translate = translate_mac16, 3337 .par = (const uint32_t[]){MAC16_MUL, MAC16_HH, 0}, 3338 }, { 3339 .name = "mul.aa.hl", 3340 .translate = translate_mac16, 3341 .par = (const uint32_t[]){MAC16_MUL, MAC16_HL, 0}, 3342 }, { 3343 .name = "mul.aa.lh", 3344 .translate = translate_mac16, 3345 .par = (const uint32_t[]){MAC16_MUL, MAC16_LH, 0}, 3346 }, { 3347 .name = "mul.aa.ll", 3348 .translate = translate_mac16, 3349 .par = (const uint32_t[]){MAC16_MUL, MAC16_LL, 0}, 3350 }, { 3351 .name = "mul.ad.hh", 3352 .translate = translate_mac16, 3353 .par = (const uint32_t[]){MAC16_MUL, MAC16_HH, 0}, 3354 }, { 3355 .name = "mul.ad.hl", 3356 .translate = translate_mac16, 3357 .par = (const uint32_t[]){MAC16_MUL, MAC16_HL, 0}, 3358 }, { 3359 .name = "mul.ad.lh", 3360 .translate = translate_mac16, 3361 .par = (const uint32_t[]){MAC16_MUL, MAC16_LH, 0}, 3362 }, { 3363 .name = "mul.ad.ll", 3364 .translate = translate_mac16, 3365 .par = (const uint32_t[]){MAC16_MUL, MAC16_LL, 0}, 3366 }, { 3367 .name = "mul.da.hh", 3368 .translate = translate_mac16, 3369 .par = (const uint32_t[]){MAC16_MUL, MAC16_HH, 0}, 3370 }, { 3371 .name = "mul.da.hl", 3372 .translate = translate_mac16, 3373 .par = (const uint32_t[]){MAC16_MUL, MAC16_HL, 0}, 3374 }, { 3375 .name = "mul.da.lh", 3376 .translate = translate_mac16, 3377 .par = (const uint32_t[]){MAC16_MUL, MAC16_LH, 0}, 3378 }, { 3379 .name = "mul.da.ll", 3380 .translate = translate_mac16, 3381 .par = (const uint32_t[]){MAC16_MUL, MAC16_LL, 0}, 3382 }, { 3383 .name = "mul.dd.hh", 3384 .translate = translate_mac16, 3385 .par = (const uint32_t[]){MAC16_MUL, MAC16_HH, 0}, 3386 }, { 3387 .name = "mul.dd.hl", 3388 .translate = translate_mac16, 3389 .par = (const uint32_t[]){MAC16_MUL, MAC16_HL, 0}, 3390 }, { 3391 .name = "mul.dd.lh", 3392 .translate = translate_mac16, 3393 .par = (const uint32_t[]){MAC16_MUL, MAC16_LH, 0}, 3394 }, { 3395 .name = "mul.dd.ll", 3396 .translate = translate_mac16, 3397 .par = (const uint32_t[]){MAC16_MUL, MAC16_LL, 0}, 3398 }, { 3399 .name = "mul16s", 3400 .translate = translate_mul16, 3401 .par = (const uint32_t[]){true}, 3402 }, { 3403 .name = "mul16u", 3404 .translate = translate_mul16, 3405 .par = (const uint32_t[]){false}, 3406 }, { 3407 .name = "mula.aa.hh", 3408 .translate = translate_mac16, 3409 .par = (const uint32_t[]){MAC16_MULA, MAC16_HH, 0}, 3410 }, { 3411 .name = "mula.aa.hl", 3412 .translate = translate_mac16, 3413 .par = (const uint32_t[]){MAC16_MULA, MAC16_HL, 0}, 3414 }, { 3415 .name = "mula.aa.lh", 3416 .translate = translate_mac16, 3417 .par = (const uint32_t[]){MAC16_MULA, MAC16_LH, 0}, 3418 }, { 3419 .name = "mula.aa.ll", 3420 .translate = translate_mac16, 3421 .par = (const uint32_t[]){MAC16_MULA, MAC16_LL, 0}, 3422 }, { 3423 .name = "mula.ad.hh", 3424 .translate = translate_mac16, 3425 .par = (const uint32_t[]){MAC16_MULA, MAC16_HH, 0}, 3426 }, { 3427 .name = "mula.ad.hl", 3428 .translate = translate_mac16, 3429 .par = (const uint32_t[]){MAC16_MULA, MAC16_HL, 0}, 3430 }, { 3431 .name = "mula.ad.lh", 3432 .translate = translate_mac16, 3433 .par = (const uint32_t[]){MAC16_MULA, MAC16_LH, 0}, 3434 }, { 3435 .name = "mula.ad.ll", 3436 .translate = translate_mac16, 3437 .par = (const uint32_t[]){MAC16_MULA, MAC16_LL, 0}, 3438 }, { 3439 .name = "mula.da.hh", 3440 .translate = translate_mac16, 3441 .par = (const uint32_t[]){MAC16_MULA, MAC16_HH, 0}, 3442 }, { 3443 .name = "mula.da.hh.lddec", 3444 .translate = translate_mac16, 3445 .par = (const uint32_t[]){MAC16_MULA, MAC16_HH, -4}, 3446 }, { 3447 .name = "mula.da.hh.ldinc", 3448 .translate = translate_mac16, 3449 .par = (const uint32_t[]){MAC16_MULA, MAC16_HH, 4}, 3450 }, { 3451 .name = "mula.da.hl", 3452 .translate = translate_mac16, 3453 .par = (const uint32_t[]){MAC16_MULA, MAC16_HL, 0}, 3454 }, { 3455 .name = "mula.da.hl.lddec", 3456 .translate = translate_mac16, 3457 .par = (const uint32_t[]){MAC16_MULA, MAC16_HL, -4}, 3458 }, { 3459 .name = "mula.da.hl.ldinc", 3460 .translate = translate_mac16, 3461 .par = (const uint32_t[]){MAC16_MULA, MAC16_HL, 4}, 3462 }, { 3463 .name = "mula.da.lh", 3464 .translate = translate_mac16, 3465 .par = (const uint32_t[]){MAC16_MULA, MAC16_LH, 0}, 3466 }, { 3467 .name = "mula.da.lh.lddec", 3468 .translate = translate_mac16, 3469 .par = (const uint32_t[]){MAC16_MULA, MAC16_LH, -4}, 3470 }, { 3471 .name = "mula.da.lh.ldinc", 3472 .translate = translate_mac16, 3473 .par = (const uint32_t[]){MAC16_MULA, MAC16_LH, 4}, 3474 }, { 3475 .name = "mula.da.ll", 3476 .translate = translate_mac16, 3477 .par = (const uint32_t[]){MAC16_MULA, MAC16_LL, 0}, 3478 }, { 3479 .name = "mula.da.ll.lddec", 3480 .translate = translate_mac16, 3481 .par = (const uint32_t[]){MAC16_MULA, MAC16_LL, -4}, 3482 }, { 3483 .name = "mula.da.ll.ldinc", 3484 .translate = translate_mac16, 3485 .par = (const uint32_t[]){MAC16_MULA, MAC16_LL, 4}, 3486 }, { 3487 .name = "mula.dd.hh", 3488 .translate = translate_mac16, 3489 .par = (const uint32_t[]){MAC16_MULA, MAC16_HH, 0}, 3490 }, { 3491 .name = "mula.dd.hh.lddec", 3492 .translate = translate_mac16, 3493 .par = (const uint32_t[]){MAC16_MULA, MAC16_HH, -4}, 3494 }, { 3495 .name = "mula.dd.hh.ldinc", 3496 .translate = translate_mac16, 3497 .par = (const uint32_t[]){MAC16_MULA, MAC16_HH, 4}, 3498 }, { 3499 .name = "mula.dd.hl", 3500 .translate = translate_mac16, 3501 .par = (const uint32_t[]){MAC16_MULA, MAC16_HL, 0}, 3502 }, { 3503 .name = "mula.dd.hl.lddec", 3504 .translate = translate_mac16, 3505 .par = (const uint32_t[]){MAC16_MULA, MAC16_HL, -4}, 3506 }, { 3507 .name = "mula.dd.hl.ldinc", 3508 .translate = translate_mac16, 3509 .par = (const uint32_t[]){MAC16_MULA, MAC16_HL, 4}, 3510 }, { 3511 .name = "mula.dd.lh", 3512 .translate = translate_mac16, 3513 .par = (const uint32_t[]){MAC16_MULA, MAC16_LH, 0}, 3514 }, { 3515 .name = "mula.dd.lh.lddec", 3516 .translate = translate_mac16, 3517 .par = (const uint32_t[]){MAC16_MULA, MAC16_LH, -4}, 3518 }, { 3519 .name = "mula.dd.lh.ldinc", 3520 .translate = translate_mac16, 3521 .par = (const uint32_t[]){MAC16_MULA, MAC16_LH, 4}, 3522 }, { 3523 .name = "mula.dd.ll", 3524 .translate = translate_mac16, 3525 .par = (const uint32_t[]){MAC16_MULA, MAC16_LL, 0}, 3526 }, { 3527 .name = "mula.dd.ll.lddec", 3528 .translate = translate_mac16, 3529 .par = (const uint32_t[]){MAC16_MULA, MAC16_LL, -4}, 3530 }, { 3531 .name = "mula.dd.ll.ldinc", 3532 .translate = translate_mac16, 3533 .par = (const uint32_t[]){MAC16_MULA, MAC16_LL, 4}, 3534 }, { 3535 .name = "mull", 3536 .translate = translate_mull, 3537 }, { 3538 .name = "muls.aa.hh", 3539 .translate = translate_mac16, 3540 .par = (const uint32_t[]){MAC16_MULS, MAC16_HH, 0}, 3541 }, { 3542 .name = "muls.aa.hl", 3543 .translate = translate_mac16, 3544 .par = (const uint32_t[]){MAC16_MULS, MAC16_HL, 0}, 3545 }, { 3546 .name = "muls.aa.lh", 3547 .translate = translate_mac16, 3548 .par = (const uint32_t[]){MAC16_MULS, MAC16_LH, 0}, 3549 }, { 3550 .name = "muls.aa.ll", 3551 .translate = translate_mac16, 3552 .par = (const uint32_t[]){MAC16_MULS, MAC16_LL, 0}, 3553 }, { 3554 .name = "muls.ad.hh", 3555 .translate = translate_mac16, 3556 .par = (const uint32_t[]){MAC16_MULS, MAC16_HH, 0}, 3557 }, { 3558 .name = "muls.ad.hl", 3559 .translate = translate_mac16, 3560 .par = (const uint32_t[]){MAC16_MULS, MAC16_HL, 0}, 3561 }, { 3562 .name = "muls.ad.lh", 3563 .translate = translate_mac16, 3564 .par = (const uint32_t[]){MAC16_MULS, MAC16_LH, 0}, 3565 }, { 3566 .name = "muls.ad.ll", 3567 .translate = translate_mac16, 3568 .par = (const uint32_t[]){MAC16_MULS, MAC16_LL, 0}, 3569 }, { 3570 .name = "muls.da.hh", 3571 .translate = translate_mac16, 3572 .par = (const uint32_t[]){MAC16_MULS, MAC16_HH, 0}, 3573 }, { 3574 .name = "muls.da.hl", 3575 .translate = translate_mac16, 3576 .par = (const uint32_t[]){MAC16_MULS, MAC16_HL, 0}, 3577 }, { 3578 .name = "muls.da.lh", 3579 .translate = translate_mac16, 3580 .par = (const uint32_t[]){MAC16_MULS, MAC16_LH, 0}, 3581 }, { 3582 .name = "muls.da.ll", 3583 .translate = translate_mac16, 3584 .par = (const uint32_t[]){MAC16_MULS, MAC16_LL, 0}, 3585 }, { 3586 .name = "muls.dd.hh", 3587 .translate = translate_mac16, 3588 .par = (const uint32_t[]){MAC16_MULS, MAC16_HH, 0}, 3589 }, { 3590 .name = "muls.dd.hl", 3591 .translate = translate_mac16, 3592 .par = (const uint32_t[]){MAC16_MULS, MAC16_HL, 0}, 3593 }, { 3594 .name = "muls.dd.lh", 3595 .translate = translate_mac16, 3596 .par = (const uint32_t[]){MAC16_MULS, MAC16_LH, 0}, 3597 }, { 3598 .name = "muls.dd.ll", 3599 .translate = translate_mac16, 3600 .par = (const uint32_t[]){MAC16_MULS, MAC16_LL, 0}, 3601 }, { 3602 .name = "mulsh", 3603 .translate = translate_mulh, 3604 .par = (const uint32_t[]){true}, 3605 }, { 3606 .name = "muluh", 3607 .translate = translate_mulh, 3608 .par = (const uint32_t[]){false}, 3609 }, { 3610 .name = "neg", 3611 .translate = translate_neg, 3612 }, { 3613 .name = (const char * const[]) { 3614 "nop", "nop.n", NULL, 3615 }, 3616 .translate = translate_nop, 3617 .op_flags = XTENSA_OP_NAME_ARRAY, 3618 }, { 3619 .name = "nsa", 3620 .translate = translate_nsa, 3621 }, { 3622 .name = "nsau", 3623 .translate = translate_nsau, 3624 }, { 3625 .name = "or", 3626 .translate = translate_or, 3627 }, { 3628 .name = "orb", 3629 .translate = translate_boolean, 3630 .par = (const uint32_t[]){BOOLEAN_OR}, 3631 }, { 3632 .name = "orbc", 3633 .translate = translate_boolean, 3634 .par = (const uint32_t[]){BOOLEAN_ORC}, 3635 }, { 3636 .name = "pdtlb", 3637 .translate = translate_ptlb, 3638 .par = (const uint32_t[]){true}, 3639 .op_flags = XTENSA_OP_PRIVILEGED, 3640 }, { 3641 .name = "pfend.a", 3642 .translate = translate_nop, 3643 }, { 3644 .name = "pfend.o", 3645 .translate = translate_nop, 3646 }, { 3647 .name = "pfnxt.f", 3648 .translate = translate_nop, 3649 }, { 3650 .name = "pfwait.a", 3651 .translate = translate_nop, 3652 }, { 3653 .name = "pfwait.r", 3654 .translate = translate_nop, 3655 }, { 3656 .name = "pitlb", 3657 .translate = translate_ptlb, 3658 .par = (const uint32_t[]){false}, 3659 .op_flags = XTENSA_OP_PRIVILEGED, 3660 }, { 3661 .name = "pptlb", 3662 .translate = translate_pptlb, 3663 .op_flags = XTENSA_OP_PRIVILEGED, 3664 }, { 3665 .name = "quos", 3666 .translate = translate_quos, 3667 .par = (const uint32_t[]){true}, 3668 .op_flags = XTENSA_OP_DIVIDE_BY_ZERO, 3669 }, { 3670 .name = "quou", 3671 .translate = translate_quou, 3672 .op_flags = XTENSA_OP_DIVIDE_BY_ZERO, 3673 }, { 3674 .name = "rdtlb0", 3675 .translate = translate_rtlb, 3676 .par = (const uint32_t[]){true, 0}, 3677 .op_flags = XTENSA_OP_PRIVILEGED, 3678 }, { 3679 .name = "rdtlb1", 3680 .translate = translate_rtlb, 3681 .par = (const uint32_t[]){true, 1}, 3682 .op_flags = XTENSA_OP_PRIVILEGED, 3683 }, { 3684 .name = "read_impwire", 3685 .translate = translate_read_impwire, 3686 }, { 3687 .name = "rems", 3688 .translate = translate_quos, 3689 .par = (const uint32_t[]){false}, 3690 .op_flags = XTENSA_OP_DIVIDE_BY_ZERO, 3691 }, { 3692 .name = "remu", 3693 .translate = translate_remu, 3694 .op_flags = XTENSA_OP_DIVIDE_BY_ZERO, 3695 }, { 3696 .name = "rer", 3697 .translate = translate_rer, 3698 .op_flags = XTENSA_OP_PRIVILEGED, 3699 }, { 3700 .name = (const char * const[]) { 3701 "ret", "ret.n", NULL, 3702 }, 3703 .translate = translate_ret, 3704 .op_flags = XTENSA_OP_NAME_ARRAY, 3705 }, { 3706 .name = (const char * const[]) { 3707 "retw", "retw.n", NULL, 3708 }, 3709 .translate = translate_retw, 3710 .test_exceptions = test_exceptions_retw, 3711 .op_flags = XTENSA_OP_UNDERFLOW | XTENSA_OP_NAME_ARRAY, 3712 }, { 3713 .name = "rfdd", 3714 .op_flags = XTENSA_OP_ILL, 3715 }, { 3716 .name = "rfde", 3717 .translate = translate_rfde, 3718 .op_flags = XTENSA_OP_PRIVILEGED, 3719 }, { 3720 .name = "rfdo", 3721 .op_flags = XTENSA_OP_ILL, 3722 }, { 3723 .name = "rfe", 3724 .translate = translate_rfe, 3725 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_CHECK_INTERRUPTS, 3726 }, { 3727 .name = "rfi", 3728 .translate = translate_rfi, 3729 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_CHECK_INTERRUPTS, 3730 }, { 3731 .name = "rfwo", 3732 .translate = translate_rfw, 3733 .par = (const uint32_t[]){true}, 3734 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_CHECK_INTERRUPTS, 3735 }, { 3736 .name = "rfwu", 3737 .translate = translate_rfw, 3738 .par = (const uint32_t[]){false}, 3739 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_CHECK_INTERRUPTS, 3740 }, { 3741 .name = "ritlb0", 3742 .translate = translate_rtlb, 3743 .par = (const uint32_t[]){false, 0}, 3744 .op_flags = XTENSA_OP_PRIVILEGED, 3745 }, { 3746 .name = "ritlb1", 3747 .translate = translate_rtlb, 3748 .par = (const uint32_t[]){false, 1}, 3749 .op_flags = XTENSA_OP_PRIVILEGED, 3750 }, { 3751 .name = "rptlb0", 3752 .translate = translate_rptlb0, 3753 .op_flags = XTENSA_OP_PRIVILEGED, 3754 }, { 3755 .name = "rptlb1", 3756 .translate = translate_rptlb1, 3757 .op_flags = XTENSA_OP_PRIVILEGED, 3758 }, { 3759 .name = "rotw", 3760 .translate = translate_rotw, 3761 .op_flags = XTENSA_OP_PRIVILEGED | 3762 XTENSA_OP_EXIT_TB_M1 | 3763 XTENSA_OP_SYNC_REGISTER_WINDOW, 3764 }, { 3765 .name = "rsil", 3766 .translate = translate_rsil, 3767 .op_flags = 3768 XTENSA_OP_PRIVILEGED | 3769 XTENSA_OP_EXIT_TB_0 | 3770 XTENSA_OP_CHECK_INTERRUPTS, 3771 }, { 3772 .name = "rsr.176", 3773 .translate = translate_rsr, 3774 .par = (const uint32_t[]){176}, 3775 .op_flags = XTENSA_OP_PRIVILEGED, 3776 }, { 3777 .name = "rsr.208", 3778 .translate = translate_rsr, 3779 .par = (const uint32_t[]){208}, 3780 .op_flags = XTENSA_OP_PRIVILEGED, 3781 }, { 3782 .name = "rsr.acchi", 3783 .translate = translate_rsr, 3784 .test_exceptions = test_exceptions_sr, 3785 .par = (const uint32_t[]){ 3786 ACCHI, 3787 XTENSA_OPTION_MAC16, 3788 }, 3789 }, { 3790 .name = "rsr.acclo", 3791 .translate = translate_rsr, 3792 .test_exceptions = test_exceptions_sr, 3793 .par = (const uint32_t[]){ 3794 ACCLO, 3795 XTENSA_OPTION_MAC16, 3796 }, 3797 }, { 3798 .name = "rsr.atomctl", 3799 .translate = translate_rsr, 3800 .test_exceptions = test_exceptions_sr, 3801 .par = (const uint32_t[]){ 3802 ATOMCTL, 3803 XTENSA_OPTION_ATOMCTL, 3804 }, 3805 .op_flags = XTENSA_OP_PRIVILEGED, 3806 }, { 3807 .name = "rsr.br", 3808 .translate = translate_rsr, 3809 .test_exceptions = test_exceptions_sr, 3810 .par = (const uint32_t[]){ 3811 BR, 3812 XTENSA_OPTION_BOOLEAN, 3813 }, 3814 }, { 3815 .name = "rsr.cacheadrdis", 3816 .translate = translate_rsr, 3817 .test_exceptions = test_exceptions_sr, 3818 .par = (const uint32_t[]){ 3819 CACHEADRDIS, 3820 XTENSA_OPTION_MPU, 3821 }, 3822 .op_flags = XTENSA_OP_PRIVILEGED, 3823 }, { 3824 .name = "rsr.cacheattr", 3825 .translate = translate_rsr, 3826 .test_exceptions = test_exceptions_sr, 3827 .par = (const uint32_t[]){ 3828 CACHEATTR, 3829 XTENSA_OPTION_CACHEATTR, 3830 }, 3831 .op_flags = XTENSA_OP_PRIVILEGED, 3832 }, { 3833 .name = "rsr.ccompare0", 3834 .translate = translate_rsr, 3835 .test_exceptions = test_exceptions_ccompare, 3836 .par = (const uint32_t[]){ 3837 CCOMPARE, 3838 XTENSA_OPTION_TIMER_INTERRUPT, 3839 }, 3840 .op_flags = XTENSA_OP_PRIVILEGED, 3841 }, { 3842 .name = "rsr.ccompare1", 3843 .translate = translate_rsr, 3844 .test_exceptions = test_exceptions_ccompare, 3845 .par = (const uint32_t[]){ 3846 CCOMPARE + 1, 3847 XTENSA_OPTION_TIMER_INTERRUPT, 3848 }, 3849 .op_flags = XTENSA_OP_PRIVILEGED, 3850 }, { 3851 .name = "rsr.ccompare2", 3852 .translate = translate_rsr, 3853 .test_exceptions = test_exceptions_ccompare, 3854 .par = (const uint32_t[]){ 3855 CCOMPARE + 2, 3856 XTENSA_OPTION_TIMER_INTERRUPT, 3857 }, 3858 .op_flags = XTENSA_OP_PRIVILEGED, 3859 }, { 3860 .name = "rsr.ccount", 3861 .translate = translate_rsr_ccount, 3862 .test_exceptions = test_exceptions_sr, 3863 .par = (const uint32_t[]){ 3864 CCOUNT, 3865 XTENSA_OPTION_TIMER_INTERRUPT, 3866 }, 3867 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_EXIT_TB_0, 3868 }, { 3869 .name = "rsr.configid0", 3870 .translate = translate_rsr, 3871 .par = (const uint32_t[]){CONFIGID0}, 3872 .op_flags = XTENSA_OP_PRIVILEGED, 3873 }, { 3874 .name = "rsr.configid1", 3875 .translate = translate_rsr, 3876 .par = (const uint32_t[]){CONFIGID1}, 3877 .op_flags = XTENSA_OP_PRIVILEGED, 3878 }, { 3879 .name = "rsr.cpenable", 3880 .translate = translate_rsr, 3881 .test_exceptions = test_exceptions_sr, 3882 .par = (const uint32_t[]){ 3883 CPENABLE, 3884 XTENSA_OPTION_COPROCESSOR, 3885 }, 3886 .op_flags = XTENSA_OP_PRIVILEGED, 3887 }, { 3888 .name = "rsr.dbreaka0", 3889 .translate = translate_rsr, 3890 .test_exceptions = test_exceptions_dbreak, 3891 .par = (const uint32_t[]){ 3892 DBREAKA, 3893 XTENSA_OPTION_DEBUG, 3894 }, 3895 .op_flags = XTENSA_OP_PRIVILEGED, 3896 }, { 3897 .name = "rsr.dbreaka1", 3898 .translate = translate_rsr, 3899 .test_exceptions = test_exceptions_dbreak, 3900 .par = (const uint32_t[]){ 3901 DBREAKA + 1, 3902 XTENSA_OPTION_DEBUG, 3903 }, 3904 .op_flags = XTENSA_OP_PRIVILEGED, 3905 }, { 3906 .name = "rsr.dbreakc0", 3907 .translate = translate_rsr, 3908 .test_exceptions = test_exceptions_dbreak, 3909 .par = (const uint32_t[]){ 3910 DBREAKC, 3911 XTENSA_OPTION_DEBUG, 3912 }, 3913 .op_flags = XTENSA_OP_PRIVILEGED, 3914 }, { 3915 .name = "rsr.dbreakc1", 3916 .translate = translate_rsr, 3917 .test_exceptions = test_exceptions_dbreak, 3918 .par = (const uint32_t[]){ 3919 DBREAKC + 1, 3920 XTENSA_OPTION_DEBUG, 3921 }, 3922 .op_flags = XTENSA_OP_PRIVILEGED, 3923 }, { 3924 .name = "rsr.ddr", 3925 .translate = translate_rsr, 3926 .test_exceptions = test_exceptions_sr, 3927 .par = (const uint32_t[]){ 3928 DDR, 3929 XTENSA_OPTION_DEBUG, 3930 }, 3931 .op_flags = XTENSA_OP_PRIVILEGED, 3932 }, { 3933 .name = "rsr.debugcause", 3934 .translate = translate_rsr, 3935 .test_exceptions = test_exceptions_sr, 3936 .par = (const uint32_t[]){ 3937 DEBUGCAUSE, 3938 XTENSA_OPTION_DEBUG, 3939 }, 3940 .op_flags = XTENSA_OP_PRIVILEGED, 3941 }, { 3942 .name = "rsr.depc", 3943 .translate = translate_rsr, 3944 .test_exceptions = test_exceptions_sr, 3945 .par = (const uint32_t[]){ 3946 DEPC, 3947 XTENSA_OPTION_EXCEPTION, 3948 }, 3949 .op_flags = XTENSA_OP_PRIVILEGED, 3950 }, { 3951 .name = "rsr.dtlbcfg", 3952 .translate = translate_rsr, 3953 .test_exceptions = test_exceptions_sr, 3954 .par = (const uint32_t[]){ 3955 DTLBCFG, 3956 XTENSA_OPTION_MMU, 3957 }, 3958 .op_flags = XTENSA_OP_PRIVILEGED, 3959 }, { 3960 .name = "rsr.epc1", 3961 .translate = translate_rsr, 3962 .test_exceptions = test_exceptions_sr, 3963 .par = (const uint32_t[]){ 3964 EPC1, 3965 XTENSA_OPTION_EXCEPTION, 3966 }, 3967 .op_flags = XTENSA_OP_PRIVILEGED, 3968 }, { 3969 .name = "rsr.epc2", 3970 .translate = translate_rsr, 3971 .test_exceptions = test_exceptions_hpi, 3972 .par = (const uint32_t[]){ 3973 EPC1 + 1, 3974 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 3975 }, 3976 .op_flags = XTENSA_OP_PRIVILEGED, 3977 }, { 3978 .name = "rsr.epc3", 3979 .translate = translate_rsr, 3980 .test_exceptions = test_exceptions_hpi, 3981 .par = (const uint32_t[]){ 3982 EPC1 + 2, 3983 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 3984 }, 3985 .op_flags = XTENSA_OP_PRIVILEGED, 3986 }, { 3987 .name = "rsr.epc4", 3988 .translate = translate_rsr, 3989 .test_exceptions = test_exceptions_hpi, 3990 .par = (const uint32_t[]){ 3991 EPC1 + 3, 3992 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 3993 }, 3994 .op_flags = XTENSA_OP_PRIVILEGED, 3995 }, { 3996 .name = "rsr.epc5", 3997 .translate = translate_rsr, 3998 .test_exceptions = test_exceptions_hpi, 3999 .par = (const uint32_t[]){ 4000 EPC1 + 4, 4001 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 4002 }, 4003 .op_flags = XTENSA_OP_PRIVILEGED, 4004 }, { 4005 .name = "rsr.epc6", 4006 .translate = translate_rsr, 4007 .test_exceptions = test_exceptions_hpi, 4008 .par = (const uint32_t[]){ 4009 EPC1 + 5, 4010 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 4011 }, 4012 .op_flags = XTENSA_OP_PRIVILEGED, 4013 }, { 4014 .name = "rsr.epc7", 4015 .translate = translate_rsr, 4016 .test_exceptions = test_exceptions_hpi, 4017 .par = (const uint32_t[]){ 4018 EPC1 + 6, 4019 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 4020 }, 4021 .op_flags = XTENSA_OP_PRIVILEGED, 4022 }, { 4023 .name = "rsr.eps2", 4024 .translate = translate_rsr, 4025 .test_exceptions = test_exceptions_hpi, 4026 .par = (const uint32_t[]){ 4027 EPS2, 4028 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 4029 }, 4030 .op_flags = XTENSA_OP_PRIVILEGED, 4031 }, { 4032 .name = "rsr.eps3", 4033 .translate = translate_rsr, 4034 .test_exceptions = test_exceptions_hpi, 4035 .par = (const uint32_t[]){ 4036 EPS2 + 1, 4037 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 4038 }, 4039 .op_flags = XTENSA_OP_PRIVILEGED, 4040 }, { 4041 .name = "rsr.eps4", 4042 .translate = translate_rsr, 4043 .test_exceptions = test_exceptions_hpi, 4044 .par = (const uint32_t[]){ 4045 EPS2 + 2, 4046 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 4047 }, 4048 .op_flags = XTENSA_OP_PRIVILEGED, 4049 }, { 4050 .name = "rsr.eps5", 4051 .translate = translate_rsr, 4052 .test_exceptions = test_exceptions_hpi, 4053 .par = (const uint32_t[]){ 4054 EPS2 + 3, 4055 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 4056 }, 4057 .op_flags = XTENSA_OP_PRIVILEGED, 4058 }, { 4059 .name = "rsr.eps6", 4060 .translate = translate_rsr, 4061 .test_exceptions = test_exceptions_hpi, 4062 .par = (const uint32_t[]){ 4063 EPS2 + 4, 4064 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 4065 }, 4066 .op_flags = XTENSA_OP_PRIVILEGED, 4067 }, { 4068 .name = "rsr.eps7", 4069 .translate = translate_rsr, 4070 .test_exceptions = test_exceptions_hpi, 4071 .par = (const uint32_t[]){ 4072 EPS2 + 5, 4073 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 4074 }, 4075 .op_flags = XTENSA_OP_PRIVILEGED, 4076 }, { 4077 .name = "rsr.eraccess", 4078 .translate = translate_rsr, 4079 .par = (const uint32_t[]){ERACCESS}, 4080 .op_flags = XTENSA_OP_PRIVILEGED, 4081 }, { 4082 .name = "rsr.exccause", 4083 .translate = translate_rsr, 4084 .test_exceptions = test_exceptions_sr, 4085 .par = (const uint32_t[]){ 4086 EXCCAUSE, 4087 XTENSA_OPTION_EXCEPTION, 4088 }, 4089 .op_flags = XTENSA_OP_PRIVILEGED, 4090 }, { 4091 .name = "rsr.excsave1", 4092 .translate = translate_rsr, 4093 .test_exceptions = test_exceptions_sr, 4094 .par = (const uint32_t[]){ 4095 EXCSAVE1, 4096 XTENSA_OPTION_EXCEPTION, 4097 }, 4098 .op_flags = XTENSA_OP_PRIVILEGED, 4099 }, { 4100 .name = "rsr.excsave2", 4101 .translate = translate_rsr, 4102 .test_exceptions = test_exceptions_hpi, 4103 .par = (const uint32_t[]){ 4104 EXCSAVE1 + 1, 4105 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 4106 }, 4107 .op_flags = XTENSA_OP_PRIVILEGED, 4108 }, { 4109 .name = "rsr.excsave3", 4110 .translate = translate_rsr, 4111 .test_exceptions = test_exceptions_hpi, 4112 .par = (const uint32_t[]){ 4113 EXCSAVE1 + 2, 4114 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 4115 }, 4116 .op_flags = XTENSA_OP_PRIVILEGED, 4117 }, { 4118 .name = "rsr.excsave4", 4119 .translate = translate_rsr, 4120 .test_exceptions = test_exceptions_hpi, 4121 .par = (const uint32_t[]){ 4122 EXCSAVE1 + 3, 4123 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 4124 }, 4125 .op_flags = XTENSA_OP_PRIVILEGED, 4126 }, { 4127 .name = "rsr.excsave5", 4128 .translate = translate_rsr, 4129 .test_exceptions = test_exceptions_hpi, 4130 .par = (const uint32_t[]){ 4131 EXCSAVE1 + 4, 4132 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 4133 }, 4134 .op_flags = XTENSA_OP_PRIVILEGED, 4135 }, { 4136 .name = "rsr.excsave6", 4137 .translate = translate_rsr, 4138 .test_exceptions = test_exceptions_hpi, 4139 .par = (const uint32_t[]){ 4140 EXCSAVE1 + 5, 4141 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 4142 }, 4143 .op_flags = XTENSA_OP_PRIVILEGED, 4144 }, { 4145 .name = "rsr.excsave7", 4146 .translate = translate_rsr, 4147 .test_exceptions = test_exceptions_hpi, 4148 .par = (const uint32_t[]){ 4149 EXCSAVE1 + 6, 4150 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 4151 }, 4152 .op_flags = XTENSA_OP_PRIVILEGED, 4153 }, { 4154 .name = "rsr.excvaddr", 4155 .translate = translate_rsr, 4156 .test_exceptions = test_exceptions_sr, 4157 .par = (const uint32_t[]){ 4158 EXCVADDR, 4159 XTENSA_OPTION_EXCEPTION, 4160 }, 4161 .op_flags = XTENSA_OP_PRIVILEGED, 4162 }, { 4163 .name = "rsr.ibreaka0", 4164 .translate = translate_rsr, 4165 .test_exceptions = test_exceptions_ibreak, 4166 .par = (const uint32_t[]){ 4167 IBREAKA, 4168 XTENSA_OPTION_DEBUG, 4169 }, 4170 .op_flags = XTENSA_OP_PRIVILEGED, 4171 }, { 4172 .name = "rsr.ibreaka1", 4173 .translate = translate_rsr, 4174 .test_exceptions = test_exceptions_ibreak, 4175 .par = (const uint32_t[]){ 4176 IBREAKA + 1, 4177 XTENSA_OPTION_DEBUG, 4178 }, 4179 .op_flags = XTENSA_OP_PRIVILEGED, 4180 }, { 4181 .name = "rsr.ibreakenable", 4182 .translate = translate_rsr, 4183 .test_exceptions = test_exceptions_sr, 4184 .par = (const uint32_t[]){ 4185 IBREAKENABLE, 4186 XTENSA_OPTION_DEBUG, 4187 }, 4188 .op_flags = XTENSA_OP_PRIVILEGED, 4189 }, { 4190 .name = "rsr.icount", 4191 .translate = translate_rsr, 4192 .test_exceptions = test_exceptions_sr, 4193 .par = (const uint32_t[]){ 4194 ICOUNT, 4195 XTENSA_OPTION_DEBUG, 4196 }, 4197 .op_flags = XTENSA_OP_PRIVILEGED, 4198 }, { 4199 .name = "rsr.icountlevel", 4200 .translate = translate_rsr, 4201 .test_exceptions = test_exceptions_sr, 4202 .par = (const uint32_t[]){ 4203 ICOUNTLEVEL, 4204 XTENSA_OPTION_DEBUG, 4205 }, 4206 .op_flags = XTENSA_OP_PRIVILEGED, 4207 }, { 4208 .name = "rsr.intclear", 4209 .translate = translate_rsr, 4210 .test_exceptions = test_exceptions_sr, 4211 .par = (const uint32_t[]){ 4212 INTCLEAR, 4213 XTENSA_OPTION_INTERRUPT, 4214 }, 4215 .op_flags = XTENSA_OP_PRIVILEGED, 4216 }, { 4217 .name = "rsr.intenable", 4218 .translate = translate_rsr, 4219 .test_exceptions = test_exceptions_sr, 4220 .par = (const uint32_t[]){ 4221 INTENABLE, 4222 XTENSA_OPTION_INTERRUPT, 4223 }, 4224 .op_flags = XTENSA_OP_PRIVILEGED, 4225 }, { 4226 .name = "rsr.interrupt", 4227 .translate = translate_rsr_ccount, 4228 .test_exceptions = test_exceptions_sr, 4229 .par = (const uint32_t[]){ 4230 INTSET, 4231 XTENSA_OPTION_INTERRUPT, 4232 }, 4233 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_EXIT_TB_0, 4234 }, { 4235 .name = "rsr.intset", 4236 .translate = translate_rsr_ccount, 4237 .test_exceptions = test_exceptions_sr, 4238 .par = (const uint32_t[]){ 4239 INTSET, 4240 XTENSA_OPTION_INTERRUPT, 4241 }, 4242 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_EXIT_TB_0, 4243 }, { 4244 .name = "rsr.itlbcfg", 4245 .translate = translate_rsr, 4246 .test_exceptions = test_exceptions_sr, 4247 .par = (const uint32_t[]){ 4248 ITLBCFG, 4249 XTENSA_OPTION_MMU, 4250 }, 4251 .op_flags = XTENSA_OP_PRIVILEGED, 4252 }, { 4253 .name = "rsr.lbeg", 4254 .translate = translate_rsr, 4255 .test_exceptions = test_exceptions_sr, 4256 .par = (const uint32_t[]){ 4257 LBEG, 4258 XTENSA_OPTION_LOOP, 4259 }, 4260 }, { 4261 .name = "rsr.lcount", 4262 .translate = translate_rsr, 4263 .test_exceptions = test_exceptions_sr, 4264 .par = (const uint32_t[]){ 4265 LCOUNT, 4266 XTENSA_OPTION_LOOP, 4267 }, 4268 }, { 4269 .name = "rsr.lend", 4270 .translate = translate_rsr, 4271 .test_exceptions = test_exceptions_sr, 4272 .par = (const uint32_t[]){ 4273 LEND, 4274 XTENSA_OPTION_LOOP, 4275 }, 4276 }, { 4277 .name = "rsr.litbase", 4278 .translate = translate_rsr, 4279 .test_exceptions = test_exceptions_sr, 4280 .par = (const uint32_t[]){ 4281 LITBASE, 4282 XTENSA_OPTION_EXTENDED_L32R, 4283 }, 4284 }, { 4285 .name = "rsr.m0", 4286 .translate = translate_rsr, 4287 .test_exceptions = test_exceptions_sr, 4288 .par = (const uint32_t[]){ 4289 MR, 4290 XTENSA_OPTION_MAC16, 4291 }, 4292 }, { 4293 .name = "rsr.m1", 4294 .translate = translate_rsr, 4295 .test_exceptions = test_exceptions_sr, 4296 .par = (const uint32_t[]){ 4297 MR + 1, 4298 XTENSA_OPTION_MAC16, 4299 }, 4300 }, { 4301 .name = "rsr.m2", 4302 .translate = translate_rsr, 4303 .test_exceptions = test_exceptions_sr, 4304 .par = (const uint32_t[]){ 4305 MR + 2, 4306 XTENSA_OPTION_MAC16, 4307 }, 4308 }, { 4309 .name = "rsr.m3", 4310 .translate = translate_rsr, 4311 .test_exceptions = test_exceptions_sr, 4312 .par = (const uint32_t[]){ 4313 MR + 3, 4314 XTENSA_OPTION_MAC16, 4315 }, 4316 }, { 4317 .name = "rsr.memctl", 4318 .translate = translate_rsr, 4319 .par = (const uint32_t[]){MEMCTL}, 4320 .op_flags = XTENSA_OP_PRIVILEGED, 4321 }, { 4322 .name = "rsr.mecr", 4323 .translate = translate_rsr, 4324 .test_exceptions = test_exceptions_sr, 4325 .par = (const uint32_t[]){ 4326 MECR, 4327 XTENSA_OPTION_MEMORY_ECC_PARITY, 4328 }, 4329 .op_flags = XTENSA_OP_PRIVILEGED, 4330 }, { 4331 .name = "rsr.mepc", 4332 .translate = translate_rsr, 4333 .test_exceptions = test_exceptions_sr, 4334 .par = (const uint32_t[]){ 4335 MEPC, 4336 XTENSA_OPTION_MEMORY_ECC_PARITY, 4337 }, 4338 .op_flags = XTENSA_OP_PRIVILEGED, 4339 }, { 4340 .name = "rsr.meps", 4341 .translate = translate_rsr, 4342 .test_exceptions = test_exceptions_sr, 4343 .par = (const uint32_t[]){ 4344 MEPS, 4345 XTENSA_OPTION_MEMORY_ECC_PARITY, 4346 }, 4347 .op_flags = XTENSA_OP_PRIVILEGED, 4348 }, { 4349 .name = "rsr.mesave", 4350 .translate = translate_rsr, 4351 .test_exceptions = test_exceptions_sr, 4352 .par = (const uint32_t[]){ 4353 MESAVE, 4354 XTENSA_OPTION_MEMORY_ECC_PARITY, 4355 }, 4356 .op_flags = XTENSA_OP_PRIVILEGED, 4357 }, { 4358 .name = "rsr.mesr", 4359 .translate = translate_rsr, 4360 .test_exceptions = test_exceptions_sr, 4361 .par = (const uint32_t[]){ 4362 MESR, 4363 XTENSA_OPTION_MEMORY_ECC_PARITY, 4364 }, 4365 .op_flags = XTENSA_OP_PRIVILEGED, 4366 }, { 4367 .name = "rsr.mevaddr", 4368 .translate = translate_rsr, 4369 .test_exceptions = test_exceptions_sr, 4370 .par = (const uint32_t[]){ 4371 MESR, 4372 XTENSA_OPTION_MEMORY_ECC_PARITY, 4373 }, 4374 .op_flags = XTENSA_OP_PRIVILEGED, 4375 }, { 4376 .name = "rsr.misc0", 4377 .translate = translate_rsr, 4378 .test_exceptions = test_exceptions_sr, 4379 .par = (const uint32_t[]){ 4380 MISC, 4381 XTENSA_OPTION_MISC_SR, 4382 }, 4383 .op_flags = XTENSA_OP_PRIVILEGED, 4384 }, { 4385 .name = "rsr.misc1", 4386 .translate = translate_rsr, 4387 .test_exceptions = test_exceptions_sr, 4388 .par = (const uint32_t[]){ 4389 MISC + 1, 4390 XTENSA_OPTION_MISC_SR, 4391 }, 4392 .op_flags = XTENSA_OP_PRIVILEGED, 4393 }, { 4394 .name = "rsr.misc2", 4395 .translate = translate_rsr, 4396 .test_exceptions = test_exceptions_sr, 4397 .par = (const uint32_t[]){ 4398 MISC + 2, 4399 XTENSA_OPTION_MISC_SR, 4400 }, 4401 .op_flags = XTENSA_OP_PRIVILEGED, 4402 }, { 4403 .name = "rsr.misc3", 4404 .translate = translate_rsr, 4405 .test_exceptions = test_exceptions_sr, 4406 .par = (const uint32_t[]){ 4407 MISC + 3, 4408 XTENSA_OPTION_MISC_SR, 4409 }, 4410 .op_flags = XTENSA_OP_PRIVILEGED, 4411 }, { 4412 .name = "rsr.mpucfg", 4413 .translate = translate_rsr, 4414 .test_exceptions = test_exceptions_sr, 4415 .par = (const uint32_t[]){ 4416 MPUCFG, 4417 XTENSA_OPTION_MPU, 4418 }, 4419 .op_flags = XTENSA_OP_PRIVILEGED, 4420 }, { 4421 .name = "rsr.mpuenb", 4422 .translate = translate_rsr, 4423 .test_exceptions = test_exceptions_sr, 4424 .par = (const uint32_t[]){ 4425 MPUENB, 4426 XTENSA_OPTION_MPU, 4427 }, 4428 .op_flags = XTENSA_OP_PRIVILEGED, 4429 }, { 4430 .name = "rsr.prefctl", 4431 .translate = translate_rsr, 4432 .par = (const uint32_t[]){PREFCTL}, 4433 }, { 4434 .name = "rsr.prid", 4435 .translate = translate_rsr, 4436 .test_exceptions = test_exceptions_sr, 4437 .par = (const uint32_t[]){ 4438 PRID, 4439 XTENSA_OPTION_PROCESSOR_ID, 4440 }, 4441 .op_flags = XTENSA_OP_PRIVILEGED, 4442 }, { 4443 .name = "rsr.ps", 4444 .translate = translate_rsr, 4445 .test_exceptions = test_exceptions_sr, 4446 .par = (const uint32_t[]){ 4447 PS, 4448 XTENSA_OPTION_EXCEPTION, 4449 }, 4450 .op_flags = XTENSA_OP_PRIVILEGED, 4451 }, { 4452 .name = "rsr.ptevaddr", 4453 .translate = translate_rsr_ptevaddr, 4454 .test_exceptions = test_exceptions_sr, 4455 .par = (const uint32_t[]){ 4456 PTEVADDR, 4457 XTENSA_OPTION_MMU, 4458 }, 4459 .op_flags = XTENSA_OP_PRIVILEGED, 4460 }, { 4461 .name = "rsr.rasid", 4462 .translate = translate_rsr, 4463 .test_exceptions = test_exceptions_sr, 4464 .par = (const uint32_t[]){ 4465 RASID, 4466 XTENSA_OPTION_MMU, 4467 }, 4468 .op_flags = XTENSA_OP_PRIVILEGED, 4469 }, { 4470 .name = "rsr.sar", 4471 .translate = translate_rsr, 4472 .par = (const uint32_t[]){SAR}, 4473 }, { 4474 .name = "rsr.scompare1", 4475 .translate = translate_rsr, 4476 .test_exceptions = test_exceptions_sr, 4477 .par = (const uint32_t[]){ 4478 SCOMPARE1, 4479 XTENSA_OPTION_CONDITIONAL_STORE, 4480 }, 4481 }, { 4482 .name = "rsr.vecbase", 4483 .translate = translate_rsr, 4484 .test_exceptions = test_exceptions_sr, 4485 .par = (const uint32_t[]){ 4486 VECBASE, 4487 XTENSA_OPTION_RELOCATABLE_VECTOR, 4488 }, 4489 .op_flags = XTENSA_OP_PRIVILEGED, 4490 }, { 4491 .name = "rsr.windowbase", 4492 .translate = translate_rsr, 4493 .test_exceptions = test_exceptions_sr, 4494 .par = (const uint32_t[]){ 4495 WINDOW_BASE, 4496 XTENSA_OPTION_WINDOWED_REGISTER, 4497 }, 4498 .op_flags = XTENSA_OP_PRIVILEGED, 4499 }, { 4500 .name = "rsr.windowstart", 4501 .translate = translate_rsr, 4502 .test_exceptions = test_exceptions_sr, 4503 .par = (const uint32_t[]){ 4504 WINDOW_START, 4505 XTENSA_OPTION_WINDOWED_REGISTER, 4506 }, 4507 .op_flags = XTENSA_OP_PRIVILEGED, 4508 }, { 4509 .name = "rsync", 4510 .translate = translate_nop, 4511 }, { 4512 .name = "rur.expstate", 4513 .translate = translate_rur, 4514 .par = (const uint32_t[]){EXPSTATE}, 4515 }, { 4516 .name = "rur.threadptr", 4517 .translate = translate_rur, 4518 .par = (const uint32_t[]){THREADPTR}, 4519 }, { 4520 .name = "s16i", 4521 .translate = translate_ldst, 4522 .par = (const uint32_t[]){MO_TEUW, false, true}, 4523 .op_flags = XTENSA_OP_STORE, 4524 }, { 4525 .name = "s32c1i", 4526 .translate = translate_s32c1i, 4527 .op_flags = XTENSA_OP_LOAD | XTENSA_OP_STORE, 4528 }, { 4529 .name = "s32e", 4530 .translate = translate_s32e, 4531 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_STORE, 4532 }, { 4533 .name = "s32ex", 4534 .translate = translate_s32ex, 4535 .op_flags = XTENSA_OP_LOAD | XTENSA_OP_STORE, 4536 }, { 4537 .name = (const char * const[]) { 4538 "s32i", "s32i.n", "s32nb", NULL, 4539 }, 4540 .translate = translate_ldst, 4541 .par = (const uint32_t[]){MO_TEUL, false, true}, 4542 .op_flags = XTENSA_OP_NAME_ARRAY | XTENSA_OP_STORE, 4543 }, { 4544 .name = "s32ri", 4545 .translate = translate_ldst, 4546 .par = (const uint32_t[]){MO_TEUL | MO_ALIGN, true, true}, 4547 .op_flags = XTENSA_OP_STORE, 4548 }, { 4549 .name = "s8i", 4550 .translate = translate_ldst, 4551 .par = (const uint32_t[]){MO_UB, false, true}, 4552 .op_flags = XTENSA_OP_STORE, 4553 }, { 4554 .name = "salt", 4555 .translate = translate_salt, 4556 .par = (const uint32_t[]){TCG_COND_LT}, 4557 }, { 4558 .name = "saltu", 4559 .translate = translate_salt, 4560 .par = (const uint32_t[]){TCG_COND_LTU}, 4561 }, { 4562 .name = "sdct", 4563 .translate = translate_nop, 4564 .op_flags = XTENSA_OP_PRIVILEGED, 4565 }, { 4566 .name = "sdcw", 4567 .translate = translate_nop, 4568 .op_flags = XTENSA_OP_PRIVILEGED, 4569 }, { 4570 .name = "setb_expstate", 4571 .translate = translate_setb_expstate, 4572 }, { 4573 .name = "sext", 4574 .translate = translate_sext, 4575 }, { 4576 .name = "sict", 4577 .translate = translate_nop, 4578 .op_flags = XTENSA_OP_PRIVILEGED, 4579 }, { 4580 .name = "sicw", 4581 .translate = translate_nop, 4582 .op_flags = XTENSA_OP_PRIVILEGED, 4583 }, { 4584 .name = "simcall", 4585 .translate = translate_simcall, 4586 .test_exceptions = test_exceptions_simcall, 4587 .op_flags = XTENSA_OP_PRIVILEGED, 4588 }, { 4589 .name = "sll", 4590 .translate = translate_sll, 4591 }, { 4592 .name = "slli", 4593 .translate = translate_slli, 4594 }, { 4595 .name = "sra", 4596 .translate = translate_sra, 4597 }, { 4598 .name = "srai", 4599 .translate = translate_srai, 4600 }, { 4601 .name = "src", 4602 .translate = translate_src, 4603 }, { 4604 .name = "srl", 4605 .translate = translate_srl, 4606 }, { 4607 .name = "srli", 4608 .translate = translate_srli, 4609 }, { 4610 .name = "ssa8b", 4611 .translate = translate_ssa8b, 4612 }, { 4613 .name = "ssa8l", 4614 .translate = translate_ssa8l, 4615 }, { 4616 .name = "ssai", 4617 .translate = translate_ssai, 4618 }, { 4619 .name = "ssl", 4620 .translate = translate_ssl, 4621 }, { 4622 .name = "ssr", 4623 .translate = translate_ssr, 4624 }, { 4625 .name = "sub", 4626 .translate = translate_sub, 4627 }, { 4628 .name = "subx2", 4629 .translate = translate_subx, 4630 .par = (const uint32_t[]){1}, 4631 }, { 4632 .name = "subx4", 4633 .translate = translate_subx, 4634 .par = (const uint32_t[]){2}, 4635 }, { 4636 .name = "subx8", 4637 .translate = translate_subx, 4638 .par = (const uint32_t[]){3}, 4639 }, { 4640 .name = "syscall", 4641 .op_flags = XTENSA_OP_SYSCALL, 4642 }, { 4643 .name = "umul.aa.hh", 4644 .translate = translate_mac16, 4645 .par = (const uint32_t[]){MAC16_UMUL, MAC16_HH, 0}, 4646 }, { 4647 .name = "umul.aa.hl", 4648 .translate = translate_mac16, 4649 .par = (const uint32_t[]){MAC16_UMUL, MAC16_HL, 0}, 4650 }, { 4651 .name = "umul.aa.lh", 4652 .translate = translate_mac16, 4653 .par = (const uint32_t[]){MAC16_UMUL, MAC16_LH, 0}, 4654 }, { 4655 .name = "umul.aa.ll", 4656 .translate = translate_mac16, 4657 .par = (const uint32_t[]){MAC16_UMUL, MAC16_LL, 0}, 4658 }, { 4659 .name = "waiti", 4660 .translate = translate_waiti, 4661 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_EXIT_TB_0, 4662 }, { 4663 .name = "wdtlb", 4664 .translate = translate_wtlb, 4665 .par = (const uint32_t[]){true}, 4666 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_EXIT_TB_M1, 4667 }, { 4668 .name = "wer", 4669 .translate = translate_wer, 4670 .op_flags = XTENSA_OP_PRIVILEGED, 4671 }, { 4672 .name = "witlb", 4673 .translate = translate_wtlb, 4674 .par = (const uint32_t[]){false}, 4675 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_EXIT_TB_M1, 4676 }, { 4677 .name = "wptlb", 4678 .translate = translate_wptlb, 4679 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_EXIT_TB_M1, 4680 }, { 4681 .name = "wrmsk_expstate", 4682 .translate = translate_wrmsk_expstate, 4683 }, { 4684 .name = "wsr.176", 4685 .op_flags = XTENSA_OP_ILL, 4686 }, { 4687 .name = "wsr.208", 4688 .op_flags = XTENSA_OP_ILL, 4689 }, { 4690 .name = "wsr.acchi", 4691 .translate = translate_wsr_acchi, 4692 .test_exceptions = test_exceptions_sr, 4693 .par = (const uint32_t[]){ 4694 ACCHI, 4695 XTENSA_OPTION_MAC16, 4696 }, 4697 }, { 4698 .name = "wsr.acclo", 4699 .translate = translate_wsr, 4700 .test_exceptions = test_exceptions_sr, 4701 .par = (const uint32_t[]){ 4702 ACCLO, 4703 XTENSA_OPTION_MAC16, 4704 }, 4705 }, { 4706 .name = "wsr.atomctl", 4707 .translate = translate_wsr_mask, 4708 .test_exceptions = test_exceptions_sr, 4709 .par = (const uint32_t[]){ 4710 ATOMCTL, 4711 XTENSA_OPTION_ATOMCTL, 4712 0x3f, 4713 }, 4714 .op_flags = XTENSA_OP_PRIVILEGED, 4715 }, { 4716 .name = "wsr.br", 4717 .translate = translate_wsr_mask, 4718 .test_exceptions = test_exceptions_sr, 4719 .par = (const uint32_t[]){ 4720 BR, 4721 XTENSA_OPTION_BOOLEAN, 4722 0xffff, 4723 }, 4724 }, { 4725 .name = "wsr.cacheadrdis", 4726 .translate = translate_wsr_mask, 4727 .test_exceptions = test_exceptions_sr, 4728 .par = (const uint32_t[]){ 4729 CACHEADRDIS, 4730 XTENSA_OPTION_MPU, 4731 0xff, 4732 }, 4733 .op_flags = XTENSA_OP_PRIVILEGED, 4734 }, { 4735 .name = "wsr.cacheattr", 4736 .translate = translate_wsr, 4737 .test_exceptions = test_exceptions_sr, 4738 .par = (const uint32_t[]){ 4739 CACHEATTR, 4740 XTENSA_OPTION_CACHEATTR, 4741 }, 4742 .op_flags = XTENSA_OP_PRIVILEGED, 4743 }, { 4744 .name = "wsr.ccompare0", 4745 .translate = translate_wsr_ccompare, 4746 .test_exceptions = test_exceptions_ccompare, 4747 .par = (const uint32_t[]){ 4748 CCOMPARE, 4749 XTENSA_OPTION_TIMER_INTERRUPT, 4750 }, 4751 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_EXIT_TB_0, 4752 }, { 4753 .name = "wsr.ccompare1", 4754 .translate = translate_wsr_ccompare, 4755 .test_exceptions = test_exceptions_ccompare, 4756 .par = (const uint32_t[]){ 4757 CCOMPARE + 1, 4758 XTENSA_OPTION_TIMER_INTERRUPT, 4759 }, 4760 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_EXIT_TB_0, 4761 }, { 4762 .name = "wsr.ccompare2", 4763 .translate = translate_wsr_ccompare, 4764 .test_exceptions = test_exceptions_ccompare, 4765 .par = (const uint32_t[]){ 4766 CCOMPARE + 2, 4767 XTENSA_OPTION_TIMER_INTERRUPT, 4768 }, 4769 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_EXIT_TB_0, 4770 }, { 4771 .name = "wsr.ccount", 4772 .translate = translate_wsr_ccount, 4773 .test_exceptions = test_exceptions_sr, 4774 .par = (const uint32_t[]){ 4775 CCOUNT, 4776 XTENSA_OPTION_TIMER_INTERRUPT, 4777 }, 4778 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_EXIT_TB_0, 4779 }, { 4780 .name = "wsr.configid0", 4781 .op_flags = XTENSA_OP_ILL, 4782 }, { 4783 .name = "wsr.configid1", 4784 .op_flags = XTENSA_OP_ILL, 4785 }, { 4786 .name = "wsr.cpenable", 4787 .translate = translate_wsr_mask, 4788 .test_exceptions = test_exceptions_sr, 4789 .par = (const uint32_t[]){ 4790 CPENABLE, 4791 XTENSA_OPTION_COPROCESSOR, 4792 0xff, 4793 }, 4794 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_EXIT_TB_M1, 4795 }, { 4796 .name = "wsr.dbreaka0", 4797 .translate = translate_wsr_dbreaka, 4798 .test_exceptions = test_exceptions_dbreak, 4799 .par = (const uint32_t[]){ 4800 DBREAKA, 4801 XTENSA_OPTION_DEBUG, 4802 }, 4803 .op_flags = XTENSA_OP_PRIVILEGED, 4804 }, { 4805 .name = "wsr.dbreaka1", 4806 .translate = translate_wsr_dbreaka, 4807 .test_exceptions = test_exceptions_dbreak, 4808 .par = (const uint32_t[]){ 4809 DBREAKA + 1, 4810 XTENSA_OPTION_DEBUG, 4811 }, 4812 .op_flags = XTENSA_OP_PRIVILEGED, 4813 }, { 4814 .name = "wsr.dbreakc0", 4815 .translate = translate_wsr_dbreakc, 4816 .test_exceptions = test_exceptions_dbreak, 4817 .par = (const uint32_t[]){ 4818 DBREAKC, 4819 XTENSA_OPTION_DEBUG, 4820 }, 4821 .op_flags = XTENSA_OP_PRIVILEGED, 4822 }, { 4823 .name = "wsr.dbreakc1", 4824 .translate = translate_wsr_dbreakc, 4825 .test_exceptions = test_exceptions_dbreak, 4826 .par = (const uint32_t[]){ 4827 DBREAKC + 1, 4828 XTENSA_OPTION_DEBUG, 4829 }, 4830 .op_flags = XTENSA_OP_PRIVILEGED, 4831 }, { 4832 .name = "wsr.ddr", 4833 .translate = translate_wsr, 4834 .test_exceptions = test_exceptions_sr, 4835 .par = (const uint32_t[]){ 4836 DDR, 4837 XTENSA_OPTION_DEBUG, 4838 }, 4839 .op_flags = XTENSA_OP_PRIVILEGED, 4840 }, { 4841 .name = "wsr.debugcause", 4842 .op_flags = XTENSA_OP_ILL, 4843 }, { 4844 .name = "wsr.depc", 4845 .translate = translate_wsr, 4846 .test_exceptions = test_exceptions_sr, 4847 .par = (const uint32_t[]){ 4848 DEPC, 4849 XTENSA_OPTION_EXCEPTION, 4850 }, 4851 .op_flags = XTENSA_OP_PRIVILEGED, 4852 }, { 4853 .name = "wsr.dtlbcfg", 4854 .translate = translate_wsr_mask, 4855 .test_exceptions = test_exceptions_sr, 4856 .par = (const uint32_t[]){ 4857 DTLBCFG, 4858 XTENSA_OPTION_MMU, 4859 0x01130000, 4860 }, 4861 .op_flags = XTENSA_OP_PRIVILEGED, 4862 }, { 4863 .name = "wsr.epc1", 4864 .translate = translate_wsr, 4865 .test_exceptions = test_exceptions_sr, 4866 .par = (const uint32_t[]){ 4867 EPC1, 4868 XTENSA_OPTION_EXCEPTION, 4869 }, 4870 .op_flags = XTENSA_OP_PRIVILEGED, 4871 }, { 4872 .name = "wsr.epc2", 4873 .translate = translate_wsr, 4874 .test_exceptions = test_exceptions_hpi, 4875 .par = (const uint32_t[]){ 4876 EPC1 + 1, 4877 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 4878 }, 4879 .op_flags = XTENSA_OP_PRIVILEGED, 4880 }, { 4881 .name = "wsr.epc3", 4882 .translate = translate_wsr, 4883 .test_exceptions = test_exceptions_hpi, 4884 .par = (const uint32_t[]){ 4885 EPC1 + 2, 4886 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 4887 }, 4888 .op_flags = XTENSA_OP_PRIVILEGED, 4889 }, { 4890 .name = "wsr.epc4", 4891 .translate = translate_wsr, 4892 .test_exceptions = test_exceptions_hpi, 4893 .par = (const uint32_t[]){ 4894 EPC1 + 3, 4895 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 4896 }, 4897 .op_flags = XTENSA_OP_PRIVILEGED, 4898 }, { 4899 .name = "wsr.epc5", 4900 .translate = translate_wsr, 4901 .test_exceptions = test_exceptions_hpi, 4902 .par = (const uint32_t[]){ 4903 EPC1 + 4, 4904 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 4905 }, 4906 .op_flags = XTENSA_OP_PRIVILEGED, 4907 }, { 4908 .name = "wsr.epc6", 4909 .translate = translate_wsr, 4910 .test_exceptions = test_exceptions_hpi, 4911 .par = (const uint32_t[]){ 4912 EPC1 + 5, 4913 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 4914 }, 4915 .op_flags = XTENSA_OP_PRIVILEGED, 4916 }, { 4917 .name = "wsr.epc7", 4918 .translate = translate_wsr, 4919 .test_exceptions = test_exceptions_hpi, 4920 .par = (const uint32_t[]){ 4921 EPC1 + 6, 4922 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 4923 }, 4924 .op_flags = XTENSA_OP_PRIVILEGED, 4925 }, { 4926 .name = "wsr.eps2", 4927 .translate = translate_wsr, 4928 .test_exceptions = test_exceptions_hpi, 4929 .par = (const uint32_t[]){ 4930 EPS2, 4931 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 4932 }, 4933 .op_flags = XTENSA_OP_PRIVILEGED, 4934 }, { 4935 .name = "wsr.eps3", 4936 .translate = translate_wsr, 4937 .test_exceptions = test_exceptions_hpi, 4938 .par = (const uint32_t[]){ 4939 EPS2 + 1, 4940 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 4941 }, 4942 .op_flags = XTENSA_OP_PRIVILEGED, 4943 }, { 4944 .name = "wsr.eps4", 4945 .translate = translate_wsr, 4946 .test_exceptions = test_exceptions_hpi, 4947 .par = (const uint32_t[]){ 4948 EPS2 + 2, 4949 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 4950 }, 4951 .op_flags = XTENSA_OP_PRIVILEGED, 4952 }, { 4953 .name = "wsr.eps5", 4954 .translate = translate_wsr, 4955 .test_exceptions = test_exceptions_hpi, 4956 .par = (const uint32_t[]){ 4957 EPS2 + 3, 4958 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 4959 }, 4960 .op_flags = XTENSA_OP_PRIVILEGED, 4961 }, { 4962 .name = "wsr.eps6", 4963 .translate = translate_wsr, 4964 .test_exceptions = test_exceptions_hpi, 4965 .par = (const uint32_t[]){ 4966 EPS2 + 4, 4967 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 4968 }, 4969 .op_flags = XTENSA_OP_PRIVILEGED, 4970 }, { 4971 .name = "wsr.eps7", 4972 .translate = translate_wsr, 4973 .test_exceptions = test_exceptions_hpi, 4974 .par = (const uint32_t[]){ 4975 EPS2 + 5, 4976 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 4977 }, 4978 .op_flags = XTENSA_OP_PRIVILEGED, 4979 }, { 4980 .name = "wsr.eraccess", 4981 .translate = translate_wsr_mask, 4982 .par = (const uint32_t[]){ 4983 ERACCESS, 4984 0, 4985 0xffff, 4986 }, 4987 .op_flags = XTENSA_OP_PRIVILEGED, 4988 }, { 4989 .name = "wsr.exccause", 4990 .translate = translate_wsr, 4991 .test_exceptions = test_exceptions_sr, 4992 .par = (const uint32_t[]){ 4993 EXCCAUSE, 4994 XTENSA_OPTION_EXCEPTION, 4995 }, 4996 .op_flags = XTENSA_OP_PRIVILEGED, 4997 }, { 4998 .name = "wsr.excsave1", 4999 .translate = translate_wsr, 5000 .test_exceptions = test_exceptions_sr, 5001 .par = (const uint32_t[]){ 5002 EXCSAVE1, 5003 XTENSA_OPTION_EXCEPTION, 5004 }, 5005 .op_flags = XTENSA_OP_PRIVILEGED, 5006 }, { 5007 .name = "wsr.excsave2", 5008 .translate = translate_wsr, 5009 .test_exceptions = test_exceptions_hpi, 5010 .par = (const uint32_t[]){ 5011 EXCSAVE1 + 1, 5012 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 5013 }, 5014 .op_flags = XTENSA_OP_PRIVILEGED, 5015 }, { 5016 .name = "wsr.excsave3", 5017 .translate = translate_wsr, 5018 .test_exceptions = test_exceptions_hpi, 5019 .par = (const uint32_t[]){ 5020 EXCSAVE1 + 2, 5021 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 5022 }, 5023 .op_flags = XTENSA_OP_PRIVILEGED, 5024 }, { 5025 .name = "wsr.excsave4", 5026 .translate = translate_wsr, 5027 .test_exceptions = test_exceptions_hpi, 5028 .par = (const uint32_t[]){ 5029 EXCSAVE1 + 3, 5030 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 5031 }, 5032 .op_flags = XTENSA_OP_PRIVILEGED, 5033 }, { 5034 .name = "wsr.excsave5", 5035 .translate = translate_wsr, 5036 .test_exceptions = test_exceptions_hpi, 5037 .par = (const uint32_t[]){ 5038 EXCSAVE1 + 4, 5039 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 5040 }, 5041 .op_flags = XTENSA_OP_PRIVILEGED, 5042 }, { 5043 .name = "wsr.excsave6", 5044 .translate = translate_wsr, 5045 .test_exceptions = test_exceptions_hpi, 5046 .par = (const uint32_t[]){ 5047 EXCSAVE1 + 5, 5048 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 5049 }, 5050 .op_flags = XTENSA_OP_PRIVILEGED, 5051 }, { 5052 .name = "wsr.excsave7", 5053 .translate = translate_wsr, 5054 .test_exceptions = test_exceptions_hpi, 5055 .par = (const uint32_t[]){ 5056 EXCSAVE1 + 6, 5057 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 5058 }, 5059 .op_flags = XTENSA_OP_PRIVILEGED, 5060 }, { 5061 .name = "wsr.excvaddr", 5062 .translate = translate_wsr, 5063 .test_exceptions = test_exceptions_sr, 5064 .par = (const uint32_t[]){ 5065 EXCVADDR, 5066 XTENSA_OPTION_EXCEPTION, 5067 }, 5068 .op_flags = XTENSA_OP_PRIVILEGED, 5069 }, { 5070 .name = "wsr.ibreaka0", 5071 .translate = translate_wsr_ibreaka, 5072 .test_exceptions = test_exceptions_ibreak, 5073 .par = (const uint32_t[]){ 5074 IBREAKA, 5075 XTENSA_OPTION_DEBUG, 5076 }, 5077 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_EXIT_TB_0, 5078 }, { 5079 .name = "wsr.ibreaka1", 5080 .translate = translate_wsr_ibreaka, 5081 .test_exceptions = test_exceptions_ibreak, 5082 .par = (const uint32_t[]){ 5083 IBREAKA + 1, 5084 XTENSA_OPTION_DEBUG, 5085 }, 5086 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_EXIT_TB_0, 5087 }, { 5088 .name = "wsr.ibreakenable", 5089 .translate = translate_wsr_ibreakenable, 5090 .test_exceptions = test_exceptions_sr, 5091 .par = (const uint32_t[]){ 5092 IBREAKENABLE, 5093 XTENSA_OPTION_DEBUG, 5094 }, 5095 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_EXIT_TB_0, 5096 }, { 5097 .name = "wsr.icount", 5098 .translate = translate_wsr_icount, 5099 .test_exceptions = test_exceptions_sr, 5100 .par = (const uint32_t[]){ 5101 ICOUNT, 5102 XTENSA_OPTION_DEBUG, 5103 }, 5104 .op_flags = XTENSA_OP_PRIVILEGED, 5105 }, { 5106 .name = "wsr.icountlevel", 5107 .translate = translate_wsr_mask, 5108 .test_exceptions = test_exceptions_sr, 5109 .par = (const uint32_t[]){ 5110 ICOUNTLEVEL, 5111 XTENSA_OPTION_DEBUG, 5112 0xf, 5113 }, 5114 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_EXIT_TB_M1, 5115 }, { 5116 .name = "wsr.intclear", 5117 .translate = translate_wsr_intclear, 5118 .test_exceptions = test_exceptions_sr, 5119 .par = (const uint32_t[]){ 5120 INTCLEAR, 5121 XTENSA_OPTION_INTERRUPT, 5122 }, 5123 .op_flags = 5124 XTENSA_OP_PRIVILEGED | 5125 XTENSA_OP_EXIT_TB_0 | 5126 XTENSA_OP_CHECK_INTERRUPTS, 5127 }, { 5128 .name = "wsr.intenable", 5129 .translate = translate_wsr, 5130 .test_exceptions = test_exceptions_sr, 5131 .par = (const uint32_t[]){ 5132 INTENABLE, 5133 XTENSA_OPTION_INTERRUPT, 5134 }, 5135 .op_flags = 5136 XTENSA_OP_PRIVILEGED | 5137 XTENSA_OP_EXIT_TB_0 | 5138 XTENSA_OP_CHECK_INTERRUPTS, 5139 }, { 5140 .name = "wsr.interrupt", 5141 .translate = translate_wsr, 5142 .test_exceptions = test_exceptions_sr, 5143 .par = (const uint32_t[]){ 5144 INTSET, 5145 XTENSA_OPTION_INTERRUPT, 5146 }, 5147 .op_flags = 5148 XTENSA_OP_PRIVILEGED | 5149 XTENSA_OP_EXIT_TB_0 | 5150 XTENSA_OP_CHECK_INTERRUPTS, 5151 }, { 5152 .name = "wsr.intset", 5153 .translate = translate_wsr_intset, 5154 .test_exceptions = test_exceptions_sr, 5155 .par = (const uint32_t[]){ 5156 INTSET, 5157 XTENSA_OPTION_INTERRUPT, 5158 }, 5159 .op_flags = 5160 XTENSA_OP_PRIVILEGED | 5161 XTENSA_OP_EXIT_TB_0 | 5162 XTENSA_OP_CHECK_INTERRUPTS, 5163 }, { 5164 .name = "wsr.itlbcfg", 5165 .translate = translate_wsr_mask, 5166 .test_exceptions = test_exceptions_sr, 5167 .par = (const uint32_t[]){ 5168 ITLBCFG, 5169 XTENSA_OPTION_MMU, 5170 0x01130000, 5171 }, 5172 .op_flags = XTENSA_OP_PRIVILEGED, 5173 }, { 5174 .name = "wsr.lbeg", 5175 .translate = translate_wsr, 5176 .test_exceptions = test_exceptions_sr, 5177 .par = (const uint32_t[]){ 5178 LBEG, 5179 XTENSA_OPTION_LOOP, 5180 }, 5181 .op_flags = XTENSA_OP_EXIT_TB_M1, 5182 }, { 5183 .name = "wsr.lcount", 5184 .translate = translate_wsr, 5185 .test_exceptions = test_exceptions_sr, 5186 .par = (const uint32_t[]){ 5187 LCOUNT, 5188 XTENSA_OPTION_LOOP, 5189 }, 5190 }, { 5191 .name = "wsr.lend", 5192 .translate = translate_wsr, 5193 .test_exceptions = test_exceptions_sr, 5194 .par = (const uint32_t[]){ 5195 LEND, 5196 XTENSA_OPTION_LOOP, 5197 }, 5198 .op_flags = XTENSA_OP_EXIT_TB_M1, 5199 }, { 5200 .name = "wsr.litbase", 5201 .translate = translate_wsr_mask, 5202 .test_exceptions = test_exceptions_sr, 5203 .par = (const uint32_t[]){ 5204 LITBASE, 5205 XTENSA_OPTION_EXTENDED_L32R, 5206 0xfffff001, 5207 }, 5208 .op_flags = XTENSA_OP_EXIT_TB_M1, 5209 }, { 5210 .name = "wsr.m0", 5211 .translate = translate_wsr, 5212 .test_exceptions = test_exceptions_sr, 5213 .par = (const uint32_t[]){ 5214 MR, 5215 XTENSA_OPTION_MAC16, 5216 }, 5217 }, { 5218 .name = "wsr.m1", 5219 .translate = translate_wsr, 5220 .test_exceptions = test_exceptions_sr, 5221 .par = (const uint32_t[]){ 5222 MR + 1, 5223 XTENSA_OPTION_MAC16, 5224 }, 5225 }, { 5226 .name = "wsr.m2", 5227 .translate = translate_wsr, 5228 .test_exceptions = test_exceptions_sr, 5229 .par = (const uint32_t[]){ 5230 MR + 2, 5231 XTENSA_OPTION_MAC16, 5232 }, 5233 }, { 5234 .name = "wsr.m3", 5235 .translate = translate_wsr, 5236 .test_exceptions = test_exceptions_sr, 5237 .par = (const uint32_t[]){ 5238 MR + 3, 5239 XTENSA_OPTION_MAC16, 5240 }, 5241 }, { 5242 .name = "wsr.memctl", 5243 .translate = translate_wsr_memctl, 5244 .par = (const uint32_t[]){MEMCTL}, 5245 .op_flags = XTENSA_OP_PRIVILEGED, 5246 }, { 5247 .name = "wsr.mecr", 5248 .translate = translate_wsr, 5249 .test_exceptions = test_exceptions_sr, 5250 .par = (const uint32_t[]){ 5251 MECR, 5252 XTENSA_OPTION_MEMORY_ECC_PARITY, 5253 }, 5254 .op_flags = XTENSA_OP_PRIVILEGED, 5255 }, { 5256 .name = "wsr.mepc", 5257 .translate = translate_wsr, 5258 .test_exceptions = test_exceptions_sr, 5259 .par = (const uint32_t[]){ 5260 MEPC, 5261 XTENSA_OPTION_MEMORY_ECC_PARITY, 5262 }, 5263 .op_flags = XTENSA_OP_PRIVILEGED, 5264 }, { 5265 .name = "wsr.meps", 5266 .translate = translate_wsr, 5267 .test_exceptions = test_exceptions_sr, 5268 .par = (const uint32_t[]){ 5269 MEPS, 5270 XTENSA_OPTION_MEMORY_ECC_PARITY, 5271 }, 5272 .op_flags = XTENSA_OP_PRIVILEGED, 5273 }, { 5274 .name = "wsr.mesave", 5275 .translate = translate_wsr, 5276 .test_exceptions = test_exceptions_sr, 5277 .par = (const uint32_t[]){ 5278 MESAVE, 5279 XTENSA_OPTION_MEMORY_ECC_PARITY, 5280 }, 5281 .op_flags = XTENSA_OP_PRIVILEGED, 5282 }, { 5283 .name = "wsr.mesr", 5284 .translate = translate_wsr, 5285 .test_exceptions = test_exceptions_sr, 5286 .par = (const uint32_t[]){ 5287 MESR, 5288 XTENSA_OPTION_MEMORY_ECC_PARITY, 5289 }, 5290 .op_flags = XTENSA_OP_PRIVILEGED, 5291 }, { 5292 .name = "wsr.mevaddr", 5293 .translate = translate_wsr, 5294 .test_exceptions = test_exceptions_sr, 5295 .par = (const uint32_t[]){ 5296 MESR, 5297 XTENSA_OPTION_MEMORY_ECC_PARITY, 5298 }, 5299 .op_flags = XTENSA_OP_PRIVILEGED, 5300 }, { 5301 .name = "wsr.misc0", 5302 .translate = translate_wsr, 5303 .test_exceptions = test_exceptions_sr, 5304 .par = (const uint32_t[]){ 5305 MISC, 5306 XTENSA_OPTION_MISC_SR, 5307 }, 5308 .op_flags = XTENSA_OP_PRIVILEGED, 5309 }, { 5310 .name = "wsr.misc1", 5311 .translate = translate_wsr, 5312 .test_exceptions = test_exceptions_sr, 5313 .par = (const uint32_t[]){ 5314 MISC + 1, 5315 XTENSA_OPTION_MISC_SR, 5316 }, 5317 .op_flags = XTENSA_OP_PRIVILEGED, 5318 }, { 5319 .name = "wsr.misc2", 5320 .translate = translate_wsr, 5321 .test_exceptions = test_exceptions_sr, 5322 .par = (const uint32_t[]){ 5323 MISC + 2, 5324 XTENSA_OPTION_MISC_SR, 5325 }, 5326 .op_flags = XTENSA_OP_PRIVILEGED, 5327 }, { 5328 .name = "wsr.misc3", 5329 .translate = translate_wsr, 5330 .test_exceptions = test_exceptions_sr, 5331 .par = (const uint32_t[]){ 5332 MISC + 3, 5333 XTENSA_OPTION_MISC_SR, 5334 }, 5335 .op_flags = XTENSA_OP_PRIVILEGED, 5336 }, { 5337 .name = "wsr.mmid", 5338 .translate = translate_wsr, 5339 .test_exceptions = test_exceptions_sr, 5340 .par = (const uint32_t[]){ 5341 MMID, 5342 XTENSA_OPTION_TRACE_PORT, 5343 }, 5344 .op_flags = XTENSA_OP_PRIVILEGED, 5345 }, { 5346 .name = "wsr.mpuenb", 5347 .translate = translate_wsr_mpuenb, 5348 .test_exceptions = test_exceptions_sr, 5349 .par = (const uint32_t[]){ 5350 MPUENB, 5351 XTENSA_OPTION_MPU, 5352 }, 5353 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_EXIT_TB_M1, 5354 }, { 5355 .name = "wsr.prefctl", 5356 .translate = translate_wsr, 5357 .par = (const uint32_t[]){PREFCTL}, 5358 }, { 5359 .name = "wsr.prid", 5360 .op_flags = XTENSA_OP_ILL, 5361 }, { 5362 .name = "wsr.ps", 5363 .translate = translate_wsr_ps, 5364 .test_exceptions = test_exceptions_sr, 5365 .par = (const uint32_t[]){ 5366 PS, 5367 XTENSA_OPTION_EXCEPTION, 5368 }, 5369 .op_flags = 5370 XTENSA_OP_PRIVILEGED | 5371 XTENSA_OP_EXIT_TB_M1 | 5372 XTENSA_OP_CHECK_INTERRUPTS, 5373 }, { 5374 .name = "wsr.ptevaddr", 5375 .translate = translate_wsr_mask, 5376 .test_exceptions = test_exceptions_sr, 5377 .par = (const uint32_t[]){ 5378 PTEVADDR, 5379 XTENSA_OPTION_MMU, 5380 0xffc00000, 5381 }, 5382 .op_flags = XTENSA_OP_PRIVILEGED, 5383 }, { 5384 .name = "wsr.rasid", 5385 .translate = translate_wsr_rasid, 5386 .test_exceptions = test_exceptions_sr, 5387 .par = (const uint32_t[]){ 5388 RASID, 5389 XTENSA_OPTION_MMU, 5390 }, 5391 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_EXIT_TB_M1, 5392 }, { 5393 .name = "wsr.sar", 5394 .translate = translate_wsr_sar, 5395 .par = (const uint32_t[]){SAR}, 5396 }, { 5397 .name = "wsr.scompare1", 5398 .translate = translate_wsr, 5399 .test_exceptions = test_exceptions_sr, 5400 .par = (const uint32_t[]){ 5401 SCOMPARE1, 5402 XTENSA_OPTION_CONDITIONAL_STORE, 5403 }, 5404 }, { 5405 .name = "wsr.vecbase", 5406 .translate = translate_wsr, 5407 .test_exceptions = test_exceptions_sr, 5408 .par = (const uint32_t[]){ 5409 VECBASE, 5410 XTENSA_OPTION_RELOCATABLE_VECTOR, 5411 }, 5412 .op_flags = XTENSA_OP_PRIVILEGED, 5413 }, { 5414 .name = "wsr.windowbase", 5415 .translate = translate_wsr_windowbase, 5416 .test_exceptions = test_exceptions_sr, 5417 .par = (const uint32_t[]){ 5418 WINDOW_BASE, 5419 XTENSA_OPTION_WINDOWED_REGISTER, 5420 }, 5421 .op_flags = XTENSA_OP_PRIVILEGED | 5422 XTENSA_OP_EXIT_TB_M1 | 5423 XTENSA_OP_SYNC_REGISTER_WINDOW, 5424 }, { 5425 .name = "wsr.windowstart", 5426 .translate = translate_wsr_windowstart, 5427 .test_exceptions = test_exceptions_sr, 5428 .par = (const uint32_t[]){ 5429 WINDOW_START, 5430 XTENSA_OPTION_WINDOWED_REGISTER, 5431 }, 5432 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_EXIT_TB_M1, 5433 }, { 5434 .name = "wur.expstate", 5435 .translate = translate_wur, 5436 .par = (const uint32_t[]){EXPSTATE}, 5437 }, { 5438 .name = "wur.threadptr", 5439 .translate = translate_wur, 5440 .par = (const uint32_t[]){THREADPTR}, 5441 }, { 5442 .name = "xor", 5443 .translate = translate_xor, 5444 }, { 5445 .name = "xorb", 5446 .translate = translate_boolean, 5447 .par = (const uint32_t[]){BOOLEAN_XOR}, 5448 }, { 5449 .name = "xsr.176", 5450 .op_flags = XTENSA_OP_ILL, 5451 }, { 5452 .name = "xsr.208", 5453 .op_flags = XTENSA_OP_ILL, 5454 }, { 5455 .name = "xsr.acchi", 5456 .translate = translate_xsr_acchi, 5457 .test_exceptions = test_exceptions_sr, 5458 .par = (const uint32_t[]){ 5459 ACCHI, 5460 XTENSA_OPTION_MAC16, 5461 }, 5462 }, { 5463 .name = "xsr.acclo", 5464 .translate = translate_xsr, 5465 .test_exceptions = test_exceptions_sr, 5466 .par = (const uint32_t[]){ 5467 ACCLO, 5468 XTENSA_OPTION_MAC16, 5469 }, 5470 }, { 5471 .name = "xsr.atomctl", 5472 .translate = translate_xsr_mask, 5473 .test_exceptions = test_exceptions_sr, 5474 .par = (const uint32_t[]){ 5475 ATOMCTL, 5476 XTENSA_OPTION_ATOMCTL, 5477 0x3f, 5478 }, 5479 .op_flags = XTENSA_OP_PRIVILEGED, 5480 }, { 5481 .name = "xsr.br", 5482 .translate = translate_xsr_mask, 5483 .test_exceptions = test_exceptions_sr, 5484 .par = (const uint32_t[]){ 5485 BR, 5486 XTENSA_OPTION_BOOLEAN, 5487 0xffff, 5488 }, 5489 }, { 5490 .name = "xsr.cacheadrdis", 5491 .translate = translate_xsr_mask, 5492 .test_exceptions = test_exceptions_sr, 5493 .par = (const uint32_t[]){ 5494 CACHEADRDIS, 5495 XTENSA_OPTION_MPU, 5496 0xff, 5497 }, 5498 .op_flags = XTENSA_OP_PRIVILEGED, 5499 }, { 5500 .name = "xsr.cacheattr", 5501 .translate = translate_xsr, 5502 .test_exceptions = test_exceptions_sr, 5503 .par = (const uint32_t[]){ 5504 CACHEATTR, 5505 XTENSA_OPTION_CACHEATTR, 5506 }, 5507 .op_flags = XTENSA_OP_PRIVILEGED, 5508 }, { 5509 .name = "xsr.ccompare0", 5510 .translate = translate_xsr_ccompare, 5511 .test_exceptions = test_exceptions_ccompare, 5512 .par = (const uint32_t[]){ 5513 CCOMPARE, 5514 XTENSA_OPTION_TIMER_INTERRUPT, 5515 }, 5516 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_EXIT_TB_0, 5517 }, { 5518 .name = "xsr.ccompare1", 5519 .translate = translate_xsr_ccompare, 5520 .test_exceptions = test_exceptions_ccompare, 5521 .par = (const uint32_t[]){ 5522 CCOMPARE + 1, 5523 XTENSA_OPTION_TIMER_INTERRUPT, 5524 }, 5525 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_EXIT_TB_0, 5526 }, { 5527 .name = "xsr.ccompare2", 5528 .translate = translate_xsr_ccompare, 5529 .test_exceptions = test_exceptions_ccompare, 5530 .par = (const uint32_t[]){ 5531 CCOMPARE + 2, 5532 XTENSA_OPTION_TIMER_INTERRUPT, 5533 }, 5534 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_EXIT_TB_0, 5535 }, { 5536 .name = "xsr.ccount", 5537 .translate = translate_xsr_ccount, 5538 .test_exceptions = test_exceptions_sr, 5539 .par = (const uint32_t[]){ 5540 CCOUNT, 5541 XTENSA_OPTION_TIMER_INTERRUPT, 5542 }, 5543 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_EXIT_TB_0, 5544 }, { 5545 .name = "xsr.configid0", 5546 .op_flags = XTENSA_OP_ILL, 5547 }, { 5548 .name = "xsr.configid1", 5549 .op_flags = XTENSA_OP_ILL, 5550 }, { 5551 .name = "xsr.cpenable", 5552 .translate = translate_xsr_mask, 5553 .test_exceptions = test_exceptions_sr, 5554 .par = (const uint32_t[]){ 5555 CPENABLE, 5556 XTENSA_OPTION_COPROCESSOR, 5557 0xff, 5558 }, 5559 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_EXIT_TB_M1, 5560 }, { 5561 .name = "xsr.dbreaka0", 5562 .translate = translate_xsr_dbreaka, 5563 .test_exceptions = test_exceptions_dbreak, 5564 .par = (const uint32_t[]){ 5565 DBREAKA, 5566 XTENSA_OPTION_DEBUG, 5567 }, 5568 .op_flags = XTENSA_OP_PRIVILEGED, 5569 }, { 5570 .name = "xsr.dbreaka1", 5571 .translate = translate_xsr_dbreaka, 5572 .test_exceptions = test_exceptions_dbreak, 5573 .par = (const uint32_t[]){ 5574 DBREAKA + 1, 5575 XTENSA_OPTION_DEBUG, 5576 }, 5577 .op_flags = XTENSA_OP_PRIVILEGED, 5578 }, { 5579 .name = "xsr.dbreakc0", 5580 .translate = translate_xsr_dbreakc, 5581 .test_exceptions = test_exceptions_dbreak, 5582 .par = (const uint32_t[]){ 5583 DBREAKC, 5584 XTENSA_OPTION_DEBUG, 5585 }, 5586 .op_flags = XTENSA_OP_PRIVILEGED, 5587 }, { 5588 .name = "xsr.dbreakc1", 5589 .translate = translate_xsr_dbreakc, 5590 .test_exceptions = test_exceptions_dbreak, 5591 .par = (const uint32_t[]){ 5592 DBREAKC + 1, 5593 XTENSA_OPTION_DEBUG, 5594 }, 5595 .op_flags = XTENSA_OP_PRIVILEGED, 5596 }, { 5597 .name = "xsr.ddr", 5598 .translate = translate_xsr, 5599 .test_exceptions = test_exceptions_sr, 5600 .par = (const uint32_t[]){ 5601 DDR, 5602 XTENSA_OPTION_DEBUG, 5603 }, 5604 .op_flags = XTENSA_OP_PRIVILEGED, 5605 }, { 5606 .name = "xsr.debugcause", 5607 .op_flags = XTENSA_OP_ILL, 5608 }, { 5609 .name = "xsr.depc", 5610 .translate = translate_xsr, 5611 .test_exceptions = test_exceptions_sr, 5612 .par = (const uint32_t[]){ 5613 DEPC, 5614 XTENSA_OPTION_EXCEPTION, 5615 }, 5616 .op_flags = XTENSA_OP_PRIVILEGED, 5617 }, { 5618 .name = "xsr.dtlbcfg", 5619 .translate = translate_xsr_mask, 5620 .test_exceptions = test_exceptions_sr, 5621 .par = (const uint32_t[]){ 5622 DTLBCFG, 5623 XTENSA_OPTION_MMU, 5624 0x01130000, 5625 }, 5626 .op_flags = XTENSA_OP_PRIVILEGED, 5627 }, { 5628 .name = "xsr.epc1", 5629 .translate = translate_xsr, 5630 .test_exceptions = test_exceptions_sr, 5631 .par = (const uint32_t[]){ 5632 EPC1, 5633 XTENSA_OPTION_EXCEPTION, 5634 }, 5635 .op_flags = XTENSA_OP_PRIVILEGED, 5636 }, { 5637 .name = "xsr.epc2", 5638 .translate = translate_xsr, 5639 .test_exceptions = test_exceptions_hpi, 5640 .par = (const uint32_t[]){ 5641 EPC1 + 1, 5642 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 5643 }, 5644 .op_flags = XTENSA_OP_PRIVILEGED, 5645 }, { 5646 .name = "xsr.epc3", 5647 .translate = translate_xsr, 5648 .test_exceptions = test_exceptions_hpi, 5649 .par = (const uint32_t[]){ 5650 EPC1 + 2, 5651 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 5652 }, 5653 .op_flags = XTENSA_OP_PRIVILEGED, 5654 }, { 5655 .name = "xsr.epc4", 5656 .translate = translate_xsr, 5657 .test_exceptions = test_exceptions_hpi, 5658 .par = (const uint32_t[]){ 5659 EPC1 + 3, 5660 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 5661 }, 5662 .op_flags = XTENSA_OP_PRIVILEGED, 5663 }, { 5664 .name = "xsr.epc5", 5665 .translate = translate_xsr, 5666 .test_exceptions = test_exceptions_hpi, 5667 .par = (const uint32_t[]){ 5668 EPC1 + 4, 5669 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 5670 }, 5671 .op_flags = XTENSA_OP_PRIVILEGED, 5672 }, { 5673 .name = "xsr.epc6", 5674 .translate = translate_xsr, 5675 .test_exceptions = test_exceptions_hpi, 5676 .par = (const uint32_t[]){ 5677 EPC1 + 5, 5678 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 5679 }, 5680 .op_flags = XTENSA_OP_PRIVILEGED, 5681 }, { 5682 .name = "xsr.epc7", 5683 .translate = translate_xsr, 5684 .test_exceptions = test_exceptions_hpi, 5685 .par = (const uint32_t[]){ 5686 EPC1 + 6, 5687 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 5688 }, 5689 .op_flags = XTENSA_OP_PRIVILEGED, 5690 }, { 5691 .name = "xsr.eps2", 5692 .translate = translate_xsr, 5693 .test_exceptions = test_exceptions_hpi, 5694 .par = (const uint32_t[]){ 5695 EPS2, 5696 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 5697 }, 5698 .op_flags = XTENSA_OP_PRIVILEGED, 5699 }, { 5700 .name = "xsr.eps3", 5701 .translate = translate_xsr, 5702 .test_exceptions = test_exceptions_hpi, 5703 .par = (const uint32_t[]){ 5704 EPS2 + 1, 5705 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 5706 }, 5707 .op_flags = XTENSA_OP_PRIVILEGED, 5708 }, { 5709 .name = "xsr.eps4", 5710 .translate = translate_xsr, 5711 .test_exceptions = test_exceptions_hpi, 5712 .par = (const uint32_t[]){ 5713 EPS2 + 2, 5714 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 5715 }, 5716 .op_flags = XTENSA_OP_PRIVILEGED, 5717 }, { 5718 .name = "xsr.eps5", 5719 .translate = translate_xsr, 5720 .test_exceptions = test_exceptions_hpi, 5721 .par = (const uint32_t[]){ 5722 EPS2 + 3, 5723 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 5724 }, 5725 .op_flags = XTENSA_OP_PRIVILEGED, 5726 }, { 5727 .name = "xsr.eps6", 5728 .translate = translate_xsr, 5729 .test_exceptions = test_exceptions_hpi, 5730 .par = (const uint32_t[]){ 5731 EPS2 + 4, 5732 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 5733 }, 5734 .op_flags = XTENSA_OP_PRIVILEGED, 5735 }, { 5736 .name = "xsr.eps7", 5737 .translate = translate_xsr, 5738 .test_exceptions = test_exceptions_hpi, 5739 .par = (const uint32_t[]){ 5740 EPS2 + 5, 5741 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 5742 }, 5743 .op_flags = XTENSA_OP_PRIVILEGED, 5744 }, { 5745 .name = "xsr.eraccess", 5746 .translate = translate_xsr_mask, 5747 .par = (const uint32_t[]){ 5748 ERACCESS, 5749 0, 5750 0xffff, 5751 }, 5752 .op_flags = XTENSA_OP_PRIVILEGED, 5753 }, { 5754 .name = "xsr.exccause", 5755 .translate = translate_xsr, 5756 .test_exceptions = test_exceptions_sr, 5757 .par = (const uint32_t[]){ 5758 EXCCAUSE, 5759 XTENSA_OPTION_EXCEPTION, 5760 }, 5761 .op_flags = XTENSA_OP_PRIVILEGED, 5762 }, { 5763 .name = "xsr.excsave1", 5764 .translate = translate_xsr, 5765 .test_exceptions = test_exceptions_sr, 5766 .par = (const uint32_t[]){ 5767 EXCSAVE1, 5768 XTENSA_OPTION_EXCEPTION, 5769 }, 5770 .op_flags = XTENSA_OP_PRIVILEGED, 5771 }, { 5772 .name = "xsr.excsave2", 5773 .translate = translate_xsr, 5774 .test_exceptions = test_exceptions_hpi, 5775 .par = (const uint32_t[]){ 5776 EXCSAVE1 + 1, 5777 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 5778 }, 5779 .op_flags = XTENSA_OP_PRIVILEGED, 5780 }, { 5781 .name = "xsr.excsave3", 5782 .translate = translate_xsr, 5783 .test_exceptions = test_exceptions_hpi, 5784 .par = (const uint32_t[]){ 5785 EXCSAVE1 + 2, 5786 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 5787 }, 5788 .op_flags = XTENSA_OP_PRIVILEGED, 5789 }, { 5790 .name = "xsr.excsave4", 5791 .translate = translate_xsr, 5792 .test_exceptions = test_exceptions_hpi, 5793 .par = (const uint32_t[]){ 5794 EXCSAVE1 + 3, 5795 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 5796 }, 5797 .op_flags = XTENSA_OP_PRIVILEGED, 5798 }, { 5799 .name = "xsr.excsave5", 5800 .translate = translate_xsr, 5801 .test_exceptions = test_exceptions_hpi, 5802 .par = (const uint32_t[]){ 5803 EXCSAVE1 + 4, 5804 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 5805 }, 5806 .op_flags = XTENSA_OP_PRIVILEGED, 5807 }, { 5808 .name = "xsr.excsave6", 5809 .translate = translate_xsr, 5810 .test_exceptions = test_exceptions_hpi, 5811 .par = (const uint32_t[]){ 5812 EXCSAVE1 + 5, 5813 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 5814 }, 5815 .op_flags = XTENSA_OP_PRIVILEGED, 5816 }, { 5817 .name = "xsr.excsave7", 5818 .translate = translate_xsr, 5819 .test_exceptions = test_exceptions_hpi, 5820 .par = (const uint32_t[]){ 5821 EXCSAVE1 + 6, 5822 XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, 5823 }, 5824 .op_flags = XTENSA_OP_PRIVILEGED, 5825 }, { 5826 .name = "xsr.excvaddr", 5827 .translate = translate_xsr, 5828 .test_exceptions = test_exceptions_sr, 5829 .par = (const uint32_t[]){ 5830 EXCVADDR, 5831 XTENSA_OPTION_EXCEPTION, 5832 }, 5833 .op_flags = XTENSA_OP_PRIVILEGED, 5834 }, { 5835 .name = "xsr.ibreaka0", 5836 .translate = translate_xsr_ibreaka, 5837 .test_exceptions = test_exceptions_ibreak, 5838 .par = (const uint32_t[]){ 5839 IBREAKA, 5840 XTENSA_OPTION_DEBUG, 5841 }, 5842 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_EXIT_TB_0, 5843 }, { 5844 .name = "xsr.ibreaka1", 5845 .translate = translate_xsr_ibreaka, 5846 .test_exceptions = test_exceptions_ibreak, 5847 .par = (const uint32_t[]){ 5848 IBREAKA + 1, 5849 XTENSA_OPTION_DEBUG, 5850 }, 5851 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_EXIT_TB_0, 5852 }, { 5853 .name = "xsr.ibreakenable", 5854 .translate = translate_xsr_ibreakenable, 5855 .test_exceptions = test_exceptions_sr, 5856 .par = (const uint32_t[]){ 5857 IBREAKENABLE, 5858 XTENSA_OPTION_DEBUG, 5859 }, 5860 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_EXIT_TB_0, 5861 }, { 5862 .name = "xsr.icount", 5863 .translate = translate_xsr_icount, 5864 .test_exceptions = test_exceptions_sr, 5865 .par = (const uint32_t[]){ 5866 ICOUNT, 5867 XTENSA_OPTION_DEBUG, 5868 }, 5869 .op_flags = XTENSA_OP_PRIVILEGED, 5870 }, { 5871 .name = "xsr.icountlevel", 5872 .translate = translate_xsr_mask, 5873 .test_exceptions = test_exceptions_sr, 5874 .par = (const uint32_t[]){ 5875 ICOUNTLEVEL, 5876 XTENSA_OPTION_DEBUG, 5877 0xf, 5878 }, 5879 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_EXIT_TB_M1, 5880 }, { 5881 .name = "xsr.intclear", 5882 .op_flags = XTENSA_OP_ILL, 5883 }, { 5884 .name = "xsr.intenable", 5885 .translate = translate_xsr, 5886 .test_exceptions = test_exceptions_sr, 5887 .par = (const uint32_t[]){ 5888 INTENABLE, 5889 XTENSA_OPTION_INTERRUPT, 5890 }, 5891 .op_flags = 5892 XTENSA_OP_PRIVILEGED | 5893 XTENSA_OP_EXIT_TB_0 | 5894 XTENSA_OP_CHECK_INTERRUPTS, 5895 }, { 5896 .name = "xsr.interrupt", 5897 .op_flags = XTENSA_OP_ILL, 5898 }, { 5899 .name = "xsr.intset", 5900 .op_flags = XTENSA_OP_ILL, 5901 }, { 5902 .name = "xsr.itlbcfg", 5903 .translate = translate_xsr_mask, 5904 .test_exceptions = test_exceptions_sr, 5905 .par = (const uint32_t[]){ 5906 ITLBCFG, 5907 XTENSA_OPTION_MMU, 5908 0x01130000, 5909 }, 5910 .op_flags = XTENSA_OP_PRIVILEGED, 5911 }, { 5912 .name = "xsr.lbeg", 5913 .translate = translate_xsr, 5914 .test_exceptions = test_exceptions_sr, 5915 .par = (const uint32_t[]){ 5916 LBEG, 5917 XTENSA_OPTION_LOOP, 5918 }, 5919 .op_flags = XTENSA_OP_EXIT_TB_M1, 5920 }, { 5921 .name = "xsr.lcount", 5922 .translate = translate_xsr, 5923 .test_exceptions = test_exceptions_sr, 5924 .par = (const uint32_t[]){ 5925 LCOUNT, 5926 XTENSA_OPTION_LOOP, 5927 }, 5928 }, { 5929 .name = "xsr.lend", 5930 .translate = translate_xsr, 5931 .test_exceptions = test_exceptions_sr, 5932 .par = (const uint32_t[]){ 5933 LEND, 5934 XTENSA_OPTION_LOOP, 5935 }, 5936 .op_flags = XTENSA_OP_EXIT_TB_M1, 5937 }, { 5938 .name = "xsr.litbase", 5939 .translate = translate_xsr_mask, 5940 .test_exceptions = test_exceptions_sr, 5941 .par = (const uint32_t[]){ 5942 LITBASE, 5943 XTENSA_OPTION_EXTENDED_L32R, 5944 0xfffff001, 5945 }, 5946 .op_flags = XTENSA_OP_EXIT_TB_M1, 5947 }, { 5948 .name = "xsr.m0", 5949 .translate = translate_xsr, 5950 .test_exceptions = test_exceptions_sr, 5951 .par = (const uint32_t[]){ 5952 MR, 5953 XTENSA_OPTION_MAC16, 5954 }, 5955 }, { 5956 .name = "xsr.m1", 5957 .translate = translate_xsr, 5958 .test_exceptions = test_exceptions_sr, 5959 .par = (const uint32_t[]){ 5960 MR + 1, 5961 XTENSA_OPTION_MAC16, 5962 }, 5963 }, { 5964 .name = "xsr.m2", 5965 .translate = translate_xsr, 5966 .test_exceptions = test_exceptions_sr, 5967 .par = (const uint32_t[]){ 5968 MR + 2, 5969 XTENSA_OPTION_MAC16, 5970 }, 5971 }, { 5972 .name = "xsr.m3", 5973 .translate = translate_xsr, 5974 .test_exceptions = test_exceptions_sr, 5975 .par = (const uint32_t[]){ 5976 MR + 3, 5977 XTENSA_OPTION_MAC16, 5978 }, 5979 }, { 5980 .name = "xsr.memctl", 5981 .translate = translate_xsr_memctl, 5982 .par = (const uint32_t[]){MEMCTL}, 5983 .op_flags = XTENSA_OP_PRIVILEGED, 5984 }, { 5985 .name = "xsr.mecr", 5986 .translate = translate_xsr, 5987 .test_exceptions = test_exceptions_sr, 5988 .par = (const uint32_t[]){ 5989 MECR, 5990 XTENSA_OPTION_MEMORY_ECC_PARITY, 5991 }, 5992 .op_flags = XTENSA_OP_PRIVILEGED, 5993 }, { 5994 .name = "xsr.mepc", 5995 .translate = translate_xsr, 5996 .test_exceptions = test_exceptions_sr, 5997 .par = (const uint32_t[]){ 5998 MEPC, 5999 XTENSA_OPTION_MEMORY_ECC_PARITY, 6000 }, 6001 .op_flags = XTENSA_OP_PRIVILEGED, 6002 }, { 6003 .name = "xsr.meps", 6004 .translate = translate_xsr, 6005 .test_exceptions = test_exceptions_sr, 6006 .par = (const uint32_t[]){ 6007 MEPS, 6008 XTENSA_OPTION_MEMORY_ECC_PARITY, 6009 }, 6010 .op_flags = XTENSA_OP_PRIVILEGED, 6011 }, { 6012 .name = "xsr.mesave", 6013 .translate = translate_xsr, 6014 .test_exceptions = test_exceptions_sr, 6015 .par = (const uint32_t[]){ 6016 MESAVE, 6017 XTENSA_OPTION_MEMORY_ECC_PARITY, 6018 }, 6019 .op_flags = XTENSA_OP_PRIVILEGED, 6020 }, { 6021 .name = "xsr.mesr", 6022 .translate = translate_xsr, 6023 .test_exceptions = test_exceptions_sr, 6024 .par = (const uint32_t[]){ 6025 MESR, 6026 XTENSA_OPTION_MEMORY_ECC_PARITY, 6027 }, 6028 .op_flags = XTENSA_OP_PRIVILEGED, 6029 }, { 6030 .name = "xsr.mevaddr", 6031 .translate = translate_xsr, 6032 .test_exceptions = test_exceptions_sr, 6033 .par = (const uint32_t[]){ 6034 MESR, 6035 XTENSA_OPTION_MEMORY_ECC_PARITY, 6036 }, 6037 .op_flags = XTENSA_OP_PRIVILEGED, 6038 }, { 6039 .name = "xsr.misc0", 6040 .translate = translate_xsr, 6041 .test_exceptions = test_exceptions_sr, 6042 .par = (const uint32_t[]){ 6043 MISC, 6044 XTENSA_OPTION_MISC_SR, 6045 }, 6046 .op_flags = XTENSA_OP_PRIVILEGED, 6047 }, { 6048 .name = "xsr.misc1", 6049 .translate = translate_xsr, 6050 .test_exceptions = test_exceptions_sr, 6051 .par = (const uint32_t[]){ 6052 MISC + 1, 6053 XTENSA_OPTION_MISC_SR, 6054 }, 6055 .op_flags = XTENSA_OP_PRIVILEGED, 6056 }, { 6057 .name = "xsr.misc2", 6058 .translate = translate_xsr, 6059 .test_exceptions = test_exceptions_sr, 6060 .par = (const uint32_t[]){ 6061 MISC + 2, 6062 XTENSA_OPTION_MISC_SR, 6063 }, 6064 .op_flags = XTENSA_OP_PRIVILEGED, 6065 }, { 6066 .name = "xsr.misc3", 6067 .translate = translate_xsr, 6068 .test_exceptions = test_exceptions_sr, 6069 .par = (const uint32_t[]){ 6070 MISC + 3, 6071 XTENSA_OPTION_MISC_SR, 6072 }, 6073 .op_flags = XTENSA_OP_PRIVILEGED, 6074 }, { 6075 .name = "xsr.mpuenb", 6076 .translate = translate_xsr_mpuenb, 6077 .test_exceptions = test_exceptions_sr, 6078 .par = (const uint32_t[]){ 6079 MPUENB, 6080 XTENSA_OPTION_MPU, 6081 }, 6082 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_EXIT_TB_M1, 6083 }, { 6084 .name = "xsr.prefctl", 6085 .translate = translate_xsr, 6086 .par = (const uint32_t[]){PREFCTL}, 6087 }, { 6088 .name = "xsr.prid", 6089 .op_flags = XTENSA_OP_ILL, 6090 }, { 6091 .name = "xsr.ps", 6092 .translate = translate_xsr_ps, 6093 .test_exceptions = test_exceptions_sr, 6094 .par = (const uint32_t[]){ 6095 PS, 6096 XTENSA_OPTION_EXCEPTION, 6097 }, 6098 .op_flags = 6099 XTENSA_OP_PRIVILEGED | 6100 XTENSA_OP_EXIT_TB_M1 | 6101 XTENSA_OP_CHECK_INTERRUPTS, 6102 }, { 6103 .name = "xsr.ptevaddr", 6104 .translate = translate_xsr_mask, 6105 .test_exceptions = test_exceptions_sr, 6106 .par = (const uint32_t[]){ 6107 PTEVADDR, 6108 XTENSA_OPTION_MMU, 6109 0xffc00000, 6110 }, 6111 .op_flags = XTENSA_OP_PRIVILEGED, 6112 }, { 6113 .name = "xsr.rasid", 6114 .translate = translate_xsr_rasid, 6115 .test_exceptions = test_exceptions_sr, 6116 .par = (const uint32_t[]){ 6117 RASID, 6118 XTENSA_OPTION_MMU, 6119 }, 6120 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_EXIT_TB_M1, 6121 }, { 6122 .name = "xsr.sar", 6123 .translate = translate_xsr_sar, 6124 .par = (const uint32_t[]){SAR}, 6125 }, { 6126 .name = "xsr.scompare1", 6127 .translate = translate_xsr, 6128 .test_exceptions = test_exceptions_sr, 6129 .par = (const uint32_t[]){ 6130 SCOMPARE1, 6131 XTENSA_OPTION_CONDITIONAL_STORE, 6132 }, 6133 }, { 6134 .name = "xsr.vecbase", 6135 .translate = translate_xsr, 6136 .test_exceptions = test_exceptions_sr, 6137 .par = (const uint32_t[]){ 6138 VECBASE, 6139 XTENSA_OPTION_RELOCATABLE_VECTOR, 6140 }, 6141 .op_flags = XTENSA_OP_PRIVILEGED, 6142 }, { 6143 .name = "xsr.windowbase", 6144 .translate = translate_xsr_windowbase, 6145 .test_exceptions = test_exceptions_sr, 6146 .par = (const uint32_t[]){ 6147 WINDOW_BASE, 6148 XTENSA_OPTION_WINDOWED_REGISTER, 6149 }, 6150 .op_flags = XTENSA_OP_PRIVILEGED | 6151 XTENSA_OP_EXIT_TB_M1 | 6152 XTENSA_OP_SYNC_REGISTER_WINDOW, 6153 }, { 6154 .name = "xsr.windowstart", 6155 .translate = translate_xsr_windowstart, 6156 .test_exceptions = test_exceptions_sr, 6157 .par = (const uint32_t[]){ 6158 WINDOW_START, 6159 XTENSA_OPTION_WINDOWED_REGISTER, 6160 }, 6161 .op_flags = XTENSA_OP_PRIVILEGED | XTENSA_OP_EXIT_TB_M1, 6162 }, 6163 }; 6164 6165 const XtensaOpcodeTranslators xtensa_core_opcodes = { 6166 .num_opcodes = ARRAY_SIZE(core_ops), 6167 .opcode = core_ops, 6168 }; 6169 6170 6171 static inline void get_f32_o1_i3(const OpcodeArg *arg, OpcodeArg *arg32, 6172 int o0, int i0, int i1, int i2) 6173 { 6174 if ((i0 >= 0 && arg[i0].num_bits == 64) || 6175 (o0 >= 0 && arg[o0].num_bits == 64)) { 6176 if (o0 >= 0) { 6177 arg32[o0].out = tcg_temp_new_i32(); 6178 } 6179 if (i0 >= 0) { 6180 arg32[i0].in = tcg_temp_new_i32(); 6181 tcg_gen_extrl_i64_i32(arg32[i0].in, arg[i0].in); 6182 } 6183 if (i1 >= 0) { 6184 arg32[i1].in = tcg_temp_new_i32(); 6185 tcg_gen_extrl_i64_i32(arg32[i1].in, arg[i1].in); 6186 } 6187 if (i2 >= 0) { 6188 arg32[i2].in = tcg_temp_new_i32(); 6189 tcg_gen_extrl_i64_i32(arg32[i2].in, arg[i2].in); 6190 } 6191 } else { 6192 if (o0 >= 0) { 6193 arg32[o0].out = arg[o0].out; 6194 } 6195 if (i0 >= 0) { 6196 arg32[i0].in = arg[i0].in; 6197 } 6198 if (i1 >= 0) { 6199 arg32[i1].in = arg[i1].in; 6200 } 6201 if (i2 >= 0) { 6202 arg32[i2].in = arg[i2].in; 6203 } 6204 } 6205 } 6206 6207 static inline void put_f32_o1_i3(const OpcodeArg *arg, const OpcodeArg *arg32, 6208 int o0, int i0, int i1, int i2) 6209 { 6210 if ((i0 >= 0 && arg[i0].num_bits == 64) || 6211 (o0 >= 0 && arg[o0].num_bits == 64)) { 6212 if (o0 >= 0) { 6213 tcg_gen_extu_i32_i64(arg[o0].out, arg32[o0].out); 6214 } 6215 } 6216 } 6217 6218 static inline void get_f32_o1_i2(const OpcodeArg *arg, OpcodeArg *arg32, 6219 int o0, int i0, int i1) 6220 { 6221 get_f32_o1_i3(arg, arg32, o0, i0, i1, -1); 6222 } 6223 6224 static inline void put_f32_o1_i2(const OpcodeArg *arg, const OpcodeArg *arg32, 6225 int o0, int i0, int i1) 6226 { 6227 put_f32_o1_i3(arg, arg32, o0, i0, i1, -1); 6228 } 6229 6230 static inline void get_f32_o1_i1(const OpcodeArg *arg, OpcodeArg *arg32, 6231 int o0, int i0) 6232 { 6233 get_f32_o1_i2(arg, arg32, o0, i0, -1); 6234 } 6235 6236 static inline void put_f32_o1_i1(const OpcodeArg *arg, const OpcodeArg *arg32, 6237 int o0, int i0) 6238 { 6239 put_f32_o1_i2(arg, arg32, o0, i0, -1); 6240 } 6241 6242 static inline void get_f32_o1(const OpcodeArg *arg, OpcodeArg *arg32, 6243 int o0) 6244 { 6245 get_f32_o1_i1(arg, arg32, o0, -1); 6246 } 6247 6248 static inline void put_f32_o1(const OpcodeArg *arg, const OpcodeArg *arg32, 6249 int o0) 6250 { 6251 put_f32_o1_i1(arg, arg32, o0, -1); 6252 } 6253 6254 static inline void get_f32_i2(const OpcodeArg *arg, OpcodeArg *arg32, 6255 int i0, int i1) 6256 { 6257 get_f32_o1_i2(arg, arg32, -1, i0, i1); 6258 } 6259 6260 static inline void put_f32_i2(const OpcodeArg *arg, const OpcodeArg *arg32, 6261 int i0, int i1) 6262 { 6263 put_f32_o1_i2(arg, arg32, -1, i0, i1); 6264 } 6265 6266 static inline void get_f32_i1(const OpcodeArg *arg, OpcodeArg *arg32, 6267 int i0) 6268 { 6269 get_f32_i2(arg, arg32, i0, -1); 6270 } 6271 6272 static inline void put_f32_i1(const OpcodeArg *arg, const OpcodeArg *arg32, 6273 int i0) 6274 { 6275 put_f32_i2(arg, arg32, i0, -1); 6276 } 6277 6278 6279 static void translate_abs_d(DisasContext *dc, const OpcodeArg arg[], 6280 const uint32_t par[]) 6281 { 6282 gen_helper_abs_d(arg[0].out, arg[1].in); 6283 } 6284 6285 static void translate_abs_s(DisasContext *dc, const OpcodeArg arg[], 6286 const uint32_t par[]) 6287 { 6288 OpcodeArg arg32[2]; 6289 6290 get_f32_o1_i1(arg, arg32, 0, 1); 6291 gen_helper_abs_s(arg32[0].out, arg32[1].in); 6292 put_f32_o1_i1(arg, arg32, 0, 1); 6293 } 6294 6295 static void translate_fpu2k_add_s(DisasContext *dc, const OpcodeArg arg[], 6296 const uint32_t par[]) 6297 { 6298 gen_helper_fpu2k_add_s(arg[0].out, tcg_env, 6299 arg[1].in, arg[2].in); 6300 } 6301 6302 enum { 6303 COMPARE_UN, 6304 COMPARE_OEQ, 6305 COMPARE_UEQ, 6306 COMPARE_OLT, 6307 COMPARE_ULT, 6308 COMPARE_OLE, 6309 COMPARE_ULE, 6310 }; 6311 6312 static void translate_compare_d(DisasContext *dc, const OpcodeArg arg[], 6313 const uint32_t par[]) 6314 { 6315 static void (* const helper[])(TCGv_i32 res, TCGv_env env, 6316 TCGv_i64 s, TCGv_i64 t) = { 6317 [COMPARE_UN] = gen_helper_un_d, 6318 [COMPARE_OEQ] = gen_helper_oeq_d, 6319 [COMPARE_UEQ] = gen_helper_ueq_d, 6320 [COMPARE_OLT] = gen_helper_olt_d, 6321 [COMPARE_ULT] = gen_helper_ult_d, 6322 [COMPARE_OLE] = gen_helper_ole_d, 6323 [COMPARE_ULE] = gen_helper_ule_d, 6324 }; 6325 TCGv_i32 zero = tcg_constant_i32(0); 6326 TCGv_i32 res = tcg_temp_new_i32(); 6327 TCGv_i32 set_br = tcg_temp_new_i32(); 6328 TCGv_i32 clr_br = tcg_temp_new_i32(); 6329 6330 tcg_gen_ori_i32(set_br, arg[0].in, 1 << arg[0].imm); 6331 tcg_gen_andi_i32(clr_br, arg[0].in, ~(1 << arg[0].imm)); 6332 6333 helper[par[0]](res, tcg_env, arg[1].in, arg[2].in); 6334 tcg_gen_movcond_i32(TCG_COND_NE, 6335 arg[0].out, res, zero, 6336 set_br, clr_br); 6337 } 6338 6339 static void translate_compare_s(DisasContext *dc, const OpcodeArg arg[], 6340 const uint32_t par[]) 6341 { 6342 static void (* const helper[])(TCGv_i32 res, TCGv_env env, 6343 TCGv_i32 s, TCGv_i32 t) = { 6344 [COMPARE_UN] = gen_helper_un_s, 6345 [COMPARE_OEQ] = gen_helper_oeq_s, 6346 [COMPARE_UEQ] = gen_helper_ueq_s, 6347 [COMPARE_OLT] = gen_helper_olt_s, 6348 [COMPARE_ULT] = gen_helper_ult_s, 6349 [COMPARE_OLE] = gen_helper_ole_s, 6350 [COMPARE_ULE] = gen_helper_ule_s, 6351 }; 6352 OpcodeArg arg32[3]; 6353 TCGv_i32 zero = tcg_constant_i32(0); 6354 TCGv_i32 res = tcg_temp_new_i32(); 6355 TCGv_i32 set_br = tcg_temp_new_i32(); 6356 TCGv_i32 clr_br = tcg_temp_new_i32(); 6357 6358 tcg_gen_ori_i32(set_br, arg[0].in, 1 << arg[0].imm); 6359 tcg_gen_andi_i32(clr_br, arg[0].in, ~(1 << arg[0].imm)); 6360 6361 get_f32_i2(arg, arg32, 1, 2); 6362 helper[par[0]](res, tcg_env, arg32[1].in, arg32[2].in); 6363 tcg_gen_movcond_i32(TCG_COND_NE, 6364 arg[0].out, res, zero, 6365 set_br, clr_br); 6366 put_f32_i2(arg, arg32, 1, 2); 6367 } 6368 6369 static void translate_const_d(DisasContext *dc, const OpcodeArg arg[], 6370 const uint32_t par[]) 6371 { 6372 static const uint64_t v[] = { 6373 UINT64_C(0x0000000000000000), 6374 UINT64_C(0x3ff0000000000000), 6375 UINT64_C(0x4000000000000000), 6376 UINT64_C(0x3fe0000000000000), 6377 }; 6378 6379 tcg_gen_movi_i64(arg[0].out, v[arg[1].imm % ARRAY_SIZE(v)]); 6380 if (arg[1].imm >= ARRAY_SIZE(v)) { 6381 qemu_log_mask(LOG_GUEST_ERROR, 6382 "const.d f%d, #%d, immediate value is reserved\n", 6383 arg[0].imm, arg[1].imm); 6384 } 6385 } 6386 6387 static void translate_const_s(DisasContext *dc, const OpcodeArg arg[], 6388 const uint32_t par[]) 6389 { 6390 static const uint32_t v[] = { 6391 0x00000000, 6392 0x3f800000, 6393 0x40000000, 6394 0x3f000000, 6395 }; 6396 6397 if (arg[0].num_bits == 32) { 6398 tcg_gen_movi_i32(arg[0].out, v[arg[1].imm % ARRAY_SIZE(v)]); 6399 } else { 6400 tcg_gen_movi_i64(arg[0].out, v[arg[1].imm % ARRAY_SIZE(v)]); 6401 } 6402 if (arg[1].imm >= ARRAY_SIZE(v)) { 6403 qemu_log_mask(LOG_GUEST_ERROR, 6404 "const.s f%d, #%d, immediate value is reserved\n", 6405 arg[0].imm, arg[1].imm); 6406 } 6407 } 6408 6409 static void translate_float_d(DisasContext *dc, const OpcodeArg arg[], 6410 const uint32_t par[]) 6411 { 6412 TCGv_i32 scale = tcg_constant_i32(-arg[2].imm); 6413 6414 if (par[0]) { 6415 gen_helper_uitof_d(arg[0].out, tcg_env, arg[1].in, scale); 6416 } else { 6417 gen_helper_itof_d(arg[0].out, tcg_env, arg[1].in, scale); 6418 } 6419 } 6420 6421 static void translate_float_s(DisasContext *dc, const OpcodeArg arg[], 6422 const uint32_t par[]) 6423 { 6424 TCGv_i32 scale = tcg_constant_i32(-arg[2].imm); 6425 OpcodeArg arg32[1]; 6426 6427 get_f32_o1(arg, arg32, 0); 6428 if (par[0]) { 6429 gen_helper_uitof_s(arg32[0].out, tcg_env, arg[1].in, scale); 6430 } else { 6431 gen_helper_itof_s(arg32[0].out, tcg_env, arg[1].in, scale); 6432 } 6433 put_f32_o1(arg, arg32, 0); 6434 } 6435 6436 static void translate_ftoi_d(DisasContext *dc, const OpcodeArg arg[], 6437 const uint32_t par[]) 6438 { 6439 TCGv_i32 rounding_mode = tcg_constant_i32(par[0]); 6440 TCGv_i32 scale = tcg_constant_i32(arg[2].imm); 6441 6442 if (par[1]) { 6443 gen_helper_ftoui_d(arg[0].out, tcg_env, arg[1].in, 6444 rounding_mode, scale); 6445 } else { 6446 gen_helper_ftoi_d(arg[0].out, tcg_env, arg[1].in, 6447 rounding_mode, scale); 6448 } 6449 } 6450 6451 static void translate_ftoi_s(DisasContext *dc, const OpcodeArg arg[], 6452 const uint32_t par[]) 6453 { 6454 TCGv_i32 rounding_mode = tcg_constant_i32(par[0]); 6455 TCGv_i32 scale = tcg_constant_i32(arg[2].imm); 6456 OpcodeArg arg32[2]; 6457 6458 get_f32_i1(arg, arg32, 1); 6459 if (par[1]) { 6460 gen_helper_ftoui_s(arg[0].out, tcg_env, arg32[1].in, 6461 rounding_mode, scale); 6462 } else { 6463 gen_helper_ftoi_s(arg[0].out, tcg_env, arg32[1].in, 6464 rounding_mode, scale); 6465 } 6466 put_f32_i1(arg, arg32, 1); 6467 } 6468 6469 static void translate_ldsti(DisasContext *dc, const OpcodeArg arg[], 6470 const uint32_t par[]) 6471 { 6472 TCGv_i32 addr = tcg_temp_new_i32(); 6473 MemOp mop; 6474 6475 tcg_gen_addi_i32(addr, arg[1].in, arg[2].imm); 6476 mop = gen_load_store_alignment(dc, MO_TEUL, addr); 6477 if (par[0]) { 6478 tcg_gen_qemu_st_tl(arg[0].in, addr, dc->cring, mop); 6479 } else { 6480 tcg_gen_qemu_ld_tl(arg[0].out, addr, dc->cring, mop); 6481 } 6482 if (par[1]) { 6483 tcg_gen_mov_i32(arg[1].out, addr); 6484 } 6485 } 6486 6487 static void translate_ldstx(DisasContext *dc, const OpcodeArg arg[], 6488 const uint32_t par[]) 6489 { 6490 TCGv_i32 addr = tcg_temp_new_i32(); 6491 MemOp mop; 6492 6493 tcg_gen_add_i32(addr, arg[1].in, arg[2].in); 6494 mop = gen_load_store_alignment(dc, MO_TEUL, addr); 6495 if (par[0]) { 6496 tcg_gen_qemu_st_tl(arg[0].in, addr, dc->cring, mop); 6497 } else { 6498 tcg_gen_qemu_ld_tl(arg[0].out, addr, dc->cring, mop); 6499 } 6500 if (par[1]) { 6501 tcg_gen_mov_i32(arg[1].out, addr); 6502 } 6503 } 6504 6505 static void translate_fpu2k_madd_s(DisasContext *dc, const OpcodeArg arg[], 6506 const uint32_t par[]) 6507 { 6508 gen_helper_fpu2k_madd_s(arg[0].out, tcg_env, 6509 arg[0].in, arg[1].in, arg[2].in); 6510 } 6511 6512 static void translate_mov_d(DisasContext *dc, const OpcodeArg arg[], 6513 const uint32_t par[]) 6514 { 6515 tcg_gen_mov_i64(arg[0].out, arg[1].in); 6516 } 6517 6518 static void translate_mov_s(DisasContext *dc, const OpcodeArg arg[], 6519 const uint32_t par[]) 6520 { 6521 if (arg[0].num_bits == 32) { 6522 tcg_gen_mov_i32(arg[0].out, arg[1].in); 6523 } else { 6524 tcg_gen_mov_i64(arg[0].out, arg[1].in); 6525 } 6526 } 6527 6528 static void translate_movcond_d(DisasContext *dc, const OpcodeArg arg[], 6529 const uint32_t par[]) 6530 { 6531 TCGv_i64 zero = tcg_constant_i64(0); 6532 TCGv_i64 arg2 = tcg_temp_new_i64(); 6533 6534 tcg_gen_ext_i32_i64(arg2, arg[2].in); 6535 tcg_gen_movcond_i64(par[0], arg[0].out, 6536 arg2, zero, 6537 arg[1].in, arg[0].in); 6538 } 6539 6540 static void translate_movcond_s(DisasContext *dc, const OpcodeArg arg[], 6541 const uint32_t par[]) 6542 { 6543 if (arg[0].num_bits == 32) { 6544 TCGv_i32 zero = tcg_constant_i32(0); 6545 6546 tcg_gen_movcond_i32(par[0], arg[0].out, 6547 arg[2].in, zero, 6548 arg[1].in, arg[0].in); 6549 } else { 6550 translate_movcond_d(dc, arg, par); 6551 } 6552 } 6553 6554 static void translate_movp_d(DisasContext *dc, const OpcodeArg arg[], 6555 const uint32_t par[]) 6556 { 6557 TCGv_i64 zero = tcg_constant_i64(0); 6558 TCGv_i32 tmp1 = tcg_temp_new_i32(); 6559 TCGv_i64 tmp2 = tcg_temp_new_i64(); 6560 6561 tcg_gen_andi_i32(tmp1, arg[2].in, 1 << arg[2].imm); 6562 tcg_gen_extu_i32_i64(tmp2, tmp1); 6563 tcg_gen_movcond_i64(par[0], 6564 arg[0].out, tmp2, zero, 6565 arg[1].in, arg[0].in); 6566 } 6567 6568 static void translate_movp_s(DisasContext *dc, const OpcodeArg arg[], 6569 const uint32_t par[]) 6570 { 6571 if (arg[0].num_bits == 32) { 6572 TCGv_i32 zero = tcg_constant_i32(0); 6573 TCGv_i32 tmp = tcg_temp_new_i32(); 6574 6575 tcg_gen_andi_i32(tmp, arg[2].in, 1 << arg[2].imm); 6576 tcg_gen_movcond_i32(par[0], 6577 arg[0].out, tmp, zero, 6578 arg[1].in, arg[0].in); 6579 } else { 6580 translate_movp_d(dc, arg, par); 6581 } 6582 } 6583 6584 static void translate_fpu2k_mul_s(DisasContext *dc, const OpcodeArg arg[], 6585 const uint32_t par[]) 6586 { 6587 gen_helper_fpu2k_mul_s(arg[0].out, tcg_env, 6588 arg[1].in, arg[2].in); 6589 } 6590 6591 static void translate_fpu2k_msub_s(DisasContext *dc, const OpcodeArg arg[], 6592 const uint32_t par[]) 6593 { 6594 gen_helper_fpu2k_msub_s(arg[0].out, tcg_env, 6595 arg[0].in, arg[1].in, arg[2].in); 6596 } 6597 6598 static void translate_neg_d(DisasContext *dc, const OpcodeArg arg[], 6599 const uint32_t par[]) 6600 { 6601 gen_helper_neg_d(arg[0].out, arg[1].in); 6602 } 6603 6604 static void translate_neg_s(DisasContext *dc, const OpcodeArg arg[], 6605 const uint32_t par[]) 6606 { 6607 OpcodeArg arg32[2]; 6608 6609 get_f32_o1_i1(arg, arg32, 0, 1); 6610 gen_helper_neg_s(arg32[0].out, arg32[1].in); 6611 put_f32_o1_i1(arg, arg32, 0, 1); 6612 } 6613 6614 static void translate_rfr_d(DisasContext *dc, const OpcodeArg arg[], 6615 const uint32_t par[]) 6616 { 6617 tcg_gen_extrh_i64_i32(arg[0].out, arg[1].in); 6618 } 6619 6620 static void translate_rfr_s(DisasContext *dc, const OpcodeArg arg[], 6621 const uint32_t par[]) 6622 { 6623 if (arg[1].num_bits == 32) { 6624 tcg_gen_mov_i32(arg[0].out, arg[1].in); 6625 } else { 6626 tcg_gen_extrl_i64_i32(arg[0].out, arg[1].in); 6627 } 6628 } 6629 6630 static void translate_fpu2k_sub_s(DisasContext *dc, const OpcodeArg arg[], 6631 const uint32_t par[]) 6632 { 6633 gen_helper_fpu2k_sub_s(arg[0].out, tcg_env, 6634 arg[1].in, arg[2].in); 6635 } 6636 6637 static void translate_wfr_d(DisasContext *dc, const OpcodeArg arg[], 6638 const uint32_t par[]) 6639 { 6640 tcg_gen_concat_i32_i64(arg[0].out, arg[2].in, arg[1].in); 6641 } 6642 6643 static void translate_wfr_s(DisasContext *dc, const OpcodeArg arg[], 6644 const uint32_t par[]) 6645 { 6646 if (arg[0].num_bits == 32) { 6647 tcg_gen_mov_i32(arg[0].out, arg[1].in); 6648 } else { 6649 tcg_gen_ext_i32_i64(arg[0].out, arg[1].in); 6650 } 6651 } 6652 6653 static void translate_wur_fpu2k_fcr(DisasContext *dc, const OpcodeArg arg[], 6654 const uint32_t par[]) 6655 { 6656 gen_helper_wur_fpu2k_fcr(tcg_env, arg[0].in); 6657 } 6658 6659 static void translate_wur_fpu2k_fsr(DisasContext *dc, const OpcodeArg arg[], 6660 const uint32_t par[]) 6661 { 6662 tcg_gen_andi_i32(cpu_UR[par[0]], arg[0].in, 0xffffff80); 6663 } 6664 6665 static const XtensaOpcodeOps fpu2000_ops[] = { 6666 { 6667 .name = "abs.s", 6668 .translate = translate_abs_s, 6669 .coprocessor = 0x1, 6670 }, { 6671 .name = "add.s", 6672 .translate = translate_fpu2k_add_s, 6673 .coprocessor = 0x1, 6674 }, { 6675 .name = "ceil.s", 6676 .translate = translate_ftoi_s, 6677 .par = (const uint32_t[]){float_round_up, false}, 6678 .coprocessor = 0x1, 6679 }, { 6680 .name = "float.s", 6681 .translate = translate_float_s, 6682 .par = (const uint32_t[]){false}, 6683 .coprocessor = 0x1, 6684 }, { 6685 .name = "floor.s", 6686 .translate = translate_ftoi_s, 6687 .par = (const uint32_t[]){float_round_down, false}, 6688 .coprocessor = 0x1, 6689 }, { 6690 .name = "lsi", 6691 .translate = translate_ldsti, 6692 .par = (const uint32_t[]){false, false}, 6693 .op_flags = XTENSA_OP_LOAD, 6694 .coprocessor = 0x1, 6695 }, { 6696 .name = "lsiu", 6697 .translate = translate_ldsti, 6698 .par = (const uint32_t[]){false, true}, 6699 .op_flags = XTENSA_OP_LOAD, 6700 .coprocessor = 0x1, 6701 }, { 6702 .name = "lsx", 6703 .translate = translate_ldstx, 6704 .par = (const uint32_t[]){false, false}, 6705 .op_flags = XTENSA_OP_LOAD, 6706 .coprocessor = 0x1, 6707 }, { 6708 .name = "lsxu", 6709 .translate = translate_ldstx, 6710 .par = (const uint32_t[]){false, true}, 6711 .op_flags = XTENSA_OP_LOAD, 6712 .coprocessor = 0x1, 6713 }, { 6714 .name = "madd.s", 6715 .translate = translate_fpu2k_madd_s, 6716 .coprocessor = 0x1, 6717 }, { 6718 .name = "mov.s", 6719 .translate = translate_mov_s, 6720 .coprocessor = 0x1, 6721 }, { 6722 .name = "moveqz.s", 6723 .translate = translate_movcond_s, 6724 .par = (const uint32_t[]){TCG_COND_EQ}, 6725 .coprocessor = 0x1, 6726 }, { 6727 .name = "movf.s", 6728 .translate = translate_movp_s, 6729 .par = (const uint32_t[]){TCG_COND_EQ}, 6730 .coprocessor = 0x1, 6731 }, { 6732 .name = "movgez.s", 6733 .translate = translate_movcond_s, 6734 .par = (const uint32_t[]){TCG_COND_GE}, 6735 .coprocessor = 0x1, 6736 }, { 6737 .name = "movltz.s", 6738 .translate = translate_movcond_s, 6739 .par = (const uint32_t[]){TCG_COND_LT}, 6740 .coprocessor = 0x1, 6741 }, { 6742 .name = "movnez.s", 6743 .translate = translate_movcond_s, 6744 .par = (const uint32_t[]){TCG_COND_NE}, 6745 .coprocessor = 0x1, 6746 }, { 6747 .name = "movt.s", 6748 .translate = translate_movp_s, 6749 .par = (const uint32_t[]){TCG_COND_NE}, 6750 .coprocessor = 0x1, 6751 }, { 6752 .name = "msub.s", 6753 .translate = translate_fpu2k_msub_s, 6754 .coprocessor = 0x1, 6755 }, { 6756 .name = "mul.s", 6757 .translate = translate_fpu2k_mul_s, 6758 .coprocessor = 0x1, 6759 }, { 6760 .name = "neg.s", 6761 .translate = translate_neg_s, 6762 .coprocessor = 0x1, 6763 }, { 6764 .name = "oeq.s", 6765 .translate = translate_compare_s, 6766 .par = (const uint32_t[]){COMPARE_OEQ}, 6767 .coprocessor = 0x1, 6768 }, { 6769 .name = "ole.s", 6770 .translate = translate_compare_s, 6771 .par = (const uint32_t[]){COMPARE_OLE}, 6772 .coprocessor = 0x1, 6773 }, { 6774 .name = "olt.s", 6775 .translate = translate_compare_s, 6776 .par = (const uint32_t[]){COMPARE_OLT}, 6777 .coprocessor = 0x1, 6778 }, { 6779 .name = "rfr", 6780 .translate = translate_rfr_s, 6781 .coprocessor = 0x1, 6782 }, { 6783 .name = "round.s", 6784 .translate = translate_ftoi_s, 6785 .par = (const uint32_t[]){float_round_nearest_even, false}, 6786 .coprocessor = 0x1, 6787 }, { 6788 .name = "rur.fcr", 6789 .translate = translate_rur, 6790 .par = (const uint32_t[]){FCR}, 6791 .coprocessor = 0x1, 6792 }, { 6793 .name = "rur.fsr", 6794 .translate = translate_rur, 6795 .par = (const uint32_t[]){FSR}, 6796 .coprocessor = 0x1, 6797 }, { 6798 .name = "ssi", 6799 .translate = translate_ldsti, 6800 .par = (const uint32_t[]){true, false}, 6801 .op_flags = XTENSA_OP_STORE, 6802 .coprocessor = 0x1, 6803 }, { 6804 .name = "ssiu", 6805 .translate = translate_ldsti, 6806 .par = (const uint32_t[]){true, true}, 6807 .op_flags = XTENSA_OP_STORE, 6808 .coprocessor = 0x1, 6809 }, { 6810 .name = "ssx", 6811 .translate = translate_ldstx, 6812 .par = (const uint32_t[]){true, false}, 6813 .op_flags = XTENSA_OP_STORE, 6814 .coprocessor = 0x1, 6815 }, { 6816 .name = "ssxu", 6817 .translate = translate_ldstx, 6818 .par = (const uint32_t[]){true, true}, 6819 .op_flags = XTENSA_OP_STORE, 6820 .coprocessor = 0x1, 6821 }, { 6822 .name = "sub.s", 6823 .translate = translate_fpu2k_sub_s, 6824 .coprocessor = 0x1, 6825 }, { 6826 .name = "trunc.s", 6827 .translate = translate_ftoi_s, 6828 .par = (const uint32_t[]){float_round_to_zero, false}, 6829 .coprocessor = 0x1, 6830 }, { 6831 .name = "ueq.s", 6832 .translate = translate_compare_s, 6833 .par = (const uint32_t[]){COMPARE_UEQ}, 6834 .coprocessor = 0x1, 6835 }, { 6836 .name = "ufloat.s", 6837 .translate = translate_float_s, 6838 .par = (const uint32_t[]){true}, 6839 .coprocessor = 0x1, 6840 }, { 6841 .name = "ule.s", 6842 .translate = translate_compare_s, 6843 .par = (const uint32_t[]){COMPARE_ULE}, 6844 .coprocessor = 0x1, 6845 }, { 6846 .name = "ult.s", 6847 .translate = translate_compare_s, 6848 .par = (const uint32_t[]){COMPARE_ULT}, 6849 .coprocessor = 0x1, 6850 }, { 6851 .name = "un.s", 6852 .translate = translate_compare_s, 6853 .par = (const uint32_t[]){COMPARE_UN}, 6854 .coprocessor = 0x1, 6855 }, { 6856 .name = "utrunc.s", 6857 .translate = translate_ftoi_s, 6858 .par = (const uint32_t[]){float_round_to_zero, true}, 6859 .coprocessor = 0x1, 6860 }, { 6861 .name = "wfr", 6862 .translate = translate_wfr_s, 6863 .coprocessor = 0x1, 6864 }, { 6865 .name = "wur.fcr", 6866 .translate = translate_wur_fpu2k_fcr, 6867 .par = (const uint32_t[]){FCR}, 6868 .coprocessor = 0x1, 6869 }, { 6870 .name = "wur.fsr", 6871 .translate = translate_wur_fpu2k_fsr, 6872 .par = (const uint32_t[]){FSR}, 6873 .coprocessor = 0x1, 6874 }, 6875 }; 6876 6877 const XtensaOpcodeTranslators xtensa_fpu2000_opcodes = { 6878 .num_opcodes = ARRAY_SIZE(fpu2000_ops), 6879 .opcode = fpu2000_ops, 6880 }; 6881 6882 static void translate_add_d(DisasContext *dc, const OpcodeArg arg[], 6883 const uint32_t par[]) 6884 { 6885 gen_helper_add_d(arg[0].out, tcg_env, arg[1].in, arg[2].in); 6886 } 6887 6888 static void translate_add_s(DisasContext *dc, const OpcodeArg arg[], 6889 const uint32_t par[]) 6890 { 6891 if (option_enabled(dc, XTENSA_OPTION_DFPU_SINGLE_ONLY)) { 6892 gen_helper_fpu2k_add_s(arg[0].out, tcg_env, 6893 arg[1].in, arg[2].in); 6894 } else { 6895 OpcodeArg arg32[3]; 6896 6897 get_f32_o1_i2(arg, arg32, 0, 1, 2); 6898 gen_helper_add_s(arg32[0].out, tcg_env, arg32[1].in, arg32[2].in); 6899 put_f32_o1_i2(arg, arg32, 0, 1, 2); 6900 } 6901 } 6902 6903 static void translate_cvtd_s(DisasContext *dc, const OpcodeArg arg[], 6904 const uint32_t par[]) 6905 { 6906 TCGv_i32 v = tcg_temp_new_i32(); 6907 6908 tcg_gen_extrl_i64_i32(v, arg[1].in); 6909 gen_helper_cvtd_s(arg[0].out, tcg_env, v); 6910 } 6911 6912 static void translate_cvts_d(DisasContext *dc, const OpcodeArg arg[], 6913 const uint32_t par[]) 6914 { 6915 TCGv_i32 v = tcg_temp_new_i32(); 6916 6917 gen_helper_cvts_d(v, tcg_env, arg[1].in); 6918 tcg_gen_extu_i32_i64(arg[0].out, v); 6919 } 6920 6921 static void translate_ldsti_d(DisasContext *dc, const OpcodeArg arg[], 6922 const uint32_t par[]) 6923 { 6924 TCGv_i32 addr; 6925 MemOp mop; 6926 6927 if (par[1]) { 6928 addr = tcg_temp_new_i32(); 6929 tcg_gen_addi_i32(addr, arg[1].in, arg[2].imm); 6930 } else { 6931 addr = arg[1].in; 6932 } 6933 mop = gen_load_store_alignment(dc, MO_TEUQ, addr); 6934 if (par[0]) { 6935 tcg_gen_qemu_st_i64(arg[0].in, addr, dc->cring, mop); 6936 } else { 6937 tcg_gen_qemu_ld_i64(arg[0].out, addr, dc->cring, mop); 6938 } 6939 if (par[2]) { 6940 if (par[1]) { 6941 tcg_gen_mov_i32(arg[1].out, addr); 6942 } else { 6943 tcg_gen_addi_i32(arg[1].out, arg[1].in, arg[2].imm); 6944 } 6945 } 6946 } 6947 6948 static void translate_ldsti_s(DisasContext *dc, const OpcodeArg arg[], 6949 const uint32_t par[]) 6950 { 6951 TCGv_i32 addr; 6952 OpcodeArg arg32[1]; 6953 MemOp mop; 6954 6955 if (par[1]) { 6956 addr = tcg_temp_new_i32(); 6957 tcg_gen_addi_i32(addr, arg[1].in, arg[2].imm); 6958 } else { 6959 addr = arg[1].in; 6960 } 6961 mop = gen_load_store_alignment(dc, MO_TEUL, addr); 6962 if (par[0]) { 6963 get_f32_i1(arg, arg32, 0); 6964 tcg_gen_qemu_st_tl(arg32[0].in, addr, dc->cring, mop); 6965 put_f32_i1(arg, arg32, 0); 6966 } else { 6967 get_f32_o1(arg, arg32, 0); 6968 tcg_gen_qemu_ld_tl(arg32[0].out, addr, dc->cring, mop); 6969 put_f32_o1(arg, arg32, 0); 6970 } 6971 if (par[2]) { 6972 if (par[1]) { 6973 tcg_gen_mov_i32(arg[1].out, addr); 6974 } else { 6975 tcg_gen_addi_i32(arg[1].out, arg[1].in, arg[2].imm); 6976 } 6977 } 6978 } 6979 6980 static void translate_ldstx_d(DisasContext *dc, const OpcodeArg arg[], 6981 const uint32_t par[]) 6982 { 6983 TCGv_i32 addr; 6984 MemOp mop; 6985 6986 if (par[1]) { 6987 addr = tcg_temp_new_i32(); 6988 tcg_gen_add_i32(addr, arg[1].in, arg[2].in); 6989 } else { 6990 addr = arg[1].in; 6991 } 6992 mop = gen_load_store_alignment(dc, MO_TEUQ, addr); 6993 if (par[0]) { 6994 tcg_gen_qemu_st_i64(arg[0].in, addr, dc->cring, mop); 6995 } else { 6996 tcg_gen_qemu_ld_i64(arg[0].out, addr, dc->cring, mop); 6997 } 6998 if (par[2]) { 6999 if (par[1]) { 7000 tcg_gen_mov_i32(arg[1].out, addr); 7001 } else { 7002 tcg_gen_add_i32(arg[1].out, arg[1].in, arg[2].in); 7003 } 7004 } 7005 } 7006 7007 static void translate_ldstx_s(DisasContext *dc, const OpcodeArg arg[], 7008 const uint32_t par[]) 7009 { 7010 TCGv_i32 addr; 7011 OpcodeArg arg32[1]; 7012 MemOp mop; 7013 7014 if (par[1]) { 7015 addr = tcg_temp_new_i32(); 7016 tcg_gen_add_i32(addr, arg[1].in, arg[2].in); 7017 } else { 7018 addr = arg[1].in; 7019 } 7020 mop = gen_load_store_alignment(dc, MO_TEUL, addr); 7021 if (par[0]) { 7022 get_f32_i1(arg, arg32, 0); 7023 tcg_gen_qemu_st_tl(arg32[0].in, addr, dc->cring, mop); 7024 put_f32_i1(arg, arg32, 0); 7025 } else { 7026 get_f32_o1(arg, arg32, 0); 7027 tcg_gen_qemu_ld_tl(arg32[0].out, addr, dc->cring, mop); 7028 put_f32_o1(arg, arg32, 0); 7029 } 7030 if (par[2]) { 7031 if (par[1]) { 7032 tcg_gen_mov_i32(arg[1].out, addr); 7033 } else { 7034 tcg_gen_add_i32(arg[1].out, arg[1].in, arg[2].in); 7035 } 7036 } 7037 } 7038 7039 static void translate_madd_d(DisasContext *dc, const OpcodeArg arg[], 7040 const uint32_t par[]) 7041 { 7042 gen_helper_madd_d(arg[0].out, tcg_env, 7043 arg[0].in, arg[1].in, arg[2].in); 7044 } 7045 7046 static void translate_madd_s(DisasContext *dc, const OpcodeArg arg[], 7047 const uint32_t par[]) 7048 { 7049 if (option_enabled(dc, XTENSA_OPTION_DFPU_SINGLE_ONLY)) { 7050 gen_helper_fpu2k_madd_s(arg[0].out, tcg_env, 7051 arg[0].in, arg[1].in, arg[2].in); 7052 } else { 7053 OpcodeArg arg32[3]; 7054 7055 get_f32_o1_i3(arg, arg32, 0, 0, 1, 2); 7056 gen_helper_madd_s(arg32[0].out, tcg_env, 7057 arg32[0].in, arg32[1].in, arg32[2].in); 7058 put_f32_o1_i3(arg, arg32, 0, 0, 1, 2); 7059 } 7060 } 7061 7062 static void translate_mul_d(DisasContext *dc, const OpcodeArg arg[], 7063 const uint32_t par[]) 7064 { 7065 gen_helper_mul_d(arg[0].out, tcg_env, arg[1].in, arg[2].in); 7066 } 7067 7068 static void translate_mul_s(DisasContext *dc, const OpcodeArg arg[], 7069 const uint32_t par[]) 7070 { 7071 if (option_enabled(dc, XTENSA_OPTION_DFPU_SINGLE_ONLY)) { 7072 gen_helper_fpu2k_mul_s(arg[0].out, tcg_env, 7073 arg[1].in, arg[2].in); 7074 } else { 7075 OpcodeArg arg32[3]; 7076 7077 get_f32_o1_i2(arg, arg32, 0, 1, 2); 7078 gen_helper_mul_s(arg32[0].out, tcg_env, arg32[1].in, arg32[2].in); 7079 put_f32_o1_i2(arg, arg32, 0, 1, 2); 7080 } 7081 } 7082 7083 static void translate_msub_d(DisasContext *dc, const OpcodeArg arg[], 7084 const uint32_t par[]) 7085 { 7086 gen_helper_msub_d(arg[0].out, tcg_env, 7087 arg[0].in, arg[1].in, arg[2].in); 7088 } 7089 7090 static void translate_msub_s(DisasContext *dc, const OpcodeArg arg[], 7091 const uint32_t par[]) 7092 { 7093 if (option_enabled(dc, XTENSA_OPTION_DFPU_SINGLE_ONLY)) { 7094 gen_helper_fpu2k_msub_s(arg[0].out, tcg_env, 7095 arg[0].in, arg[1].in, arg[2].in); 7096 } else { 7097 OpcodeArg arg32[3]; 7098 7099 get_f32_o1_i3(arg, arg32, 0, 0, 1, 2); 7100 gen_helper_msub_s(arg32[0].out, tcg_env, 7101 arg32[0].in, arg32[1].in, arg32[2].in); 7102 put_f32_o1_i3(arg, arg32, 0, 0, 1, 2); 7103 } 7104 } 7105 7106 static void translate_sub_d(DisasContext *dc, const OpcodeArg arg[], 7107 const uint32_t par[]) 7108 { 7109 gen_helper_sub_d(arg[0].out, tcg_env, arg[1].in, arg[2].in); 7110 } 7111 7112 static void translate_sub_s(DisasContext *dc, const OpcodeArg arg[], 7113 const uint32_t par[]) 7114 { 7115 if (option_enabled(dc, XTENSA_OPTION_DFPU_SINGLE_ONLY)) { 7116 gen_helper_fpu2k_sub_s(arg[0].out, tcg_env, 7117 arg[1].in, arg[2].in); 7118 } else { 7119 OpcodeArg arg32[3]; 7120 7121 get_f32_o1_i2(arg, arg32, 0, 1, 2); 7122 gen_helper_sub_s(arg32[0].out, tcg_env, arg32[1].in, arg32[2].in); 7123 put_f32_o1_i2(arg, arg32, 0, 1, 2); 7124 } 7125 } 7126 7127 static void translate_mkdadj_d(DisasContext *dc, const OpcodeArg arg[], 7128 const uint32_t par[]) 7129 { 7130 gen_helper_mkdadj_d(arg[0].out, tcg_env, arg[0].in, arg[1].in); 7131 } 7132 7133 static void translate_mkdadj_s(DisasContext *dc, const OpcodeArg arg[], 7134 const uint32_t par[]) 7135 { 7136 OpcodeArg arg32[2]; 7137 7138 get_f32_o1_i2(arg, arg32, 0, 0, 1); 7139 gen_helper_mkdadj_s(arg32[0].out, tcg_env, arg32[0].in, arg32[1].in); 7140 put_f32_o1_i2(arg, arg32, 0, 0, 1); 7141 } 7142 7143 static void translate_mksadj_d(DisasContext *dc, const OpcodeArg arg[], 7144 const uint32_t par[]) 7145 { 7146 gen_helper_mksadj_d(arg[0].out, tcg_env, arg[1].in); 7147 } 7148 7149 static void translate_mksadj_s(DisasContext *dc, const OpcodeArg arg[], 7150 const uint32_t par[]) 7151 { 7152 OpcodeArg arg32[2]; 7153 7154 get_f32_o1_i1(arg, arg32, 0, 1); 7155 gen_helper_mksadj_s(arg32[0].out, tcg_env, arg32[1].in); 7156 put_f32_o1_i1(arg, arg32, 0, 1); 7157 } 7158 7159 static void translate_wur_fpu_fcr(DisasContext *dc, const OpcodeArg arg[], 7160 const uint32_t par[]) 7161 { 7162 gen_helper_wur_fpu_fcr(tcg_env, arg[0].in); 7163 } 7164 7165 static void translate_rur_fpu_fsr(DisasContext *dc, const OpcodeArg arg[], 7166 const uint32_t par[]) 7167 { 7168 gen_helper_rur_fpu_fsr(arg[0].out, tcg_env); 7169 } 7170 7171 static void translate_wur_fpu_fsr(DisasContext *dc, const OpcodeArg arg[], 7172 const uint32_t par[]) 7173 { 7174 gen_helper_wur_fpu_fsr(tcg_env, arg[0].in); 7175 } 7176 7177 static const XtensaOpcodeOps fpu_ops[] = { 7178 { 7179 .name = "abs.d", 7180 .translate = translate_abs_d, 7181 .coprocessor = 0x1, 7182 }, { 7183 .name = "abs.s", 7184 .translate = translate_abs_s, 7185 .coprocessor = 0x1, 7186 }, { 7187 .name = "add.d", 7188 .translate = translate_add_d, 7189 .coprocessor = 0x1, 7190 }, { 7191 .name = "add.s", 7192 .translate = translate_add_s, 7193 .coprocessor = 0x1, 7194 }, { 7195 .name = "addexp.d", 7196 .translate = translate_nop, 7197 .coprocessor = 0x1, 7198 }, { 7199 .name = "addexp.s", 7200 .translate = translate_nop, 7201 .coprocessor = 0x1, 7202 }, { 7203 .name = "addexpm.d", 7204 .translate = translate_mov_s, 7205 .coprocessor = 0x1, 7206 }, { 7207 .name = "addexpm.s", 7208 .translate = translate_mov_s, 7209 .coprocessor = 0x1, 7210 }, { 7211 .name = "ceil.d", 7212 .translate = translate_ftoi_d, 7213 .par = (const uint32_t[]){float_round_up, false}, 7214 .coprocessor = 0x1, 7215 }, { 7216 .name = "ceil.s", 7217 .translate = translate_ftoi_s, 7218 .par = (const uint32_t[]){float_round_up, false}, 7219 .coprocessor = 0x1, 7220 }, { 7221 .name = "const.d", 7222 .translate = translate_const_d, 7223 .coprocessor = 0x1, 7224 }, { 7225 .name = "const.s", 7226 .translate = translate_const_s, 7227 .coprocessor = 0x1, 7228 }, { 7229 .name = "cvtd.s", 7230 .translate = translate_cvtd_s, 7231 .coprocessor = 0x1, 7232 }, { 7233 .name = "cvts.d", 7234 .translate = translate_cvts_d, 7235 .coprocessor = 0x1, 7236 }, { 7237 .name = "div0.d", 7238 .translate = translate_nop, 7239 .coprocessor = 0x1, 7240 }, { 7241 .name = "div0.s", 7242 .translate = translate_nop, 7243 .coprocessor = 0x1, 7244 }, { 7245 .name = "divn.d", 7246 .translate = translate_nop, 7247 .coprocessor = 0x1, 7248 }, { 7249 .name = "divn.s", 7250 .translate = translate_nop, 7251 .coprocessor = 0x1, 7252 }, { 7253 .name = "float.d", 7254 .translate = translate_float_d, 7255 .par = (const uint32_t[]){false}, 7256 .coprocessor = 0x1, 7257 }, { 7258 .name = "float.s", 7259 .translate = translate_float_s, 7260 .par = (const uint32_t[]){false}, 7261 .coprocessor = 0x1, 7262 }, { 7263 .name = "floor.d", 7264 .translate = translate_ftoi_d, 7265 .par = (const uint32_t[]){float_round_down, false}, 7266 .coprocessor = 0x1, 7267 }, { 7268 .name = "floor.s", 7269 .translate = translate_ftoi_s, 7270 .par = (const uint32_t[]){float_round_down, false}, 7271 .coprocessor = 0x1, 7272 }, { 7273 .name = "ldi", 7274 .translate = translate_ldsti_d, 7275 .par = (const uint32_t[]){false, true, false}, 7276 .op_flags = XTENSA_OP_LOAD, 7277 .coprocessor = 0x1, 7278 }, { 7279 .name = "ldip", 7280 .translate = translate_ldsti_d, 7281 .par = (const uint32_t[]){false, false, true}, 7282 .op_flags = XTENSA_OP_LOAD, 7283 .coprocessor = 0x1, 7284 }, { 7285 .name = "ldiu", 7286 .translate = translate_ldsti_d, 7287 .par = (const uint32_t[]){false, true, true}, 7288 .op_flags = XTENSA_OP_LOAD, 7289 .coprocessor = 0x1, 7290 }, { 7291 .name = "ldx", 7292 .translate = translate_ldstx_d, 7293 .par = (const uint32_t[]){false, true, false}, 7294 .op_flags = XTENSA_OP_LOAD, 7295 .coprocessor = 0x1, 7296 }, { 7297 .name = "ldxp", 7298 .translate = translate_ldstx_d, 7299 .par = (const uint32_t[]){false, false, true}, 7300 .op_flags = XTENSA_OP_LOAD, 7301 .coprocessor = 0x1, 7302 }, { 7303 .name = "ldxu", 7304 .translate = translate_ldstx_d, 7305 .par = (const uint32_t[]){false, true, true}, 7306 .op_flags = XTENSA_OP_LOAD, 7307 .coprocessor = 0x1, 7308 }, { 7309 .name = "lsi", 7310 .translate = translate_ldsti_s, 7311 .par = (const uint32_t[]){false, true, false}, 7312 .op_flags = XTENSA_OP_LOAD, 7313 .coprocessor = 0x1, 7314 }, { 7315 .name = "lsip", 7316 .translate = translate_ldsti_s, 7317 .par = (const uint32_t[]){false, false, true}, 7318 .op_flags = XTENSA_OP_LOAD, 7319 .coprocessor = 0x1, 7320 }, { 7321 .name = "lsiu", 7322 .translate = translate_ldsti_s, 7323 .par = (const uint32_t[]){false, true, true}, 7324 .op_flags = XTENSA_OP_LOAD, 7325 .coprocessor = 0x1, 7326 }, { 7327 .name = "lsx", 7328 .translate = translate_ldstx_s, 7329 .par = (const uint32_t[]){false, true, false}, 7330 .op_flags = XTENSA_OP_LOAD, 7331 .coprocessor = 0x1, 7332 }, { 7333 .name = "lsxp", 7334 .translate = translate_ldstx_s, 7335 .par = (const uint32_t[]){false, false, true}, 7336 .op_flags = XTENSA_OP_LOAD, 7337 .coprocessor = 0x1, 7338 }, { 7339 .name = "lsxu", 7340 .translate = translate_ldstx_s, 7341 .par = (const uint32_t[]){false, true, true}, 7342 .op_flags = XTENSA_OP_LOAD, 7343 .coprocessor = 0x1, 7344 }, { 7345 .name = "madd.d", 7346 .translate = translate_madd_d, 7347 .coprocessor = 0x1, 7348 }, { 7349 .name = "madd.s", 7350 .translate = translate_madd_s, 7351 .coprocessor = 0x1, 7352 }, { 7353 .name = "maddn.d", 7354 .translate = translate_nop, 7355 .coprocessor = 0x1, 7356 }, { 7357 .name = "maddn.s", 7358 .translate = translate_nop, 7359 .coprocessor = 0x1, 7360 }, { 7361 .name = "mkdadj.d", 7362 .translate = translate_mkdadj_d, 7363 .coprocessor = 0x1, 7364 }, { 7365 .name = "mkdadj.s", 7366 .translate = translate_mkdadj_s, 7367 .coprocessor = 0x1, 7368 }, { 7369 .name = "mksadj.d", 7370 .translate = translate_mksadj_d, 7371 .coprocessor = 0x1, 7372 }, { 7373 .name = "mksadj.s", 7374 .translate = translate_mksadj_s, 7375 .coprocessor = 0x1, 7376 }, { 7377 .name = "mov.d", 7378 .translate = translate_mov_d, 7379 .coprocessor = 0x1, 7380 }, { 7381 .name = "mov.s", 7382 .translate = translate_mov_s, 7383 .coprocessor = 0x1, 7384 }, { 7385 .name = "moveqz.d", 7386 .translate = translate_movcond_d, 7387 .par = (const uint32_t[]){TCG_COND_EQ}, 7388 .coprocessor = 0x1, 7389 }, { 7390 .name = "moveqz.s", 7391 .translate = translate_movcond_s, 7392 .par = (const uint32_t[]){TCG_COND_EQ}, 7393 .coprocessor = 0x1, 7394 }, { 7395 .name = "movf.d", 7396 .translate = translate_movp_d, 7397 .par = (const uint32_t[]){TCG_COND_EQ}, 7398 .coprocessor = 0x1, 7399 }, { 7400 .name = "movf.s", 7401 .translate = translate_movp_s, 7402 .par = (const uint32_t[]){TCG_COND_EQ}, 7403 .coprocessor = 0x1, 7404 }, { 7405 .name = "movgez.d", 7406 .translate = translate_movcond_d, 7407 .par = (const uint32_t[]){TCG_COND_GE}, 7408 .coprocessor = 0x1, 7409 }, { 7410 .name = "movgez.s", 7411 .translate = translate_movcond_s, 7412 .par = (const uint32_t[]){TCG_COND_GE}, 7413 .coprocessor = 0x1, 7414 }, { 7415 .name = "movltz.d", 7416 .translate = translate_movcond_d, 7417 .par = (const uint32_t[]){TCG_COND_LT}, 7418 .coprocessor = 0x1, 7419 }, { 7420 .name = "movltz.s", 7421 .translate = translate_movcond_s, 7422 .par = (const uint32_t[]){TCG_COND_LT}, 7423 .coprocessor = 0x1, 7424 }, { 7425 .name = "movnez.d", 7426 .translate = translate_movcond_d, 7427 .par = (const uint32_t[]){TCG_COND_NE}, 7428 .coprocessor = 0x1, 7429 }, { 7430 .name = "movnez.s", 7431 .translate = translate_movcond_s, 7432 .par = (const uint32_t[]){TCG_COND_NE}, 7433 .coprocessor = 0x1, 7434 }, { 7435 .name = "movt.d", 7436 .translate = translate_movp_d, 7437 .par = (const uint32_t[]){TCG_COND_NE}, 7438 .coprocessor = 0x1, 7439 }, { 7440 .name = "movt.s", 7441 .translate = translate_movp_s, 7442 .par = (const uint32_t[]){TCG_COND_NE}, 7443 .coprocessor = 0x1, 7444 }, { 7445 .name = "msub.d", 7446 .translate = translate_msub_d, 7447 .coprocessor = 0x1, 7448 }, { 7449 .name = "msub.s", 7450 .translate = translate_msub_s, 7451 .coprocessor = 0x1, 7452 }, { 7453 .name = "mul.d", 7454 .translate = translate_mul_d, 7455 .coprocessor = 0x1, 7456 }, { 7457 .name = "mul.s", 7458 .translate = translate_mul_s, 7459 .coprocessor = 0x1, 7460 }, { 7461 .name = "neg.d", 7462 .translate = translate_neg_d, 7463 .coprocessor = 0x1, 7464 }, { 7465 .name = "neg.s", 7466 .translate = translate_neg_s, 7467 .coprocessor = 0x1, 7468 }, { 7469 .name = "nexp01.d", 7470 .translate = translate_nop, 7471 .coprocessor = 0x1, 7472 }, { 7473 .name = "nexp01.s", 7474 .translate = translate_nop, 7475 .coprocessor = 0x1, 7476 }, { 7477 .name = "oeq.d", 7478 .translate = translate_compare_d, 7479 .par = (const uint32_t[]){COMPARE_OEQ}, 7480 .coprocessor = 0x1, 7481 }, { 7482 .name = "oeq.s", 7483 .translate = translate_compare_s, 7484 .par = (const uint32_t[]){COMPARE_OEQ}, 7485 .coprocessor = 0x1, 7486 }, { 7487 .name = "ole.d", 7488 .translate = translate_compare_d, 7489 .par = (const uint32_t[]){COMPARE_OLE}, 7490 .coprocessor = 0x1, 7491 }, { 7492 .name = "ole.s", 7493 .translate = translate_compare_s, 7494 .par = (const uint32_t[]){COMPARE_OLE}, 7495 .coprocessor = 0x1, 7496 }, { 7497 .name = "olt.d", 7498 .translate = translate_compare_d, 7499 .par = (const uint32_t[]){COMPARE_OLT}, 7500 .coprocessor = 0x1, 7501 }, { 7502 .name = "olt.s", 7503 .translate = translate_compare_s, 7504 .par = (const uint32_t[]){COMPARE_OLT}, 7505 .coprocessor = 0x1, 7506 }, { 7507 .name = "rfr", 7508 .translate = translate_rfr_s, 7509 .coprocessor = 0x1, 7510 }, { 7511 .name = "rfrd", 7512 .translate = translate_rfr_d, 7513 .coprocessor = 0x1, 7514 }, { 7515 .name = "round.d", 7516 .translate = translate_ftoi_d, 7517 .par = (const uint32_t[]){float_round_nearest_even, false}, 7518 .coprocessor = 0x1, 7519 }, { 7520 .name = "round.s", 7521 .translate = translate_ftoi_s, 7522 .par = (const uint32_t[]){float_round_nearest_even, false}, 7523 .coprocessor = 0x1, 7524 }, { 7525 .name = "rur.fcr", 7526 .translate = translate_rur, 7527 .par = (const uint32_t[]){FCR}, 7528 .coprocessor = 0x1, 7529 }, { 7530 .name = "rur.fsr", 7531 .translate = translate_rur_fpu_fsr, 7532 .coprocessor = 0x1, 7533 }, { 7534 .name = "sdi", 7535 .translate = translate_ldsti_d, 7536 .par = (const uint32_t[]){true, true, false}, 7537 .op_flags = XTENSA_OP_STORE, 7538 .coprocessor = 0x1, 7539 }, { 7540 .name = "sdip", 7541 .translate = translate_ldsti_d, 7542 .par = (const uint32_t[]){true, false, true}, 7543 .op_flags = XTENSA_OP_STORE, 7544 .coprocessor = 0x1, 7545 }, { 7546 .name = "sdiu", 7547 .translate = translate_ldsti_d, 7548 .par = (const uint32_t[]){true, true, true}, 7549 .op_flags = XTENSA_OP_STORE, 7550 .coprocessor = 0x1, 7551 }, { 7552 .name = "sdx", 7553 .translate = translate_ldstx_d, 7554 .par = (const uint32_t[]){true, true, false}, 7555 .op_flags = XTENSA_OP_STORE, 7556 .coprocessor = 0x1, 7557 }, { 7558 .name = "sdxp", 7559 .translate = translate_ldstx_d, 7560 .par = (const uint32_t[]){true, false, true}, 7561 .op_flags = XTENSA_OP_STORE, 7562 .coprocessor = 0x1, 7563 }, { 7564 .name = "sdxu", 7565 .translate = translate_ldstx_d, 7566 .par = (const uint32_t[]){true, true, true}, 7567 .op_flags = XTENSA_OP_STORE, 7568 .coprocessor = 0x1, 7569 }, { 7570 .name = "sqrt0.d", 7571 .translate = translate_nop, 7572 .coprocessor = 0x1, 7573 }, { 7574 .name = "sqrt0.s", 7575 .translate = translate_nop, 7576 .coprocessor = 0x1, 7577 }, { 7578 .name = "ssi", 7579 .translate = translate_ldsti_s, 7580 .par = (const uint32_t[]){true, true, false}, 7581 .op_flags = XTENSA_OP_STORE, 7582 .coprocessor = 0x1, 7583 }, { 7584 .name = "ssip", 7585 .translate = translate_ldsti_s, 7586 .par = (const uint32_t[]){true, false, true}, 7587 .op_flags = XTENSA_OP_STORE, 7588 .coprocessor = 0x1, 7589 }, { 7590 .name = "ssiu", 7591 .translate = translate_ldsti_s, 7592 .par = (const uint32_t[]){true, true, true}, 7593 .op_flags = XTENSA_OP_STORE, 7594 .coprocessor = 0x1, 7595 }, { 7596 .name = "ssx", 7597 .translate = translate_ldstx_s, 7598 .par = (const uint32_t[]){true, true, false}, 7599 .op_flags = XTENSA_OP_STORE, 7600 .coprocessor = 0x1, 7601 }, { 7602 .name = "ssxp", 7603 .translate = translate_ldstx_s, 7604 .par = (const uint32_t[]){true, false, true}, 7605 .op_flags = XTENSA_OP_STORE, 7606 .coprocessor = 0x1, 7607 }, { 7608 .name = "ssxu", 7609 .translate = translate_ldstx_s, 7610 .par = (const uint32_t[]){true, true, true}, 7611 .op_flags = XTENSA_OP_STORE, 7612 .coprocessor = 0x1, 7613 }, { 7614 .name = "sub.d", 7615 .translate = translate_sub_d, 7616 .coprocessor = 0x1, 7617 }, { 7618 .name = "sub.s", 7619 .translate = translate_sub_s, 7620 .coprocessor = 0x1, 7621 }, { 7622 .name = "trunc.d", 7623 .translate = translate_ftoi_d, 7624 .par = (const uint32_t[]){float_round_to_zero, false}, 7625 .coprocessor = 0x1, 7626 }, { 7627 .name = "trunc.s", 7628 .translate = translate_ftoi_s, 7629 .par = (const uint32_t[]){float_round_to_zero, false}, 7630 .coprocessor = 0x1, 7631 }, { 7632 .name = "ueq.d", 7633 .translate = translate_compare_d, 7634 .par = (const uint32_t[]){COMPARE_UEQ}, 7635 .coprocessor = 0x1, 7636 }, { 7637 .name = "ueq.s", 7638 .translate = translate_compare_s, 7639 .par = (const uint32_t[]){COMPARE_UEQ}, 7640 .coprocessor = 0x1, 7641 }, { 7642 .name = "ufloat.d", 7643 .translate = translate_float_d, 7644 .par = (const uint32_t[]){true}, 7645 .coprocessor = 0x1, 7646 }, { 7647 .name = "ufloat.s", 7648 .translate = translate_float_s, 7649 .par = (const uint32_t[]){true}, 7650 .coprocessor = 0x1, 7651 }, { 7652 .name = "ule.d", 7653 .translate = translate_compare_d, 7654 .par = (const uint32_t[]){COMPARE_ULE}, 7655 .coprocessor = 0x1, 7656 }, { 7657 .name = "ule.s", 7658 .translate = translate_compare_s, 7659 .par = (const uint32_t[]){COMPARE_ULE}, 7660 .coprocessor = 0x1, 7661 }, { 7662 .name = "ult.d", 7663 .translate = translate_compare_d, 7664 .par = (const uint32_t[]){COMPARE_ULT}, 7665 .coprocessor = 0x1, 7666 }, { 7667 .name = "ult.s", 7668 .translate = translate_compare_s, 7669 .par = (const uint32_t[]){COMPARE_ULT}, 7670 .coprocessor = 0x1, 7671 }, { 7672 .name = "un.d", 7673 .translate = translate_compare_d, 7674 .par = (const uint32_t[]){COMPARE_UN}, 7675 .coprocessor = 0x1, 7676 }, { 7677 .name = "un.s", 7678 .translate = translate_compare_s, 7679 .par = (const uint32_t[]){COMPARE_UN}, 7680 .coprocessor = 0x1, 7681 }, { 7682 .name = "utrunc.d", 7683 .translate = translate_ftoi_d, 7684 .par = (const uint32_t[]){float_round_to_zero, true}, 7685 .coprocessor = 0x1, 7686 }, { 7687 .name = "utrunc.s", 7688 .translate = translate_ftoi_s, 7689 .par = (const uint32_t[]){float_round_to_zero, true}, 7690 .coprocessor = 0x1, 7691 }, { 7692 .name = "wfr", 7693 .translate = translate_wfr_s, 7694 .coprocessor = 0x1, 7695 }, { 7696 .name = "wfrd", 7697 .translate = translate_wfr_d, 7698 .coprocessor = 0x1, 7699 }, { 7700 .name = "wur.fcr", 7701 .translate = translate_wur_fpu_fcr, 7702 .par = (const uint32_t[]){FCR}, 7703 .coprocessor = 0x1, 7704 }, { 7705 .name = "wur.fsr", 7706 .translate = translate_wur_fpu_fsr, 7707 .coprocessor = 0x1, 7708 }, 7709 }; 7710 7711 const XtensaOpcodeTranslators xtensa_fpu_opcodes = { 7712 .num_opcodes = ARRAY_SIZE(fpu_ops), 7713 .opcode = fpu_ops, 7714 }; 7715