1 /* 2 * m68k translation 3 * 4 * Copyright (c) 2005-2007 CodeSourcery 5 * Written by Paul Brook 6 * 7 * This library is free software; you can redistribute it and/or 8 * modify it under the terms of the GNU Lesser General Public 9 * License as published by the Free Software Foundation; either 10 * version 2.1 of the License, or (at your option) any later version. 11 * 12 * This library is distributed in the hope that it will be useful, 13 * but WITHOUT ANY WARRANTY; without even the implied warranty of 14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 15 * Lesser General Public License for more details. 16 * 17 * You should have received a copy of the GNU Lesser General Public 18 * License along with this library; if not, see <http://www.gnu.org/licenses/>. 19 */ 20 21 #include "qemu/osdep.h" 22 #include "cpu.h" 23 #include "disas/disas.h" 24 #include "exec/exec-all.h" 25 #include "tcg/tcg-op.h" 26 #include "qemu/log.h" 27 #include "qemu/qemu-print.h" 28 #include "exec/translator.h" 29 30 #include "exec/helper-proto.h" 31 #include "exec/helper-gen.h" 32 33 #include "exec/log.h" 34 #include "fpu/softfloat.h" 35 36 #define HELPER_H "helper.h" 37 #include "exec/helper-info.c.inc" 38 #undef HELPER_H 39 40 //#define DEBUG_DISPATCH 1 41 42 #define DEFO32(name, offset) static TCGv QREG_##name; 43 #define DEFO64(name, offset) static TCGv_i64 QREG_##name; 44 #include "qregs.h.inc" 45 #undef DEFO32 46 #undef DEFO64 47 48 static TCGv_i32 cpu_halted; 49 static TCGv_i32 cpu_exception_index; 50 51 static char cpu_reg_names[2 * 8 * 3 + 5 * 4]; 52 static TCGv cpu_dregs[8]; 53 static TCGv cpu_aregs[8]; 54 static TCGv_i64 cpu_macc[4]; 55 56 #define REG(insn, pos) (((insn) >> (pos)) & 7) 57 #define DREG(insn, pos) cpu_dregs[REG(insn, pos)] 58 #define AREG(insn, pos) get_areg(s, REG(insn, pos)) 59 #define MACREG(acc) cpu_macc[acc] 60 #define QREG_SP get_areg(s, 7) 61 62 static TCGv NULL_QREG; 63 #define IS_NULL_QREG(t) (t == NULL_QREG) 64 /* Used to distinguish stores from bad addressing modes. */ 65 static TCGv store_dummy; 66 67 void m68k_tcg_init(void) 68 { 69 char *p; 70 int i; 71 72 #define DEFO32(name, offset) \ 73 QREG_##name = tcg_global_mem_new_i32(tcg_env, \ 74 offsetof(CPUM68KState, offset), #name); 75 #define DEFO64(name, offset) \ 76 QREG_##name = tcg_global_mem_new_i64(tcg_env, \ 77 offsetof(CPUM68KState, offset), #name); 78 #include "qregs.h.inc" 79 #undef DEFO32 80 #undef DEFO64 81 82 cpu_halted = tcg_global_mem_new_i32(tcg_env, 83 -offsetof(M68kCPU, env) + 84 offsetof(CPUState, halted), "HALTED"); 85 cpu_exception_index = tcg_global_mem_new_i32(tcg_env, 86 -offsetof(M68kCPU, env) + 87 offsetof(CPUState, exception_index), 88 "EXCEPTION"); 89 90 p = cpu_reg_names; 91 for (i = 0; i < 8; i++) { 92 sprintf(p, "D%d", i); 93 cpu_dregs[i] = tcg_global_mem_new(tcg_env, 94 offsetof(CPUM68KState, dregs[i]), p); 95 p += 3; 96 sprintf(p, "A%d", i); 97 cpu_aregs[i] = tcg_global_mem_new(tcg_env, 98 offsetof(CPUM68KState, aregs[i]), p); 99 p += 3; 100 } 101 for (i = 0; i < 4; i++) { 102 sprintf(p, "ACC%d", i); 103 cpu_macc[i] = tcg_global_mem_new_i64(tcg_env, 104 offsetof(CPUM68KState, macc[i]), p); 105 p += 5; 106 } 107 108 NULL_QREG = tcg_global_mem_new(tcg_env, -4, "NULL"); 109 store_dummy = tcg_global_mem_new(tcg_env, -8, "NULL"); 110 } 111 112 /* internal defines */ 113 typedef struct DisasContext { 114 DisasContextBase base; 115 CPUM68KState *env; 116 target_ulong pc; 117 target_ulong pc_prev; 118 CCOp cc_op; /* Current CC operation */ 119 int cc_op_synced; 120 TCGv_i64 mactmp; 121 int done_mac; 122 int writeback_mask; 123 TCGv writeback[8]; 124 bool ss_active; 125 } DisasContext; 126 127 static TCGv get_areg(DisasContext *s, unsigned regno) 128 { 129 if (s->writeback_mask & (1 << regno)) { 130 return s->writeback[regno]; 131 } else { 132 return cpu_aregs[regno]; 133 } 134 } 135 136 static void delay_set_areg(DisasContext *s, unsigned regno, 137 TCGv val, bool give_temp) 138 { 139 if (s->writeback_mask & (1 << regno)) { 140 if (give_temp) { 141 s->writeback[regno] = val; 142 } else { 143 tcg_gen_mov_i32(s->writeback[regno], val); 144 } 145 } else { 146 s->writeback_mask |= 1 << regno; 147 if (give_temp) { 148 s->writeback[regno] = val; 149 } else { 150 TCGv tmp = tcg_temp_new(); 151 s->writeback[regno] = tmp; 152 tcg_gen_mov_i32(tmp, val); 153 } 154 } 155 } 156 157 static void do_writebacks(DisasContext *s) 158 { 159 unsigned mask = s->writeback_mask; 160 if (mask) { 161 s->writeback_mask = 0; 162 do { 163 unsigned regno = ctz32(mask); 164 tcg_gen_mov_i32(cpu_aregs[regno], s->writeback[regno]); 165 mask &= mask - 1; 166 } while (mask); 167 } 168 } 169 170 /* is_jmp field values */ 171 #define DISAS_JUMP DISAS_TARGET_0 /* only pc was modified dynamically */ 172 #define DISAS_EXIT DISAS_TARGET_1 /* cpu state was modified dynamically */ 173 174 #if defined(CONFIG_USER_ONLY) 175 #define IS_USER(s) 1 176 #else 177 #define IS_USER(s) (!(s->base.tb->flags & TB_FLAGS_MSR_S)) 178 #define SFC_INDEX(s) ((s->base.tb->flags & TB_FLAGS_SFC_S) ? \ 179 MMU_KERNEL_IDX : MMU_USER_IDX) 180 #define DFC_INDEX(s) ((s->base.tb->flags & TB_FLAGS_DFC_S) ? \ 181 MMU_KERNEL_IDX : MMU_USER_IDX) 182 #endif 183 184 typedef void (*disas_proc)(CPUM68KState *env, DisasContext *s, uint16_t insn); 185 186 #ifdef DEBUG_DISPATCH 187 #define DISAS_INSN(name) \ 188 static void real_disas_##name(CPUM68KState *env, DisasContext *s, \ 189 uint16_t insn); \ 190 static void disas_##name(CPUM68KState *env, DisasContext *s, \ 191 uint16_t insn) \ 192 { \ 193 qemu_log("Dispatch " #name "\n"); \ 194 real_disas_##name(env, s, insn); \ 195 } \ 196 static void real_disas_##name(CPUM68KState *env, DisasContext *s, \ 197 uint16_t insn) 198 #else 199 #define DISAS_INSN(name) \ 200 static void disas_##name(CPUM68KState *env, DisasContext *s, \ 201 uint16_t insn) 202 #endif 203 204 static const uint8_t cc_op_live[CC_OP_NB] = { 205 [CC_OP_DYNAMIC] = CCF_C | CCF_V | CCF_Z | CCF_N | CCF_X, 206 [CC_OP_FLAGS] = CCF_C | CCF_V | CCF_Z | CCF_N | CCF_X, 207 [CC_OP_ADDB ... CC_OP_ADDL] = CCF_X | CCF_N | CCF_V, 208 [CC_OP_SUBB ... CC_OP_SUBL] = CCF_X | CCF_N | CCF_V, 209 [CC_OP_CMPB ... CC_OP_CMPL] = CCF_X | CCF_N | CCF_V, 210 [CC_OP_LOGIC] = CCF_X | CCF_N 211 }; 212 213 static void set_cc_op(DisasContext *s, CCOp op) 214 { 215 CCOp old_op = s->cc_op; 216 int dead; 217 218 if (old_op == op) { 219 return; 220 } 221 s->cc_op = op; 222 s->cc_op_synced = 0; 223 224 /* 225 * Discard CC computation that will no longer be used. 226 * Note that X and N are never dead. 227 */ 228 dead = cc_op_live[old_op] & ~cc_op_live[op]; 229 if (dead & CCF_C) { 230 tcg_gen_discard_i32(QREG_CC_C); 231 } 232 if (dead & CCF_Z) { 233 tcg_gen_discard_i32(QREG_CC_Z); 234 } 235 if (dead & CCF_V) { 236 tcg_gen_discard_i32(QREG_CC_V); 237 } 238 } 239 240 /* Update the CPU env CC_OP state. */ 241 static void update_cc_op(DisasContext *s) 242 { 243 if (!s->cc_op_synced) { 244 s->cc_op_synced = 1; 245 tcg_gen_movi_i32(QREG_CC_OP, s->cc_op); 246 } 247 } 248 249 /* Generate a jump to an immediate address. */ 250 static void gen_jmp_im(DisasContext *s, uint32_t dest) 251 { 252 update_cc_op(s); 253 tcg_gen_movi_i32(QREG_PC, dest); 254 s->base.is_jmp = DISAS_JUMP; 255 } 256 257 /* Generate a jump to the address in qreg DEST. */ 258 static void gen_jmp(DisasContext *s, TCGv dest) 259 { 260 update_cc_op(s); 261 tcg_gen_mov_i32(QREG_PC, dest); 262 s->base.is_jmp = DISAS_JUMP; 263 } 264 265 static void gen_raise_exception(int nr) 266 { 267 gen_helper_raise_exception(tcg_env, tcg_constant_i32(nr)); 268 } 269 270 static void gen_raise_exception_format2(DisasContext *s, int nr, 271 target_ulong this_pc) 272 { 273 /* 274 * Pass the address of the insn to the exception handler, 275 * for recording in the Format $2 (6-word) stack frame. 276 * Re-use mmu.ar for the purpose, since that's only valid 277 * after tlb_fill. 278 */ 279 tcg_gen_st_i32(tcg_constant_i32(this_pc), tcg_env, 280 offsetof(CPUM68KState, mmu.ar)); 281 gen_raise_exception(nr); 282 s->base.is_jmp = DISAS_NORETURN; 283 } 284 285 static void gen_exception(DisasContext *s, uint32_t dest, int nr) 286 { 287 update_cc_op(s); 288 tcg_gen_movi_i32(QREG_PC, dest); 289 290 gen_raise_exception(nr); 291 292 s->base.is_jmp = DISAS_NORETURN; 293 } 294 295 static inline void gen_addr_fault(DisasContext *s) 296 { 297 gen_exception(s, s->base.pc_next, EXCP_ADDRESS); 298 } 299 300 /* 301 * Generate a load from the specified address. Narrow values are 302 * sign extended to full register width. 303 */ 304 static inline TCGv gen_load(DisasContext *s, int opsize, TCGv addr, 305 int sign, int index) 306 { 307 TCGv tmp = tcg_temp_new_i32(); 308 309 switch (opsize) { 310 case OS_BYTE: 311 case OS_WORD: 312 case OS_LONG: 313 tcg_gen_qemu_ld_tl(tmp, addr, index, 314 opsize | (sign ? MO_SIGN : 0) | MO_TE); 315 break; 316 default: 317 g_assert_not_reached(); 318 } 319 return tmp; 320 } 321 322 /* Generate a store. */ 323 static inline void gen_store(DisasContext *s, int opsize, TCGv addr, TCGv val, 324 int index) 325 { 326 switch (opsize) { 327 case OS_BYTE: 328 case OS_WORD: 329 case OS_LONG: 330 tcg_gen_qemu_st_tl(val, addr, index, opsize | MO_TE); 331 break; 332 default: 333 g_assert_not_reached(); 334 } 335 } 336 337 typedef enum { 338 EA_STORE, 339 EA_LOADU, 340 EA_LOADS 341 } ea_what; 342 343 /* 344 * Generate an unsigned load if VAL is 0 a signed load if val is -1, 345 * otherwise generate a store. 346 */ 347 static TCGv gen_ldst(DisasContext *s, int opsize, TCGv addr, TCGv val, 348 ea_what what, int index) 349 { 350 if (what == EA_STORE) { 351 gen_store(s, opsize, addr, val, index); 352 return store_dummy; 353 } else { 354 return gen_load(s, opsize, addr, what == EA_LOADS, index); 355 } 356 } 357 358 /* Read a 16-bit immediate constant */ 359 static inline uint16_t read_im16(CPUM68KState *env, DisasContext *s) 360 { 361 uint16_t im; 362 im = translator_lduw(env, &s->base, s->pc); 363 s->pc += 2; 364 return im; 365 } 366 367 /* Read an 8-bit immediate constant */ 368 static inline uint8_t read_im8(CPUM68KState *env, DisasContext *s) 369 { 370 return read_im16(env, s); 371 } 372 373 /* Read a 32-bit immediate constant. */ 374 static inline uint32_t read_im32(CPUM68KState *env, DisasContext *s) 375 { 376 uint32_t im; 377 im = read_im16(env, s) << 16; 378 im |= 0xffff & read_im16(env, s); 379 return im; 380 } 381 382 /* Read a 64-bit immediate constant. */ 383 static inline uint64_t read_im64(CPUM68KState *env, DisasContext *s) 384 { 385 uint64_t im; 386 im = (uint64_t)read_im32(env, s) << 32; 387 im |= (uint64_t)read_im32(env, s); 388 return im; 389 } 390 391 /* Calculate and address index. */ 392 static TCGv gen_addr_index(DisasContext *s, uint16_t ext, TCGv tmp) 393 { 394 TCGv add; 395 int scale; 396 397 add = (ext & 0x8000) ? AREG(ext, 12) : DREG(ext, 12); 398 if ((ext & 0x800) == 0) { 399 tcg_gen_ext16s_i32(tmp, add); 400 add = tmp; 401 } 402 scale = (ext >> 9) & 3; 403 if (scale != 0) { 404 tcg_gen_shli_i32(tmp, add, scale); 405 add = tmp; 406 } 407 return add; 408 } 409 410 /* 411 * Handle a base + index + displacement effective address. 412 * A NULL_QREG base means pc-relative. 413 */ 414 static TCGv gen_lea_indexed(CPUM68KState *env, DisasContext *s, TCGv base) 415 { 416 uint32_t offset; 417 uint16_t ext; 418 TCGv add; 419 TCGv tmp; 420 uint32_t bd, od; 421 422 offset = s->pc; 423 ext = read_im16(env, s); 424 425 if ((ext & 0x800) == 0 && !m68k_feature(s->env, M68K_FEATURE_WORD_INDEX)) 426 return NULL_QREG; 427 428 if (m68k_feature(s->env, M68K_FEATURE_M68K) && 429 !m68k_feature(s->env, M68K_FEATURE_SCALED_INDEX)) { 430 ext &= ~(3 << 9); 431 } 432 433 if (ext & 0x100) { 434 /* full extension word format */ 435 if (!m68k_feature(s->env, M68K_FEATURE_EXT_FULL)) 436 return NULL_QREG; 437 438 if ((ext & 0x30) > 0x10) { 439 /* base displacement */ 440 if ((ext & 0x30) == 0x20) { 441 bd = (int16_t)read_im16(env, s); 442 } else { 443 bd = read_im32(env, s); 444 } 445 } else { 446 bd = 0; 447 } 448 tmp = tcg_temp_new(); 449 if ((ext & 0x44) == 0) { 450 /* pre-index */ 451 add = gen_addr_index(s, ext, tmp); 452 } else { 453 add = NULL_QREG; 454 } 455 if ((ext & 0x80) == 0) { 456 /* base not suppressed */ 457 if (IS_NULL_QREG(base)) { 458 base = tcg_constant_i32(offset + bd); 459 bd = 0; 460 } 461 if (!IS_NULL_QREG(add)) { 462 tcg_gen_add_i32(tmp, add, base); 463 add = tmp; 464 } else { 465 add = base; 466 } 467 } 468 if (!IS_NULL_QREG(add)) { 469 if (bd != 0) { 470 tcg_gen_addi_i32(tmp, add, bd); 471 add = tmp; 472 } 473 } else { 474 add = tcg_constant_i32(bd); 475 } 476 if ((ext & 3) != 0) { 477 /* memory indirect */ 478 base = gen_load(s, OS_LONG, add, 0, IS_USER(s)); 479 if ((ext & 0x44) == 4) { 480 add = gen_addr_index(s, ext, tmp); 481 tcg_gen_add_i32(tmp, add, base); 482 add = tmp; 483 } else { 484 add = base; 485 } 486 if ((ext & 3) > 1) { 487 /* outer displacement */ 488 if ((ext & 3) == 2) { 489 od = (int16_t)read_im16(env, s); 490 } else { 491 od = read_im32(env, s); 492 } 493 } else { 494 od = 0; 495 } 496 if (od != 0) { 497 tcg_gen_addi_i32(tmp, add, od); 498 add = tmp; 499 } 500 } 501 } else { 502 /* brief extension word format */ 503 tmp = tcg_temp_new(); 504 add = gen_addr_index(s, ext, tmp); 505 if (!IS_NULL_QREG(base)) { 506 tcg_gen_add_i32(tmp, add, base); 507 if ((int8_t)ext) 508 tcg_gen_addi_i32(tmp, tmp, (int8_t)ext); 509 } else { 510 tcg_gen_addi_i32(tmp, add, offset + (int8_t)ext); 511 } 512 add = tmp; 513 } 514 return add; 515 } 516 517 /* Sign or zero extend a value. */ 518 519 static inline void gen_ext(TCGv res, TCGv val, int opsize, int sign) 520 { 521 switch (opsize) { 522 case OS_BYTE: 523 case OS_WORD: 524 case OS_LONG: 525 tcg_gen_ext_i32(res, val, opsize | (sign ? MO_SIGN : 0)); 526 break; 527 default: 528 g_assert_not_reached(); 529 } 530 } 531 532 /* Evaluate all the CC flags. */ 533 534 static void gen_flush_flags(DisasContext *s) 535 { 536 TCGv t0, t1; 537 538 switch (s->cc_op) { 539 case CC_OP_FLAGS: 540 return; 541 542 case CC_OP_ADDB: 543 case CC_OP_ADDW: 544 case CC_OP_ADDL: 545 tcg_gen_mov_i32(QREG_CC_C, QREG_CC_X); 546 tcg_gen_mov_i32(QREG_CC_Z, QREG_CC_N); 547 /* Compute signed overflow for addition. */ 548 t0 = tcg_temp_new(); 549 t1 = tcg_temp_new(); 550 tcg_gen_sub_i32(t0, QREG_CC_N, QREG_CC_V); 551 gen_ext(t0, t0, s->cc_op - CC_OP_ADDB, 1); 552 tcg_gen_xor_i32(t1, QREG_CC_N, QREG_CC_V); 553 tcg_gen_xor_i32(QREG_CC_V, QREG_CC_V, t0); 554 tcg_gen_andc_i32(QREG_CC_V, t1, QREG_CC_V); 555 break; 556 557 case CC_OP_SUBB: 558 case CC_OP_SUBW: 559 case CC_OP_SUBL: 560 tcg_gen_mov_i32(QREG_CC_C, QREG_CC_X); 561 tcg_gen_mov_i32(QREG_CC_Z, QREG_CC_N); 562 /* Compute signed overflow for subtraction. */ 563 t0 = tcg_temp_new(); 564 t1 = tcg_temp_new(); 565 tcg_gen_add_i32(t0, QREG_CC_N, QREG_CC_V); 566 gen_ext(t0, t0, s->cc_op - CC_OP_SUBB, 1); 567 tcg_gen_xor_i32(t1, QREG_CC_N, t0); 568 tcg_gen_xor_i32(QREG_CC_V, QREG_CC_V, t0); 569 tcg_gen_and_i32(QREG_CC_V, QREG_CC_V, t1); 570 break; 571 572 case CC_OP_CMPB: 573 case CC_OP_CMPW: 574 case CC_OP_CMPL: 575 tcg_gen_setcond_i32(TCG_COND_LTU, QREG_CC_C, QREG_CC_N, QREG_CC_V); 576 tcg_gen_sub_i32(QREG_CC_Z, QREG_CC_N, QREG_CC_V); 577 gen_ext(QREG_CC_Z, QREG_CC_Z, s->cc_op - CC_OP_CMPB, 1); 578 /* Compute signed overflow for subtraction. */ 579 t0 = tcg_temp_new(); 580 tcg_gen_xor_i32(t0, QREG_CC_Z, QREG_CC_N); 581 tcg_gen_xor_i32(QREG_CC_V, QREG_CC_V, QREG_CC_N); 582 tcg_gen_and_i32(QREG_CC_V, QREG_CC_V, t0); 583 tcg_gen_mov_i32(QREG_CC_N, QREG_CC_Z); 584 break; 585 586 case CC_OP_LOGIC: 587 tcg_gen_mov_i32(QREG_CC_Z, QREG_CC_N); 588 tcg_gen_movi_i32(QREG_CC_C, 0); 589 tcg_gen_movi_i32(QREG_CC_V, 0); 590 break; 591 592 case CC_OP_DYNAMIC: 593 gen_helper_flush_flags(tcg_env, QREG_CC_OP); 594 s->cc_op_synced = 1; 595 break; 596 597 default: 598 gen_helper_flush_flags(tcg_env, tcg_constant_i32(s->cc_op)); 599 s->cc_op_synced = 1; 600 break; 601 } 602 603 /* Note that flush_flags also assigned to env->cc_op. */ 604 s->cc_op = CC_OP_FLAGS; 605 } 606 607 static inline TCGv gen_extend(DisasContext *s, TCGv val, int opsize, int sign) 608 { 609 TCGv tmp; 610 611 if (opsize == OS_LONG) { 612 tmp = val; 613 } else { 614 tmp = tcg_temp_new(); 615 gen_ext(tmp, val, opsize, sign); 616 } 617 618 return tmp; 619 } 620 621 static void gen_logic_cc(DisasContext *s, TCGv val, int opsize) 622 { 623 gen_ext(QREG_CC_N, val, opsize, 1); 624 set_cc_op(s, CC_OP_LOGIC); 625 } 626 627 static void gen_update_cc_cmp(DisasContext *s, TCGv dest, TCGv src, int opsize) 628 { 629 tcg_gen_mov_i32(QREG_CC_N, dest); 630 tcg_gen_mov_i32(QREG_CC_V, src); 631 set_cc_op(s, CC_OP_CMPB + opsize); 632 } 633 634 static void gen_update_cc_add(TCGv dest, TCGv src, int opsize) 635 { 636 gen_ext(QREG_CC_N, dest, opsize, 1); 637 tcg_gen_mov_i32(QREG_CC_V, src); 638 } 639 640 static inline int opsize_bytes(int opsize) 641 { 642 switch (opsize) { 643 case OS_BYTE: return 1; 644 case OS_WORD: return 2; 645 case OS_LONG: return 4; 646 case OS_SINGLE: return 4; 647 case OS_DOUBLE: return 8; 648 case OS_EXTENDED: return 12; 649 case OS_PACKED: return 12; 650 default: 651 g_assert_not_reached(); 652 } 653 } 654 655 static inline int insn_opsize(int insn) 656 { 657 switch ((insn >> 6) & 3) { 658 case 0: return OS_BYTE; 659 case 1: return OS_WORD; 660 case 2: return OS_LONG; 661 default: 662 g_assert_not_reached(); 663 } 664 } 665 666 static inline int ext_opsize(int ext, int pos) 667 { 668 switch ((ext >> pos) & 7) { 669 case 0: return OS_LONG; 670 case 1: return OS_SINGLE; 671 case 2: return OS_EXTENDED; 672 case 3: return OS_PACKED; 673 case 4: return OS_WORD; 674 case 5: return OS_DOUBLE; 675 case 6: return OS_BYTE; 676 default: 677 g_assert_not_reached(); 678 } 679 } 680 681 /* 682 * Assign value to a register. If the width is less than the register width 683 * only the low part of the register is set. 684 */ 685 static void gen_partset_reg(int opsize, TCGv reg, TCGv val) 686 { 687 switch (opsize) { 688 case OS_BYTE: 689 tcg_gen_deposit_i32(reg, reg, val, 0, 8); 690 break; 691 case OS_WORD: 692 tcg_gen_deposit_i32(reg, reg, val, 0, 16); 693 break; 694 case OS_LONG: 695 case OS_SINGLE: 696 tcg_gen_mov_i32(reg, val); 697 break; 698 default: 699 g_assert_not_reached(); 700 } 701 } 702 703 /* 704 * Generate code for an "effective address". Does not adjust the base 705 * register for autoincrement addressing modes. 706 */ 707 static TCGv gen_lea_mode(CPUM68KState *env, DisasContext *s, 708 int mode, int reg0, int opsize) 709 { 710 TCGv reg; 711 TCGv tmp; 712 uint16_t ext; 713 uint32_t offset; 714 715 switch (mode) { 716 case 0: /* Data register direct. */ 717 case 1: /* Address register direct. */ 718 return NULL_QREG; 719 case 3: /* Indirect postincrement. */ 720 if (opsize == OS_UNSIZED) { 721 return NULL_QREG; 722 } 723 /* fallthru */ 724 case 2: /* Indirect register */ 725 return get_areg(s, reg0); 726 case 4: /* Indirect predecrememnt. */ 727 if (opsize == OS_UNSIZED) { 728 return NULL_QREG; 729 } 730 reg = get_areg(s, reg0); 731 tmp = tcg_temp_new(); 732 if (reg0 == 7 && opsize == OS_BYTE && 733 m68k_feature(s->env, M68K_FEATURE_M68K)) { 734 tcg_gen_subi_i32(tmp, reg, 2); 735 } else { 736 tcg_gen_subi_i32(tmp, reg, opsize_bytes(opsize)); 737 } 738 return tmp; 739 case 5: /* Indirect displacement. */ 740 reg = get_areg(s, reg0); 741 tmp = tcg_temp_new(); 742 ext = read_im16(env, s); 743 tcg_gen_addi_i32(tmp, reg, (int16_t)ext); 744 return tmp; 745 case 6: /* Indirect index + displacement. */ 746 reg = get_areg(s, reg0); 747 return gen_lea_indexed(env, s, reg); 748 case 7: /* Other */ 749 switch (reg0) { 750 case 0: /* Absolute short. */ 751 offset = (int16_t)read_im16(env, s); 752 return tcg_constant_i32(offset); 753 case 1: /* Absolute long. */ 754 offset = read_im32(env, s); 755 return tcg_constant_i32(offset); 756 case 2: /* pc displacement */ 757 offset = s->pc; 758 offset += (int16_t)read_im16(env, s); 759 return tcg_constant_i32(offset); 760 case 3: /* pc index+displacement. */ 761 return gen_lea_indexed(env, s, NULL_QREG); 762 case 4: /* Immediate. */ 763 default: 764 return NULL_QREG; 765 } 766 } 767 /* Should never happen. */ 768 return NULL_QREG; 769 } 770 771 static TCGv gen_lea(CPUM68KState *env, DisasContext *s, uint16_t insn, 772 int opsize) 773 { 774 int mode = extract32(insn, 3, 3); 775 int reg0 = REG(insn, 0); 776 return gen_lea_mode(env, s, mode, reg0, opsize); 777 } 778 779 /* 780 * Generate code to load/store a value from/into an EA. If WHAT > 0 this is 781 * a write otherwise it is a read (0 == sign extend, -1 == zero extend). 782 * ADDRP is non-null for readwrite operands. 783 */ 784 static TCGv gen_ea_mode(CPUM68KState *env, DisasContext *s, int mode, int reg0, 785 int opsize, TCGv val, TCGv *addrp, ea_what what, 786 int index) 787 { 788 TCGv reg, tmp, result; 789 int32_t offset; 790 791 switch (mode) { 792 case 0: /* Data register direct. */ 793 reg = cpu_dregs[reg0]; 794 if (what == EA_STORE) { 795 gen_partset_reg(opsize, reg, val); 796 return store_dummy; 797 } else { 798 return gen_extend(s, reg, opsize, what == EA_LOADS); 799 } 800 case 1: /* Address register direct. */ 801 reg = get_areg(s, reg0); 802 if (what == EA_STORE) { 803 tcg_gen_mov_i32(reg, val); 804 return store_dummy; 805 } else { 806 return gen_extend(s, reg, opsize, what == EA_LOADS); 807 } 808 case 2: /* Indirect register */ 809 reg = get_areg(s, reg0); 810 return gen_ldst(s, opsize, reg, val, what, index); 811 case 3: /* Indirect postincrement. */ 812 reg = get_areg(s, reg0); 813 result = gen_ldst(s, opsize, reg, val, what, index); 814 if (what == EA_STORE || !addrp) { 815 tmp = tcg_temp_new(); 816 if (reg0 == 7 && opsize == OS_BYTE && 817 m68k_feature(s->env, M68K_FEATURE_M68K)) { 818 tcg_gen_addi_i32(tmp, reg, 2); 819 } else { 820 tcg_gen_addi_i32(tmp, reg, opsize_bytes(opsize)); 821 } 822 delay_set_areg(s, reg0, tmp, true); 823 } 824 return result; 825 case 4: /* Indirect predecrememnt. */ 826 if (addrp && what == EA_STORE) { 827 tmp = *addrp; 828 } else { 829 tmp = gen_lea_mode(env, s, mode, reg0, opsize); 830 if (IS_NULL_QREG(tmp)) { 831 return tmp; 832 } 833 if (addrp) { 834 *addrp = tmp; 835 } 836 } 837 result = gen_ldst(s, opsize, tmp, val, what, index); 838 if (what == EA_STORE || !addrp) { 839 delay_set_areg(s, reg0, tmp, false); 840 } 841 return result; 842 case 5: /* Indirect displacement. */ 843 case 6: /* Indirect index + displacement. */ 844 do_indirect: 845 if (addrp && what == EA_STORE) { 846 tmp = *addrp; 847 } else { 848 tmp = gen_lea_mode(env, s, mode, reg0, opsize); 849 if (IS_NULL_QREG(tmp)) { 850 return tmp; 851 } 852 if (addrp) { 853 *addrp = tmp; 854 } 855 } 856 return gen_ldst(s, opsize, tmp, val, what, index); 857 case 7: /* Other */ 858 switch (reg0) { 859 case 0: /* Absolute short. */ 860 case 1: /* Absolute long. */ 861 case 2: /* pc displacement */ 862 case 3: /* pc index+displacement. */ 863 goto do_indirect; 864 case 4: /* Immediate. */ 865 /* Sign extend values for consistency. */ 866 switch (opsize) { 867 case OS_BYTE: 868 if (what == EA_LOADS) { 869 offset = (int8_t)read_im8(env, s); 870 } else { 871 offset = read_im8(env, s); 872 } 873 break; 874 case OS_WORD: 875 if (what == EA_LOADS) { 876 offset = (int16_t)read_im16(env, s); 877 } else { 878 offset = read_im16(env, s); 879 } 880 break; 881 case OS_LONG: 882 offset = read_im32(env, s); 883 break; 884 default: 885 g_assert_not_reached(); 886 } 887 return tcg_constant_i32(offset); 888 default: 889 return NULL_QREG; 890 } 891 } 892 /* Should never happen. */ 893 return NULL_QREG; 894 } 895 896 static TCGv gen_ea(CPUM68KState *env, DisasContext *s, uint16_t insn, 897 int opsize, TCGv val, TCGv *addrp, ea_what what, int index) 898 { 899 int mode = extract32(insn, 3, 3); 900 int reg0 = REG(insn, 0); 901 return gen_ea_mode(env, s, mode, reg0, opsize, val, addrp, what, index); 902 } 903 904 static TCGv_ptr gen_fp_ptr(int freg) 905 { 906 TCGv_ptr fp = tcg_temp_new_ptr(); 907 tcg_gen_addi_ptr(fp, tcg_env, offsetof(CPUM68KState, fregs[freg])); 908 return fp; 909 } 910 911 static TCGv_ptr gen_fp_result_ptr(void) 912 { 913 TCGv_ptr fp = tcg_temp_new_ptr(); 914 tcg_gen_addi_ptr(fp, tcg_env, offsetof(CPUM68KState, fp_result)); 915 return fp; 916 } 917 918 static void gen_fp_move(TCGv_ptr dest, TCGv_ptr src) 919 { 920 TCGv t32; 921 TCGv_i64 t64; 922 923 t32 = tcg_temp_new(); 924 tcg_gen_ld16u_i32(t32, src, offsetof(FPReg, l.upper)); 925 tcg_gen_st16_i32(t32, dest, offsetof(FPReg, l.upper)); 926 927 t64 = tcg_temp_new_i64(); 928 tcg_gen_ld_i64(t64, src, offsetof(FPReg, l.lower)); 929 tcg_gen_st_i64(t64, dest, offsetof(FPReg, l.lower)); 930 } 931 932 static void gen_load_fp(DisasContext *s, int opsize, TCGv addr, TCGv_ptr fp, 933 int index) 934 { 935 TCGv tmp; 936 TCGv_i64 t64; 937 938 t64 = tcg_temp_new_i64(); 939 tmp = tcg_temp_new(); 940 switch (opsize) { 941 case OS_BYTE: 942 case OS_WORD: 943 case OS_LONG: 944 tcg_gen_qemu_ld_tl(tmp, addr, index, opsize | MO_SIGN | MO_TE); 945 gen_helper_exts32(tcg_env, fp, tmp); 946 break; 947 case OS_SINGLE: 948 tcg_gen_qemu_ld_tl(tmp, addr, index, MO_TEUL); 949 gen_helper_extf32(tcg_env, fp, tmp); 950 break; 951 case OS_DOUBLE: 952 tcg_gen_qemu_ld_i64(t64, addr, index, MO_TEUQ); 953 gen_helper_extf64(tcg_env, fp, t64); 954 break; 955 case OS_EXTENDED: 956 if (m68k_feature(s->env, M68K_FEATURE_CF_FPU)) { 957 gen_exception(s, s->base.pc_next, EXCP_FP_UNIMP); 958 break; 959 } 960 tcg_gen_qemu_ld_i32(tmp, addr, index, MO_TEUL); 961 tcg_gen_shri_i32(tmp, tmp, 16); 962 tcg_gen_st16_i32(tmp, fp, offsetof(FPReg, l.upper)); 963 tcg_gen_addi_i32(tmp, addr, 4); 964 tcg_gen_qemu_ld_i64(t64, tmp, index, MO_TEUQ); 965 tcg_gen_st_i64(t64, fp, offsetof(FPReg, l.lower)); 966 break; 967 case OS_PACKED: 968 /* 969 * unimplemented data type on 68040/ColdFire 970 * FIXME if needed for another FPU 971 */ 972 gen_exception(s, s->base.pc_next, EXCP_FP_UNIMP); 973 break; 974 default: 975 g_assert_not_reached(); 976 } 977 } 978 979 static void gen_store_fp(DisasContext *s, int opsize, TCGv addr, TCGv_ptr fp, 980 int index) 981 { 982 TCGv tmp; 983 TCGv_i64 t64; 984 985 t64 = tcg_temp_new_i64(); 986 tmp = tcg_temp_new(); 987 switch (opsize) { 988 case OS_BYTE: 989 case OS_WORD: 990 case OS_LONG: 991 gen_helper_reds32(tmp, tcg_env, fp); 992 tcg_gen_qemu_st_tl(tmp, addr, index, opsize | MO_TE); 993 break; 994 case OS_SINGLE: 995 gen_helper_redf32(tmp, tcg_env, fp); 996 tcg_gen_qemu_st_tl(tmp, addr, index, MO_TEUL); 997 break; 998 case OS_DOUBLE: 999 gen_helper_redf64(t64, tcg_env, fp); 1000 tcg_gen_qemu_st_i64(t64, addr, index, MO_TEUQ); 1001 break; 1002 case OS_EXTENDED: 1003 if (m68k_feature(s->env, M68K_FEATURE_CF_FPU)) { 1004 gen_exception(s, s->base.pc_next, EXCP_FP_UNIMP); 1005 break; 1006 } 1007 tcg_gen_ld16u_i32(tmp, fp, offsetof(FPReg, l.upper)); 1008 tcg_gen_shli_i32(tmp, tmp, 16); 1009 tcg_gen_qemu_st_i32(tmp, addr, index, MO_TEUL); 1010 tcg_gen_addi_i32(tmp, addr, 4); 1011 tcg_gen_ld_i64(t64, fp, offsetof(FPReg, l.lower)); 1012 tcg_gen_qemu_st_i64(t64, tmp, index, MO_TEUQ); 1013 break; 1014 case OS_PACKED: 1015 /* 1016 * unimplemented data type on 68040/ColdFire 1017 * FIXME if needed for another FPU 1018 */ 1019 gen_exception(s, s->base.pc_next, EXCP_FP_UNIMP); 1020 break; 1021 default: 1022 g_assert_not_reached(); 1023 } 1024 } 1025 1026 static void gen_ldst_fp(DisasContext *s, int opsize, TCGv addr, 1027 TCGv_ptr fp, ea_what what, int index) 1028 { 1029 if (what == EA_STORE) { 1030 gen_store_fp(s, opsize, addr, fp, index); 1031 } else { 1032 gen_load_fp(s, opsize, addr, fp, index); 1033 } 1034 } 1035 1036 static int gen_ea_mode_fp(CPUM68KState *env, DisasContext *s, int mode, 1037 int reg0, int opsize, TCGv_ptr fp, ea_what what, 1038 int index) 1039 { 1040 TCGv reg, addr, tmp; 1041 TCGv_i64 t64; 1042 1043 switch (mode) { 1044 case 0: /* Data register direct. */ 1045 reg = cpu_dregs[reg0]; 1046 if (what == EA_STORE) { 1047 switch (opsize) { 1048 case OS_BYTE: 1049 case OS_WORD: 1050 case OS_LONG: 1051 gen_helper_reds32(reg, tcg_env, fp); 1052 break; 1053 case OS_SINGLE: 1054 gen_helper_redf32(reg, tcg_env, fp); 1055 break; 1056 default: 1057 g_assert_not_reached(); 1058 } 1059 } else { 1060 tmp = tcg_temp_new(); 1061 switch (opsize) { 1062 case OS_BYTE: 1063 case OS_WORD: 1064 case OS_LONG: 1065 tcg_gen_ext_i32(tmp, reg, opsize | MO_SIGN); 1066 gen_helper_exts32(tcg_env, fp, tmp); 1067 break; 1068 case OS_SINGLE: 1069 gen_helper_extf32(tcg_env, fp, reg); 1070 break; 1071 default: 1072 g_assert_not_reached(); 1073 } 1074 } 1075 return 0; 1076 case 1: /* Address register direct. */ 1077 return -1; 1078 case 2: /* Indirect register */ 1079 addr = get_areg(s, reg0); 1080 gen_ldst_fp(s, opsize, addr, fp, what, index); 1081 return 0; 1082 case 3: /* Indirect postincrement. */ 1083 addr = cpu_aregs[reg0]; 1084 gen_ldst_fp(s, opsize, addr, fp, what, index); 1085 tcg_gen_addi_i32(addr, addr, opsize_bytes(opsize)); 1086 return 0; 1087 case 4: /* Indirect predecrememnt. */ 1088 addr = gen_lea_mode(env, s, mode, reg0, opsize); 1089 if (IS_NULL_QREG(addr)) { 1090 return -1; 1091 } 1092 gen_ldst_fp(s, opsize, addr, fp, what, index); 1093 tcg_gen_mov_i32(cpu_aregs[reg0], addr); 1094 return 0; 1095 case 5: /* Indirect displacement. */ 1096 case 6: /* Indirect index + displacement. */ 1097 do_indirect: 1098 addr = gen_lea_mode(env, s, mode, reg0, opsize); 1099 if (IS_NULL_QREG(addr)) { 1100 return -1; 1101 } 1102 gen_ldst_fp(s, opsize, addr, fp, what, index); 1103 return 0; 1104 case 7: /* Other */ 1105 switch (reg0) { 1106 case 0: /* Absolute short. */ 1107 case 1: /* Absolute long. */ 1108 case 2: /* pc displacement */ 1109 case 3: /* pc index+displacement. */ 1110 goto do_indirect; 1111 case 4: /* Immediate. */ 1112 if (what == EA_STORE) { 1113 return -1; 1114 } 1115 switch (opsize) { 1116 case OS_BYTE: 1117 tmp = tcg_constant_i32((int8_t)read_im8(env, s)); 1118 gen_helper_exts32(tcg_env, fp, tmp); 1119 break; 1120 case OS_WORD: 1121 tmp = tcg_constant_i32((int16_t)read_im16(env, s)); 1122 gen_helper_exts32(tcg_env, fp, tmp); 1123 break; 1124 case OS_LONG: 1125 tmp = tcg_constant_i32(read_im32(env, s)); 1126 gen_helper_exts32(tcg_env, fp, tmp); 1127 break; 1128 case OS_SINGLE: 1129 tmp = tcg_constant_i32(read_im32(env, s)); 1130 gen_helper_extf32(tcg_env, fp, tmp); 1131 break; 1132 case OS_DOUBLE: 1133 t64 = tcg_constant_i64(read_im64(env, s)); 1134 gen_helper_extf64(tcg_env, fp, t64); 1135 break; 1136 case OS_EXTENDED: 1137 if (m68k_feature(s->env, M68K_FEATURE_CF_FPU)) { 1138 gen_exception(s, s->base.pc_next, EXCP_FP_UNIMP); 1139 break; 1140 } 1141 tmp = tcg_constant_i32(read_im32(env, s) >> 16); 1142 tcg_gen_st16_i32(tmp, fp, offsetof(FPReg, l.upper)); 1143 t64 = tcg_constant_i64(read_im64(env, s)); 1144 tcg_gen_st_i64(t64, fp, offsetof(FPReg, l.lower)); 1145 break; 1146 case OS_PACKED: 1147 /* 1148 * unimplemented data type on 68040/ColdFire 1149 * FIXME if needed for another FPU 1150 */ 1151 gen_exception(s, s->base.pc_next, EXCP_FP_UNIMP); 1152 break; 1153 default: 1154 g_assert_not_reached(); 1155 } 1156 return 0; 1157 default: 1158 return -1; 1159 } 1160 } 1161 return -1; 1162 } 1163 1164 static int gen_ea_fp(CPUM68KState *env, DisasContext *s, uint16_t insn, 1165 int opsize, TCGv_ptr fp, ea_what what, int index) 1166 { 1167 int mode = extract32(insn, 3, 3); 1168 int reg0 = REG(insn, 0); 1169 return gen_ea_mode_fp(env, s, mode, reg0, opsize, fp, what, index); 1170 } 1171 1172 typedef struct { 1173 TCGCond tcond; 1174 TCGv v1; 1175 TCGv v2; 1176 } DisasCompare; 1177 1178 static void gen_cc_cond(DisasCompare *c, DisasContext *s, int cond) 1179 { 1180 TCGv tmp, tmp2; 1181 TCGCond tcond; 1182 CCOp op = s->cc_op; 1183 1184 /* The CC_OP_CMP form can handle most normal comparisons directly. */ 1185 if (op == CC_OP_CMPB || op == CC_OP_CMPW || op == CC_OP_CMPL) { 1186 c->v1 = QREG_CC_N; 1187 c->v2 = QREG_CC_V; 1188 switch (cond) { 1189 case 2: /* HI */ 1190 case 3: /* LS */ 1191 tcond = TCG_COND_LEU; 1192 goto done; 1193 case 4: /* CC */ 1194 case 5: /* CS */ 1195 tcond = TCG_COND_LTU; 1196 goto done; 1197 case 6: /* NE */ 1198 case 7: /* EQ */ 1199 tcond = TCG_COND_EQ; 1200 goto done; 1201 case 10: /* PL */ 1202 case 11: /* MI */ 1203 c->v2 = tcg_constant_i32(0); 1204 c->v1 = tmp = tcg_temp_new(); 1205 tcg_gen_sub_i32(tmp, QREG_CC_N, QREG_CC_V); 1206 gen_ext(tmp, tmp, op - CC_OP_CMPB, 1); 1207 /* fallthru */ 1208 case 12: /* GE */ 1209 case 13: /* LT */ 1210 tcond = TCG_COND_LT; 1211 goto done; 1212 case 14: /* GT */ 1213 case 15: /* LE */ 1214 tcond = TCG_COND_LE; 1215 goto done; 1216 } 1217 } 1218 1219 c->v2 = tcg_constant_i32(0); 1220 1221 switch (cond) { 1222 case 0: /* T */ 1223 case 1: /* F */ 1224 c->v1 = c->v2; 1225 tcond = TCG_COND_NEVER; 1226 goto done; 1227 case 14: /* GT (!(Z || (N ^ V))) */ 1228 case 15: /* LE (Z || (N ^ V)) */ 1229 /* 1230 * Logic operations clear V, which simplifies LE to (Z || N), 1231 * and since Z and N are co-located, this becomes a normal 1232 * comparison vs N. 1233 */ 1234 if (op == CC_OP_LOGIC) { 1235 c->v1 = QREG_CC_N; 1236 tcond = TCG_COND_LE; 1237 goto done; 1238 } 1239 break; 1240 case 12: /* GE (!(N ^ V)) */ 1241 case 13: /* LT (N ^ V) */ 1242 /* Logic operations clear V, which simplifies this to N. */ 1243 if (op != CC_OP_LOGIC) { 1244 break; 1245 } 1246 /* fallthru */ 1247 case 10: /* PL (!N) */ 1248 case 11: /* MI (N) */ 1249 /* Several cases represent N normally. */ 1250 if (op == CC_OP_ADDB || op == CC_OP_ADDW || op == CC_OP_ADDL || 1251 op == CC_OP_SUBB || op == CC_OP_SUBW || op == CC_OP_SUBL || 1252 op == CC_OP_LOGIC) { 1253 c->v1 = QREG_CC_N; 1254 tcond = TCG_COND_LT; 1255 goto done; 1256 } 1257 break; 1258 case 6: /* NE (!Z) */ 1259 case 7: /* EQ (Z) */ 1260 /* Some cases fold Z into N. */ 1261 if (op == CC_OP_ADDB || op == CC_OP_ADDW || op == CC_OP_ADDL || 1262 op == CC_OP_SUBB || op == CC_OP_SUBW || op == CC_OP_SUBL || 1263 op == CC_OP_LOGIC) { 1264 tcond = TCG_COND_EQ; 1265 c->v1 = QREG_CC_N; 1266 goto done; 1267 } 1268 break; 1269 case 4: /* CC (!C) */ 1270 case 5: /* CS (C) */ 1271 /* Some cases fold C into X. */ 1272 if (op == CC_OP_ADDB || op == CC_OP_ADDW || op == CC_OP_ADDL || 1273 op == CC_OP_SUBB || op == CC_OP_SUBW || op == CC_OP_SUBL) { 1274 tcond = TCG_COND_NE; 1275 c->v1 = QREG_CC_X; 1276 goto done; 1277 } 1278 /* fallthru */ 1279 case 8: /* VC (!V) */ 1280 case 9: /* VS (V) */ 1281 /* Logic operations clear V and C. */ 1282 if (op == CC_OP_LOGIC) { 1283 tcond = TCG_COND_NEVER; 1284 c->v1 = c->v2; 1285 goto done; 1286 } 1287 break; 1288 } 1289 1290 /* Otherwise, flush flag state to CC_OP_FLAGS. */ 1291 gen_flush_flags(s); 1292 1293 switch (cond) { 1294 case 0: /* T */ 1295 case 1: /* F */ 1296 default: 1297 /* Invalid, or handled above. */ 1298 abort(); 1299 case 2: /* HI (!C && !Z) -> !(C || Z)*/ 1300 case 3: /* LS (C || Z) */ 1301 c->v1 = tmp = tcg_temp_new(); 1302 tcg_gen_setcond_i32(TCG_COND_EQ, tmp, QREG_CC_Z, c->v2); 1303 tcg_gen_or_i32(tmp, tmp, QREG_CC_C); 1304 tcond = TCG_COND_NE; 1305 break; 1306 case 4: /* CC (!C) */ 1307 case 5: /* CS (C) */ 1308 c->v1 = QREG_CC_C; 1309 tcond = TCG_COND_NE; 1310 break; 1311 case 6: /* NE (!Z) */ 1312 case 7: /* EQ (Z) */ 1313 c->v1 = QREG_CC_Z; 1314 tcond = TCG_COND_EQ; 1315 break; 1316 case 8: /* VC (!V) */ 1317 case 9: /* VS (V) */ 1318 c->v1 = QREG_CC_V; 1319 tcond = TCG_COND_LT; 1320 break; 1321 case 10: /* PL (!N) */ 1322 case 11: /* MI (N) */ 1323 c->v1 = QREG_CC_N; 1324 tcond = TCG_COND_LT; 1325 break; 1326 case 12: /* GE (!(N ^ V)) */ 1327 case 13: /* LT (N ^ V) */ 1328 c->v1 = tmp = tcg_temp_new(); 1329 tcg_gen_xor_i32(tmp, QREG_CC_N, QREG_CC_V); 1330 tcond = TCG_COND_LT; 1331 break; 1332 case 14: /* GT (!(Z || (N ^ V))) */ 1333 case 15: /* LE (Z || (N ^ V)) */ 1334 c->v1 = tmp = tcg_temp_new(); 1335 tcg_gen_negsetcond_i32(TCG_COND_EQ, tmp, QREG_CC_Z, c->v2); 1336 tmp2 = tcg_temp_new(); 1337 tcg_gen_xor_i32(tmp2, QREG_CC_N, QREG_CC_V); 1338 tcg_gen_or_i32(tmp, tmp, tmp2); 1339 tcond = TCG_COND_LT; 1340 break; 1341 } 1342 1343 done: 1344 if ((cond & 1) == 0) { 1345 tcond = tcg_invert_cond(tcond); 1346 } 1347 c->tcond = tcond; 1348 } 1349 1350 static void gen_jmpcc(DisasContext *s, int cond, TCGLabel *l1) 1351 { 1352 DisasCompare c; 1353 1354 gen_cc_cond(&c, s, cond); 1355 update_cc_op(s); 1356 tcg_gen_brcond_i32(c.tcond, c.v1, c.v2, l1); 1357 } 1358 1359 /* Force a TB lookup after an instruction that changes the CPU state. */ 1360 static void gen_exit_tb(DisasContext *s) 1361 { 1362 update_cc_op(s); 1363 tcg_gen_movi_i32(QREG_PC, s->pc); 1364 s->base.is_jmp = DISAS_EXIT; 1365 } 1366 1367 #define SRC_EA(env, result, opsize, op_sign, addrp) do { \ 1368 result = gen_ea(env, s, insn, opsize, NULL_QREG, addrp, \ 1369 op_sign ? EA_LOADS : EA_LOADU, IS_USER(s)); \ 1370 if (IS_NULL_QREG(result)) { \ 1371 gen_addr_fault(s); \ 1372 return; \ 1373 } \ 1374 } while (0) 1375 1376 #define DEST_EA(env, insn, opsize, val, addrp) do { \ 1377 TCGv ea_result = gen_ea(env, s, insn, opsize, val, addrp, \ 1378 EA_STORE, IS_USER(s)); \ 1379 if (IS_NULL_QREG(ea_result)) { \ 1380 gen_addr_fault(s); \ 1381 return; \ 1382 } \ 1383 } while (0) 1384 1385 /* Generate a jump to an immediate address. */ 1386 static void gen_jmp_tb(DisasContext *s, int n, target_ulong dest, 1387 target_ulong src) 1388 { 1389 if (unlikely(s->ss_active)) { 1390 update_cc_op(s); 1391 tcg_gen_movi_i32(QREG_PC, dest); 1392 gen_raise_exception_format2(s, EXCP_TRACE, src); 1393 } else if (translator_use_goto_tb(&s->base, dest)) { 1394 tcg_gen_goto_tb(n); 1395 tcg_gen_movi_i32(QREG_PC, dest); 1396 tcg_gen_exit_tb(s->base.tb, n); 1397 } else { 1398 gen_jmp_im(s, dest); 1399 tcg_gen_exit_tb(NULL, 0); 1400 } 1401 s->base.is_jmp = DISAS_NORETURN; 1402 } 1403 1404 DISAS_INSN(scc) 1405 { 1406 DisasCompare c; 1407 int cond; 1408 TCGv tmp; 1409 1410 cond = (insn >> 8) & 0xf; 1411 gen_cc_cond(&c, s, cond); 1412 1413 tmp = tcg_temp_new(); 1414 tcg_gen_negsetcond_i32(c.tcond, tmp, c.v1, c.v2); 1415 1416 DEST_EA(env, insn, OS_BYTE, tmp, NULL); 1417 } 1418 1419 DISAS_INSN(dbcc) 1420 { 1421 TCGLabel *l1; 1422 TCGv reg; 1423 TCGv tmp; 1424 int16_t offset; 1425 uint32_t base; 1426 1427 reg = DREG(insn, 0); 1428 base = s->pc; 1429 offset = (int16_t)read_im16(env, s); 1430 l1 = gen_new_label(); 1431 gen_jmpcc(s, (insn >> 8) & 0xf, l1); 1432 1433 tmp = tcg_temp_new(); 1434 tcg_gen_ext16s_i32(tmp, reg); 1435 tcg_gen_addi_i32(tmp, tmp, -1); 1436 gen_partset_reg(OS_WORD, reg, tmp); 1437 tcg_gen_brcondi_i32(TCG_COND_EQ, tmp, -1, l1); 1438 gen_jmp_tb(s, 1, base + offset, s->base.pc_next); 1439 gen_set_label(l1); 1440 gen_jmp_tb(s, 0, s->pc, s->base.pc_next); 1441 } 1442 1443 DISAS_INSN(undef_mac) 1444 { 1445 gen_exception(s, s->base.pc_next, EXCP_LINEA); 1446 } 1447 1448 DISAS_INSN(undef_fpu) 1449 { 1450 gen_exception(s, s->base.pc_next, EXCP_LINEF); 1451 } 1452 1453 DISAS_INSN(undef) 1454 { 1455 /* 1456 * ??? This is both instructions that are as yet unimplemented 1457 * for the 680x0 series, as well as those that are implemented 1458 * but actually illegal for CPU32 or pre-68020. 1459 */ 1460 qemu_log_mask(LOG_UNIMP, "Illegal instruction: %04x @ %08x\n", 1461 insn, s->base.pc_next); 1462 gen_exception(s, s->base.pc_next, EXCP_ILLEGAL); 1463 } 1464 1465 DISAS_INSN(mulw) 1466 { 1467 TCGv reg; 1468 TCGv tmp; 1469 TCGv src; 1470 int sign; 1471 1472 sign = (insn & 0x100) != 0; 1473 reg = DREG(insn, 9); 1474 tmp = tcg_temp_new(); 1475 if (sign) 1476 tcg_gen_ext16s_i32(tmp, reg); 1477 else 1478 tcg_gen_ext16u_i32(tmp, reg); 1479 SRC_EA(env, src, OS_WORD, sign, NULL); 1480 tcg_gen_mul_i32(tmp, tmp, src); 1481 tcg_gen_mov_i32(reg, tmp); 1482 gen_logic_cc(s, tmp, OS_LONG); 1483 } 1484 1485 DISAS_INSN(divw) 1486 { 1487 int sign; 1488 TCGv src; 1489 TCGv destr; 1490 TCGv ilen; 1491 1492 /* divX.w <EA>,Dn 32/16 -> 16r:16q */ 1493 1494 sign = (insn & 0x100) != 0; 1495 1496 /* dest.l / src.w */ 1497 1498 SRC_EA(env, src, OS_WORD, sign, NULL); 1499 destr = tcg_constant_i32(REG(insn, 9)); 1500 ilen = tcg_constant_i32(s->pc - s->base.pc_next); 1501 if (sign) { 1502 gen_helper_divsw(tcg_env, destr, src, ilen); 1503 } else { 1504 gen_helper_divuw(tcg_env, destr, src, ilen); 1505 } 1506 1507 set_cc_op(s, CC_OP_FLAGS); 1508 } 1509 1510 DISAS_INSN(divl) 1511 { 1512 TCGv num, reg, den, ilen; 1513 int sign; 1514 uint16_t ext; 1515 1516 ext = read_im16(env, s); 1517 1518 sign = (ext & 0x0800) != 0; 1519 1520 if (ext & 0x400) { 1521 if (!m68k_feature(s->env, M68K_FEATURE_QUAD_MULDIV)) { 1522 gen_exception(s, s->base.pc_next, EXCP_ILLEGAL); 1523 return; 1524 } 1525 1526 /* divX.l <EA>, Dr:Dq 64/32 -> 32r:32q */ 1527 1528 SRC_EA(env, den, OS_LONG, 0, NULL); 1529 num = tcg_constant_i32(REG(ext, 12)); 1530 reg = tcg_constant_i32(REG(ext, 0)); 1531 ilen = tcg_constant_i32(s->pc - s->base.pc_next); 1532 if (sign) { 1533 gen_helper_divsll(tcg_env, num, reg, den, ilen); 1534 } else { 1535 gen_helper_divull(tcg_env, num, reg, den, ilen); 1536 } 1537 set_cc_op(s, CC_OP_FLAGS); 1538 return; 1539 } 1540 1541 /* divX.l <EA>, Dq 32/32 -> 32q */ 1542 /* divXl.l <EA>, Dr:Dq 32/32 -> 32r:32q */ 1543 1544 SRC_EA(env, den, OS_LONG, 0, NULL); 1545 num = tcg_constant_i32(REG(ext, 12)); 1546 reg = tcg_constant_i32(REG(ext, 0)); 1547 ilen = tcg_constant_i32(s->pc - s->base.pc_next); 1548 if (sign) { 1549 gen_helper_divsl(tcg_env, num, reg, den, ilen); 1550 } else { 1551 gen_helper_divul(tcg_env, num, reg, den, ilen); 1552 } 1553 1554 set_cc_op(s, CC_OP_FLAGS); 1555 } 1556 1557 static void bcd_add(TCGv dest, TCGv src) 1558 { 1559 TCGv t0, t1; 1560 1561 /* 1562 * dest10 = dest10 + src10 + X 1563 * 1564 * t1 = src 1565 * t2 = t1 + 0x066 1566 * t3 = t2 + dest + X 1567 * t4 = t2 ^ dest 1568 * t5 = t3 ^ t4 1569 * t6 = ~t5 & 0x110 1570 * t7 = (t6 >> 2) | (t6 >> 3) 1571 * return t3 - t7 1572 */ 1573 1574 /* 1575 * t1 = (src + 0x066) + dest + X 1576 * = result with some possible exceeding 0x6 1577 */ 1578 1579 t0 = tcg_temp_new(); 1580 tcg_gen_addi_i32(t0, src, 0x066); 1581 1582 t1 = tcg_temp_new(); 1583 tcg_gen_add_i32(t1, t0, dest); 1584 tcg_gen_add_i32(t1, t1, QREG_CC_X); 1585 1586 /* we will remove exceeding 0x6 where there is no carry */ 1587 1588 /* 1589 * t0 = (src + 0x0066) ^ dest 1590 * = t1 without carries 1591 */ 1592 1593 tcg_gen_xor_i32(t0, t0, dest); 1594 1595 /* 1596 * extract the carries 1597 * t0 = t0 ^ t1 1598 * = only the carries 1599 */ 1600 1601 tcg_gen_xor_i32(t0, t0, t1); 1602 1603 /* 1604 * generate 0x1 where there is no carry 1605 * and for each 0x10, generate a 0x6 1606 */ 1607 1608 tcg_gen_shri_i32(t0, t0, 3); 1609 tcg_gen_not_i32(t0, t0); 1610 tcg_gen_andi_i32(t0, t0, 0x22); 1611 tcg_gen_add_i32(dest, t0, t0); 1612 tcg_gen_add_i32(dest, dest, t0); 1613 1614 /* 1615 * remove the exceeding 0x6 1616 * for digits that have not generated a carry 1617 */ 1618 1619 tcg_gen_sub_i32(dest, t1, dest); 1620 } 1621 1622 static void bcd_sub(TCGv dest, TCGv src) 1623 { 1624 TCGv t0, t1, t2; 1625 1626 /* 1627 * dest10 = dest10 - src10 - X 1628 * = bcd_add(dest + 1 - X, 0x199 - src) 1629 */ 1630 1631 /* t0 = 0x066 + (0x199 - src) */ 1632 1633 t0 = tcg_temp_new(); 1634 tcg_gen_subfi_i32(t0, 0x1ff, src); 1635 1636 /* t1 = t0 + dest + 1 - X*/ 1637 1638 t1 = tcg_temp_new(); 1639 tcg_gen_add_i32(t1, t0, dest); 1640 tcg_gen_addi_i32(t1, t1, 1); 1641 tcg_gen_sub_i32(t1, t1, QREG_CC_X); 1642 1643 /* t2 = t0 ^ dest */ 1644 1645 t2 = tcg_temp_new(); 1646 tcg_gen_xor_i32(t2, t0, dest); 1647 1648 /* t0 = t1 ^ t2 */ 1649 1650 tcg_gen_xor_i32(t0, t1, t2); 1651 1652 /* 1653 * t2 = ~t0 & 0x110 1654 * t0 = (t2 >> 2) | (t2 >> 3) 1655 * 1656 * to fit on 8bit operands, changed in: 1657 * 1658 * t2 = ~(t0 >> 3) & 0x22 1659 * t0 = t2 + t2 1660 * t0 = t0 + t2 1661 */ 1662 1663 tcg_gen_shri_i32(t2, t0, 3); 1664 tcg_gen_not_i32(t2, t2); 1665 tcg_gen_andi_i32(t2, t2, 0x22); 1666 tcg_gen_add_i32(t0, t2, t2); 1667 tcg_gen_add_i32(t0, t0, t2); 1668 1669 /* return t1 - t0 */ 1670 1671 tcg_gen_sub_i32(dest, t1, t0); 1672 } 1673 1674 static void bcd_flags(TCGv val) 1675 { 1676 tcg_gen_andi_i32(QREG_CC_C, val, 0x0ff); 1677 tcg_gen_or_i32(QREG_CC_Z, QREG_CC_Z, QREG_CC_C); 1678 1679 tcg_gen_extract_i32(QREG_CC_C, val, 8, 1); 1680 1681 tcg_gen_mov_i32(QREG_CC_X, QREG_CC_C); 1682 } 1683 1684 DISAS_INSN(abcd_reg) 1685 { 1686 TCGv src; 1687 TCGv dest; 1688 1689 gen_flush_flags(s); /* !Z is sticky */ 1690 1691 src = gen_extend(s, DREG(insn, 0), OS_BYTE, 0); 1692 dest = gen_extend(s, DREG(insn, 9), OS_BYTE, 0); 1693 bcd_add(dest, src); 1694 gen_partset_reg(OS_BYTE, DREG(insn, 9), dest); 1695 1696 bcd_flags(dest); 1697 } 1698 1699 DISAS_INSN(abcd_mem) 1700 { 1701 TCGv src, dest, addr; 1702 1703 gen_flush_flags(s); /* !Z is sticky */ 1704 1705 /* Indirect pre-decrement load (mode 4) */ 1706 1707 src = gen_ea_mode(env, s, 4, REG(insn, 0), OS_BYTE, 1708 NULL_QREG, NULL, EA_LOADU, IS_USER(s)); 1709 dest = gen_ea_mode(env, s, 4, REG(insn, 9), OS_BYTE, 1710 NULL_QREG, &addr, EA_LOADU, IS_USER(s)); 1711 1712 bcd_add(dest, src); 1713 1714 gen_ea_mode(env, s, 4, REG(insn, 9), OS_BYTE, dest, &addr, 1715 EA_STORE, IS_USER(s)); 1716 1717 bcd_flags(dest); 1718 } 1719 1720 DISAS_INSN(sbcd_reg) 1721 { 1722 TCGv src, dest; 1723 1724 gen_flush_flags(s); /* !Z is sticky */ 1725 1726 src = gen_extend(s, DREG(insn, 0), OS_BYTE, 0); 1727 dest = gen_extend(s, DREG(insn, 9), OS_BYTE, 0); 1728 1729 bcd_sub(dest, src); 1730 1731 gen_partset_reg(OS_BYTE, DREG(insn, 9), dest); 1732 1733 bcd_flags(dest); 1734 } 1735 1736 DISAS_INSN(sbcd_mem) 1737 { 1738 TCGv src, dest, addr; 1739 1740 gen_flush_flags(s); /* !Z is sticky */ 1741 1742 /* Indirect pre-decrement load (mode 4) */ 1743 1744 src = gen_ea_mode(env, s, 4, REG(insn, 0), OS_BYTE, 1745 NULL_QREG, NULL, EA_LOADU, IS_USER(s)); 1746 dest = gen_ea_mode(env, s, 4, REG(insn, 9), OS_BYTE, 1747 NULL_QREG, &addr, EA_LOADU, IS_USER(s)); 1748 1749 bcd_sub(dest, src); 1750 1751 gen_ea_mode(env, s, 4, REG(insn, 9), OS_BYTE, dest, &addr, 1752 EA_STORE, IS_USER(s)); 1753 1754 bcd_flags(dest); 1755 } 1756 1757 DISAS_INSN(nbcd) 1758 { 1759 TCGv src, dest; 1760 TCGv addr; 1761 1762 gen_flush_flags(s); /* !Z is sticky */ 1763 1764 SRC_EA(env, src, OS_BYTE, 0, &addr); 1765 1766 dest = tcg_temp_new(); 1767 tcg_gen_movi_i32(dest, 0); 1768 bcd_sub(dest, src); 1769 1770 DEST_EA(env, insn, OS_BYTE, dest, &addr); 1771 1772 bcd_flags(dest); 1773 } 1774 1775 DISAS_INSN(addsub) 1776 { 1777 TCGv reg; 1778 TCGv dest; 1779 TCGv src; 1780 TCGv tmp; 1781 TCGv addr; 1782 int add; 1783 int opsize; 1784 1785 add = (insn & 0x4000) != 0; 1786 opsize = insn_opsize(insn); 1787 reg = gen_extend(s, DREG(insn, 9), opsize, 1); 1788 dest = tcg_temp_new(); 1789 if (insn & 0x100) { 1790 SRC_EA(env, tmp, opsize, 1, &addr); 1791 src = reg; 1792 } else { 1793 tmp = reg; 1794 SRC_EA(env, src, opsize, 1, NULL); 1795 } 1796 if (add) { 1797 tcg_gen_add_i32(dest, tmp, src); 1798 tcg_gen_setcond_i32(TCG_COND_LTU, QREG_CC_X, dest, src); 1799 set_cc_op(s, CC_OP_ADDB + opsize); 1800 } else { 1801 tcg_gen_setcond_i32(TCG_COND_LTU, QREG_CC_X, tmp, src); 1802 tcg_gen_sub_i32(dest, tmp, src); 1803 set_cc_op(s, CC_OP_SUBB + opsize); 1804 } 1805 gen_update_cc_add(dest, src, opsize); 1806 if (insn & 0x100) { 1807 DEST_EA(env, insn, opsize, dest, &addr); 1808 } else { 1809 gen_partset_reg(opsize, DREG(insn, 9), dest); 1810 } 1811 } 1812 1813 /* Reverse the order of the bits in REG. */ 1814 DISAS_INSN(bitrev) 1815 { 1816 TCGv reg; 1817 reg = DREG(insn, 0); 1818 gen_helper_bitrev(reg, reg); 1819 } 1820 1821 DISAS_INSN(bitop_reg) 1822 { 1823 int opsize; 1824 int op; 1825 TCGv src1; 1826 TCGv src2; 1827 TCGv tmp; 1828 TCGv addr; 1829 TCGv dest; 1830 1831 if ((insn & 0x38) != 0) 1832 opsize = OS_BYTE; 1833 else 1834 opsize = OS_LONG; 1835 op = (insn >> 6) & 3; 1836 SRC_EA(env, src1, opsize, 0, op ? &addr: NULL); 1837 1838 gen_flush_flags(s); 1839 src2 = tcg_temp_new(); 1840 if (opsize == OS_BYTE) 1841 tcg_gen_andi_i32(src2, DREG(insn, 9), 7); 1842 else 1843 tcg_gen_andi_i32(src2, DREG(insn, 9), 31); 1844 1845 tmp = tcg_temp_new(); 1846 tcg_gen_shl_i32(tmp, tcg_constant_i32(1), src2); 1847 1848 tcg_gen_and_i32(QREG_CC_Z, src1, tmp); 1849 1850 dest = tcg_temp_new(); 1851 switch (op) { 1852 case 1: /* bchg */ 1853 tcg_gen_xor_i32(dest, src1, tmp); 1854 break; 1855 case 2: /* bclr */ 1856 tcg_gen_andc_i32(dest, src1, tmp); 1857 break; 1858 case 3: /* bset */ 1859 tcg_gen_or_i32(dest, src1, tmp); 1860 break; 1861 default: /* btst */ 1862 break; 1863 } 1864 if (op) { 1865 DEST_EA(env, insn, opsize, dest, &addr); 1866 } 1867 } 1868 1869 DISAS_INSN(sats) 1870 { 1871 TCGv reg; 1872 reg = DREG(insn, 0); 1873 gen_flush_flags(s); 1874 gen_helper_sats(reg, reg, QREG_CC_V); 1875 gen_logic_cc(s, reg, OS_LONG); 1876 } 1877 1878 static void gen_push(DisasContext *s, TCGv val) 1879 { 1880 TCGv tmp; 1881 1882 tmp = tcg_temp_new(); 1883 tcg_gen_subi_i32(tmp, QREG_SP, 4); 1884 gen_store(s, OS_LONG, tmp, val, IS_USER(s)); 1885 tcg_gen_mov_i32(QREG_SP, tmp); 1886 } 1887 1888 static TCGv mreg(int reg) 1889 { 1890 if (reg < 8) { 1891 /* Dx */ 1892 return cpu_dregs[reg]; 1893 } 1894 /* Ax */ 1895 return cpu_aregs[reg & 7]; 1896 } 1897 1898 DISAS_INSN(movem) 1899 { 1900 TCGv addr, incr, tmp, r[16]; 1901 int is_load = (insn & 0x0400) != 0; 1902 int opsize = (insn & 0x40) != 0 ? OS_LONG : OS_WORD; 1903 uint16_t mask = read_im16(env, s); 1904 int mode = extract32(insn, 3, 3); 1905 int reg0 = REG(insn, 0); 1906 int i; 1907 1908 tmp = cpu_aregs[reg0]; 1909 1910 switch (mode) { 1911 case 0: /* data register direct */ 1912 case 1: /* addr register direct */ 1913 do_addr_fault: 1914 gen_addr_fault(s); 1915 return; 1916 1917 case 2: /* indirect */ 1918 break; 1919 1920 case 3: /* indirect post-increment */ 1921 if (!is_load) { 1922 /* post-increment is not allowed */ 1923 goto do_addr_fault; 1924 } 1925 break; 1926 1927 case 4: /* indirect pre-decrement */ 1928 if (is_load) { 1929 /* pre-decrement is not allowed */ 1930 goto do_addr_fault; 1931 } 1932 /* 1933 * We want a bare copy of the address reg, without any pre-decrement 1934 * adjustment, as gen_lea would provide. 1935 */ 1936 break; 1937 1938 default: 1939 tmp = gen_lea_mode(env, s, mode, reg0, opsize); 1940 if (IS_NULL_QREG(tmp)) { 1941 goto do_addr_fault; 1942 } 1943 break; 1944 } 1945 1946 addr = tcg_temp_new(); 1947 tcg_gen_mov_i32(addr, tmp); 1948 incr = tcg_constant_i32(opsize_bytes(opsize)); 1949 1950 if (is_load) { 1951 /* memory to register */ 1952 for (i = 0; i < 16; i++) { 1953 if (mask & (1 << i)) { 1954 r[i] = gen_load(s, opsize, addr, 1, IS_USER(s)); 1955 tcg_gen_add_i32(addr, addr, incr); 1956 } 1957 } 1958 for (i = 0; i < 16; i++) { 1959 if (mask & (1 << i)) { 1960 tcg_gen_mov_i32(mreg(i), r[i]); 1961 } 1962 } 1963 if (mode == 3) { 1964 /* post-increment: movem (An)+,X */ 1965 tcg_gen_mov_i32(cpu_aregs[reg0], addr); 1966 } 1967 } else { 1968 /* register to memory */ 1969 if (mode == 4) { 1970 /* pre-decrement: movem X,-(An) */ 1971 for (i = 15; i >= 0; i--) { 1972 if ((mask << i) & 0x8000) { 1973 tcg_gen_sub_i32(addr, addr, incr); 1974 if (reg0 + 8 == i && 1975 m68k_feature(s->env, M68K_FEATURE_EXT_FULL)) { 1976 /* 1977 * M68020+: if the addressing register is the 1978 * register moved to memory, the value written 1979 * is the initial value decremented by the size of 1980 * the operation, regardless of how many actual 1981 * stores have been performed until this point. 1982 * M68000/M68010: the value is the initial value. 1983 */ 1984 tmp = tcg_temp_new(); 1985 tcg_gen_sub_i32(tmp, cpu_aregs[reg0], incr); 1986 gen_store(s, opsize, addr, tmp, IS_USER(s)); 1987 } else { 1988 gen_store(s, opsize, addr, mreg(i), IS_USER(s)); 1989 } 1990 } 1991 } 1992 tcg_gen_mov_i32(cpu_aregs[reg0], addr); 1993 } else { 1994 for (i = 0; i < 16; i++) { 1995 if (mask & (1 << i)) { 1996 gen_store(s, opsize, addr, mreg(i), IS_USER(s)); 1997 tcg_gen_add_i32(addr, addr, incr); 1998 } 1999 } 2000 } 2001 } 2002 } 2003 2004 DISAS_INSN(movep) 2005 { 2006 uint8_t i; 2007 int16_t displ; 2008 TCGv reg; 2009 TCGv addr; 2010 TCGv abuf; 2011 TCGv dbuf; 2012 2013 displ = read_im16(env, s); 2014 2015 addr = AREG(insn, 0); 2016 reg = DREG(insn, 9); 2017 2018 abuf = tcg_temp_new(); 2019 tcg_gen_addi_i32(abuf, addr, displ); 2020 dbuf = tcg_temp_new(); 2021 2022 if (insn & 0x40) { 2023 i = 4; 2024 } else { 2025 i = 2; 2026 } 2027 2028 if (insn & 0x80) { 2029 for ( ; i > 0 ; i--) { 2030 tcg_gen_shri_i32(dbuf, reg, (i - 1) * 8); 2031 tcg_gen_qemu_st_i32(dbuf, abuf, IS_USER(s), MO_UB); 2032 if (i > 1) { 2033 tcg_gen_addi_i32(abuf, abuf, 2); 2034 } 2035 } 2036 } else { 2037 for ( ; i > 0 ; i--) { 2038 tcg_gen_qemu_ld_tl(dbuf, abuf, IS_USER(s), MO_UB); 2039 tcg_gen_deposit_i32(reg, reg, dbuf, (i - 1) * 8, 8); 2040 if (i > 1) { 2041 tcg_gen_addi_i32(abuf, abuf, 2); 2042 } 2043 } 2044 } 2045 } 2046 2047 DISAS_INSN(bitop_im) 2048 { 2049 int opsize; 2050 int op; 2051 TCGv src1; 2052 uint32_t mask; 2053 int bitnum; 2054 TCGv tmp; 2055 TCGv addr; 2056 2057 if ((insn & 0x38) != 0) 2058 opsize = OS_BYTE; 2059 else 2060 opsize = OS_LONG; 2061 op = (insn >> 6) & 3; 2062 2063 bitnum = read_im16(env, s); 2064 if (m68k_feature(s->env, M68K_FEATURE_M68K)) { 2065 if (bitnum & 0xfe00) { 2066 disas_undef(env, s, insn); 2067 return; 2068 } 2069 } else { 2070 if (bitnum & 0xff00) { 2071 disas_undef(env, s, insn); 2072 return; 2073 } 2074 } 2075 2076 SRC_EA(env, src1, opsize, 0, op ? &addr: NULL); 2077 2078 gen_flush_flags(s); 2079 if (opsize == OS_BYTE) 2080 bitnum &= 7; 2081 else 2082 bitnum &= 31; 2083 mask = 1 << bitnum; 2084 2085 tcg_gen_andi_i32(QREG_CC_Z, src1, mask); 2086 2087 if (op) { 2088 tmp = tcg_temp_new(); 2089 switch (op) { 2090 case 1: /* bchg */ 2091 tcg_gen_xori_i32(tmp, src1, mask); 2092 break; 2093 case 2: /* bclr */ 2094 tcg_gen_andi_i32(tmp, src1, ~mask); 2095 break; 2096 case 3: /* bset */ 2097 tcg_gen_ori_i32(tmp, src1, mask); 2098 break; 2099 default: /* btst */ 2100 break; 2101 } 2102 DEST_EA(env, insn, opsize, tmp, &addr); 2103 } 2104 } 2105 2106 static TCGv gen_get_ccr(DisasContext *s) 2107 { 2108 TCGv dest; 2109 2110 update_cc_op(s); 2111 dest = tcg_temp_new(); 2112 gen_helper_get_ccr(dest, tcg_env); 2113 return dest; 2114 } 2115 2116 static TCGv gen_get_sr(DisasContext *s) 2117 { 2118 TCGv ccr; 2119 TCGv sr; 2120 2121 ccr = gen_get_ccr(s); 2122 sr = tcg_temp_new(); 2123 tcg_gen_andi_i32(sr, QREG_SR, 0xffe0); 2124 tcg_gen_or_i32(sr, sr, ccr); 2125 return sr; 2126 } 2127 2128 static void gen_set_sr_im(DisasContext *s, uint16_t val, int ccr_only) 2129 { 2130 if (ccr_only) { 2131 tcg_gen_movi_i32(QREG_CC_C, val & CCF_C ? 1 : 0); 2132 tcg_gen_movi_i32(QREG_CC_V, val & CCF_V ? -1 : 0); 2133 tcg_gen_movi_i32(QREG_CC_Z, val & CCF_Z ? 0 : 1); 2134 tcg_gen_movi_i32(QREG_CC_N, val & CCF_N ? -1 : 0); 2135 tcg_gen_movi_i32(QREG_CC_X, val & CCF_X ? 1 : 0); 2136 } else { 2137 /* Must writeback before changing security state. */ 2138 do_writebacks(s); 2139 gen_helper_set_sr(tcg_env, tcg_constant_i32(val)); 2140 } 2141 set_cc_op(s, CC_OP_FLAGS); 2142 } 2143 2144 static void gen_set_sr(DisasContext *s, TCGv val, int ccr_only) 2145 { 2146 if (ccr_only) { 2147 gen_helper_set_ccr(tcg_env, val); 2148 } else { 2149 /* Must writeback before changing security state. */ 2150 do_writebacks(s); 2151 gen_helper_set_sr(tcg_env, val); 2152 } 2153 set_cc_op(s, CC_OP_FLAGS); 2154 } 2155 2156 static void gen_move_to_sr(CPUM68KState *env, DisasContext *s, uint16_t insn, 2157 bool ccr_only) 2158 { 2159 if ((insn & 0x3f) == 0x3c) { 2160 uint16_t val; 2161 val = read_im16(env, s); 2162 gen_set_sr_im(s, val, ccr_only); 2163 } else { 2164 TCGv src; 2165 SRC_EA(env, src, OS_WORD, 0, NULL); 2166 gen_set_sr(s, src, ccr_only); 2167 } 2168 } 2169 2170 DISAS_INSN(arith_im) 2171 { 2172 int op; 2173 TCGv im; 2174 TCGv src1; 2175 TCGv dest; 2176 TCGv addr; 2177 int opsize; 2178 bool with_SR = ((insn & 0x3f) == 0x3c); 2179 2180 op = (insn >> 9) & 7; 2181 opsize = insn_opsize(insn); 2182 switch (opsize) { 2183 case OS_BYTE: 2184 im = tcg_constant_i32((int8_t)read_im8(env, s)); 2185 break; 2186 case OS_WORD: 2187 im = tcg_constant_i32((int16_t)read_im16(env, s)); 2188 break; 2189 case OS_LONG: 2190 im = tcg_constant_i32(read_im32(env, s)); 2191 break; 2192 default: 2193 g_assert_not_reached(); 2194 } 2195 2196 if (with_SR) { 2197 /* SR/CCR can only be used with andi/eori/ori */ 2198 if (op == 2 || op == 3 || op == 6) { 2199 disas_undef(env, s, insn); 2200 return; 2201 } 2202 switch (opsize) { 2203 case OS_BYTE: 2204 src1 = gen_get_ccr(s); 2205 break; 2206 case OS_WORD: 2207 if (IS_USER(s)) { 2208 gen_exception(s, s->base.pc_next, EXCP_PRIVILEGE); 2209 return; 2210 } 2211 src1 = gen_get_sr(s); 2212 break; 2213 default: 2214 /* OS_LONG; others already g_assert_not_reached. */ 2215 disas_undef(env, s, insn); 2216 return; 2217 } 2218 } else { 2219 SRC_EA(env, src1, opsize, 1, (op == 6) ? NULL : &addr); 2220 } 2221 dest = tcg_temp_new(); 2222 switch (op) { 2223 case 0: /* ori */ 2224 tcg_gen_or_i32(dest, src1, im); 2225 if (with_SR) { 2226 gen_set_sr(s, dest, opsize == OS_BYTE); 2227 gen_exit_tb(s); 2228 } else { 2229 DEST_EA(env, insn, opsize, dest, &addr); 2230 gen_logic_cc(s, dest, opsize); 2231 } 2232 break; 2233 case 1: /* andi */ 2234 tcg_gen_and_i32(dest, src1, im); 2235 if (with_SR) { 2236 gen_set_sr(s, dest, opsize == OS_BYTE); 2237 gen_exit_tb(s); 2238 } else { 2239 DEST_EA(env, insn, opsize, dest, &addr); 2240 gen_logic_cc(s, dest, opsize); 2241 } 2242 break; 2243 case 2: /* subi */ 2244 tcg_gen_setcond_i32(TCG_COND_LTU, QREG_CC_X, src1, im); 2245 tcg_gen_sub_i32(dest, src1, im); 2246 gen_update_cc_add(dest, im, opsize); 2247 set_cc_op(s, CC_OP_SUBB + opsize); 2248 DEST_EA(env, insn, opsize, dest, &addr); 2249 break; 2250 case 3: /* addi */ 2251 tcg_gen_add_i32(dest, src1, im); 2252 gen_update_cc_add(dest, im, opsize); 2253 tcg_gen_setcond_i32(TCG_COND_LTU, QREG_CC_X, dest, im); 2254 set_cc_op(s, CC_OP_ADDB + opsize); 2255 DEST_EA(env, insn, opsize, dest, &addr); 2256 break; 2257 case 5: /* eori */ 2258 tcg_gen_xor_i32(dest, src1, im); 2259 if (with_SR) { 2260 gen_set_sr(s, dest, opsize == OS_BYTE); 2261 gen_exit_tb(s); 2262 } else { 2263 DEST_EA(env, insn, opsize, dest, &addr); 2264 gen_logic_cc(s, dest, opsize); 2265 } 2266 break; 2267 case 6: /* cmpi */ 2268 gen_update_cc_cmp(s, src1, im, opsize); 2269 break; 2270 default: 2271 abort(); 2272 } 2273 } 2274 2275 DISAS_INSN(cas) 2276 { 2277 int opsize; 2278 TCGv addr; 2279 uint16_t ext; 2280 TCGv load; 2281 TCGv cmp; 2282 MemOp opc; 2283 2284 switch ((insn >> 9) & 3) { 2285 case 1: 2286 opsize = OS_BYTE; 2287 opc = MO_SB; 2288 break; 2289 case 2: 2290 opsize = OS_WORD; 2291 opc = MO_TESW; 2292 break; 2293 case 3: 2294 opsize = OS_LONG; 2295 opc = MO_TESL; 2296 break; 2297 default: 2298 g_assert_not_reached(); 2299 } 2300 2301 ext = read_im16(env, s); 2302 2303 /* cas Dc,Du,<EA> */ 2304 2305 addr = gen_lea(env, s, insn, opsize); 2306 if (IS_NULL_QREG(addr)) { 2307 gen_addr_fault(s); 2308 return; 2309 } 2310 2311 cmp = gen_extend(s, DREG(ext, 0), opsize, 1); 2312 2313 /* 2314 * if <EA> == Dc then 2315 * <EA> = Du 2316 * Dc = <EA> (because <EA> == Dc) 2317 * else 2318 * Dc = <EA> 2319 */ 2320 2321 load = tcg_temp_new(); 2322 tcg_gen_atomic_cmpxchg_i32(load, addr, cmp, DREG(ext, 6), 2323 IS_USER(s), opc); 2324 /* update flags before setting cmp to load */ 2325 gen_update_cc_cmp(s, load, cmp, opsize); 2326 gen_partset_reg(opsize, DREG(ext, 0), load); 2327 2328 switch (extract32(insn, 3, 3)) { 2329 case 3: /* Indirect postincrement. */ 2330 tcg_gen_addi_i32(AREG(insn, 0), addr, opsize_bytes(opsize)); 2331 break; 2332 case 4: /* Indirect predecrememnt. */ 2333 tcg_gen_mov_i32(AREG(insn, 0), addr); 2334 break; 2335 } 2336 } 2337 2338 DISAS_INSN(cas2w) 2339 { 2340 uint16_t ext1, ext2; 2341 TCGv addr1, addr2; 2342 2343 /* cas2 Dc1:Dc2,Du1:Du2,(Rn1):(Rn2) */ 2344 2345 ext1 = read_im16(env, s); 2346 2347 if (ext1 & 0x8000) { 2348 /* Address Register */ 2349 addr1 = AREG(ext1, 12); 2350 } else { 2351 /* Data Register */ 2352 addr1 = DREG(ext1, 12); 2353 } 2354 2355 ext2 = read_im16(env, s); 2356 if (ext2 & 0x8000) { 2357 /* Address Register */ 2358 addr2 = AREG(ext2, 12); 2359 } else { 2360 /* Data Register */ 2361 addr2 = DREG(ext2, 12); 2362 } 2363 2364 /* 2365 * if (R1) == Dc1 && (R2) == Dc2 then 2366 * (R1) = Du1 2367 * (R2) = Du2 2368 * else 2369 * Dc1 = (R1) 2370 * Dc2 = (R2) 2371 */ 2372 2373 if (tb_cflags(s->base.tb) & CF_PARALLEL) { 2374 gen_helper_exit_atomic(tcg_env); 2375 } else { 2376 TCGv regs = tcg_constant_i32(REG(ext2, 6) | 2377 (REG(ext1, 6) << 3) | 2378 (REG(ext2, 0) << 6) | 2379 (REG(ext1, 0) << 9)); 2380 gen_helper_cas2w(tcg_env, regs, addr1, addr2); 2381 } 2382 2383 /* Note that cas2w also assigned to env->cc_op. */ 2384 s->cc_op = CC_OP_CMPW; 2385 s->cc_op_synced = 1; 2386 } 2387 2388 DISAS_INSN(cas2l) 2389 { 2390 uint16_t ext1, ext2; 2391 TCGv addr1, addr2, regs; 2392 2393 /* cas2 Dc1:Dc2,Du1:Du2,(Rn1):(Rn2) */ 2394 2395 ext1 = read_im16(env, s); 2396 2397 if (ext1 & 0x8000) { 2398 /* Address Register */ 2399 addr1 = AREG(ext1, 12); 2400 } else { 2401 /* Data Register */ 2402 addr1 = DREG(ext1, 12); 2403 } 2404 2405 ext2 = read_im16(env, s); 2406 if (ext2 & 0x8000) { 2407 /* Address Register */ 2408 addr2 = AREG(ext2, 12); 2409 } else { 2410 /* Data Register */ 2411 addr2 = DREG(ext2, 12); 2412 } 2413 2414 /* 2415 * if (R1) == Dc1 && (R2) == Dc2 then 2416 * (R1) = Du1 2417 * (R2) = Du2 2418 * else 2419 * Dc1 = (R1) 2420 * Dc2 = (R2) 2421 */ 2422 2423 regs = tcg_constant_i32(REG(ext2, 6) | 2424 (REG(ext1, 6) << 3) | 2425 (REG(ext2, 0) << 6) | 2426 (REG(ext1, 0) << 9)); 2427 if (tb_cflags(s->base.tb) & CF_PARALLEL) { 2428 gen_helper_cas2l_parallel(tcg_env, regs, addr1, addr2); 2429 } else { 2430 gen_helper_cas2l(tcg_env, regs, addr1, addr2); 2431 } 2432 2433 /* Note that cas2l also assigned to env->cc_op. */ 2434 s->cc_op = CC_OP_CMPL; 2435 s->cc_op_synced = 1; 2436 } 2437 2438 DISAS_INSN(byterev) 2439 { 2440 TCGv reg; 2441 2442 reg = DREG(insn, 0); 2443 tcg_gen_bswap32_i32(reg, reg); 2444 } 2445 2446 DISAS_INSN(move) 2447 { 2448 TCGv src; 2449 TCGv dest; 2450 int op; 2451 int opsize; 2452 2453 switch (insn >> 12) { 2454 case 1: /* move.b */ 2455 opsize = OS_BYTE; 2456 break; 2457 case 2: /* move.l */ 2458 opsize = OS_LONG; 2459 break; 2460 case 3: /* move.w */ 2461 opsize = OS_WORD; 2462 break; 2463 default: 2464 abort(); 2465 } 2466 SRC_EA(env, src, opsize, 1, NULL); 2467 op = (insn >> 6) & 7; 2468 if (op == 1) { 2469 /* movea */ 2470 /* The value will already have been sign extended. */ 2471 dest = AREG(insn, 9); 2472 tcg_gen_mov_i32(dest, src); 2473 } else { 2474 /* normal move */ 2475 uint16_t dest_ea; 2476 dest_ea = ((insn >> 9) & 7) | (op << 3); 2477 DEST_EA(env, dest_ea, opsize, src, NULL); 2478 /* This will be correct because loads sign extend. */ 2479 gen_logic_cc(s, src, opsize); 2480 } 2481 } 2482 2483 DISAS_INSN(negx) 2484 { 2485 TCGv z; 2486 TCGv src; 2487 TCGv addr; 2488 int opsize; 2489 2490 opsize = insn_opsize(insn); 2491 SRC_EA(env, src, opsize, 1, &addr); 2492 2493 gen_flush_flags(s); /* compute old Z */ 2494 2495 /* 2496 * Perform subtract with borrow. 2497 * (X, N) = -(src + X); 2498 */ 2499 2500 z = tcg_constant_i32(0); 2501 tcg_gen_add2_i32(QREG_CC_N, QREG_CC_X, src, z, QREG_CC_X, z); 2502 tcg_gen_sub2_i32(QREG_CC_N, QREG_CC_X, z, z, QREG_CC_N, QREG_CC_X); 2503 gen_ext(QREG_CC_N, QREG_CC_N, opsize, 1); 2504 2505 tcg_gen_andi_i32(QREG_CC_X, QREG_CC_X, 1); 2506 2507 /* 2508 * Compute signed-overflow for negation. The normal formula for 2509 * subtraction is (res ^ src) & (src ^ dest), but with dest==0 2510 * this simplifies to res & src. 2511 */ 2512 2513 tcg_gen_and_i32(QREG_CC_V, QREG_CC_N, src); 2514 2515 /* Copy the rest of the results into place. */ 2516 tcg_gen_or_i32(QREG_CC_Z, QREG_CC_Z, QREG_CC_N); /* !Z is sticky */ 2517 tcg_gen_mov_i32(QREG_CC_C, QREG_CC_X); 2518 2519 set_cc_op(s, CC_OP_FLAGS); 2520 2521 /* result is in QREG_CC_N */ 2522 2523 DEST_EA(env, insn, opsize, QREG_CC_N, &addr); 2524 } 2525 2526 DISAS_INSN(lea) 2527 { 2528 TCGv reg; 2529 TCGv tmp; 2530 2531 reg = AREG(insn, 9); 2532 tmp = gen_lea(env, s, insn, OS_LONG); 2533 if (IS_NULL_QREG(tmp)) { 2534 gen_addr_fault(s); 2535 return; 2536 } 2537 tcg_gen_mov_i32(reg, tmp); 2538 } 2539 2540 DISAS_INSN(clr) 2541 { 2542 int opsize; 2543 TCGv zero; 2544 2545 zero = tcg_constant_i32(0); 2546 opsize = insn_opsize(insn); 2547 DEST_EA(env, insn, opsize, zero, NULL); 2548 gen_logic_cc(s, zero, opsize); 2549 } 2550 2551 DISAS_INSN(move_from_ccr) 2552 { 2553 TCGv ccr; 2554 2555 ccr = gen_get_ccr(s); 2556 DEST_EA(env, insn, OS_WORD, ccr, NULL); 2557 } 2558 2559 DISAS_INSN(neg) 2560 { 2561 TCGv src1; 2562 TCGv dest; 2563 TCGv addr; 2564 int opsize; 2565 2566 opsize = insn_opsize(insn); 2567 SRC_EA(env, src1, opsize, 1, &addr); 2568 dest = tcg_temp_new(); 2569 tcg_gen_neg_i32(dest, src1); 2570 set_cc_op(s, CC_OP_SUBB + opsize); 2571 gen_update_cc_add(dest, src1, opsize); 2572 tcg_gen_setcondi_i32(TCG_COND_NE, QREG_CC_X, dest, 0); 2573 DEST_EA(env, insn, opsize, dest, &addr); 2574 } 2575 2576 DISAS_INSN(move_to_ccr) 2577 { 2578 gen_move_to_sr(env, s, insn, true); 2579 } 2580 2581 DISAS_INSN(not) 2582 { 2583 TCGv src1; 2584 TCGv dest; 2585 TCGv addr; 2586 int opsize; 2587 2588 opsize = insn_opsize(insn); 2589 SRC_EA(env, src1, opsize, 1, &addr); 2590 dest = tcg_temp_new(); 2591 tcg_gen_not_i32(dest, src1); 2592 DEST_EA(env, insn, opsize, dest, &addr); 2593 gen_logic_cc(s, dest, opsize); 2594 } 2595 2596 DISAS_INSN(swap) 2597 { 2598 TCGv src1; 2599 TCGv src2; 2600 TCGv reg; 2601 2602 src1 = tcg_temp_new(); 2603 src2 = tcg_temp_new(); 2604 reg = DREG(insn, 0); 2605 tcg_gen_shli_i32(src1, reg, 16); 2606 tcg_gen_shri_i32(src2, reg, 16); 2607 tcg_gen_or_i32(reg, src1, src2); 2608 gen_logic_cc(s, reg, OS_LONG); 2609 } 2610 2611 DISAS_INSN(bkpt) 2612 { 2613 #if defined(CONFIG_USER_ONLY) 2614 gen_exception(s, s->base.pc_next, EXCP_DEBUG); 2615 #else 2616 gen_exception(s, s->base.pc_next, EXCP_ILLEGAL); 2617 #endif 2618 } 2619 2620 DISAS_INSN(pea) 2621 { 2622 TCGv tmp; 2623 2624 tmp = gen_lea(env, s, insn, OS_LONG); 2625 if (IS_NULL_QREG(tmp)) { 2626 gen_addr_fault(s); 2627 return; 2628 } 2629 gen_push(s, tmp); 2630 } 2631 2632 DISAS_INSN(ext) 2633 { 2634 int op; 2635 TCGv reg; 2636 TCGv tmp; 2637 2638 reg = DREG(insn, 0); 2639 op = (insn >> 6) & 7; 2640 tmp = tcg_temp_new(); 2641 if (op == 3) 2642 tcg_gen_ext16s_i32(tmp, reg); 2643 else 2644 tcg_gen_ext8s_i32(tmp, reg); 2645 if (op == 2) 2646 gen_partset_reg(OS_WORD, reg, tmp); 2647 else 2648 tcg_gen_mov_i32(reg, tmp); 2649 gen_logic_cc(s, tmp, OS_LONG); 2650 } 2651 2652 DISAS_INSN(tst) 2653 { 2654 int opsize; 2655 TCGv tmp; 2656 2657 opsize = insn_opsize(insn); 2658 SRC_EA(env, tmp, opsize, 1, NULL); 2659 gen_logic_cc(s, tmp, opsize); 2660 } 2661 2662 DISAS_INSN(pulse) 2663 { 2664 /* Implemented as a NOP. */ 2665 } 2666 2667 DISAS_INSN(illegal) 2668 { 2669 gen_exception(s, s->base.pc_next, EXCP_ILLEGAL); 2670 } 2671 2672 DISAS_INSN(tas) 2673 { 2674 int mode = extract32(insn, 3, 3); 2675 int reg0 = REG(insn, 0); 2676 2677 if (mode == 0) { 2678 /* data register direct */ 2679 TCGv dest = cpu_dregs[reg0]; 2680 gen_logic_cc(s, dest, OS_BYTE); 2681 tcg_gen_ori_tl(dest, dest, 0x80); 2682 } else { 2683 TCGv src1, addr; 2684 2685 addr = gen_lea_mode(env, s, mode, reg0, OS_BYTE); 2686 if (IS_NULL_QREG(addr)) { 2687 gen_addr_fault(s); 2688 return; 2689 } 2690 src1 = tcg_temp_new(); 2691 tcg_gen_atomic_fetch_or_tl(src1, addr, tcg_constant_tl(0x80), 2692 IS_USER(s), MO_SB); 2693 gen_logic_cc(s, src1, OS_BYTE); 2694 2695 switch (mode) { 2696 case 3: /* Indirect postincrement. */ 2697 tcg_gen_addi_i32(AREG(insn, 0), addr, 1); 2698 break; 2699 case 4: /* Indirect predecrememnt. */ 2700 tcg_gen_mov_i32(AREG(insn, 0), addr); 2701 break; 2702 } 2703 } 2704 } 2705 2706 DISAS_INSN(mull) 2707 { 2708 uint16_t ext; 2709 TCGv src1; 2710 int sign; 2711 2712 ext = read_im16(env, s); 2713 2714 sign = ext & 0x800; 2715 2716 if (ext & 0x400) { 2717 if (!m68k_feature(s->env, M68K_FEATURE_QUAD_MULDIV)) { 2718 gen_exception(s, s->base.pc_next, EXCP_ILLEGAL); 2719 return; 2720 } 2721 2722 SRC_EA(env, src1, OS_LONG, 0, NULL); 2723 2724 if (sign) { 2725 tcg_gen_muls2_i32(QREG_CC_Z, QREG_CC_N, src1, DREG(ext, 12)); 2726 } else { 2727 tcg_gen_mulu2_i32(QREG_CC_Z, QREG_CC_N, src1, DREG(ext, 12)); 2728 } 2729 /* if Dl == Dh, 68040 returns low word */ 2730 tcg_gen_mov_i32(DREG(ext, 0), QREG_CC_N); 2731 tcg_gen_mov_i32(DREG(ext, 12), QREG_CC_Z); 2732 tcg_gen_or_i32(QREG_CC_Z, QREG_CC_Z, QREG_CC_N); 2733 2734 tcg_gen_movi_i32(QREG_CC_V, 0); 2735 tcg_gen_movi_i32(QREG_CC_C, 0); 2736 2737 set_cc_op(s, CC_OP_FLAGS); 2738 return; 2739 } 2740 SRC_EA(env, src1, OS_LONG, 0, NULL); 2741 if (m68k_feature(s->env, M68K_FEATURE_M68K)) { 2742 tcg_gen_movi_i32(QREG_CC_C, 0); 2743 if (sign) { 2744 tcg_gen_muls2_i32(QREG_CC_N, QREG_CC_V, src1, DREG(ext, 12)); 2745 /* QREG_CC_V is -(QREG_CC_V != (QREG_CC_N >> 31)) */ 2746 tcg_gen_sari_i32(QREG_CC_Z, QREG_CC_N, 31); 2747 tcg_gen_negsetcond_i32(TCG_COND_NE, QREG_CC_V, 2748 QREG_CC_V, QREG_CC_Z); 2749 } else { 2750 tcg_gen_mulu2_i32(QREG_CC_N, QREG_CC_V, src1, DREG(ext, 12)); 2751 /* QREG_CC_V is -(QREG_CC_V != 0), use QREG_CC_C as 0 */ 2752 tcg_gen_negsetcond_i32(TCG_COND_NE, QREG_CC_V, 2753 QREG_CC_V, QREG_CC_C); 2754 } 2755 tcg_gen_mov_i32(DREG(ext, 12), QREG_CC_N); 2756 2757 tcg_gen_mov_i32(QREG_CC_Z, QREG_CC_N); 2758 2759 set_cc_op(s, CC_OP_FLAGS); 2760 } else { 2761 /* 2762 * The upper 32 bits of the product are discarded, so 2763 * muls.l and mulu.l are functionally equivalent. 2764 */ 2765 tcg_gen_mul_i32(DREG(ext, 12), src1, DREG(ext, 12)); 2766 gen_logic_cc(s, DREG(ext, 12), OS_LONG); 2767 } 2768 } 2769 2770 static void gen_link(DisasContext *s, uint16_t insn, int32_t offset) 2771 { 2772 TCGv reg; 2773 TCGv tmp; 2774 2775 reg = AREG(insn, 0); 2776 tmp = tcg_temp_new(); 2777 tcg_gen_subi_i32(tmp, QREG_SP, 4); 2778 gen_store(s, OS_LONG, tmp, reg, IS_USER(s)); 2779 if ((insn & 7) != 7) { 2780 tcg_gen_mov_i32(reg, tmp); 2781 } 2782 tcg_gen_addi_i32(QREG_SP, tmp, offset); 2783 } 2784 2785 DISAS_INSN(link) 2786 { 2787 int16_t offset; 2788 2789 offset = read_im16(env, s); 2790 gen_link(s, insn, offset); 2791 } 2792 2793 DISAS_INSN(linkl) 2794 { 2795 int32_t offset; 2796 2797 offset = read_im32(env, s); 2798 gen_link(s, insn, offset); 2799 } 2800 2801 DISAS_INSN(unlk) 2802 { 2803 TCGv src; 2804 TCGv reg; 2805 TCGv tmp; 2806 2807 src = tcg_temp_new(); 2808 reg = AREG(insn, 0); 2809 tcg_gen_mov_i32(src, reg); 2810 tmp = gen_load(s, OS_LONG, src, 0, IS_USER(s)); 2811 tcg_gen_mov_i32(reg, tmp); 2812 tcg_gen_addi_i32(QREG_SP, src, 4); 2813 } 2814 2815 #if !defined(CONFIG_USER_ONLY) 2816 DISAS_INSN(reset) 2817 { 2818 if (IS_USER(s)) { 2819 gen_exception(s, s->base.pc_next, EXCP_PRIVILEGE); 2820 return; 2821 } 2822 2823 gen_helper_reset(tcg_env); 2824 } 2825 #endif 2826 2827 DISAS_INSN(nop) 2828 { 2829 } 2830 2831 DISAS_INSN(rtd) 2832 { 2833 TCGv tmp; 2834 int16_t offset = read_im16(env, s); 2835 2836 tmp = gen_load(s, OS_LONG, QREG_SP, 0, IS_USER(s)); 2837 tcg_gen_addi_i32(QREG_SP, QREG_SP, offset + 4); 2838 gen_jmp(s, tmp); 2839 } 2840 2841 DISAS_INSN(rtr) 2842 { 2843 TCGv tmp; 2844 TCGv ccr; 2845 TCGv sp; 2846 2847 sp = tcg_temp_new(); 2848 ccr = gen_load(s, OS_WORD, QREG_SP, 0, IS_USER(s)); 2849 tcg_gen_addi_i32(sp, QREG_SP, 2); 2850 tmp = gen_load(s, OS_LONG, sp, 0, IS_USER(s)); 2851 tcg_gen_addi_i32(QREG_SP, sp, 4); 2852 2853 gen_set_sr(s, ccr, true); 2854 2855 gen_jmp(s, tmp); 2856 } 2857 2858 DISAS_INSN(rts) 2859 { 2860 TCGv tmp; 2861 2862 tmp = gen_load(s, OS_LONG, QREG_SP, 0, IS_USER(s)); 2863 tcg_gen_addi_i32(QREG_SP, QREG_SP, 4); 2864 gen_jmp(s, tmp); 2865 } 2866 2867 DISAS_INSN(jump) 2868 { 2869 TCGv tmp; 2870 2871 /* 2872 * Load the target address first to ensure correct exception 2873 * behavior. 2874 */ 2875 tmp = gen_lea(env, s, insn, OS_LONG); 2876 if (IS_NULL_QREG(tmp)) { 2877 gen_addr_fault(s); 2878 return; 2879 } 2880 if ((insn & 0x40) == 0) { 2881 /* jsr */ 2882 gen_push(s, tcg_constant_i32(s->pc)); 2883 } 2884 gen_jmp(s, tmp); 2885 } 2886 2887 DISAS_INSN(addsubq) 2888 { 2889 TCGv src; 2890 TCGv dest; 2891 TCGv val; 2892 int imm; 2893 TCGv addr; 2894 int opsize; 2895 2896 if ((insn & 070) == 010) { 2897 /* Operation on address register is always long. */ 2898 opsize = OS_LONG; 2899 } else { 2900 opsize = insn_opsize(insn); 2901 } 2902 SRC_EA(env, src, opsize, 1, &addr); 2903 imm = (insn >> 9) & 7; 2904 if (imm == 0) { 2905 imm = 8; 2906 } 2907 val = tcg_constant_i32(imm); 2908 dest = tcg_temp_new(); 2909 tcg_gen_mov_i32(dest, src); 2910 if ((insn & 0x38) == 0x08) { 2911 /* 2912 * Don't update condition codes if the destination is an 2913 * address register. 2914 */ 2915 if (insn & 0x0100) { 2916 tcg_gen_sub_i32(dest, dest, val); 2917 } else { 2918 tcg_gen_add_i32(dest, dest, val); 2919 } 2920 } else { 2921 if (insn & 0x0100) { 2922 tcg_gen_setcond_i32(TCG_COND_LTU, QREG_CC_X, dest, val); 2923 tcg_gen_sub_i32(dest, dest, val); 2924 set_cc_op(s, CC_OP_SUBB + opsize); 2925 } else { 2926 tcg_gen_add_i32(dest, dest, val); 2927 tcg_gen_setcond_i32(TCG_COND_LTU, QREG_CC_X, dest, val); 2928 set_cc_op(s, CC_OP_ADDB + opsize); 2929 } 2930 gen_update_cc_add(dest, val, opsize); 2931 } 2932 DEST_EA(env, insn, opsize, dest, &addr); 2933 } 2934 2935 DISAS_INSN(branch) 2936 { 2937 int32_t offset; 2938 uint32_t base; 2939 int op; 2940 2941 base = s->pc; 2942 op = (insn >> 8) & 0xf; 2943 offset = (int8_t)insn; 2944 if (offset == 0) { 2945 offset = (int16_t)read_im16(env, s); 2946 } else if (offset == -1) { 2947 offset = read_im32(env, s); 2948 } 2949 if (op == 1) { 2950 /* bsr */ 2951 gen_push(s, tcg_constant_i32(s->pc)); 2952 } 2953 if (op > 1) { 2954 /* Bcc */ 2955 TCGLabel *l1 = gen_new_label(); 2956 gen_jmpcc(s, ((insn >> 8) & 0xf) ^ 1, l1); 2957 gen_jmp_tb(s, 1, base + offset, s->base.pc_next); 2958 gen_set_label(l1); 2959 gen_jmp_tb(s, 0, s->pc, s->base.pc_next); 2960 } else { 2961 /* Unconditional branch. */ 2962 update_cc_op(s); 2963 gen_jmp_tb(s, 0, base + offset, s->base.pc_next); 2964 } 2965 } 2966 2967 DISAS_INSN(moveq) 2968 { 2969 tcg_gen_movi_i32(DREG(insn, 9), (int8_t)insn); 2970 gen_logic_cc(s, DREG(insn, 9), OS_LONG); 2971 } 2972 2973 DISAS_INSN(mvzs) 2974 { 2975 int opsize; 2976 TCGv src; 2977 TCGv reg; 2978 2979 if (insn & 0x40) 2980 opsize = OS_WORD; 2981 else 2982 opsize = OS_BYTE; 2983 SRC_EA(env, src, opsize, (insn & 0x80) == 0, NULL); 2984 reg = DREG(insn, 9); 2985 tcg_gen_mov_i32(reg, src); 2986 gen_logic_cc(s, src, opsize); 2987 } 2988 2989 DISAS_INSN(or) 2990 { 2991 TCGv reg; 2992 TCGv dest; 2993 TCGv src; 2994 TCGv addr; 2995 int opsize; 2996 2997 opsize = insn_opsize(insn); 2998 reg = gen_extend(s, DREG(insn, 9), opsize, 0); 2999 dest = tcg_temp_new(); 3000 if (insn & 0x100) { 3001 SRC_EA(env, src, opsize, 0, &addr); 3002 tcg_gen_or_i32(dest, src, reg); 3003 DEST_EA(env, insn, opsize, dest, &addr); 3004 } else { 3005 SRC_EA(env, src, opsize, 0, NULL); 3006 tcg_gen_or_i32(dest, src, reg); 3007 gen_partset_reg(opsize, DREG(insn, 9), dest); 3008 } 3009 gen_logic_cc(s, dest, opsize); 3010 } 3011 3012 DISAS_INSN(suba) 3013 { 3014 TCGv src; 3015 TCGv reg; 3016 3017 SRC_EA(env, src, (insn & 0x100) ? OS_LONG : OS_WORD, 1, NULL); 3018 reg = AREG(insn, 9); 3019 tcg_gen_sub_i32(reg, reg, src); 3020 } 3021 3022 static inline void gen_subx(DisasContext *s, TCGv src, TCGv dest, int opsize) 3023 { 3024 TCGv tmp, zero; 3025 3026 gen_flush_flags(s); /* compute old Z */ 3027 3028 /* 3029 * Perform subtract with borrow. 3030 * (X, N) = dest - (src + X); 3031 */ 3032 3033 zero = tcg_constant_i32(0); 3034 tcg_gen_add2_i32(QREG_CC_N, QREG_CC_X, src, zero, QREG_CC_X, zero); 3035 tcg_gen_sub2_i32(QREG_CC_N, QREG_CC_X, dest, zero, QREG_CC_N, QREG_CC_X); 3036 gen_ext(QREG_CC_N, QREG_CC_N, opsize, 1); 3037 tcg_gen_andi_i32(QREG_CC_X, QREG_CC_X, 1); 3038 3039 /* Compute signed-overflow for subtract. */ 3040 3041 tmp = tcg_temp_new(); 3042 tcg_gen_xor_i32(QREG_CC_V, QREG_CC_N, dest); 3043 tcg_gen_xor_i32(tmp, dest, src); 3044 tcg_gen_and_i32(QREG_CC_V, QREG_CC_V, tmp); 3045 3046 /* Copy the rest of the results into place. */ 3047 tcg_gen_or_i32(QREG_CC_Z, QREG_CC_Z, QREG_CC_N); /* !Z is sticky */ 3048 tcg_gen_mov_i32(QREG_CC_C, QREG_CC_X); 3049 3050 set_cc_op(s, CC_OP_FLAGS); 3051 3052 /* result is in QREG_CC_N */ 3053 } 3054 3055 DISAS_INSN(subx_reg) 3056 { 3057 TCGv dest; 3058 TCGv src; 3059 int opsize; 3060 3061 opsize = insn_opsize(insn); 3062 3063 src = gen_extend(s, DREG(insn, 0), opsize, 1); 3064 dest = gen_extend(s, DREG(insn, 9), opsize, 1); 3065 3066 gen_subx(s, src, dest, opsize); 3067 3068 gen_partset_reg(opsize, DREG(insn, 9), QREG_CC_N); 3069 } 3070 3071 DISAS_INSN(subx_mem) 3072 { 3073 TCGv src; 3074 TCGv addr_src; 3075 TCGv dest; 3076 TCGv addr_dest; 3077 int opsize; 3078 3079 opsize = insn_opsize(insn); 3080 3081 addr_src = AREG(insn, 0); 3082 tcg_gen_subi_i32(addr_src, addr_src, opsize_bytes(opsize)); 3083 src = gen_load(s, opsize, addr_src, 1, IS_USER(s)); 3084 3085 addr_dest = AREG(insn, 9); 3086 tcg_gen_subi_i32(addr_dest, addr_dest, opsize_bytes(opsize)); 3087 dest = gen_load(s, opsize, addr_dest, 1, IS_USER(s)); 3088 3089 gen_subx(s, src, dest, opsize); 3090 3091 gen_store(s, opsize, addr_dest, QREG_CC_N, IS_USER(s)); 3092 } 3093 3094 DISAS_INSN(mov3q) 3095 { 3096 TCGv src; 3097 int val; 3098 3099 val = (insn >> 9) & 7; 3100 if (val == 0) { 3101 val = -1; 3102 } 3103 src = tcg_constant_i32(val); 3104 gen_logic_cc(s, src, OS_LONG); 3105 DEST_EA(env, insn, OS_LONG, src, NULL); 3106 } 3107 3108 DISAS_INSN(cmp) 3109 { 3110 TCGv src; 3111 TCGv reg; 3112 int opsize; 3113 3114 opsize = insn_opsize(insn); 3115 SRC_EA(env, src, opsize, 1, NULL); 3116 reg = gen_extend(s, DREG(insn, 9), opsize, 1); 3117 gen_update_cc_cmp(s, reg, src, opsize); 3118 } 3119 3120 DISAS_INSN(cmpa) 3121 { 3122 int opsize; 3123 TCGv src; 3124 TCGv reg; 3125 3126 if (insn & 0x100) { 3127 opsize = OS_LONG; 3128 } else { 3129 opsize = OS_WORD; 3130 } 3131 SRC_EA(env, src, opsize, 1, NULL); 3132 reg = AREG(insn, 9); 3133 gen_update_cc_cmp(s, reg, src, OS_LONG); 3134 } 3135 3136 DISAS_INSN(cmpm) 3137 { 3138 int opsize = insn_opsize(insn); 3139 TCGv src, dst; 3140 3141 /* Post-increment load (mode 3) from Ay. */ 3142 src = gen_ea_mode(env, s, 3, REG(insn, 0), opsize, 3143 NULL_QREG, NULL, EA_LOADS, IS_USER(s)); 3144 /* Post-increment load (mode 3) from Ax. */ 3145 dst = gen_ea_mode(env, s, 3, REG(insn, 9), opsize, 3146 NULL_QREG, NULL, EA_LOADS, IS_USER(s)); 3147 3148 gen_update_cc_cmp(s, dst, src, opsize); 3149 } 3150 3151 DISAS_INSN(eor) 3152 { 3153 TCGv src; 3154 TCGv dest; 3155 TCGv addr; 3156 int opsize; 3157 3158 opsize = insn_opsize(insn); 3159 3160 SRC_EA(env, src, opsize, 0, &addr); 3161 dest = tcg_temp_new(); 3162 tcg_gen_xor_i32(dest, src, DREG(insn, 9)); 3163 gen_logic_cc(s, dest, opsize); 3164 DEST_EA(env, insn, opsize, dest, &addr); 3165 } 3166 3167 static void do_exg(TCGv reg1, TCGv reg2) 3168 { 3169 TCGv temp = tcg_temp_new(); 3170 tcg_gen_mov_i32(temp, reg1); 3171 tcg_gen_mov_i32(reg1, reg2); 3172 tcg_gen_mov_i32(reg2, temp); 3173 } 3174 3175 DISAS_INSN(exg_dd) 3176 { 3177 /* exchange Dx and Dy */ 3178 do_exg(DREG(insn, 9), DREG(insn, 0)); 3179 } 3180 3181 DISAS_INSN(exg_aa) 3182 { 3183 /* exchange Ax and Ay */ 3184 do_exg(AREG(insn, 9), AREG(insn, 0)); 3185 } 3186 3187 DISAS_INSN(exg_da) 3188 { 3189 /* exchange Dx and Ay */ 3190 do_exg(DREG(insn, 9), AREG(insn, 0)); 3191 } 3192 3193 DISAS_INSN(and) 3194 { 3195 TCGv src; 3196 TCGv reg; 3197 TCGv dest; 3198 TCGv addr; 3199 int opsize; 3200 3201 dest = tcg_temp_new(); 3202 3203 opsize = insn_opsize(insn); 3204 reg = DREG(insn, 9); 3205 if (insn & 0x100) { 3206 SRC_EA(env, src, opsize, 0, &addr); 3207 tcg_gen_and_i32(dest, src, reg); 3208 DEST_EA(env, insn, opsize, dest, &addr); 3209 } else { 3210 SRC_EA(env, src, opsize, 0, NULL); 3211 tcg_gen_and_i32(dest, src, reg); 3212 gen_partset_reg(opsize, reg, dest); 3213 } 3214 gen_logic_cc(s, dest, opsize); 3215 } 3216 3217 DISAS_INSN(adda) 3218 { 3219 TCGv src; 3220 TCGv reg; 3221 3222 SRC_EA(env, src, (insn & 0x100) ? OS_LONG : OS_WORD, 1, NULL); 3223 reg = AREG(insn, 9); 3224 tcg_gen_add_i32(reg, reg, src); 3225 } 3226 3227 static inline void gen_addx(DisasContext *s, TCGv src, TCGv dest, int opsize) 3228 { 3229 TCGv tmp, zero; 3230 3231 gen_flush_flags(s); /* compute old Z */ 3232 3233 /* 3234 * Perform addition with carry. 3235 * (X, N) = src + dest + X; 3236 */ 3237 3238 zero = tcg_constant_i32(0); 3239 tcg_gen_add2_i32(QREG_CC_N, QREG_CC_X, QREG_CC_X, zero, dest, zero); 3240 tcg_gen_add2_i32(QREG_CC_N, QREG_CC_X, QREG_CC_N, QREG_CC_X, src, zero); 3241 gen_ext(QREG_CC_N, QREG_CC_N, opsize, 1); 3242 3243 /* Compute signed-overflow for addition. */ 3244 3245 tmp = tcg_temp_new(); 3246 tcg_gen_xor_i32(QREG_CC_V, QREG_CC_N, src); 3247 tcg_gen_xor_i32(tmp, dest, src); 3248 tcg_gen_andc_i32(QREG_CC_V, QREG_CC_V, tmp); 3249 3250 /* Copy the rest of the results into place. */ 3251 tcg_gen_or_i32(QREG_CC_Z, QREG_CC_Z, QREG_CC_N); /* !Z is sticky */ 3252 tcg_gen_mov_i32(QREG_CC_C, QREG_CC_X); 3253 3254 set_cc_op(s, CC_OP_FLAGS); 3255 3256 /* result is in QREG_CC_N */ 3257 } 3258 3259 DISAS_INSN(addx_reg) 3260 { 3261 TCGv dest; 3262 TCGv src; 3263 int opsize; 3264 3265 opsize = insn_opsize(insn); 3266 3267 dest = gen_extend(s, DREG(insn, 9), opsize, 1); 3268 src = gen_extend(s, DREG(insn, 0), opsize, 1); 3269 3270 gen_addx(s, src, dest, opsize); 3271 3272 gen_partset_reg(opsize, DREG(insn, 9), QREG_CC_N); 3273 } 3274 3275 DISAS_INSN(addx_mem) 3276 { 3277 TCGv src; 3278 TCGv addr_src; 3279 TCGv dest; 3280 TCGv addr_dest; 3281 int opsize; 3282 3283 opsize = insn_opsize(insn); 3284 3285 addr_src = AREG(insn, 0); 3286 tcg_gen_subi_i32(addr_src, addr_src, opsize_bytes(opsize)); 3287 src = gen_load(s, opsize, addr_src, 1, IS_USER(s)); 3288 3289 addr_dest = AREG(insn, 9); 3290 tcg_gen_subi_i32(addr_dest, addr_dest, opsize_bytes(opsize)); 3291 dest = gen_load(s, opsize, addr_dest, 1, IS_USER(s)); 3292 3293 gen_addx(s, src, dest, opsize); 3294 3295 gen_store(s, opsize, addr_dest, QREG_CC_N, IS_USER(s)); 3296 } 3297 3298 static inline void shift_im(DisasContext *s, uint16_t insn, int opsize) 3299 { 3300 int count = (insn >> 9) & 7; 3301 int logical = insn & 8; 3302 int left = insn & 0x100; 3303 int bits = opsize_bytes(opsize) * 8; 3304 TCGv reg = gen_extend(s, DREG(insn, 0), opsize, !logical); 3305 3306 if (count == 0) { 3307 count = 8; 3308 } 3309 3310 tcg_gen_movi_i32(QREG_CC_V, 0); 3311 if (left) { 3312 tcg_gen_shri_i32(QREG_CC_C, reg, bits - count); 3313 tcg_gen_shli_i32(QREG_CC_N, reg, count); 3314 3315 /* 3316 * Note that ColdFire always clears V (done above), 3317 * while M68000 sets if the most significant bit is changed at 3318 * any time during the shift operation. 3319 */ 3320 if (!logical && m68k_feature(s->env, M68K_FEATURE_M68K)) { 3321 /* if shift count >= bits, V is (reg != 0) */ 3322 if (count >= bits) { 3323 tcg_gen_negsetcond_i32(TCG_COND_NE, QREG_CC_V, reg, QREG_CC_V); 3324 } else { 3325 TCGv t0 = tcg_temp_new(); 3326 tcg_gen_sari_i32(QREG_CC_V, reg, bits - 1); 3327 tcg_gen_sari_i32(t0, reg, bits - count - 1); 3328 tcg_gen_negsetcond_i32(TCG_COND_NE, QREG_CC_V, QREG_CC_V, t0); 3329 } 3330 } 3331 } else { 3332 tcg_gen_shri_i32(QREG_CC_C, reg, count - 1); 3333 if (logical) { 3334 tcg_gen_shri_i32(QREG_CC_N, reg, count); 3335 } else { 3336 tcg_gen_sari_i32(QREG_CC_N, reg, count); 3337 } 3338 } 3339 3340 gen_ext(QREG_CC_N, QREG_CC_N, opsize, 1); 3341 tcg_gen_andi_i32(QREG_CC_C, QREG_CC_C, 1); 3342 tcg_gen_mov_i32(QREG_CC_Z, QREG_CC_N); 3343 tcg_gen_mov_i32(QREG_CC_X, QREG_CC_C); 3344 3345 gen_partset_reg(opsize, DREG(insn, 0), QREG_CC_N); 3346 set_cc_op(s, CC_OP_FLAGS); 3347 } 3348 3349 static inline void shift_reg(DisasContext *s, uint16_t insn, int opsize) 3350 { 3351 int logical = insn & 8; 3352 int left = insn & 0x100; 3353 int bits = opsize_bytes(opsize) * 8; 3354 TCGv reg = gen_extend(s, DREG(insn, 0), opsize, !logical); 3355 TCGv s32; 3356 TCGv_i64 t64, s64; 3357 3358 t64 = tcg_temp_new_i64(); 3359 s64 = tcg_temp_new_i64(); 3360 s32 = tcg_temp_new(); 3361 3362 /* 3363 * Note that m68k truncates the shift count modulo 64, not 32. 3364 * In addition, a 64-bit shift makes it easy to find "the last 3365 * bit shifted out", for the carry flag. 3366 */ 3367 tcg_gen_andi_i32(s32, DREG(insn, 9), 63); 3368 tcg_gen_extu_i32_i64(s64, s32); 3369 tcg_gen_extu_i32_i64(t64, reg); 3370 3371 /* Optimistically set V=0. Also used as a zero source below. */ 3372 tcg_gen_movi_i32(QREG_CC_V, 0); 3373 if (left) { 3374 tcg_gen_shl_i64(t64, t64, s64); 3375 3376 if (opsize == OS_LONG) { 3377 tcg_gen_extr_i64_i32(QREG_CC_N, QREG_CC_C, t64); 3378 /* Note that C=0 if shift count is 0, and we get that for free. */ 3379 } else { 3380 TCGv zero = tcg_constant_i32(0); 3381 tcg_gen_extrl_i64_i32(QREG_CC_N, t64); 3382 tcg_gen_shri_i32(QREG_CC_C, QREG_CC_N, bits); 3383 tcg_gen_movcond_i32(TCG_COND_EQ, QREG_CC_C, 3384 s32, zero, zero, QREG_CC_C); 3385 } 3386 tcg_gen_andi_i32(QREG_CC_C, QREG_CC_C, 1); 3387 3388 /* X = C, but only if the shift count was non-zero. */ 3389 tcg_gen_movcond_i32(TCG_COND_NE, QREG_CC_X, s32, QREG_CC_V, 3390 QREG_CC_C, QREG_CC_X); 3391 3392 /* 3393 * M68000 sets V if the most significant bit is changed at 3394 * any time during the shift operation. Do this via creating 3395 * an extension of the sign bit, comparing, and discarding 3396 * the bits below the sign bit. I.e. 3397 * int64_t s = (intN_t)reg; 3398 * int64_t t = (int64_t)(intN_t)reg << count; 3399 * V = ((s ^ t) & (-1 << (bits - 1))) != 0 3400 */ 3401 if (!logical && m68k_feature(s->env, M68K_FEATURE_M68K)) { 3402 TCGv_i64 tt = tcg_constant_i64(32); 3403 /* if shift is greater than 32, use 32 */ 3404 tcg_gen_movcond_i64(TCG_COND_GT, s64, s64, tt, tt, s64); 3405 /* Sign extend the input to 64 bits; re-do the shift. */ 3406 tcg_gen_ext_i32_i64(t64, reg); 3407 tcg_gen_shl_i64(s64, t64, s64); 3408 /* Clear all bits that are unchanged. */ 3409 tcg_gen_xor_i64(t64, t64, s64); 3410 /* Ignore the bits below the sign bit. */ 3411 tcg_gen_andi_i64(t64, t64, -1ULL << (bits - 1)); 3412 /* If any bits remain set, we have overflow. */ 3413 tcg_gen_negsetcond_i64(TCG_COND_NE, t64, t64, tcg_constant_i64(0)); 3414 tcg_gen_extrl_i64_i32(QREG_CC_V, t64); 3415 } 3416 } else { 3417 tcg_gen_shli_i64(t64, t64, 32); 3418 if (logical) { 3419 tcg_gen_shr_i64(t64, t64, s64); 3420 } else { 3421 tcg_gen_sar_i64(t64, t64, s64); 3422 } 3423 tcg_gen_extr_i64_i32(QREG_CC_C, QREG_CC_N, t64); 3424 3425 /* Note that C=0 if shift count is 0, and we get that for free. */ 3426 tcg_gen_shri_i32(QREG_CC_C, QREG_CC_C, 31); 3427 3428 /* X = C, but only if the shift count was non-zero. */ 3429 tcg_gen_movcond_i32(TCG_COND_NE, QREG_CC_X, s32, QREG_CC_V, 3430 QREG_CC_C, QREG_CC_X); 3431 } 3432 gen_ext(QREG_CC_N, QREG_CC_N, opsize, 1); 3433 tcg_gen_mov_i32(QREG_CC_Z, QREG_CC_N); 3434 3435 /* Write back the result. */ 3436 gen_partset_reg(opsize, DREG(insn, 0), QREG_CC_N); 3437 set_cc_op(s, CC_OP_FLAGS); 3438 } 3439 3440 DISAS_INSN(shift8_im) 3441 { 3442 shift_im(s, insn, OS_BYTE); 3443 } 3444 3445 DISAS_INSN(shift16_im) 3446 { 3447 shift_im(s, insn, OS_WORD); 3448 } 3449 3450 DISAS_INSN(shift_im) 3451 { 3452 shift_im(s, insn, OS_LONG); 3453 } 3454 3455 DISAS_INSN(shift8_reg) 3456 { 3457 shift_reg(s, insn, OS_BYTE); 3458 } 3459 3460 DISAS_INSN(shift16_reg) 3461 { 3462 shift_reg(s, insn, OS_WORD); 3463 } 3464 3465 DISAS_INSN(shift_reg) 3466 { 3467 shift_reg(s, insn, OS_LONG); 3468 } 3469 3470 DISAS_INSN(shift_mem) 3471 { 3472 int logical = insn & 8; 3473 int left = insn & 0x100; 3474 TCGv src; 3475 TCGv addr; 3476 3477 SRC_EA(env, src, OS_WORD, !logical, &addr); 3478 tcg_gen_movi_i32(QREG_CC_V, 0); 3479 if (left) { 3480 tcg_gen_shri_i32(QREG_CC_C, src, 15); 3481 tcg_gen_shli_i32(QREG_CC_N, src, 1); 3482 3483 /* 3484 * Note that ColdFire always clears V, 3485 * while M68000 sets if the most significant bit is changed at 3486 * any time during the shift operation 3487 */ 3488 if (!logical && m68k_feature(s->env, M68K_FEATURE_M68K)) { 3489 src = gen_extend(s, src, OS_WORD, 1); 3490 tcg_gen_xor_i32(QREG_CC_V, QREG_CC_N, src); 3491 } 3492 } else { 3493 tcg_gen_mov_i32(QREG_CC_C, src); 3494 if (logical) { 3495 tcg_gen_shri_i32(QREG_CC_N, src, 1); 3496 } else { 3497 tcg_gen_sari_i32(QREG_CC_N, src, 1); 3498 } 3499 } 3500 3501 gen_ext(QREG_CC_N, QREG_CC_N, OS_WORD, 1); 3502 tcg_gen_andi_i32(QREG_CC_C, QREG_CC_C, 1); 3503 tcg_gen_mov_i32(QREG_CC_Z, QREG_CC_N); 3504 tcg_gen_mov_i32(QREG_CC_X, QREG_CC_C); 3505 3506 DEST_EA(env, insn, OS_WORD, QREG_CC_N, &addr); 3507 set_cc_op(s, CC_OP_FLAGS); 3508 } 3509 3510 static void rotate(TCGv reg, TCGv shift, int left, int size) 3511 { 3512 switch (size) { 3513 case 8: 3514 /* Replicate the 8-bit input so that a 32-bit rotate works. */ 3515 tcg_gen_ext8u_i32(reg, reg); 3516 tcg_gen_muli_i32(reg, reg, 0x01010101); 3517 goto do_long; 3518 case 16: 3519 /* Replicate the 16-bit input so that a 32-bit rotate works. */ 3520 tcg_gen_deposit_i32(reg, reg, reg, 16, 16); 3521 goto do_long; 3522 do_long: 3523 default: 3524 if (left) { 3525 tcg_gen_rotl_i32(reg, reg, shift); 3526 } else { 3527 tcg_gen_rotr_i32(reg, reg, shift); 3528 } 3529 } 3530 3531 /* compute flags */ 3532 3533 switch (size) { 3534 case 8: 3535 tcg_gen_ext8s_i32(reg, reg); 3536 break; 3537 case 16: 3538 tcg_gen_ext16s_i32(reg, reg); 3539 break; 3540 default: 3541 break; 3542 } 3543 3544 /* QREG_CC_X is not affected */ 3545 3546 tcg_gen_mov_i32(QREG_CC_N, reg); 3547 tcg_gen_mov_i32(QREG_CC_Z, reg); 3548 3549 if (left) { 3550 tcg_gen_andi_i32(QREG_CC_C, reg, 1); 3551 } else { 3552 tcg_gen_shri_i32(QREG_CC_C, reg, 31); 3553 } 3554 3555 tcg_gen_movi_i32(QREG_CC_V, 0); /* always cleared */ 3556 } 3557 3558 static void rotate_x_flags(TCGv reg, TCGv X, int size) 3559 { 3560 switch (size) { 3561 case 8: 3562 tcg_gen_ext8s_i32(reg, reg); 3563 break; 3564 case 16: 3565 tcg_gen_ext16s_i32(reg, reg); 3566 break; 3567 default: 3568 break; 3569 } 3570 tcg_gen_mov_i32(QREG_CC_N, reg); 3571 tcg_gen_mov_i32(QREG_CC_Z, reg); 3572 tcg_gen_mov_i32(QREG_CC_X, X); 3573 tcg_gen_mov_i32(QREG_CC_C, X); 3574 tcg_gen_movi_i32(QREG_CC_V, 0); 3575 } 3576 3577 /* Result of rotate_x() is valid if 0 <= shift <= size */ 3578 static TCGv rotate_x(TCGv reg, TCGv shift, int left, int size) 3579 { 3580 TCGv X, shl, shr, shx, sz, zero; 3581 3582 sz = tcg_constant_i32(size); 3583 3584 shr = tcg_temp_new(); 3585 shl = tcg_temp_new(); 3586 shx = tcg_temp_new(); 3587 if (left) { 3588 tcg_gen_mov_i32(shl, shift); /* shl = shift */ 3589 tcg_gen_movi_i32(shr, size + 1); 3590 tcg_gen_sub_i32(shr, shr, shift); /* shr = size + 1 - shift */ 3591 tcg_gen_subi_i32(shx, shift, 1); /* shx = shift - 1 */ 3592 /* shx = shx < 0 ? size : shx; */ 3593 zero = tcg_constant_i32(0); 3594 tcg_gen_movcond_i32(TCG_COND_LT, shx, shx, zero, sz, shx); 3595 } else { 3596 tcg_gen_mov_i32(shr, shift); /* shr = shift */ 3597 tcg_gen_movi_i32(shl, size + 1); 3598 tcg_gen_sub_i32(shl, shl, shift); /* shl = size + 1 - shift */ 3599 tcg_gen_sub_i32(shx, sz, shift); /* shx = size - shift */ 3600 } 3601 3602 /* reg = (reg << shl) | (reg >> shr) | (x << shx); */ 3603 3604 tcg_gen_shl_i32(shl, reg, shl); 3605 tcg_gen_shr_i32(shr, reg, shr); 3606 tcg_gen_or_i32(reg, shl, shr); 3607 tcg_gen_shl_i32(shx, QREG_CC_X, shx); 3608 tcg_gen_or_i32(reg, reg, shx); 3609 3610 /* X = (reg >> size) & 1 */ 3611 3612 X = tcg_temp_new(); 3613 tcg_gen_extract_i32(X, reg, size, 1); 3614 3615 return X; 3616 } 3617 3618 /* Result of rotate32_x() is valid if 0 <= shift < 33 */ 3619 static TCGv rotate32_x(TCGv reg, TCGv shift, int left) 3620 { 3621 TCGv_i64 t0, shift64; 3622 TCGv X, lo, hi, zero; 3623 3624 shift64 = tcg_temp_new_i64(); 3625 tcg_gen_extu_i32_i64(shift64, shift); 3626 3627 t0 = tcg_temp_new_i64(); 3628 3629 X = tcg_temp_new(); 3630 lo = tcg_temp_new(); 3631 hi = tcg_temp_new(); 3632 3633 if (left) { 3634 /* create [reg:X:..] */ 3635 3636 tcg_gen_shli_i32(lo, QREG_CC_X, 31); 3637 tcg_gen_concat_i32_i64(t0, lo, reg); 3638 3639 /* rotate */ 3640 3641 tcg_gen_rotl_i64(t0, t0, shift64); 3642 3643 /* result is [reg:..:reg:X] */ 3644 3645 tcg_gen_extr_i64_i32(lo, hi, t0); 3646 tcg_gen_andi_i32(X, lo, 1); 3647 3648 tcg_gen_shri_i32(lo, lo, 1); 3649 } else { 3650 /* create [..:X:reg] */ 3651 3652 tcg_gen_concat_i32_i64(t0, reg, QREG_CC_X); 3653 3654 tcg_gen_rotr_i64(t0, t0, shift64); 3655 3656 /* result is value: [X:reg:..:reg] */ 3657 3658 tcg_gen_extr_i64_i32(lo, hi, t0); 3659 3660 /* extract X */ 3661 3662 tcg_gen_shri_i32(X, hi, 31); 3663 3664 /* extract result */ 3665 3666 tcg_gen_shli_i32(hi, hi, 1); 3667 } 3668 tcg_gen_or_i32(lo, lo, hi); 3669 3670 /* if shift == 0, register and X are not affected */ 3671 3672 zero = tcg_constant_i32(0); 3673 tcg_gen_movcond_i32(TCG_COND_EQ, X, shift, zero, QREG_CC_X, X); 3674 tcg_gen_movcond_i32(TCG_COND_EQ, reg, shift, zero, reg, lo); 3675 3676 return X; 3677 } 3678 3679 DISAS_INSN(rotate_im) 3680 { 3681 TCGv shift; 3682 int tmp; 3683 int left = (insn & 0x100); 3684 3685 tmp = (insn >> 9) & 7; 3686 if (tmp == 0) { 3687 tmp = 8; 3688 } 3689 3690 shift = tcg_constant_i32(tmp); 3691 if (insn & 8) { 3692 rotate(DREG(insn, 0), shift, left, 32); 3693 } else { 3694 TCGv X = rotate32_x(DREG(insn, 0), shift, left); 3695 rotate_x_flags(DREG(insn, 0), X, 32); 3696 } 3697 3698 set_cc_op(s, CC_OP_FLAGS); 3699 } 3700 3701 DISAS_INSN(rotate8_im) 3702 { 3703 int left = (insn & 0x100); 3704 TCGv reg; 3705 TCGv shift; 3706 int tmp; 3707 3708 reg = gen_extend(s, DREG(insn, 0), OS_BYTE, 0); 3709 3710 tmp = (insn >> 9) & 7; 3711 if (tmp == 0) { 3712 tmp = 8; 3713 } 3714 3715 shift = tcg_constant_i32(tmp); 3716 if (insn & 8) { 3717 rotate(reg, shift, left, 8); 3718 } else { 3719 TCGv X = rotate_x(reg, shift, left, 8); 3720 rotate_x_flags(reg, X, 8); 3721 } 3722 gen_partset_reg(OS_BYTE, DREG(insn, 0), reg); 3723 set_cc_op(s, CC_OP_FLAGS); 3724 } 3725 3726 DISAS_INSN(rotate16_im) 3727 { 3728 int left = (insn & 0x100); 3729 TCGv reg; 3730 TCGv shift; 3731 int tmp; 3732 3733 reg = gen_extend(s, DREG(insn, 0), OS_WORD, 0); 3734 tmp = (insn >> 9) & 7; 3735 if (tmp == 0) { 3736 tmp = 8; 3737 } 3738 3739 shift = tcg_constant_i32(tmp); 3740 if (insn & 8) { 3741 rotate(reg, shift, left, 16); 3742 } else { 3743 TCGv X = rotate_x(reg, shift, left, 16); 3744 rotate_x_flags(reg, X, 16); 3745 } 3746 gen_partset_reg(OS_WORD, DREG(insn, 0), reg); 3747 set_cc_op(s, CC_OP_FLAGS); 3748 } 3749 3750 DISAS_INSN(rotate_reg) 3751 { 3752 TCGv reg; 3753 TCGv src; 3754 TCGv t0, t1; 3755 int left = (insn & 0x100); 3756 3757 reg = DREG(insn, 0); 3758 src = DREG(insn, 9); 3759 /* shift in [0..63] */ 3760 t0 = tcg_temp_new(); 3761 tcg_gen_andi_i32(t0, src, 63); 3762 t1 = tcg_temp_new_i32(); 3763 if (insn & 8) { 3764 tcg_gen_andi_i32(t1, src, 31); 3765 rotate(reg, t1, left, 32); 3766 /* if shift == 0, clear C */ 3767 tcg_gen_movcond_i32(TCG_COND_EQ, QREG_CC_C, 3768 t0, QREG_CC_V /* 0 */, 3769 QREG_CC_V /* 0 */, QREG_CC_C); 3770 } else { 3771 TCGv X; 3772 /* modulo 33 */ 3773 tcg_gen_movi_i32(t1, 33); 3774 tcg_gen_remu_i32(t1, t0, t1); 3775 X = rotate32_x(DREG(insn, 0), t1, left); 3776 rotate_x_flags(DREG(insn, 0), X, 32); 3777 } 3778 set_cc_op(s, CC_OP_FLAGS); 3779 } 3780 3781 DISAS_INSN(rotate8_reg) 3782 { 3783 TCGv reg; 3784 TCGv src; 3785 TCGv t0, t1; 3786 int left = (insn & 0x100); 3787 3788 reg = gen_extend(s, DREG(insn, 0), OS_BYTE, 0); 3789 src = DREG(insn, 9); 3790 /* shift in [0..63] */ 3791 t0 = tcg_temp_new_i32(); 3792 tcg_gen_andi_i32(t0, src, 63); 3793 t1 = tcg_temp_new_i32(); 3794 if (insn & 8) { 3795 tcg_gen_andi_i32(t1, src, 7); 3796 rotate(reg, t1, left, 8); 3797 /* if shift == 0, clear C */ 3798 tcg_gen_movcond_i32(TCG_COND_EQ, QREG_CC_C, 3799 t0, QREG_CC_V /* 0 */, 3800 QREG_CC_V /* 0 */, QREG_CC_C); 3801 } else { 3802 TCGv X; 3803 /* modulo 9 */ 3804 tcg_gen_movi_i32(t1, 9); 3805 tcg_gen_remu_i32(t1, t0, t1); 3806 X = rotate_x(reg, t1, left, 8); 3807 rotate_x_flags(reg, X, 8); 3808 } 3809 gen_partset_reg(OS_BYTE, DREG(insn, 0), reg); 3810 set_cc_op(s, CC_OP_FLAGS); 3811 } 3812 3813 DISAS_INSN(rotate16_reg) 3814 { 3815 TCGv reg; 3816 TCGv src; 3817 TCGv t0, t1; 3818 int left = (insn & 0x100); 3819 3820 reg = gen_extend(s, DREG(insn, 0), OS_WORD, 0); 3821 src = DREG(insn, 9); 3822 /* shift in [0..63] */ 3823 t0 = tcg_temp_new_i32(); 3824 tcg_gen_andi_i32(t0, src, 63); 3825 t1 = tcg_temp_new_i32(); 3826 if (insn & 8) { 3827 tcg_gen_andi_i32(t1, src, 15); 3828 rotate(reg, t1, left, 16); 3829 /* if shift == 0, clear C */ 3830 tcg_gen_movcond_i32(TCG_COND_EQ, QREG_CC_C, 3831 t0, QREG_CC_V /* 0 */, 3832 QREG_CC_V /* 0 */, QREG_CC_C); 3833 } else { 3834 TCGv X; 3835 /* modulo 17 */ 3836 tcg_gen_movi_i32(t1, 17); 3837 tcg_gen_remu_i32(t1, t0, t1); 3838 X = rotate_x(reg, t1, left, 16); 3839 rotate_x_flags(reg, X, 16); 3840 } 3841 gen_partset_reg(OS_WORD, DREG(insn, 0), reg); 3842 set_cc_op(s, CC_OP_FLAGS); 3843 } 3844 3845 DISAS_INSN(rotate_mem) 3846 { 3847 TCGv src; 3848 TCGv addr; 3849 TCGv shift; 3850 int left = (insn & 0x100); 3851 3852 SRC_EA(env, src, OS_WORD, 0, &addr); 3853 3854 shift = tcg_constant_i32(1); 3855 if (insn & 0x0200) { 3856 rotate(src, shift, left, 16); 3857 } else { 3858 TCGv X = rotate_x(src, shift, left, 16); 3859 rotate_x_flags(src, X, 16); 3860 } 3861 DEST_EA(env, insn, OS_WORD, src, &addr); 3862 set_cc_op(s, CC_OP_FLAGS); 3863 } 3864 3865 DISAS_INSN(bfext_reg) 3866 { 3867 int ext = read_im16(env, s); 3868 int is_sign = insn & 0x200; 3869 TCGv src = DREG(insn, 0); 3870 TCGv dst = DREG(ext, 12); 3871 int len = ((extract32(ext, 0, 5) - 1) & 31) + 1; 3872 int ofs = extract32(ext, 6, 5); /* big bit-endian */ 3873 int pos = 32 - ofs - len; /* little bit-endian */ 3874 TCGv tmp = tcg_temp_new(); 3875 TCGv shift; 3876 3877 /* 3878 * In general, we're going to rotate the field so that it's at the 3879 * top of the word and then right-shift by the complement of the 3880 * width to extend the field. 3881 */ 3882 if (ext & 0x20) { 3883 /* Variable width. */ 3884 if (ext & 0x800) { 3885 /* Variable offset. */ 3886 tcg_gen_andi_i32(tmp, DREG(ext, 6), 31); 3887 tcg_gen_rotl_i32(tmp, src, tmp); 3888 } else { 3889 tcg_gen_rotli_i32(tmp, src, ofs); 3890 } 3891 3892 shift = tcg_temp_new(); 3893 tcg_gen_neg_i32(shift, DREG(ext, 0)); 3894 tcg_gen_andi_i32(shift, shift, 31); 3895 tcg_gen_sar_i32(QREG_CC_N, tmp, shift); 3896 if (is_sign) { 3897 tcg_gen_mov_i32(dst, QREG_CC_N); 3898 } else { 3899 tcg_gen_shr_i32(dst, tmp, shift); 3900 } 3901 } else { 3902 /* Immediate width. */ 3903 if (ext & 0x800) { 3904 /* Variable offset */ 3905 tcg_gen_andi_i32(tmp, DREG(ext, 6), 31); 3906 tcg_gen_rotl_i32(tmp, src, tmp); 3907 src = tmp; 3908 pos = 32 - len; 3909 } else { 3910 /* 3911 * Immediate offset. If the field doesn't wrap around the 3912 * end of the word, rely on (s)extract completely. 3913 */ 3914 if (pos < 0) { 3915 tcg_gen_rotli_i32(tmp, src, ofs); 3916 src = tmp; 3917 pos = 32 - len; 3918 } 3919 } 3920 3921 tcg_gen_sextract_i32(QREG_CC_N, src, pos, len); 3922 if (is_sign) { 3923 tcg_gen_mov_i32(dst, QREG_CC_N); 3924 } else { 3925 tcg_gen_extract_i32(dst, src, pos, len); 3926 } 3927 } 3928 3929 set_cc_op(s, CC_OP_LOGIC); 3930 } 3931 3932 DISAS_INSN(bfext_mem) 3933 { 3934 int ext = read_im16(env, s); 3935 int is_sign = insn & 0x200; 3936 TCGv dest = DREG(ext, 12); 3937 TCGv addr, len, ofs; 3938 3939 addr = gen_lea(env, s, insn, OS_UNSIZED); 3940 if (IS_NULL_QREG(addr)) { 3941 gen_addr_fault(s); 3942 return; 3943 } 3944 3945 if (ext & 0x20) { 3946 len = DREG(ext, 0); 3947 } else { 3948 len = tcg_constant_i32(extract32(ext, 0, 5)); 3949 } 3950 if (ext & 0x800) { 3951 ofs = DREG(ext, 6); 3952 } else { 3953 ofs = tcg_constant_i32(extract32(ext, 6, 5)); 3954 } 3955 3956 if (is_sign) { 3957 gen_helper_bfexts_mem(dest, tcg_env, addr, ofs, len); 3958 tcg_gen_mov_i32(QREG_CC_N, dest); 3959 } else { 3960 TCGv_i64 tmp = tcg_temp_new_i64(); 3961 gen_helper_bfextu_mem(tmp, tcg_env, addr, ofs, len); 3962 tcg_gen_extr_i64_i32(dest, QREG_CC_N, tmp); 3963 } 3964 set_cc_op(s, CC_OP_LOGIC); 3965 } 3966 3967 DISAS_INSN(bfop_reg) 3968 { 3969 int ext = read_im16(env, s); 3970 TCGv src = DREG(insn, 0); 3971 int len = ((extract32(ext, 0, 5) - 1) & 31) + 1; 3972 int ofs = extract32(ext, 6, 5); /* big bit-endian */ 3973 TCGv mask, tofs = NULL, tlen = NULL; 3974 bool is_bfffo = (insn & 0x0f00) == 0x0d00; 3975 3976 if ((ext & 0x820) == 0) { 3977 /* Immediate width and offset. */ 3978 uint32_t maski = 0x7fffffffu >> (len - 1); 3979 if (ofs + len <= 32) { 3980 tcg_gen_shli_i32(QREG_CC_N, src, ofs); 3981 } else { 3982 tcg_gen_rotli_i32(QREG_CC_N, src, ofs); 3983 } 3984 tcg_gen_andi_i32(QREG_CC_N, QREG_CC_N, ~maski); 3985 3986 mask = tcg_constant_i32(ror32(maski, ofs)); 3987 if (is_bfffo) { 3988 tofs = tcg_constant_i32(ofs); 3989 tlen = tcg_constant_i32(len); 3990 } 3991 } else { 3992 TCGv tmp = tcg_temp_new(); 3993 3994 mask = tcg_temp_new(); 3995 if (ext & 0x20) { 3996 /* Variable width */ 3997 tcg_gen_subi_i32(tmp, DREG(ext, 0), 1); 3998 tcg_gen_andi_i32(tmp, tmp, 31); 3999 tcg_gen_shr_i32(mask, tcg_constant_i32(0x7fffffffu), tmp); 4000 if (is_bfffo) { 4001 tlen = tcg_temp_new(); 4002 tcg_gen_addi_i32(tlen, tmp, 1); 4003 } 4004 } else { 4005 /* Immediate width */ 4006 tcg_gen_movi_i32(mask, 0x7fffffffu >> (len - 1)); 4007 if (is_bfffo) { 4008 tlen = tcg_constant_i32(len); 4009 } 4010 } 4011 4012 if (ext & 0x800) { 4013 /* Variable offset */ 4014 tcg_gen_andi_i32(tmp, DREG(ext, 6), 31); 4015 tcg_gen_rotl_i32(QREG_CC_N, src, tmp); 4016 tcg_gen_andc_i32(QREG_CC_N, QREG_CC_N, mask); 4017 tcg_gen_rotr_i32(mask, mask, tmp); 4018 if (is_bfffo) { 4019 tofs = tmp; 4020 } 4021 } else { 4022 /* Immediate offset (and variable width) */ 4023 tcg_gen_rotli_i32(QREG_CC_N, src, ofs); 4024 tcg_gen_andc_i32(QREG_CC_N, QREG_CC_N, mask); 4025 tcg_gen_rotri_i32(mask, mask, ofs); 4026 if (is_bfffo) { 4027 tofs = tcg_constant_i32(ofs); 4028 } 4029 } 4030 } 4031 set_cc_op(s, CC_OP_LOGIC); 4032 4033 switch (insn & 0x0f00) { 4034 case 0x0a00: /* bfchg */ 4035 tcg_gen_eqv_i32(src, src, mask); 4036 break; 4037 case 0x0c00: /* bfclr */ 4038 tcg_gen_and_i32(src, src, mask); 4039 break; 4040 case 0x0d00: /* bfffo */ 4041 gen_helper_bfffo_reg(DREG(ext, 12), QREG_CC_N, tofs, tlen); 4042 break; 4043 case 0x0e00: /* bfset */ 4044 tcg_gen_orc_i32(src, src, mask); 4045 break; 4046 case 0x0800: /* bftst */ 4047 /* flags already set; no other work to do. */ 4048 break; 4049 default: 4050 g_assert_not_reached(); 4051 } 4052 } 4053 4054 DISAS_INSN(bfop_mem) 4055 { 4056 int ext = read_im16(env, s); 4057 TCGv addr, len, ofs; 4058 TCGv_i64 t64; 4059 4060 addr = gen_lea(env, s, insn, OS_UNSIZED); 4061 if (IS_NULL_QREG(addr)) { 4062 gen_addr_fault(s); 4063 return; 4064 } 4065 4066 if (ext & 0x20) { 4067 len = DREG(ext, 0); 4068 } else { 4069 len = tcg_constant_i32(extract32(ext, 0, 5)); 4070 } 4071 if (ext & 0x800) { 4072 ofs = DREG(ext, 6); 4073 } else { 4074 ofs = tcg_constant_i32(extract32(ext, 6, 5)); 4075 } 4076 4077 switch (insn & 0x0f00) { 4078 case 0x0a00: /* bfchg */ 4079 gen_helper_bfchg_mem(QREG_CC_N, tcg_env, addr, ofs, len); 4080 break; 4081 case 0x0c00: /* bfclr */ 4082 gen_helper_bfclr_mem(QREG_CC_N, tcg_env, addr, ofs, len); 4083 break; 4084 case 0x0d00: /* bfffo */ 4085 t64 = tcg_temp_new_i64(); 4086 gen_helper_bfffo_mem(t64, tcg_env, addr, ofs, len); 4087 tcg_gen_extr_i64_i32(DREG(ext, 12), QREG_CC_N, t64); 4088 break; 4089 case 0x0e00: /* bfset */ 4090 gen_helper_bfset_mem(QREG_CC_N, tcg_env, addr, ofs, len); 4091 break; 4092 case 0x0800: /* bftst */ 4093 gen_helper_bfexts_mem(QREG_CC_N, tcg_env, addr, ofs, len); 4094 break; 4095 default: 4096 g_assert_not_reached(); 4097 } 4098 set_cc_op(s, CC_OP_LOGIC); 4099 } 4100 4101 DISAS_INSN(bfins_reg) 4102 { 4103 int ext = read_im16(env, s); 4104 TCGv dst = DREG(insn, 0); 4105 TCGv src = DREG(ext, 12); 4106 int len = ((extract32(ext, 0, 5) - 1) & 31) + 1; 4107 int ofs = extract32(ext, 6, 5); /* big bit-endian */ 4108 int pos = 32 - ofs - len; /* little bit-endian */ 4109 TCGv tmp; 4110 4111 tmp = tcg_temp_new(); 4112 4113 if (ext & 0x20) { 4114 /* Variable width */ 4115 tcg_gen_neg_i32(tmp, DREG(ext, 0)); 4116 tcg_gen_andi_i32(tmp, tmp, 31); 4117 tcg_gen_shl_i32(QREG_CC_N, src, tmp); 4118 } else { 4119 /* Immediate width */ 4120 tcg_gen_shli_i32(QREG_CC_N, src, 32 - len); 4121 } 4122 set_cc_op(s, CC_OP_LOGIC); 4123 4124 /* Immediate width and offset */ 4125 if ((ext & 0x820) == 0) { 4126 /* Check for suitability for deposit. */ 4127 if (pos >= 0) { 4128 tcg_gen_deposit_i32(dst, dst, src, pos, len); 4129 } else { 4130 uint32_t maski = -2U << (len - 1); 4131 uint32_t roti = (ofs + len) & 31; 4132 tcg_gen_andi_i32(tmp, src, ~maski); 4133 tcg_gen_rotri_i32(tmp, tmp, roti); 4134 tcg_gen_andi_i32(dst, dst, ror32(maski, roti)); 4135 tcg_gen_or_i32(dst, dst, tmp); 4136 } 4137 } else { 4138 TCGv mask = tcg_temp_new(); 4139 TCGv rot = tcg_temp_new(); 4140 4141 if (ext & 0x20) { 4142 /* Variable width */ 4143 tcg_gen_subi_i32(rot, DREG(ext, 0), 1); 4144 tcg_gen_andi_i32(rot, rot, 31); 4145 tcg_gen_movi_i32(mask, -2); 4146 tcg_gen_shl_i32(mask, mask, rot); 4147 tcg_gen_mov_i32(rot, DREG(ext, 0)); 4148 tcg_gen_andc_i32(tmp, src, mask); 4149 } else { 4150 /* Immediate width (variable offset) */ 4151 uint32_t maski = -2U << (len - 1); 4152 tcg_gen_andi_i32(tmp, src, ~maski); 4153 tcg_gen_movi_i32(mask, maski); 4154 tcg_gen_movi_i32(rot, len & 31); 4155 } 4156 if (ext & 0x800) { 4157 /* Variable offset */ 4158 tcg_gen_add_i32(rot, rot, DREG(ext, 6)); 4159 } else { 4160 /* Immediate offset (variable width) */ 4161 tcg_gen_addi_i32(rot, rot, ofs); 4162 } 4163 tcg_gen_andi_i32(rot, rot, 31); 4164 tcg_gen_rotr_i32(mask, mask, rot); 4165 tcg_gen_rotr_i32(tmp, tmp, rot); 4166 tcg_gen_and_i32(dst, dst, mask); 4167 tcg_gen_or_i32(dst, dst, tmp); 4168 } 4169 } 4170 4171 DISAS_INSN(bfins_mem) 4172 { 4173 int ext = read_im16(env, s); 4174 TCGv src = DREG(ext, 12); 4175 TCGv addr, len, ofs; 4176 4177 addr = gen_lea(env, s, insn, OS_UNSIZED); 4178 if (IS_NULL_QREG(addr)) { 4179 gen_addr_fault(s); 4180 return; 4181 } 4182 4183 if (ext & 0x20) { 4184 len = DREG(ext, 0); 4185 } else { 4186 len = tcg_constant_i32(extract32(ext, 0, 5)); 4187 } 4188 if (ext & 0x800) { 4189 ofs = DREG(ext, 6); 4190 } else { 4191 ofs = tcg_constant_i32(extract32(ext, 6, 5)); 4192 } 4193 4194 gen_helper_bfins_mem(QREG_CC_N, tcg_env, addr, src, ofs, len); 4195 set_cc_op(s, CC_OP_LOGIC); 4196 } 4197 4198 DISAS_INSN(ff1) 4199 { 4200 TCGv reg; 4201 reg = DREG(insn, 0); 4202 gen_logic_cc(s, reg, OS_LONG); 4203 gen_helper_ff1(reg, reg); 4204 } 4205 4206 DISAS_INSN(chk) 4207 { 4208 TCGv src, reg; 4209 int opsize; 4210 4211 switch ((insn >> 7) & 3) { 4212 case 3: 4213 opsize = OS_WORD; 4214 break; 4215 case 2: 4216 if (m68k_feature(env, M68K_FEATURE_CHK2)) { 4217 opsize = OS_LONG; 4218 break; 4219 } 4220 /* fallthru */ 4221 default: 4222 gen_exception(s, s->base.pc_next, EXCP_ILLEGAL); 4223 return; 4224 } 4225 SRC_EA(env, src, opsize, 1, NULL); 4226 reg = gen_extend(s, DREG(insn, 9), opsize, 1); 4227 4228 gen_flush_flags(s); 4229 gen_helper_chk(tcg_env, reg, src); 4230 } 4231 4232 DISAS_INSN(chk2) 4233 { 4234 uint16_t ext; 4235 TCGv addr1, addr2, bound1, bound2, reg; 4236 int opsize; 4237 4238 switch ((insn >> 9) & 3) { 4239 case 0: 4240 opsize = OS_BYTE; 4241 break; 4242 case 1: 4243 opsize = OS_WORD; 4244 break; 4245 case 2: 4246 opsize = OS_LONG; 4247 break; 4248 default: 4249 gen_exception(s, s->base.pc_next, EXCP_ILLEGAL); 4250 return; 4251 } 4252 4253 ext = read_im16(env, s); 4254 if ((ext & 0x0800) == 0) { 4255 gen_exception(s, s->base.pc_next, EXCP_ILLEGAL); 4256 return; 4257 } 4258 4259 addr1 = gen_lea(env, s, insn, OS_UNSIZED); 4260 addr2 = tcg_temp_new(); 4261 tcg_gen_addi_i32(addr2, addr1, opsize_bytes(opsize)); 4262 4263 bound1 = gen_load(s, opsize, addr1, 1, IS_USER(s)); 4264 bound2 = gen_load(s, opsize, addr2, 1, IS_USER(s)); 4265 4266 reg = tcg_temp_new(); 4267 if (ext & 0x8000) { 4268 tcg_gen_mov_i32(reg, AREG(ext, 12)); 4269 } else { 4270 gen_ext(reg, DREG(ext, 12), opsize, 1); 4271 } 4272 4273 gen_flush_flags(s); 4274 gen_helper_chk2(tcg_env, reg, bound1, bound2); 4275 } 4276 4277 static void m68k_copy_line(TCGv dst, TCGv src, int index) 4278 { 4279 TCGv addr; 4280 TCGv_i64 t0, t1; 4281 4282 addr = tcg_temp_new(); 4283 4284 t0 = tcg_temp_new_i64(); 4285 t1 = tcg_temp_new_i64(); 4286 4287 tcg_gen_andi_i32(addr, src, ~15); 4288 tcg_gen_qemu_ld_i64(t0, addr, index, MO_TEUQ); 4289 tcg_gen_addi_i32(addr, addr, 8); 4290 tcg_gen_qemu_ld_i64(t1, addr, index, MO_TEUQ); 4291 4292 tcg_gen_andi_i32(addr, dst, ~15); 4293 tcg_gen_qemu_st_i64(t0, addr, index, MO_TEUQ); 4294 tcg_gen_addi_i32(addr, addr, 8); 4295 tcg_gen_qemu_st_i64(t1, addr, index, MO_TEUQ); 4296 } 4297 4298 DISAS_INSN(move16_reg) 4299 { 4300 int index = IS_USER(s); 4301 TCGv tmp; 4302 uint16_t ext; 4303 4304 ext = read_im16(env, s); 4305 if ((ext & (1 << 15)) == 0) { 4306 gen_exception(s, s->base.pc_next, EXCP_ILLEGAL); 4307 } 4308 4309 m68k_copy_line(AREG(ext, 12), AREG(insn, 0), index); 4310 4311 /* Ax can be Ay, so save Ay before incrementing Ax */ 4312 tmp = tcg_temp_new(); 4313 tcg_gen_mov_i32(tmp, AREG(ext, 12)); 4314 tcg_gen_addi_i32(AREG(insn, 0), AREG(insn, 0), 16); 4315 tcg_gen_addi_i32(AREG(ext, 12), tmp, 16); 4316 } 4317 4318 DISAS_INSN(move16_mem) 4319 { 4320 int index = IS_USER(s); 4321 TCGv reg, addr; 4322 4323 reg = AREG(insn, 0); 4324 addr = tcg_constant_i32(read_im32(env, s)); 4325 4326 if ((insn >> 3) & 1) { 4327 /* MOVE16 (xxx).L, (Ay) */ 4328 m68k_copy_line(reg, addr, index); 4329 } else { 4330 /* MOVE16 (Ay), (xxx).L */ 4331 m68k_copy_line(addr, reg, index); 4332 } 4333 4334 if (((insn >> 3) & 2) == 0) { 4335 /* (Ay)+ */ 4336 tcg_gen_addi_i32(reg, reg, 16); 4337 } 4338 } 4339 4340 DISAS_INSN(strldsr) 4341 { 4342 uint16_t ext; 4343 uint32_t addr; 4344 4345 addr = s->pc - 2; 4346 ext = read_im16(env, s); 4347 if (ext != 0x46FC) { 4348 gen_exception(s, addr, EXCP_ILLEGAL); 4349 return; 4350 } 4351 ext = read_im16(env, s); 4352 if (IS_USER(s) || (ext & SR_S) == 0) { 4353 gen_exception(s, addr, EXCP_PRIVILEGE); 4354 return; 4355 } 4356 gen_push(s, gen_get_sr(s)); 4357 gen_set_sr_im(s, ext, 0); 4358 gen_exit_tb(s); 4359 } 4360 4361 DISAS_INSN(move_from_sr) 4362 { 4363 TCGv sr; 4364 4365 if (IS_USER(s) && m68k_feature(env, M68K_FEATURE_MOVEFROMSR_PRIV)) { 4366 gen_exception(s, s->base.pc_next, EXCP_PRIVILEGE); 4367 return; 4368 } 4369 sr = gen_get_sr(s); 4370 DEST_EA(env, insn, OS_WORD, sr, NULL); 4371 } 4372 4373 #if !defined(CONFIG_USER_ONLY) 4374 DISAS_INSN(moves) 4375 { 4376 int opsize; 4377 uint16_t ext; 4378 TCGv reg; 4379 TCGv addr; 4380 int extend; 4381 4382 if (IS_USER(s)) { 4383 gen_exception(s, s->base.pc_next, EXCP_PRIVILEGE); 4384 return; 4385 } 4386 4387 ext = read_im16(env, s); 4388 4389 opsize = insn_opsize(insn); 4390 4391 if (ext & 0x8000) { 4392 /* address register */ 4393 reg = AREG(ext, 12); 4394 extend = 1; 4395 } else { 4396 /* data register */ 4397 reg = DREG(ext, 12); 4398 extend = 0; 4399 } 4400 4401 addr = gen_lea(env, s, insn, opsize); 4402 if (IS_NULL_QREG(addr)) { 4403 gen_addr_fault(s); 4404 return; 4405 } 4406 4407 if (ext & 0x0800) { 4408 /* from reg to ea */ 4409 gen_store(s, opsize, addr, reg, DFC_INDEX(s)); 4410 } else { 4411 /* from ea to reg */ 4412 TCGv tmp = gen_load(s, opsize, addr, 0, SFC_INDEX(s)); 4413 if (extend) { 4414 gen_ext(reg, tmp, opsize, 1); 4415 } else { 4416 gen_partset_reg(opsize, reg, tmp); 4417 } 4418 } 4419 switch (extract32(insn, 3, 3)) { 4420 case 3: /* Indirect postincrement. */ 4421 tcg_gen_addi_i32(AREG(insn, 0), addr, 4422 REG(insn, 0) == 7 && opsize == OS_BYTE 4423 ? 2 4424 : opsize_bytes(opsize)); 4425 break; 4426 case 4: /* Indirect predecrememnt. */ 4427 tcg_gen_mov_i32(AREG(insn, 0), addr); 4428 break; 4429 } 4430 } 4431 4432 DISAS_INSN(move_to_sr) 4433 { 4434 if (IS_USER(s)) { 4435 gen_exception(s, s->base.pc_next, EXCP_PRIVILEGE); 4436 return; 4437 } 4438 gen_move_to_sr(env, s, insn, false); 4439 gen_exit_tb(s); 4440 } 4441 4442 DISAS_INSN(move_from_usp) 4443 { 4444 if (IS_USER(s)) { 4445 gen_exception(s, s->base.pc_next, EXCP_PRIVILEGE); 4446 return; 4447 } 4448 tcg_gen_ld_i32(AREG(insn, 0), tcg_env, 4449 offsetof(CPUM68KState, sp[M68K_USP])); 4450 } 4451 4452 DISAS_INSN(move_to_usp) 4453 { 4454 if (IS_USER(s)) { 4455 gen_exception(s, s->base.pc_next, EXCP_PRIVILEGE); 4456 return; 4457 } 4458 tcg_gen_st_i32(AREG(insn, 0), tcg_env, 4459 offsetof(CPUM68KState, sp[M68K_USP])); 4460 } 4461 4462 DISAS_INSN(halt) 4463 { 4464 if (IS_USER(s)) { 4465 gen_exception(s, s->base.pc_next, EXCP_PRIVILEGE); 4466 return; 4467 } 4468 4469 gen_exception(s, s->pc, EXCP_HALT_INSN); 4470 } 4471 4472 DISAS_INSN(stop) 4473 { 4474 uint16_t ext; 4475 4476 if (IS_USER(s)) { 4477 gen_exception(s, s->base.pc_next, EXCP_PRIVILEGE); 4478 return; 4479 } 4480 4481 ext = read_im16(env, s); 4482 4483 gen_set_sr_im(s, ext, 0); 4484 tcg_gen_movi_i32(cpu_halted, 1); 4485 gen_exception(s, s->pc, EXCP_HLT); 4486 } 4487 4488 DISAS_INSN(rte) 4489 { 4490 if (IS_USER(s)) { 4491 gen_exception(s, s->base.pc_next, EXCP_PRIVILEGE); 4492 return; 4493 } 4494 gen_exception(s, s->base.pc_next, EXCP_RTE); 4495 } 4496 4497 DISAS_INSN(cf_movec) 4498 { 4499 uint16_t ext; 4500 TCGv reg; 4501 4502 if (IS_USER(s)) { 4503 gen_exception(s, s->base.pc_next, EXCP_PRIVILEGE); 4504 return; 4505 } 4506 4507 ext = read_im16(env, s); 4508 4509 if (ext & 0x8000) { 4510 reg = AREG(ext, 12); 4511 } else { 4512 reg = DREG(ext, 12); 4513 } 4514 gen_helper_cf_movec_to(tcg_env, tcg_constant_i32(ext & 0xfff), reg); 4515 gen_exit_tb(s); 4516 } 4517 4518 DISAS_INSN(m68k_movec) 4519 { 4520 uint16_t ext; 4521 TCGv reg, creg; 4522 4523 if (IS_USER(s)) { 4524 gen_exception(s, s->base.pc_next, EXCP_PRIVILEGE); 4525 return; 4526 } 4527 4528 ext = read_im16(env, s); 4529 4530 if (ext & 0x8000) { 4531 reg = AREG(ext, 12); 4532 } else { 4533 reg = DREG(ext, 12); 4534 } 4535 creg = tcg_constant_i32(ext & 0xfff); 4536 if (insn & 1) { 4537 gen_helper_m68k_movec_to(tcg_env, creg, reg); 4538 } else { 4539 gen_helper_m68k_movec_from(reg, tcg_env, creg); 4540 } 4541 gen_exit_tb(s); 4542 } 4543 4544 DISAS_INSN(intouch) 4545 { 4546 if (IS_USER(s)) { 4547 gen_exception(s, s->base.pc_next, EXCP_PRIVILEGE); 4548 return; 4549 } 4550 /* ICache fetch. Implement as no-op. */ 4551 } 4552 4553 DISAS_INSN(cpushl) 4554 { 4555 if (IS_USER(s)) { 4556 gen_exception(s, s->base.pc_next, EXCP_PRIVILEGE); 4557 return; 4558 } 4559 /* Cache push/invalidate. Implement as no-op. */ 4560 } 4561 4562 DISAS_INSN(cpush) 4563 { 4564 if (IS_USER(s)) { 4565 gen_exception(s, s->base.pc_next, EXCP_PRIVILEGE); 4566 return; 4567 } 4568 /* Cache push/invalidate. Implement as no-op. */ 4569 } 4570 4571 DISAS_INSN(cinv) 4572 { 4573 if (IS_USER(s)) { 4574 gen_exception(s, s->base.pc_next, EXCP_PRIVILEGE); 4575 return; 4576 } 4577 /* Invalidate cache line. Implement as no-op. */ 4578 } 4579 4580 #if !defined(CONFIG_USER_ONLY) 4581 DISAS_INSN(pflush) 4582 { 4583 TCGv opmode; 4584 4585 if (IS_USER(s)) { 4586 gen_exception(s, s->base.pc_next, EXCP_PRIVILEGE); 4587 return; 4588 } 4589 4590 opmode = tcg_constant_i32((insn >> 3) & 3); 4591 gen_helper_pflush(tcg_env, AREG(insn, 0), opmode); 4592 } 4593 4594 DISAS_INSN(ptest) 4595 { 4596 TCGv is_read; 4597 4598 if (IS_USER(s)) { 4599 gen_exception(s, s->base.pc_next, EXCP_PRIVILEGE); 4600 return; 4601 } 4602 is_read = tcg_constant_i32((insn >> 5) & 1); 4603 gen_helper_ptest(tcg_env, AREG(insn, 0), is_read); 4604 } 4605 #endif 4606 4607 DISAS_INSN(wddata) 4608 { 4609 gen_exception(s, s->base.pc_next, EXCP_PRIVILEGE); 4610 } 4611 4612 DISAS_INSN(wdebug) 4613 { 4614 if (IS_USER(s)) { 4615 gen_exception(s, s->base.pc_next, EXCP_PRIVILEGE); 4616 return; 4617 } 4618 /* TODO: Implement wdebug. */ 4619 cpu_abort(env_cpu(env), "WDEBUG not implemented"); 4620 } 4621 #endif 4622 4623 DISAS_INSN(trap) 4624 { 4625 gen_exception(s, s->pc, EXCP_TRAP0 + (insn & 0xf)); 4626 } 4627 4628 static void do_trapcc(DisasContext *s, DisasCompare *c) 4629 { 4630 if (c->tcond != TCG_COND_NEVER) { 4631 TCGLabel *over = NULL; 4632 4633 update_cc_op(s); 4634 4635 if (c->tcond != TCG_COND_ALWAYS) { 4636 /* Jump over if !c. */ 4637 over = gen_new_label(); 4638 tcg_gen_brcond_i32(tcg_invert_cond(c->tcond), c->v1, c->v2, over); 4639 } 4640 4641 tcg_gen_movi_i32(QREG_PC, s->pc); 4642 gen_raise_exception_format2(s, EXCP_TRAPCC, s->base.pc_next); 4643 4644 if (over != NULL) { 4645 gen_set_label(over); 4646 s->base.is_jmp = DISAS_NEXT; 4647 } 4648 } 4649 } 4650 4651 DISAS_INSN(trapcc) 4652 { 4653 DisasCompare c; 4654 4655 /* Consume and discard the immediate operand. */ 4656 switch (extract32(insn, 0, 3)) { 4657 case 2: /* trapcc.w */ 4658 (void)read_im16(env, s); 4659 break; 4660 case 3: /* trapcc.l */ 4661 (void)read_im32(env, s); 4662 break; 4663 case 4: /* trapcc (no operand) */ 4664 break; 4665 default: 4666 /* trapcc registered with only valid opmodes */ 4667 g_assert_not_reached(); 4668 } 4669 4670 gen_cc_cond(&c, s, extract32(insn, 8, 4)); 4671 do_trapcc(s, &c); 4672 } 4673 4674 DISAS_INSN(trapv) 4675 { 4676 DisasCompare c; 4677 4678 gen_cc_cond(&c, s, 9); /* V set */ 4679 do_trapcc(s, &c); 4680 } 4681 4682 static void gen_load_fcr(DisasContext *s, TCGv res, int reg) 4683 { 4684 switch (reg) { 4685 case M68K_FPIAR: 4686 tcg_gen_movi_i32(res, 0); 4687 break; 4688 case M68K_FPSR: 4689 tcg_gen_ld_i32(res, tcg_env, offsetof(CPUM68KState, fpsr)); 4690 break; 4691 case M68K_FPCR: 4692 tcg_gen_ld_i32(res, tcg_env, offsetof(CPUM68KState, fpcr)); 4693 break; 4694 } 4695 } 4696 4697 static void gen_store_fcr(DisasContext *s, TCGv val, int reg) 4698 { 4699 switch (reg) { 4700 case M68K_FPIAR: 4701 break; 4702 case M68K_FPSR: 4703 tcg_gen_st_i32(val, tcg_env, offsetof(CPUM68KState, fpsr)); 4704 break; 4705 case M68K_FPCR: 4706 gen_helper_set_fpcr(tcg_env, val); 4707 break; 4708 } 4709 } 4710 4711 static void gen_qemu_store_fcr(DisasContext *s, TCGv addr, int reg) 4712 { 4713 int index = IS_USER(s); 4714 TCGv tmp; 4715 4716 tmp = tcg_temp_new(); 4717 gen_load_fcr(s, tmp, reg); 4718 tcg_gen_qemu_st_tl(tmp, addr, index, MO_TEUL); 4719 } 4720 4721 static void gen_qemu_load_fcr(DisasContext *s, TCGv addr, int reg) 4722 { 4723 int index = IS_USER(s); 4724 TCGv tmp; 4725 4726 tmp = tcg_temp_new(); 4727 tcg_gen_qemu_ld_tl(tmp, addr, index, MO_TEUL); 4728 gen_store_fcr(s, tmp, reg); 4729 } 4730 4731 4732 static void gen_op_fmove_fcr(CPUM68KState *env, DisasContext *s, 4733 uint32_t insn, uint32_t ext) 4734 { 4735 int mask = (ext >> 10) & 7; 4736 int is_write = (ext >> 13) & 1; 4737 int mode = extract32(insn, 3, 3); 4738 int i; 4739 TCGv addr, tmp; 4740 4741 switch (mode) { 4742 case 0: /* Dn */ 4743 if (mask != M68K_FPIAR && mask != M68K_FPSR && mask != M68K_FPCR) { 4744 gen_exception(s, s->base.pc_next, EXCP_ILLEGAL); 4745 return; 4746 } 4747 if (is_write) { 4748 gen_load_fcr(s, DREG(insn, 0), mask); 4749 } else { 4750 gen_store_fcr(s, DREG(insn, 0), mask); 4751 } 4752 return; 4753 case 1: /* An, only with FPIAR */ 4754 if (mask != M68K_FPIAR) { 4755 gen_exception(s, s->base.pc_next, EXCP_ILLEGAL); 4756 return; 4757 } 4758 if (is_write) { 4759 gen_load_fcr(s, AREG(insn, 0), mask); 4760 } else { 4761 gen_store_fcr(s, AREG(insn, 0), mask); 4762 } 4763 return; 4764 case 7: /* Immediate */ 4765 if (REG(insn, 0) == 4) { 4766 if (is_write || 4767 (mask != M68K_FPIAR && mask != M68K_FPSR && 4768 mask != M68K_FPCR)) { 4769 gen_exception(s, s->base.pc_next, EXCP_ILLEGAL); 4770 return; 4771 } 4772 tmp = tcg_constant_i32(read_im32(env, s)); 4773 gen_store_fcr(s, tmp, mask); 4774 return; 4775 } 4776 break; 4777 default: 4778 break; 4779 } 4780 4781 tmp = gen_lea(env, s, insn, OS_LONG); 4782 if (IS_NULL_QREG(tmp)) { 4783 gen_addr_fault(s); 4784 return; 4785 } 4786 4787 addr = tcg_temp_new(); 4788 tcg_gen_mov_i32(addr, tmp); 4789 4790 /* 4791 * mask: 4792 * 4793 * 0b100 Floating-Point Control Register 4794 * 0b010 Floating-Point Status Register 4795 * 0b001 Floating-Point Instruction Address Register 4796 * 4797 */ 4798 4799 if (is_write && mode == 4) { 4800 for (i = 2; i >= 0; i--, mask >>= 1) { 4801 if (mask & 1) { 4802 gen_qemu_store_fcr(s, addr, 1 << i); 4803 if (mask != 1) { 4804 tcg_gen_subi_i32(addr, addr, opsize_bytes(OS_LONG)); 4805 } 4806 } 4807 } 4808 tcg_gen_mov_i32(AREG(insn, 0), addr); 4809 } else { 4810 for (i = 0; i < 3; i++, mask >>= 1) { 4811 if (mask & 1) { 4812 if (is_write) { 4813 gen_qemu_store_fcr(s, addr, 1 << i); 4814 } else { 4815 gen_qemu_load_fcr(s, addr, 1 << i); 4816 } 4817 if (mask != 1 || mode == 3) { 4818 tcg_gen_addi_i32(addr, addr, opsize_bytes(OS_LONG)); 4819 } 4820 } 4821 } 4822 if (mode == 3) { 4823 tcg_gen_mov_i32(AREG(insn, 0), addr); 4824 } 4825 } 4826 } 4827 4828 static void gen_op_fmovem(CPUM68KState *env, DisasContext *s, 4829 uint32_t insn, uint32_t ext) 4830 { 4831 int opsize; 4832 TCGv addr, tmp; 4833 int mode = (ext >> 11) & 0x3; 4834 int is_load = ((ext & 0x2000) == 0); 4835 4836 if (m68k_feature(s->env, M68K_FEATURE_FPU)) { 4837 opsize = OS_EXTENDED; 4838 } else { 4839 opsize = OS_DOUBLE; /* FIXME */ 4840 } 4841 4842 addr = gen_lea(env, s, insn, opsize); 4843 if (IS_NULL_QREG(addr)) { 4844 gen_addr_fault(s); 4845 return; 4846 } 4847 4848 tmp = tcg_temp_new(); 4849 if (mode & 0x1) { 4850 /* Dynamic register list */ 4851 tcg_gen_ext8u_i32(tmp, DREG(ext, 4)); 4852 } else { 4853 /* Static register list */ 4854 tcg_gen_movi_i32(tmp, ext & 0xff); 4855 } 4856 4857 if (!is_load && (mode & 2) == 0) { 4858 /* 4859 * predecrement addressing mode 4860 * only available to store register to memory 4861 */ 4862 if (opsize == OS_EXTENDED) { 4863 gen_helper_fmovemx_st_predec(tmp, tcg_env, addr, tmp); 4864 } else { 4865 gen_helper_fmovemd_st_predec(tmp, tcg_env, addr, tmp); 4866 } 4867 } else { 4868 /* postincrement addressing mode */ 4869 if (opsize == OS_EXTENDED) { 4870 if (is_load) { 4871 gen_helper_fmovemx_ld_postinc(tmp, tcg_env, addr, tmp); 4872 } else { 4873 gen_helper_fmovemx_st_postinc(tmp, tcg_env, addr, tmp); 4874 } 4875 } else { 4876 if (is_load) { 4877 gen_helper_fmovemd_ld_postinc(tmp, tcg_env, addr, tmp); 4878 } else { 4879 gen_helper_fmovemd_st_postinc(tmp, tcg_env, addr, tmp); 4880 } 4881 } 4882 } 4883 if ((insn & 070) == 030 || (insn & 070) == 040) { 4884 tcg_gen_mov_i32(AREG(insn, 0), tmp); 4885 } 4886 } 4887 4888 /* 4889 * ??? FP exceptions are not implemented. Most exceptions are deferred until 4890 * immediately before the next FP instruction is executed. 4891 */ 4892 DISAS_INSN(fpu) 4893 { 4894 uint16_t ext; 4895 int opmode; 4896 int opsize; 4897 TCGv_ptr cpu_src, cpu_dest; 4898 4899 ext = read_im16(env, s); 4900 opmode = ext & 0x7f; 4901 switch ((ext >> 13) & 7) { 4902 case 0: 4903 break; 4904 case 1: 4905 goto undef; 4906 case 2: 4907 if (insn == 0xf200 && (ext & 0xfc00) == 0x5c00) { 4908 /* fmovecr */ 4909 TCGv rom_offset = tcg_constant_i32(opmode); 4910 cpu_dest = gen_fp_ptr(REG(ext, 7)); 4911 gen_helper_fconst(tcg_env, cpu_dest, rom_offset); 4912 return; 4913 } 4914 break; 4915 case 3: /* fmove out */ 4916 cpu_src = gen_fp_ptr(REG(ext, 7)); 4917 opsize = ext_opsize(ext, 10); 4918 if (gen_ea_fp(env, s, insn, opsize, cpu_src, 4919 EA_STORE, IS_USER(s)) == -1) { 4920 gen_addr_fault(s); 4921 } 4922 gen_helper_ftst(tcg_env, cpu_src); 4923 return; 4924 case 4: /* fmove to control register. */ 4925 case 5: /* fmove from control register. */ 4926 gen_op_fmove_fcr(env, s, insn, ext); 4927 return; 4928 case 6: /* fmovem */ 4929 case 7: 4930 if ((ext & 0x1000) == 0 && !m68k_feature(s->env, M68K_FEATURE_FPU)) { 4931 goto undef; 4932 } 4933 gen_op_fmovem(env, s, insn, ext); 4934 return; 4935 } 4936 if (ext & (1 << 14)) { 4937 /* Source effective address. */ 4938 opsize = ext_opsize(ext, 10); 4939 cpu_src = gen_fp_result_ptr(); 4940 if (gen_ea_fp(env, s, insn, opsize, cpu_src, 4941 EA_LOADS, IS_USER(s)) == -1) { 4942 gen_addr_fault(s); 4943 return; 4944 } 4945 } else { 4946 /* Source register. */ 4947 opsize = OS_EXTENDED; 4948 cpu_src = gen_fp_ptr(REG(ext, 10)); 4949 } 4950 cpu_dest = gen_fp_ptr(REG(ext, 7)); 4951 switch (opmode) { 4952 case 0: /* fmove */ 4953 gen_fp_move(cpu_dest, cpu_src); 4954 break; 4955 case 0x40: /* fsmove */ 4956 gen_helper_fsround(tcg_env, cpu_dest, cpu_src); 4957 break; 4958 case 0x44: /* fdmove */ 4959 gen_helper_fdround(tcg_env, cpu_dest, cpu_src); 4960 break; 4961 case 1: /* fint */ 4962 gen_helper_firound(tcg_env, cpu_dest, cpu_src); 4963 break; 4964 case 2: /* fsinh */ 4965 gen_helper_fsinh(tcg_env, cpu_dest, cpu_src); 4966 break; 4967 case 3: /* fintrz */ 4968 gen_helper_fitrunc(tcg_env, cpu_dest, cpu_src); 4969 break; 4970 case 4: /* fsqrt */ 4971 gen_helper_fsqrt(tcg_env, cpu_dest, cpu_src); 4972 break; 4973 case 0x41: /* fssqrt */ 4974 gen_helper_fssqrt(tcg_env, cpu_dest, cpu_src); 4975 break; 4976 case 0x45: /* fdsqrt */ 4977 gen_helper_fdsqrt(tcg_env, cpu_dest, cpu_src); 4978 break; 4979 case 0x06: /* flognp1 */ 4980 gen_helper_flognp1(tcg_env, cpu_dest, cpu_src); 4981 break; 4982 case 0x08: /* fetoxm1 */ 4983 gen_helper_fetoxm1(tcg_env, cpu_dest, cpu_src); 4984 break; 4985 case 0x09: /* ftanh */ 4986 gen_helper_ftanh(tcg_env, cpu_dest, cpu_src); 4987 break; 4988 case 0x0a: /* fatan */ 4989 gen_helper_fatan(tcg_env, cpu_dest, cpu_src); 4990 break; 4991 case 0x0c: /* fasin */ 4992 gen_helper_fasin(tcg_env, cpu_dest, cpu_src); 4993 break; 4994 case 0x0d: /* fatanh */ 4995 gen_helper_fatanh(tcg_env, cpu_dest, cpu_src); 4996 break; 4997 case 0x0e: /* fsin */ 4998 gen_helper_fsin(tcg_env, cpu_dest, cpu_src); 4999 break; 5000 case 0x0f: /* ftan */ 5001 gen_helper_ftan(tcg_env, cpu_dest, cpu_src); 5002 break; 5003 case 0x10: /* fetox */ 5004 gen_helper_fetox(tcg_env, cpu_dest, cpu_src); 5005 break; 5006 case 0x11: /* ftwotox */ 5007 gen_helper_ftwotox(tcg_env, cpu_dest, cpu_src); 5008 break; 5009 case 0x12: /* ftentox */ 5010 gen_helper_ftentox(tcg_env, cpu_dest, cpu_src); 5011 break; 5012 case 0x14: /* flogn */ 5013 gen_helper_flogn(tcg_env, cpu_dest, cpu_src); 5014 break; 5015 case 0x15: /* flog10 */ 5016 gen_helper_flog10(tcg_env, cpu_dest, cpu_src); 5017 break; 5018 case 0x16: /* flog2 */ 5019 gen_helper_flog2(tcg_env, cpu_dest, cpu_src); 5020 break; 5021 case 0x18: /* fabs */ 5022 gen_helper_fabs(tcg_env, cpu_dest, cpu_src); 5023 break; 5024 case 0x58: /* fsabs */ 5025 gen_helper_fsabs(tcg_env, cpu_dest, cpu_src); 5026 break; 5027 case 0x5c: /* fdabs */ 5028 gen_helper_fdabs(tcg_env, cpu_dest, cpu_src); 5029 break; 5030 case 0x19: /* fcosh */ 5031 gen_helper_fcosh(tcg_env, cpu_dest, cpu_src); 5032 break; 5033 case 0x1a: /* fneg */ 5034 gen_helper_fneg(tcg_env, cpu_dest, cpu_src); 5035 break; 5036 case 0x5a: /* fsneg */ 5037 gen_helper_fsneg(tcg_env, cpu_dest, cpu_src); 5038 break; 5039 case 0x5e: /* fdneg */ 5040 gen_helper_fdneg(tcg_env, cpu_dest, cpu_src); 5041 break; 5042 case 0x1c: /* facos */ 5043 gen_helper_facos(tcg_env, cpu_dest, cpu_src); 5044 break; 5045 case 0x1d: /* fcos */ 5046 gen_helper_fcos(tcg_env, cpu_dest, cpu_src); 5047 break; 5048 case 0x1e: /* fgetexp */ 5049 gen_helper_fgetexp(tcg_env, cpu_dest, cpu_src); 5050 break; 5051 case 0x1f: /* fgetman */ 5052 gen_helper_fgetman(tcg_env, cpu_dest, cpu_src); 5053 break; 5054 case 0x20: /* fdiv */ 5055 gen_helper_fdiv(tcg_env, cpu_dest, cpu_src, cpu_dest); 5056 break; 5057 case 0x60: /* fsdiv */ 5058 gen_helper_fsdiv(tcg_env, cpu_dest, cpu_src, cpu_dest); 5059 break; 5060 case 0x64: /* fddiv */ 5061 gen_helper_fddiv(tcg_env, cpu_dest, cpu_src, cpu_dest); 5062 break; 5063 case 0x21: /* fmod */ 5064 gen_helper_fmod(tcg_env, cpu_dest, cpu_src, cpu_dest); 5065 break; 5066 case 0x22: /* fadd */ 5067 gen_helper_fadd(tcg_env, cpu_dest, cpu_src, cpu_dest); 5068 break; 5069 case 0x62: /* fsadd */ 5070 gen_helper_fsadd(tcg_env, cpu_dest, cpu_src, cpu_dest); 5071 break; 5072 case 0x66: /* fdadd */ 5073 gen_helper_fdadd(tcg_env, cpu_dest, cpu_src, cpu_dest); 5074 break; 5075 case 0x23: /* fmul */ 5076 gen_helper_fmul(tcg_env, cpu_dest, cpu_src, cpu_dest); 5077 break; 5078 case 0x63: /* fsmul */ 5079 gen_helper_fsmul(tcg_env, cpu_dest, cpu_src, cpu_dest); 5080 break; 5081 case 0x67: /* fdmul */ 5082 gen_helper_fdmul(tcg_env, cpu_dest, cpu_src, cpu_dest); 5083 break; 5084 case 0x24: /* fsgldiv */ 5085 gen_helper_fsgldiv(tcg_env, cpu_dest, cpu_src, cpu_dest); 5086 break; 5087 case 0x25: /* frem */ 5088 gen_helper_frem(tcg_env, cpu_dest, cpu_src, cpu_dest); 5089 break; 5090 case 0x26: /* fscale */ 5091 gen_helper_fscale(tcg_env, cpu_dest, cpu_src, cpu_dest); 5092 break; 5093 case 0x27: /* fsglmul */ 5094 gen_helper_fsglmul(tcg_env, cpu_dest, cpu_src, cpu_dest); 5095 break; 5096 case 0x28: /* fsub */ 5097 gen_helper_fsub(tcg_env, cpu_dest, cpu_src, cpu_dest); 5098 break; 5099 case 0x68: /* fssub */ 5100 gen_helper_fssub(tcg_env, cpu_dest, cpu_src, cpu_dest); 5101 break; 5102 case 0x6c: /* fdsub */ 5103 gen_helper_fdsub(tcg_env, cpu_dest, cpu_src, cpu_dest); 5104 break; 5105 case 0x30: case 0x31: case 0x32: 5106 case 0x33: case 0x34: case 0x35: 5107 case 0x36: case 0x37: { 5108 TCGv_ptr cpu_dest2 = gen_fp_ptr(REG(ext, 0)); 5109 gen_helper_fsincos(tcg_env, cpu_dest, cpu_dest2, cpu_src); 5110 } 5111 break; 5112 case 0x38: /* fcmp */ 5113 gen_helper_fcmp(tcg_env, cpu_src, cpu_dest); 5114 return; 5115 case 0x3a: /* ftst */ 5116 gen_helper_ftst(tcg_env, cpu_src); 5117 return; 5118 default: 5119 goto undef; 5120 } 5121 gen_helper_ftst(tcg_env, cpu_dest); 5122 return; 5123 undef: 5124 /* FIXME: Is this right for offset addressing modes? */ 5125 s->pc -= 2; 5126 disas_undef_fpu(env, s, insn); 5127 } 5128 5129 static void gen_fcc_cond(DisasCompare *c, DisasContext *s, int cond) 5130 { 5131 TCGv fpsr; 5132 5133 c->v2 = tcg_constant_i32(0); 5134 /* TODO: Raise BSUN exception. */ 5135 fpsr = tcg_temp_new(); 5136 gen_load_fcr(s, fpsr, M68K_FPSR); 5137 switch (cond) { 5138 case 0: /* False */ 5139 case 16: /* Signaling False */ 5140 c->v1 = c->v2; 5141 c->tcond = TCG_COND_NEVER; 5142 break; 5143 case 1: /* EQual Z */ 5144 case 17: /* Signaling EQual Z */ 5145 c->v1 = tcg_temp_new(); 5146 tcg_gen_andi_i32(c->v1, fpsr, FPSR_CC_Z); 5147 c->tcond = TCG_COND_NE; 5148 break; 5149 case 2: /* Ordered Greater Than !(A || Z || N) */ 5150 case 18: /* Greater Than !(A || Z || N) */ 5151 c->v1 = tcg_temp_new(); 5152 tcg_gen_andi_i32(c->v1, fpsr, 5153 FPSR_CC_A | FPSR_CC_Z | FPSR_CC_N); 5154 c->tcond = TCG_COND_EQ; 5155 break; 5156 case 3: /* Ordered Greater than or Equal Z || !(A || N) */ 5157 case 19: /* Greater than or Equal Z || !(A || N) */ 5158 c->v1 = tcg_temp_new(); 5159 tcg_gen_andi_i32(c->v1, fpsr, FPSR_CC_A); 5160 tcg_gen_shli_i32(c->v1, c->v1, ctz32(FPSR_CC_N) - ctz32(FPSR_CC_A)); 5161 tcg_gen_andi_i32(fpsr, fpsr, FPSR_CC_Z | FPSR_CC_N); 5162 tcg_gen_or_i32(c->v1, c->v1, fpsr); 5163 tcg_gen_xori_i32(c->v1, c->v1, FPSR_CC_N); 5164 c->tcond = TCG_COND_NE; 5165 break; 5166 case 4: /* Ordered Less Than !(!N || A || Z); */ 5167 case 20: /* Less Than !(!N || A || Z); */ 5168 c->v1 = tcg_temp_new(); 5169 tcg_gen_xori_i32(c->v1, fpsr, FPSR_CC_N); 5170 tcg_gen_andi_i32(c->v1, c->v1, FPSR_CC_N | FPSR_CC_A | FPSR_CC_Z); 5171 c->tcond = TCG_COND_EQ; 5172 break; 5173 case 5: /* Ordered Less than or Equal Z || (N && !A) */ 5174 case 21: /* Less than or Equal Z || (N && !A) */ 5175 c->v1 = tcg_temp_new(); 5176 tcg_gen_andi_i32(c->v1, fpsr, FPSR_CC_A); 5177 tcg_gen_shli_i32(c->v1, c->v1, ctz32(FPSR_CC_N) - ctz32(FPSR_CC_A)); 5178 tcg_gen_andc_i32(c->v1, fpsr, c->v1); 5179 tcg_gen_andi_i32(c->v1, c->v1, FPSR_CC_Z | FPSR_CC_N); 5180 c->tcond = TCG_COND_NE; 5181 break; 5182 case 6: /* Ordered Greater or Less than !(A || Z) */ 5183 case 22: /* Greater or Less than !(A || Z) */ 5184 c->v1 = tcg_temp_new(); 5185 tcg_gen_andi_i32(c->v1, fpsr, FPSR_CC_A | FPSR_CC_Z); 5186 c->tcond = TCG_COND_EQ; 5187 break; 5188 case 7: /* Ordered !A */ 5189 case 23: /* Greater, Less or Equal !A */ 5190 c->v1 = tcg_temp_new(); 5191 tcg_gen_andi_i32(c->v1, fpsr, FPSR_CC_A); 5192 c->tcond = TCG_COND_EQ; 5193 break; 5194 case 8: /* Unordered A */ 5195 case 24: /* Not Greater, Less or Equal A */ 5196 c->v1 = tcg_temp_new(); 5197 tcg_gen_andi_i32(c->v1, fpsr, FPSR_CC_A); 5198 c->tcond = TCG_COND_NE; 5199 break; 5200 case 9: /* Unordered or Equal A || Z */ 5201 case 25: /* Not Greater or Less then A || Z */ 5202 c->v1 = tcg_temp_new(); 5203 tcg_gen_andi_i32(c->v1, fpsr, FPSR_CC_A | FPSR_CC_Z); 5204 c->tcond = TCG_COND_NE; 5205 break; 5206 case 10: /* Unordered or Greater Than A || !(N || Z)) */ 5207 case 26: /* Not Less or Equal A || !(N || Z)) */ 5208 c->v1 = tcg_temp_new(); 5209 tcg_gen_andi_i32(c->v1, fpsr, FPSR_CC_Z); 5210 tcg_gen_shli_i32(c->v1, c->v1, ctz32(FPSR_CC_N) - ctz32(FPSR_CC_Z)); 5211 tcg_gen_andi_i32(fpsr, fpsr, FPSR_CC_A | FPSR_CC_N); 5212 tcg_gen_or_i32(c->v1, c->v1, fpsr); 5213 tcg_gen_xori_i32(c->v1, c->v1, FPSR_CC_N); 5214 c->tcond = TCG_COND_NE; 5215 break; 5216 case 11: /* Unordered or Greater or Equal A || Z || !N */ 5217 case 27: /* Not Less Than A || Z || !N */ 5218 c->v1 = tcg_temp_new(); 5219 tcg_gen_andi_i32(c->v1, fpsr, FPSR_CC_A | FPSR_CC_Z | FPSR_CC_N); 5220 tcg_gen_xori_i32(c->v1, c->v1, FPSR_CC_N); 5221 c->tcond = TCG_COND_NE; 5222 break; 5223 case 12: /* Unordered or Less Than A || (N && !Z) */ 5224 case 28: /* Not Greater than or Equal A || (N && !Z) */ 5225 c->v1 = tcg_temp_new(); 5226 tcg_gen_andi_i32(c->v1, fpsr, FPSR_CC_Z); 5227 tcg_gen_shli_i32(c->v1, c->v1, ctz32(FPSR_CC_N) - ctz32(FPSR_CC_Z)); 5228 tcg_gen_andc_i32(c->v1, fpsr, c->v1); 5229 tcg_gen_andi_i32(c->v1, c->v1, FPSR_CC_A | FPSR_CC_N); 5230 c->tcond = TCG_COND_NE; 5231 break; 5232 case 13: /* Unordered or Less or Equal A || Z || N */ 5233 case 29: /* Not Greater Than A || Z || N */ 5234 c->v1 = tcg_temp_new(); 5235 tcg_gen_andi_i32(c->v1, fpsr, FPSR_CC_A | FPSR_CC_Z | FPSR_CC_N); 5236 c->tcond = TCG_COND_NE; 5237 break; 5238 case 14: /* Not Equal !Z */ 5239 case 30: /* Signaling Not Equal !Z */ 5240 c->v1 = tcg_temp_new(); 5241 tcg_gen_andi_i32(c->v1, fpsr, FPSR_CC_Z); 5242 c->tcond = TCG_COND_EQ; 5243 break; 5244 case 15: /* True */ 5245 case 31: /* Signaling True */ 5246 c->v1 = c->v2; 5247 c->tcond = TCG_COND_ALWAYS; 5248 break; 5249 } 5250 } 5251 5252 static void gen_fjmpcc(DisasContext *s, int cond, TCGLabel *l1) 5253 { 5254 DisasCompare c; 5255 5256 gen_fcc_cond(&c, s, cond); 5257 update_cc_op(s); 5258 tcg_gen_brcond_i32(c.tcond, c.v1, c.v2, l1); 5259 } 5260 5261 DISAS_INSN(fbcc) 5262 { 5263 uint32_t offset; 5264 uint32_t base; 5265 TCGLabel *l1; 5266 5267 base = s->pc; 5268 offset = (int16_t)read_im16(env, s); 5269 if (insn & (1 << 6)) { 5270 offset = (offset << 16) | read_im16(env, s); 5271 } 5272 5273 l1 = gen_new_label(); 5274 update_cc_op(s); 5275 gen_fjmpcc(s, insn & 0x3f, l1); 5276 gen_jmp_tb(s, 0, s->pc, s->base.pc_next); 5277 gen_set_label(l1); 5278 gen_jmp_tb(s, 1, base + offset, s->base.pc_next); 5279 } 5280 5281 DISAS_INSN(fscc) 5282 { 5283 DisasCompare c; 5284 int cond; 5285 TCGv tmp; 5286 uint16_t ext; 5287 5288 ext = read_im16(env, s); 5289 cond = ext & 0x3f; 5290 gen_fcc_cond(&c, s, cond); 5291 5292 tmp = tcg_temp_new(); 5293 tcg_gen_negsetcond_i32(c.tcond, tmp, c.v1, c.v2); 5294 5295 DEST_EA(env, insn, OS_BYTE, tmp, NULL); 5296 } 5297 5298 DISAS_INSN(ftrapcc) 5299 { 5300 DisasCompare c; 5301 uint16_t ext; 5302 int cond; 5303 5304 ext = read_im16(env, s); 5305 cond = ext & 0x3f; 5306 5307 /* Consume and discard the immediate operand. */ 5308 switch (extract32(insn, 0, 3)) { 5309 case 2: /* ftrapcc.w */ 5310 (void)read_im16(env, s); 5311 break; 5312 case 3: /* ftrapcc.l */ 5313 (void)read_im32(env, s); 5314 break; 5315 case 4: /* ftrapcc (no operand) */ 5316 break; 5317 default: 5318 /* ftrapcc registered with only valid opmodes */ 5319 g_assert_not_reached(); 5320 } 5321 5322 gen_fcc_cond(&c, s, cond); 5323 do_trapcc(s, &c); 5324 } 5325 5326 #if !defined(CONFIG_USER_ONLY) 5327 DISAS_INSN(frestore) 5328 { 5329 TCGv addr; 5330 5331 if (IS_USER(s)) { 5332 gen_exception(s, s->base.pc_next, EXCP_PRIVILEGE); 5333 return; 5334 } 5335 if (m68k_feature(s->env, M68K_FEATURE_M68040)) { 5336 SRC_EA(env, addr, OS_LONG, 0, NULL); 5337 /* FIXME: check the state frame */ 5338 } else { 5339 disas_undef(env, s, insn); 5340 } 5341 } 5342 5343 DISAS_INSN(fsave) 5344 { 5345 if (IS_USER(s)) { 5346 gen_exception(s, s->base.pc_next, EXCP_PRIVILEGE); 5347 return; 5348 } 5349 5350 if (m68k_feature(s->env, M68K_FEATURE_M68040)) { 5351 /* always write IDLE */ 5352 TCGv idle = tcg_constant_i32(0x41000000); 5353 DEST_EA(env, insn, OS_LONG, idle, NULL); 5354 } else { 5355 disas_undef(env, s, insn); 5356 } 5357 } 5358 #endif 5359 5360 static inline TCGv gen_mac_extract_word(DisasContext *s, TCGv val, int upper) 5361 { 5362 TCGv tmp = tcg_temp_new(); 5363 if (s->env->macsr & MACSR_FI) { 5364 if (upper) 5365 tcg_gen_andi_i32(tmp, val, 0xffff0000); 5366 else 5367 tcg_gen_shli_i32(tmp, val, 16); 5368 } else if (s->env->macsr & MACSR_SU) { 5369 if (upper) 5370 tcg_gen_sari_i32(tmp, val, 16); 5371 else 5372 tcg_gen_ext16s_i32(tmp, val); 5373 } else { 5374 if (upper) 5375 tcg_gen_shri_i32(tmp, val, 16); 5376 else 5377 tcg_gen_ext16u_i32(tmp, val); 5378 } 5379 return tmp; 5380 } 5381 5382 static void gen_mac_clear_flags(void) 5383 { 5384 tcg_gen_andi_i32(QREG_MACSR, QREG_MACSR, 5385 ~(MACSR_V | MACSR_Z | MACSR_N | MACSR_EV)); 5386 } 5387 5388 DISAS_INSN(mac) 5389 { 5390 TCGv rx; 5391 TCGv ry; 5392 uint16_t ext; 5393 int acc; 5394 TCGv tmp; 5395 TCGv addr; 5396 TCGv loadval; 5397 int dual; 5398 TCGv saved_flags; 5399 5400 if (!s->done_mac) { 5401 s->mactmp = tcg_temp_new_i64(); 5402 s->done_mac = 1; 5403 } 5404 5405 ext = read_im16(env, s); 5406 5407 acc = ((insn >> 7) & 1) | ((ext >> 3) & 2); 5408 dual = ((insn & 0x30) != 0 && (ext & 3) != 0); 5409 if (dual && !m68k_feature(s->env, M68K_FEATURE_CF_EMAC_B)) { 5410 disas_undef(env, s, insn); 5411 return; 5412 } 5413 if (insn & 0x30) { 5414 /* MAC with load. */ 5415 tmp = gen_lea(env, s, insn, OS_LONG); 5416 addr = tcg_temp_new(); 5417 tcg_gen_and_i32(addr, tmp, QREG_MAC_MASK); 5418 /* 5419 * Load the value now to ensure correct exception behavior. 5420 * Perform writeback after reading the MAC inputs. 5421 */ 5422 loadval = gen_load(s, OS_LONG, addr, 0, IS_USER(s)); 5423 5424 acc ^= 1; 5425 rx = (ext & 0x8000) ? AREG(ext, 12) : DREG(insn, 12); 5426 ry = (ext & 8) ? AREG(ext, 0) : DREG(ext, 0); 5427 } else { 5428 loadval = addr = NULL_QREG; 5429 rx = (insn & 0x40) ? AREG(insn, 9) : DREG(insn, 9); 5430 ry = (insn & 8) ? AREG(insn, 0) : DREG(insn, 0); 5431 } 5432 5433 gen_mac_clear_flags(); 5434 #if 0 5435 l1 = -1; 5436 /* Disabled because conditional branches clobber temporary vars. */ 5437 if ((s->env->macsr & MACSR_OMC) != 0 && !dual) { 5438 /* Skip the multiply if we know we will ignore it. */ 5439 l1 = gen_new_label(); 5440 tmp = tcg_temp_new(); 5441 tcg_gen_andi_i32(tmp, QREG_MACSR, 1 << (acc + 8)); 5442 gen_op_jmp_nz32(tmp, l1); 5443 } 5444 #endif 5445 5446 if ((ext & 0x0800) == 0) { 5447 /* Word. */ 5448 rx = gen_mac_extract_word(s, rx, (ext & 0x80) != 0); 5449 ry = gen_mac_extract_word(s, ry, (ext & 0x40) != 0); 5450 } 5451 if (s->env->macsr & MACSR_FI) { 5452 gen_helper_macmulf(s->mactmp, tcg_env, rx, ry); 5453 } else { 5454 if (s->env->macsr & MACSR_SU) 5455 gen_helper_macmuls(s->mactmp, tcg_env, rx, ry); 5456 else 5457 gen_helper_macmulu(s->mactmp, tcg_env, rx, ry); 5458 switch ((ext >> 9) & 3) { 5459 case 1: 5460 tcg_gen_shli_i64(s->mactmp, s->mactmp, 1); 5461 break; 5462 case 3: 5463 tcg_gen_shri_i64(s->mactmp, s->mactmp, 1); 5464 break; 5465 } 5466 } 5467 5468 if (dual) { 5469 /* Save the overflow flag from the multiply. */ 5470 saved_flags = tcg_temp_new(); 5471 tcg_gen_mov_i32(saved_flags, QREG_MACSR); 5472 } else { 5473 saved_flags = NULL_QREG; 5474 } 5475 5476 #if 0 5477 /* Disabled because conditional branches clobber temporary vars. */ 5478 if ((s->env->macsr & MACSR_OMC) != 0 && dual) { 5479 /* Skip the accumulate if the value is already saturated. */ 5480 l1 = gen_new_label(); 5481 tmp = tcg_temp_new(); 5482 gen_op_and32(tmp, QREG_MACSR, tcg_constant_i32(MACSR_PAV0 << acc)); 5483 gen_op_jmp_nz32(tmp, l1); 5484 } 5485 #endif 5486 5487 if (insn & 0x100) 5488 tcg_gen_sub_i64(MACREG(acc), MACREG(acc), s->mactmp); 5489 else 5490 tcg_gen_add_i64(MACREG(acc), MACREG(acc), s->mactmp); 5491 5492 if (s->env->macsr & MACSR_FI) 5493 gen_helper_macsatf(tcg_env, tcg_constant_i32(acc)); 5494 else if (s->env->macsr & MACSR_SU) 5495 gen_helper_macsats(tcg_env, tcg_constant_i32(acc)); 5496 else 5497 gen_helper_macsatu(tcg_env, tcg_constant_i32(acc)); 5498 5499 #if 0 5500 /* Disabled because conditional branches clobber temporary vars. */ 5501 if (l1 != -1) 5502 gen_set_label(l1); 5503 #endif 5504 5505 if (dual) { 5506 /* Dual accumulate variant. */ 5507 acc = (ext >> 2) & 3; 5508 /* Restore the overflow flag from the multiplier. */ 5509 tcg_gen_mov_i32(QREG_MACSR, saved_flags); 5510 #if 0 5511 /* Disabled because conditional branches clobber temporary vars. */ 5512 if ((s->env->macsr & MACSR_OMC) != 0) { 5513 /* Skip the accumulate if the value is already saturated. */ 5514 l1 = gen_new_label(); 5515 tmp = tcg_temp_new(); 5516 gen_op_and32(tmp, QREG_MACSR, tcg_constant_i32(MACSR_PAV0 << acc)); 5517 gen_op_jmp_nz32(tmp, l1); 5518 } 5519 #endif 5520 if (ext & 2) 5521 tcg_gen_sub_i64(MACREG(acc), MACREG(acc), s->mactmp); 5522 else 5523 tcg_gen_add_i64(MACREG(acc), MACREG(acc), s->mactmp); 5524 if (s->env->macsr & MACSR_FI) 5525 gen_helper_macsatf(tcg_env, tcg_constant_i32(acc)); 5526 else if (s->env->macsr & MACSR_SU) 5527 gen_helper_macsats(tcg_env, tcg_constant_i32(acc)); 5528 else 5529 gen_helper_macsatu(tcg_env, tcg_constant_i32(acc)); 5530 #if 0 5531 /* Disabled because conditional branches clobber temporary vars. */ 5532 if (l1 != -1) 5533 gen_set_label(l1); 5534 #endif 5535 } 5536 gen_helper_mac_set_flags(tcg_env, tcg_constant_i32(acc)); 5537 5538 if (insn & 0x30) { 5539 TCGv rw; 5540 rw = (insn & 0x40) ? AREG(insn, 9) : DREG(insn, 9); 5541 tcg_gen_mov_i32(rw, loadval); 5542 /* 5543 * FIXME: Should address writeback happen with the masked or 5544 * unmasked value? 5545 */ 5546 switch ((insn >> 3) & 7) { 5547 case 3: /* Post-increment. */ 5548 tcg_gen_addi_i32(AREG(insn, 0), addr, 4); 5549 break; 5550 case 4: /* Pre-decrement. */ 5551 tcg_gen_mov_i32(AREG(insn, 0), addr); 5552 } 5553 } 5554 } 5555 5556 DISAS_INSN(from_mac) 5557 { 5558 TCGv rx; 5559 TCGv_i64 acc; 5560 int accnum; 5561 5562 rx = (insn & 8) ? AREG(insn, 0) : DREG(insn, 0); 5563 accnum = (insn >> 9) & 3; 5564 acc = MACREG(accnum); 5565 if (s->env->macsr & MACSR_FI) { 5566 gen_helper_get_macf(rx, tcg_env, acc); 5567 } else if ((s->env->macsr & MACSR_OMC) == 0) { 5568 tcg_gen_extrl_i64_i32(rx, acc); 5569 } else if (s->env->macsr & MACSR_SU) { 5570 gen_helper_get_macs(rx, acc); 5571 } else { 5572 gen_helper_get_macu(rx, acc); 5573 } 5574 if (insn & 0x40) { 5575 tcg_gen_movi_i64(acc, 0); 5576 tcg_gen_andi_i32(QREG_MACSR, QREG_MACSR, ~(MACSR_PAV0 << accnum)); 5577 } 5578 } 5579 5580 DISAS_INSN(move_mac) 5581 { 5582 /* FIXME: This can be done without a helper. */ 5583 int src; 5584 TCGv dest; 5585 src = insn & 3; 5586 dest = tcg_constant_i32((insn >> 9) & 3); 5587 gen_helper_mac_move(tcg_env, dest, tcg_constant_i32(src)); 5588 gen_mac_clear_flags(); 5589 gen_helper_mac_set_flags(tcg_env, dest); 5590 } 5591 5592 DISAS_INSN(from_macsr) 5593 { 5594 TCGv reg; 5595 5596 reg = (insn & 8) ? AREG(insn, 0) : DREG(insn, 0); 5597 tcg_gen_mov_i32(reg, QREG_MACSR); 5598 } 5599 5600 DISAS_INSN(from_mask) 5601 { 5602 TCGv reg; 5603 reg = (insn & 8) ? AREG(insn, 0) : DREG(insn, 0); 5604 tcg_gen_mov_i32(reg, QREG_MAC_MASK); 5605 } 5606 5607 DISAS_INSN(from_mext) 5608 { 5609 TCGv reg; 5610 TCGv acc; 5611 reg = (insn & 8) ? AREG(insn, 0) : DREG(insn, 0); 5612 acc = tcg_constant_i32((insn & 0x400) ? 2 : 0); 5613 if (s->env->macsr & MACSR_FI) 5614 gen_helper_get_mac_extf(reg, tcg_env, acc); 5615 else 5616 gen_helper_get_mac_exti(reg, tcg_env, acc); 5617 } 5618 5619 DISAS_INSN(macsr_to_ccr) 5620 { 5621 TCGv tmp = tcg_temp_new(); 5622 5623 /* Note that X and C are always cleared. */ 5624 tcg_gen_andi_i32(tmp, QREG_MACSR, CCF_N | CCF_Z | CCF_V); 5625 gen_helper_set_ccr(tcg_env, tmp); 5626 set_cc_op(s, CC_OP_FLAGS); 5627 } 5628 5629 DISAS_INSN(to_mac) 5630 { 5631 TCGv_i64 acc; 5632 TCGv val; 5633 int accnum; 5634 accnum = (insn >> 9) & 3; 5635 acc = MACREG(accnum); 5636 SRC_EA(env, val, OS_LONG, 0, NULL); 5637 if (s->env->macsr & MACSR_FI) { 5638 tcg_gen_ext_i32_i64(acc, val); 5639 tcg_gen_shli_i64(acc, acc, 8); 5640 } else if (s->env->macsr & MACSR_SU) { 5641 tcg_gen_ext_i32_i64(acc, val); 5642 } else { 5643 tcg_gen_extu_i32_i64(acc, val); 5644 } 5645 tcg_gen_andi_i32(QREG_MACSR, QREG_MACSR, ~(MACSR_PAV0 << accnum)); 5646 gen_mac_clear_flags(); 5647 gen_helper_mac_set_flags(tcg_env, tcg_constant_i32(accnum)); 5648 } 5649 5650 DISAS_INSN(to_macsr) 5651 { 5652 TCGv val; 5653 SRC_EA(env, val, OS_LONG, 0, NULL); 5654 gen_helper_set_macsr(tcg_env, val); 5655 gen_exit_tb(s); 5656 } 5657 5658 DISAS_INSN(to_mask) 5659 { 5660 TCGv val; 5661 SRC_EA(env, val, OS_LONG, 0, NULL); 5662 tcg_gen_ori_i32(QREG_MAC_MASK, val, 0xffff0000); 5663 } 5664 5665 DISAS_INSN(to_mext) 5666 { 5667 TCGv val; 5668 TCGv acc; 5669 SRC_EA(env, val, OS_LONG, 0, NULL); 5670 acc = tcg_constant_i32((insn & 0x400) ? 2 : 0); 5671 if (s->env->macsr & MACSR_FI) 5672 gen_helper_set_mac_extf(tcg_env, val, acc); 5673 else if (s->env->macsr & MACSR_SU) 5674 gen_helper_set_mac_exts(tcg_env, val, acc); 5675 else 5676 gen_helper_set_mac_extu(tcg_env, val, acc); 5677 } 5678 5679 static disas_proc opcode_table[65536]; 5680 5681 static void 5682 register_opcode (disas_proc proc, uint16_t opcode, uint16_t mask) 5683 { 5684 int i; 5685 int from; 5686 int to; 5687 5688 /* Sanity check. All set bits must be included in the mask. */ 5689 if (opcode & ~mask) { 5690 fprintf(stderr, 5691 "qemu internal error: bogus opcode definition %04x/%04x\n", 5692 opcode, mask); 5693 abort(); 5694 } 5695 /* 5696 * This could probably be cleverer. For now just optimize the case where 5697 * the top bits are known. 5698 */ 5699 /* Find the first zero bit in the mask. */ 5700 i = 0x8000; 5701 while ((i & mask) != 0) 5702 i >>= 1; 5703 /* Iterate over all combinations of this and lower bits. */ 5704 if (i == 0) 5705 i = 1; 5706 else 5707 i <<= 1; 5708 from = opcode & ~(i - 1); 5709 to = from + i; 5710 for (i = from; i < to; i++) { 5711 if ((i & mask) == opcode) 5712 opcode_table[i] = proc; 5713 } 5714 } 5715 5716 /* 5717 * Register m68k opcode handlers. Order is important. 5718 * Later insn override earlier ones. 5719 */ 5720 void register_m68k_insns (CPUM68KState *env) 5721 { 5722 /* 5723 * Build the opcode table only once to avoid 5724 * multithreading issues. 5725 */ 5726 if (opcode_table[0] != NULL) { 5727 return; 5728 } 5729 5730 /* 5731 * use BASE() for instruction available 5732 * for CF_ISA_A and M68000. 5733 */ 5734 #define BASE(name, opcode, mask) \ 5735 register_opcode(disas_##name, 0x##opcode, 0x##mask) 5736 #define INSN(name, opcode, mask, feature) do { \ 5737 if (m68k_feature(env, M68K_FEATURE_##feature)) \ 5738 BASE(name, opcode, mask); \ 5739 } while(0) 5740 BASE(undef, 0000, 0000); 5741 INSN(arith_im, 0080, fff8, CF_ISA_A); 5742 INSN(arith_im, 0000, ff00, M68K); 5743 INSN(chk2, 00c0, f9c0, CHK2); 5744 INSN(bitrev, 00c0, fff8, CF_ISA_APLUSC); 5745 BASE(bitop_reg, 0100, f1c0); 5746 BASE(bitop_reg, 0140, f1c0); 5747 BASE(bitop_reg, 0180, f1c0); 5748 BASE(bitop_reg, 01c0, f1c0); 5749 INSN(movep, 0108, f138, MOVEP); 5750 INSN(arith_im, 0280, fff8, CF_ISA_A); 5751 INSN(arith_im, 0200, ff00, M68K); 5752 INSN(undef, 02c0, ffc0, M68K); 5753 INSN(byterev, 02c0, fff8, CF_ISA_APLUSC); 5754 INSN(arith_im, 0480, fff8, CF_ISA_A); 5755 INSN(arith_im, 0400, ff00, M68K); 5756 INSN(undef, 04c0, ffc0, M68K); 5757 INSN(arith_im, 0600, ff00, M68K); 5758 INSN(undef, 06c0, ffc0, M68K); 5759 INSN(ff1, 04c0, fff8, CF_ISA_APLUSC); 5760 INSN(arith_im, 0680, fff8, CF_ISA_A); 5761 INSN(arith_im, 0c00, ff38, CF_ISA_A); 5762 INSN(arith_im, 0c00, ff00, M68K); 5763 BASE(bitop_im, 0800, ffc0); 5764 BASE(bitop_im, 0840, ffc0); 5765 BASE(bitop_im, 0880, ffc0); 5766 BASE(bitop_im, 08c0, ffc0); 5767 INSN(arith_im, 0a80, fff8, CF_ISA_A); 5768 INSN(arith_im, 0a00, ff00, M68K); 5769 #if !defined(CONFIG_USER_ONLY) 5770 INSN(moves, 0e00, ff00, M68K); 5771 #endif 5772 INSN(cas, 0ac0, ffc0, CAS); 5773 INSN(cas, 0cc0, ffc0, CAS); 5774 INSN(cas, 0ec0, ffc0, CAS); 5775 INSN(cas2w, 0cfc, ffff, CAS); 5776 INSN(cas2l, 0efc, ffff, CAS); 5777 BASE(move, 1000, f000); 5778 BASE(move, 2000, f000); 5779 BASE(move, 3000, f000); 5780 INSN(chk, 4000, f040, M68K); 5781 INSN(strldsr, 40e7, ffff, CF_ISA_APLUSC); 5782 INSN(negx, 4080, fff8, CF_ISA_A); 5783 INSN(negx, 4000, ff00, M68K); 5784 INSN(undef, 40c0, ffc0, M68K); 5785 INSN(move_from_sr, 40c0, fff8, CF_ISA_A); 5786 INSN(move_from_sr, 40c0, ffc0, M68K); 5787 BASE(lea, 41c0, f1c0); 5788 BASE(clr, 4200, ff00); 5789 BASE(undef, 42c0, ffc0); 5790 INSN(move_from_ccr, 42c0, fff8, CF_ISA_A); 5791 INSN(move_from_ccr, 42c0, ffc0, M68K); 5792 INSN(neg, 4480, fff8, CF_ISA_A); 5793 INSN(neg, 4400, ff00, M68K); 5794 INSN(undef, 44c0, ffc0, M68K); 5795 BASE(move_to_ccr, 44c0, ffc0); 5796 INSN(not, 4680, fff8, CF_ISA_A); 5797 INSN(not, 4600, ff00, M68K); 5798 #if !defined(CONFIG_USER_ONLY) 5799 BASE(move_to_sr, 46c0, ffc0); 5800 #endif 5801 INSN(nbcd, 4800, ffc0, M68K); 5802 INSN(linkl, 4808, fff8, M68K); 5803 BASE(pea, 4840, ffc0); 5804 BASE(swap, 4840, fff8); 5805 INSN(bkpt, 4848, fff8, BKPT); 5806 INSN(movem, 48d0, fbf8, CF_ISA_A); 5807 INSN(movem, 48e8, fbf8, CF_ISA_A); 5808 INSN(movem, 4880, fb80, M68K); 5809 BASE(ext, 4880, fff8); 5810 BASE(ext, 48c0, fff8); 5811 BASE(ext, 49c0, fff8); 5812 BASE(tst, 4a00, ff00); 5813 INSN(tas, 4ac0, ffc0, CF_ISA_B); 5814 INSN(tas, 4ac0, ffc0, M68K); 5815 #if !defined(CONFIG_USER_ONLY) 5816 INSN(halt, 4ac8, ffff, CF_ISA_A); 5817 INSN(halt, 4ac8, ffff, M68K); 5818 #endif 5819 INSN(pulse, 4acc, ffff, CF_ISA_A); 5820 BASE(illegal, 4afc, ffff); 5821 INSN(mull, 4c00, ffc0, CF_ISA_A); 5822 INSN(mull, 4c00, ffc0, LONG_MULDIV); 5823 INSN(divl, 4c40, ffc0, CF_ISA_A); 5824 INSN(divl, 4c40, ffc0, LONG_MULDIV); 5825 INSN(sats, 4c80, fff8, CF_ISA_B); 5826 BASE(trap, 4e40, fff0); 5827 BASE(link, 4e50, fff8); 5828 BASE(unlk, 4e58, fff8); 5829 #if !defined(CONFIG_USER_ONLY) 5830 INSN(move_to_usp, 4e60, fff8, USP); 5831 INSN(move_from_usp, 4e68, fff8, USP); 5832 INSN(reset, 4e70, ffff, M68K); 5833 BASE(stop, 4e72, ffff); 5834 BASE(rte, 4e73, ffff); 5835 INSN(cf_movec, 4e7b, ffff, CF_ISA_A); 5836 INSN(m68k_movec, 4e7a, fffe, MOVEC); 5837 #endif 5838 BASE(nop, 4e71, ffff); 5839 INSN(rtd, 4e74, ffff, RTD); 5840 BASE(rts, 4e75, ffff); 5841 INSN(trapv, 4e76, ffff, M68K); 5842 INSN(rtr, 4e77, ffff, M68K); 5843 BASE(jump, 4e80, ffc0); 5844 BASE(jump, 4ec0, ffc0); 5845 INSN(addsubq, 5000, f080, M68K); 5846 BASE(addsubq, 5080, f0c0); 5847 INSN(scc, 50c0, f0f8, CF_ISA_A); /* Scc.B Dx */ 5848 INSN(scc, 50c0, f0c0, M68K); /* Scc.B <EA> */ 5849 INSN(dbcc, 50c8, f0f8, M68K); 5850 INSN(trapcc, 50fa, f0fe, TRAPCC); /* opmode 010, 011 */ 5851 INSN(trapcc, 50fc, f0ff, TRAPCC); /* opmode 100 */ 5852 INSN(trapcc, 51fa, fffe, CF_ISA_A); /* TPF (trapf) opmode 010, 011 */ 5853 INSN(trapcc, 51fc, ffff, CF_ISA_A); /* TPF (trapf) opmode 100 */ 5854 5855 /* Branch instructions. */ 5856 BASE(branch, 6000, f000); 5857 /* Disable long branch instructions, then add back the ones we want. */ 5858 BASE(undef, 60ff, f0ff); /* All long branches. */ 5859 INSN(branch, 60ff, f0ff, CF_ISA_B); 5860 INSN(undef, 60ff, ffff, CF_ISA_B); /* bra.l */ 5861 INSN(branch, 60ff, ffff, BRAL); 5862 INSN(branch, 60ff, f0ff, BCCL); 5863 5864 BASE(moveq, 7000, f100); 5865 INSN(mvzs, 7100, f100, CF_ISA_B); 5866 BASE(or, 8000, f000); 5867 BASE(divw, 80c0, f0c0); 5868 INSN(sbcd_reg, 8100, f1f8, M68K); 5869 INSN(sbcd_mem, 8108, f1f8, M68K); 5870 BASE(addsub, 9000, f000); 5871 INSN(undef, 90c0, f0c0, CF_ISA_A); 5872 INSN(subx_reg, 9180, f1f8, CF_ISA_A); 5873 INSN(subx_reg, 9100, f138, M68K); 5874 INSN(subx_mem, 9108, f138, M68K); 5875 INSN(suba, 91c0, f1c0, CF_ISA_A); 5876 INSN(suba, 90c0, f0c0, M68K); 5877 5878 BASE(undef_mac, a000, f000); 5879 INSN(mac, a000, f100, CF_EMAC); 5880 INSN(from_mac, a180, f9b0, CF_EMAC); 5881 INSN(move_mac, a110, f9fc, CF_EMAC); 5882 INSN(from_macsr,a980, f9f0, CF_EMAC); 5883 INSN(from_mask, ad80, fff0, CF_EMAC); 5884 INSN(from_mext, ab80, fbf0, CF_EMAC); 5885 INSN(macsr_to_ccr, a9c0, ffff, CF_EMAC); 5886 INSN(to_mac, a100, f9c0, CF_EMAC); 5887 INSN(to_macsr, a900, ffc0, CF_EMAC); 5888 INSN(to_mext, ab00, fbc0, CF_EMAC); 5889 INSN(to_mask, ad00, ffc0, CF_EMAC); 5890 5891 INSN(mov3q, a140, f1c0, CF_ISA_B); 5892 INSN(cmp, b000, f1c0, CF_ISA_B); /* cmp.b */ 5893 INSN(cmp, b040, f1c0, CF_ISA_B); /* cmp.w */ 5894 INSN(cmpa, b0c0, f1c0, CF_ISA_B); /* cmpa.w */ 5895 INSN(cmp, b080, f1c0, CF_ISA_A); 5896 INSN(cmpa, b1c0, f1c0, CF_ISA_A); 5897 INSN(cmp, b000, f100, M68K); 5898 INSN(eor, b100, f100, M68K); 5899 INSN(cmpm, b108, f138, M68K); 5900 INSN(cmpa, b0c0, f0c0, M68K); 5901 INSN(eor, b180, f1c0, CF_ISA_A); 5902 BASE(and, c000, f000); 5903 INSN(exg_dd, c140, f1f8, M68K); 5904 INSN(exg_aa, c148, f1f8, M68K); 5905 INSN(exg_da, c188, f1f8, M68K); 5906 BASE(mulw, c0c0, f0c0); 5907 INSN(abcd_reg, c100, f1f8, M68K); 5908 INSN(abcd_mem, c108, f1f8, M68K); 5909 BASE(addsub, d000, f000); 5910 INSN(undef, d0c0, f0c0, CF_ISA_A); 5911 INSN(addx_reg, d180, f1f8, CF_ISA_A); 5912 INSN(addx_reg, d100, f138, M68K); 5913 INSN(addx_mem, d108, f138, M68K); 5914 INSN(adda, d1c0, f1c0, CF_ISA_A); 5915 INSN(adda, d0c0, f0c0, M68K); 5916 INSN(shift_im, e080, f0f0, CF_ISA_A); 5917 INSN(shift_reg, e0a0, f0f0, CF_ISA_A); 5918 INSN(shift8_im, e000, f0f0, M68K); 5919 INSN(shift16_im, e040, f0f0, M68K); 5920 INSN(shift_im, e080, f0f0, M68K); 5921 INSN(shift8_reg, e020, f0f0, M68K); 5922 INSN(shift16_reg, e060, f0f0, M68K); 5923 INSN(shift_reg, e0a0, f0f0, M68K); 5924 INSN(shift_mem, e0c0, fcc0, M68K); 5925 INSN(rotate_im, e090, f0f0, M68K); 5926 INSN(rotate8_im, e010, f0f0, M68K); 5927 INSN(rotate16_im, e050, f0f0, M68K); 5928 INSN(rotate_reg, e0b0, f0f0, M68K); 5929 INSN(rotate8_reg, e030, f0f0, M68K); 5930 INSN(rotate16_reg, e070, f0f0, M68K); 5931 INSN(rotate_mem, e4c0, fcc0, M68K); 5932 INSN(bfext_mem, e9c0, fdc0, BITFIELD); /* bfextu & bfexts */ 5933 INSN(bfext_reg, e9c0, fdf8, BITFIELD); 5934 INSN(bfins_mem, efc0, ffc0, BITFIELD); 5935 INSN(bfins_reg, efc0, fff8, BITFIELD); 5936 INSN(bfop_mem, eac0, ffc0, BITFIELD); /* bfchg */ 5937 INSN(bfop_reg, eac0, fff8, BITFIELD); /* bfchg */ 5938 INSN(bfop_mem, ecc0, ffc0, BITFIELD); /* bfclr */ 5939 INSN(bfop_reg, ecc0, fff8, BITFIELD); /* bfclr */ 5940 INSN(bfop_mem, edc0, ffc0, BITFIELD); /* bfffo */ 5941 INSN(bfop_reg, edc0, fff8, BITFIELD); /* bfffo */ 5942 INSN(bfop_mem, eec0, ffc0, BITFIELD); /* bfset */ 5943 INSN(bfop_reg, eec0, fff8, BITFIELD); /* bfset */ 5944 INSN(bfop_mem, e8c0, ffc0, BITFIELD); /* bftst */ 5945 INSN(bfop_reg, e8c0, fff8, BITFIELD); /* bftst */ 5946 BASE(undef_fpu, f000, f000); 5947 INSN(fpu, f200, ffc0, CF_FPU); 5948 INSN(fbcc, f280, ffc0, CF_FPU); 5949 INSN(fpu, f200, ffc0, FPU); 5950 INSN(fscc, f240, ffc0, FPU); 5951 INSN(ftrapcc, f27a, fffe, FPU); /* opmode 010, 011 */ 5952 INSN(ftrapcc, f27c, ffff, FPU); /* opmode 100 */ 5953 INSN(fbcc, f280, ff80, FPU); 5954 #if !defined(CONFIG_USER_ONLY) 5955 INSN(frestore, f340, ffc0, CF_FPU); 5956 INSN(fsave, f300, ffc0, CF_FPU); 5957 INSN(frestore, f340, ffc0, FPU); 5958 INSN(fsave, f300, ffc0, FPU); 5959 INSN(intouch, f340, ffc0, CF_ISA_A); 5960 INSN(cpushl, f428, ff38, CF_ISA_A); 5961 INSN(cpush, f420, ff20, M68040); 5962 INSN(cinv, f400, ff20, M68040); 5963 INSN(pflush, f500, ffe0, M68040); 5964 INSN(ptest, f548, ffd8, M68040); 5965 INSN(wddata, fb00, ff00, CF_ISA_A); 5966 INSN(wdebug, fbc0, ffc0, CF_ISA_A); 5967 #endif 5968 INSN(move16_mem, f600, ffe0, M68040); 5969 INSN(move16_reg, f620, fff8, M68040); 5970 #undef INSN 5971 } 5972 5973 static void m68k_tr_init_disas_context(DisasContextBase *dcbase, CPUState *cpu) 5974 { 5975 DisasContext *dc = container_of(dcbase, DisasContext, base); 5976 CPUM68KState *env = cpu_env(cpu); 5977 5978 dc->env = env; 5979 dc->pc = dc->base.pc_first; 5980 /* This value will always be filled in properly before m68k_tr_tb_stop. */ 5981 dc->pc_prev = 0xdeadbeef; 5982 dc->cc_op = CC_OP_DYNAMIC; 5983 dc->cc_op_synced = 1; 5984 dc->done_mac = 0; 5985 dc->writeback_mask = 0; 5986 5987 dc->ss_active = (M68K_SR_TRACE(env->sr) == M68K_SR_TRACE_ANY_INS); 5988 /* If architectural single step active, limit to 1 */ 5989 if (dc->ss_active) { 5990 dc->base.max_insns = 1; 5991 } 5992 } 5993 5994 static void m68k_tr_tb_start(DisasContextBase *dcbase, CPUState *cpu) 5995 { 5996 } 5997 5998 static void m68k_tr_insn_start(DisasContextBase *dcbase, CPUState *cpu) 5999 { 6000 DisasContext *dc = container_of(dcbase, DisasContext, base); 6001 tcg_gen_insn_start(dc->base.pc_next, dc->cc_op); 6002 } 6003 6004 static void m68k_tr_translate_insn(DisasContextBase *dcbase, CPUState *cpu) 6005 { 6006 DisasContext *dc = container_of(dcbase, DisasContext, base); 6007 CPUM68KState *env = cpu_env(cpu); 6008 uint16_t insn = read_im16(env, dc); 6009 6010 opcode_table[insn](env, dc, insn); 6011 do_writebacks(dc); 6012 6013 dc->pc_prev = dc->base.pc_next; 6014 dc->base.pc_next = dc->pc; 6015 6016 if (dc->base.is_jmp == DISAS_NEXT) { 6017 /* 6018 * Stop translation when the next insn might touch a new page. 6019 * This ensures that prefetch aborts at the right place. 6020 * 6021 * We cannot determine the size of the next insn without 6022 * completely decoding it. However, the maximum insn size 6023 * is 32 bytes, so end if we do not have that much remaining. 6024 * This may produce several small TBs at the end of each page, 6025 * but they will all be linked with goto_tb. 6026 * 6027 * ??? ColdFire maximum is 4 bytes; MC68000's maximum is also 6028 * smaller than MC68020's. 6029 */ 6030 target_ulong start_page_offset 6031 = dc->pc - (dc->base.pc_first & TARGET_PAGE_MASK); 6032 6033 if (start_page_offset >= TARGET_PAGE_SIZE - 32) { 6034 dc->base.is_jmp = DISAS_TOO_MANY; 6035 } 6036 } 6037 } 6038 6039 static void m68k_tr_tb_stop(DisasContextBase *dcbase, CPUState *cpu) 6040 { 6041 DisasContext *dc = container_of(dcbase, DisasContext, base); 6042 6043 switch (dc->base.is_jmp) { 6044 case DISAS_NORETURN: 6045 break; 6046 case DISAS_TOO_MANY: 6047 update_cc_op(dc); 6048 gen_jmp_tb(dc, 0, dc->pc, dc->pc_prev); 6049 break; 6050 case DISAS_JUMP: 6051 /* We updated CC_OP and PC in gen_jmp/gen_jmp_im. */ 6052 if (dc->ss_active) { 6053 gen_raise_exception_format2(dc, EXCP_TRACE, dc->pc_prev); 6054 } else { 6055 tcg_gen_lookup_and_goto_ptr(); 6056 } 6057 break; 6058 case DISAS_EXIT: 6059 /* 6060 * We updated CC_OP and PC in gen_exit_tb, but also modified 6061 * other state that may require returning to the main loop. 6062 */ 6063 if (dc->ss_active) { 6064 gen_raise_exception_format2(dc, EXCP_TRACE, dc->pc_prev); 6065 } else { 6066 tcg_gen_exit_tb(NULL, 0); 6067 } 6068 break; 6069 default: 6070 g_assert_not_reached(); 6071 } 6072 } 6073 6074 static void m68k_tr_disas_log(const DisasContextBase *dcbase, 6075 CPUState *cpu, FILE *logfile) 6076 { 6077 fprintf(logfile, "IN: %s\n", lookup_symbol(dcbase->pc_first)); 6078 target_disas(logfile, cpu, dcbase->pc_first, dcbase->tb->size); 6079 } 6080 6081 static const TranslatorOps m68k_tr_ops = { 6082 .init_disas_context = m68k_tr_init_disas_context, 6083 .tb_start = m68k_tr_tb_start, 6084 .insn_start = m68k_tr_insn_start, 6085 .translate_insn = m68k_tr_translate_insn, 6086 .tb_stop = m68k_tr_tb_stop, 6087 .disas_log = m68k_tr_disas_log, 6088 }; 6089 6090 void gen_intermediate_code(CPUState *cpu, TranslationBlock *tb, int *max_insns, 6091 vaddr pc, void *host_pc) 6092 { 6093 DisasContext dc; 6094 translator_loop(cpu, tb, max_insns, pc, host_pc, &m68k_tr_ops, &dc.base); 6095 } 6096 6097 static double floatx80_to_double(CPUM68KState *env, uint16_t high, uint64_t low) 6098 { 6099 floatx80 a = { .high = high, .low = low }; 6100 union { 6101 float64 f64; 6102 double d; 6103 } u; 6104 6105 u.f64 = floatx80_to_float64(a, &env->fp_status); 6106 return u.d; 6107 } 6108 6109 void m68k_cpu_dump_state(CPUState *cs, FILE *f, int flags) 6110 { 6111 M68kCPU *cpu = M68K_CPU(cs); 6112 CPUM68KState *env = &cpu->env; 6113 int i; 6114 uint16_t sr; 6115 for (i = 0; i < 8; i++) { 6116 qemu_fprintf(f, "D%d = %08x A%d = %08x " 6117 "F%d = %04x %016"PRIx64" (%12g)\n", 6118 i, env->dregs[i], i, env->aregs[i], 6119 i, env->fregs[i].l.upper, env->fregs[i].l.lower, 6120 floatx80_to_double(env, env->fregs[i].l.upper, 6121 env->fregs[i].l.lower)); 6122 } 6123 qemu_fprintf(f, "PC = %08x ", env->pc); 6124 sr = env->sr | cpu_m68k_get_ccr(env); 6125 qemu_fprintf(f, "SR = %04x T:%x I:%x %c%c %c%c%c%c%c\n", 6126 sr, (sr & SR_T) >> SR_T_SHIFT, (sr & SR_I) >> SR_I_SHIFT, 6127 (sr & SR_S) ? 'S' : 'U', (sr & SR_M) ? '%' : 'I', 6128 (sr & CCF_X) ? 'X' : '-', (sr & CCF_N) ? 'N' : '-', 6129 (sr & CCF_Z) ? 'Z' : '-', (sr & CCF_V) ? 'V' : '-', 6130 (sr & CCF_C) ? 'C' : '-'); 6131 qemu_fprintf(f, "FPSR = %08x %c%c%c%c ", env->fpsr, 6132 (env->fpsr & FPSR_CC_A) ? 'A' : '-', 6133 (env->fpsr & FPSR_CC_I) ? 'I' : '-', 6134 (env->fpsr & FPSR_CC_Z) ? 'Z' : '-', 6135 (env->fpsr & FPSR_CC_N) ? 'N' : '-'); 6136 qemu_fprintf(f, "\n " 6137 "FPCR = %04x ", env->fpcr); 6138 switch (env->fpcr & FPCR_PREC_MASK) { 6139 case FPCR_PREC_X: 6140 qemu_fprintf(f, "X "); 6141 break; 6142 case FPCR_PREC_S: 6143 qemu_fprintf(f, "S "); 6144 break; 6145 case FPCR_PREC_D: 6146 qemu_fprintf(f, "D "); 6147 break; 6148 } 6149 switch (env->fpcr & FPCR_RND_MASK) { 6150 case FPCR_RND_N: 6151 qemu_fprintf(f, "RN "); 6152 break; 6153 case FPCR_RND_Z: 6154 qemu_fprintf(f, "RZ "); 6155 break; 6156 case FPCR_RND_M: 6157 qemu_fprintf(f, "RM "); 6158 break; 6159 case FPCR_RND_P: 6160 qemu_fprintf(f, "RP "); 6161 break; 6162 } 6163 qemu_fprintf(f, "\n"); 6164 #ifndef CONFIG_USER_ONLY 6165 qemu_fprintf(f, "%sA7(MSP) = %08x %sA7(USP) = %08x %sA7(ISP) = %08x\n", 6166 env->current_sp == M68K_SSP ? "->" : " ", env->sp[M68K_SSP], 6167 env->current_sp == M68K_USP ? "->" : " ", env->sp[M68K_USP], 6168 env->current_sp == M68K_ISP ? "->" : " ", env->sp[M68K_ISP]); 6169 qemu_fprintf(f, "VBR = 0x%08x\n", env->vbr); 6170 qemu_fprintf(f, "SFC = %x DFC %x\n", env->sfc, env->dfc); 6171 qemu_fprintf(f, "SSW %08x TCR %08x URP %08x SRP %08x\n", 6172 env->mmu.ssw, env->mmu.tcr, env->mmu.urp, env->mmu.srp); 6173 qemu_fprintf(f, "DTTR0/1: %08x/%08x ITTR0/1: %08x/%08x\n", 6174 env->mmu.ttr[M68K_DTTR0], env->mmu.ttr[M68K_DTTR1], 6175 env->mmu.ttr[M68K_ITTR0], env->mmu.ttr[M68K_ITTR1]); 6176 qemu_fprintf(f, "MMUSR %08x, fault at %08x\n", 6177 env->mmu.mmusr, env->mmu.ar); 6178 #endif /* !CONFIG_USER_ONLY */ 6179 } 6180