1 /* 2 * Xilinx MicroBlaze emulation for qemu: main translation routines. 3 * 4 * Copyright (c) 2009 Edgar E. Iglesias. 5 * Copyright (c) 2009-2012 PetaLogix Qld Pty Ltd. 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 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-op.h" 26 #include "exec/helper-proto.h" 27 #include "microblaze-decode.h" 28 #include "exec/cpu_ldst.h" 29 #include "exec/helper-gen.h" 30 #include "exec/translator.h" 31 32 #include "trace-tcg.h" 33 #include "exec/log.h" 34 35 36 #define SIM_COMPAT 0 37 #define DISAS_GNU 1 38 #define DISAS_MB 1 39 #if DISAS_MB && !SIM_COMPAT 40 # define LOG_DIS(...) qemu_log_mask(CPU_LOG_TB_IN_ASM, ## __VA_ARGS__) 41 #else 42 # define LOG_DIS(...) do { } while (0) 43 #endif 44 45 #define D(x) 46 47 #define EXTRACT_FIELD(src, start, end) \ 48 (((src) >> start) & ((1 << (end - start + 1)) - 1)) 49 50 /* is_jmp field values */ 51 #define DISAS_JUMP DISAS_TARGET_0 /* only pc was modified dynamically */ 52 #define DISAS_UPDATE DISAS_TARGET_1 /* cpu state was modified dynamically */ 53 #define DISAS_TB_JUMP DISAS_TARGET_2 /* only pc was modified statically */ 54 55 static TCGv env_debug; 56 static TCGv cpu_R[32]; 57 static TCGv cpu_SR[18]; 58 static TCGv env_imm; 59 static TCGv env_btaken; 60 static TCGv env_btarget; 61 static TCGv env_iflags; 62 static TCGv env_res_addr; 63 static TCGv env_res_val; 64 65 #include "exec/gen-icount.h" 66 67 /* This is the state at translation time. */ 68 typedef struct DisasContext { 69 MicroBlazeCPU *cpu; 70 target_ulong pc; 71 72 /* Decoder. */ 73 int type_b; 74 uint32_t ir; 75 uint8_t opcode; 76 uint8_t rd, ra, rb; 77 uint16_t imm; 78 79 unsigned int cpustate_changed; 80 unsigned int delayed_branch; 81 unsigned int tb_flags, synced_flags; /* tb dependent flags. */ 82 unsigned int clear_imm; 83 int is_jmp; 84 85 #define JMP_NOJMP 0 86 #define JMP_DIRECT 1 87 #define JMP_DIRECT_CC 2 88 #define JMP_INDIRECT 3 89 unsigned int jmp; 90 uint32_t jmp_pc; 91 92 int abort_at_next_insn; 93 int nr_nops; 94 struct TranslationBlock *tb; 95 int singlestep_enabled; 96 } DisasContext; 97 98 static const char *regnames[] = 99 { 100 "r0", "r1", "r2", "r3", "r4", "r5", "r6", "r7", 101 "r8", "r9", "r10", "r11", "r12", "r13", "r14", "r15", 102 "r16", "r17", "r18", "r19", "r20", "r21", "r22", "r23", 103 "r24", "r25", "r26", "r27", "r28", "r29", "r30", "r31", 104 }; 105 106 static const char *special_regnames[] = 107 { 108 "rpc", "rmsr", "sr2", "sr3", "sr4", "sr5", "sr6", "sr7", 109 "sr8", "sr9", "sr10", "sr11", "sr12", "sr13", "sr14", "sr15", 110 "sr16", "sr17", "sr18" 111 }; 112 113 static inline void t_sync_flags(DisasContext *dc) 114 { 115 /* Synch the tb dependent flags between translator and runtime. */ 116 if (dc->tb_flags != dc->synced_flags) { 117 tcg_gen_movi_tl(env_iflags, dc->tb_flags); 118 dc->synced_flags = dc->tb_flags; 119 } 120 } 121 122 static inline void t_gen_raise_exception(DisasContext *dc, uint32_t index) 123 { 124 TCGv_i32 tmp = tcg_const_i32(index); 125 126 t_sync_flags(dc); 127 tcg_gen_movi_tl(cpu_SR[SR_PC], dc->pc); 128 gen_helper_raise_exception(cpu_env, tmp); 129 tcg_temp_free_i32(tmp); 130 dc->is_jmp = DISAS_UPDATE; 131 } 132 133 static inline bool use_goto_tb(DisasContext *dc, target_ulong dest) 134 { 135 #ifndef CONFIG_USER_ONLY 136 return (dc->tb->pc & TARGET_PAGE_MASK) == (dest & TARGET_PAGE_MASK); 137 #else 138 return true; 139 #endif 140 } 141 142 static void gen_goto_tb(DisasContext *dc, int n, target_ulong dest) 143 { 144 if (use_goto_tb(dc, dest)) { 145 tcg_gen_goto_tb(n); 146 tcg_gen_movi_tl(cpu_SR[SR_PC], dest); 147 tcg_gen_exit_tb((uintptr_t)dc->tb + n); 148 } else { 149 tcg_gen_movi_tl(cpu_SR[SR_PC], dest); 150 tcg_gen_exit_tb(0); 151 } 152 } 153 154 static void read_carry(DisasContext *dc, TCGv d) 155 { 156 tcg_gen_shri_tl(d, cpu_SR[SR_MSR], 31); 157 } 158 159 /* 160 * write_carry sets the carry bits in MSR based on bit 0 of v. 161 * v[31:1] are ignored. 162 */ 163 static void write_carry(DisasContext *dc, TCGv v) 164 { 165 TCGv t0 = tcg_temp_new(); 166 tcg_gen_shli_tl(t0, v, 31); 167 tcg_gen_sari_tl(t0, t0, 31); 168 tcg_gen_andi_tl(t0, t0, (MSR_C | MSR_CC)); 169 tcg_gen_andi_tl(cpu_SR[SR_MSR], cpu_SR[SR_MSR], 170 ~(MSR_C | MSR_CC)); 171 tcg_gen_or_tl(cpu_SR[SR_MSR], cpu_SR[SR_MSR], t0); 172 tcg_temp_free(t0); 173 } 174 175 static void write_carryi(DisasContext *dc, bool carry) 176 { 177 TCGv t0 = tcg_temp_new(); 178 tcg_gen_movi_tl(t0, carry); 179 write_carry(dc, t0); 180 tcg_temp_free(t0); 181 } 182 183 /* True if ALU operand b is a small immediate that may deserve 184 faster treatment. */ 185 static inline int dec_alu_op_b_is_small_imm(DisasContext *dc) 186 { 187 /* Immediate insn without the imm prefix ? */ 188 return dc->type_b && !(dc->tb_flags & IMM_FLAG); 189 } 190 191 static inline TCGv *dec_alu_op_b(DisasContext *dc) 192 { 193 if (dc->type_b) { 194 if (dc->tb_flags & IMM_FLAG) 195 tcg_gen_ori_tl(env_imm, env_imm, dc->imm); 196 else 197 tcg_gen_movi_tl(env_imm, (int32_t)((int16_t)dc->imm)); 198 return &env_imm; 199 } else 200 return &cpu_R[dc->rb]; 201 } 202 203 static void dec_add(DisasContext *dc) 204 { 205 unsigned int k, c; 206 TCGv cf; 207 208 k = dc->opcode & 4; 209 c = dc->opcode & 2; 210 211 LOG_DIS("add%s%s%s r%d r%d r%d\n", 212 dc->type_b ? "i" : "", k ? "k" : "", c ? "c" : "", 213 dc->rd, dc->ra, dc->rb); 214 215 /* Take care of the easy cases first. */ 216 if (k) { 217 /* k - keep carry, no need to update MSR. */ 218 /* If rd == r0, it's a nop. */ 219 if (dc->rd) { 220 tcg_gen_add_tl(cpu_R[dc->rd], cpu_R[dc->ra], *(dec_alu_op_b(dc))); 221 222 if (c) { 223 /* c - Add carry into the result. */ 224 cf = tcg_temp_new(); 225 226 read_carry(dc, cf); 227 tcg_gen_add_tl(cpu_R[dc->rd], cpu_R[dc->rd], cf); 228 tcg_temp_free(cf); 229 } 230 } 231 return; 232 } 233 234 /* From now on, we can assume k is zero. So we need to update MSR. */ 235 /* Extract carry. */ 236 cf = tcg_temp_new(); 237 if (c) { 238 read_carry(dc, cf); 239 } else { 240 tcg_gen_movi_tl(cf, 0); 241 } 242 243 if (dc->rd) { 244 TCGv ncf = tcg_temp_new(); 245 gen_helper_carry(ncf, cpu_R[dc->ra], *(dec_alu_op_b(dc)), cf); 246 tcg_gen_add_tl(cpu_R[dc->rd], cpu_R[dc->ra], *(dec_alu_op_b(dc))); 247 tcg_gen_add_tl(cpu_R[dc->rd], cpu_R[dc->rd], cf); 248 write_carry(dc, ncf); 249 tcg_temp_free(ncf); 250 } else { 251 gen_helper_carry(cf, cpu_R[dc->ra], *(dec_alu_op_b(dc)), cf); 252 write_carry(dc, cf); 253 } 254 tcg_temp_free(cf); 255 } 256 257 static void dec_sub(DisasContext *dc) 258 { 259 unsigned int u, cmp, k, c; 260 TCGv cf, na; 261 262 u = dc->imm & 2; 263 k = dc->opcode & 4; 264 c = dc->opcode & 2; 265 cmp = (dc->imm & 1) && (!dc->type_b) && k; 266 267 if (cmp) { 268 LOG_DIS("cmp%s r%d, r%d ir=%x\n", u ? "u" : "", dc->rd, dc->ra, dc->ir); 269 if (dc->rd) { 270 if (u) 271 gen_helper_cmpu(cpu_R[dc->rd], cpu_R[dc->ra], cpu_R[dc->rb]); 272 else 273 gen_helper_cmp(cpu_R[dc->rd], cpu_R[dc->ra], cpu_R[dc->rb]); 274 } 275 return; 276 } 277 278 LOG_DIS("sub%s%s r%d, r%d r%d\n", 279 k ? "k" : "", c ? "c" : "", dc->rd, dc->ra, dc->rb); 280 281 /* Take care of the easy cases first. */ 282 if (k) { 283 /* k - keep carry, no need to update MSR. */ 284 /* If rd == r0, it's a nop. */ 285 if (dc->rd) { 286 tcg_gen_sub_tl(cpu_R[dc->rd], *(dec_alu_op_b(dc)), cpu_R[dc->ra]); 287 288 if (c) { 289 /* c - Add carry into the result. */ 290 cf = tcg_temp_new(); 291 292 read_carry(dc, cf); 293 tcg_gen_add_tl(cpu_R[dc->rd], cpu_R[dc->rd], cf); 294 tcg_temp_free(cf); 295 } 296 } 297 return; 298 } 299 300 /* From now on, we can assume k is zero. So we need to update MSR. */ 301 /* Extract carry. And complement a into na. */ 302 cf = tcg_temp_new(); 303 na = tcg_temp_new(); 304 if (c) { 305 read_carry(dc, cf); 306 } else { 307 tcg_gen_movi_tl(cf, 1); 308 } 309 310 /* d = b + ~a + c. carry defaults to 1. */ 311 tcg_gen_not_tl(na, cpu_R[dc->ra]); 312 313 if (dc->rd) { 314 TCGv ncf = tcg_temp_new(); 315 gen_helper_carry(ncf, na, *(dec_alu_op_b(dc)), cf); 316 tcg_gen_add_tl(cpu_R[dc->rd], na, *(dec_alu_op_b(dc))); 317 tcg_gen_add_tl(cpu_R[dc->rd], cpu_R[dc->rd], cf); 318 write_carry(dc, ncf); 319 tcg_temp_free(ncf); 320 } else { 321 gen_helper_carry(cf, na, *(dec_alu_op_b(dc)), cf); 322 write_carry(dc, cf); 323 } 324 tcg_temp_free(cf); 325 tcg_temp_free(na); 326 } 327 328 static void dec_pattern(DisasContext *dc) 329 { 330 unsigned int mode; 331 332 if ((dc->tb_flags & MSR_EE_FLAG) 333 && (dc->cpu->env.pvr.regs[2] & PVR2_ILL_OPCODE_EXC_MASK) 334 && !dc->cpu->cfg.use_pcmp_instr) { 335 tcg_gen_movi_tl(cpu_SR[SR_ESR], ESR_EC_ILLEGAL_OP); 336 t_gen_raise_exception(dc, EXCP_HW_EXCP); 337 } 338 339 mode = dc->opcode & 3; 340 switch (mode) { 341 case 0: 342 /* pcmpbf. */ 343 LOG_DIS("pcmpbf r%d r%d r%d\n", dc->rd, dc->ra, dc->rb); 344 if (dc->rd) 345 gen_helper_pcmpbf(cpu_R[dc->rd], cpu_R[dc->ra], cpu_R[dc->rb]); 346 break; 347 case 2: 348 LOG_DIS("pcmpeq r%d r%d r%d\n", dc->rd, dc->ra, dc->rb); 349 if (dc->rd) { 350 tcg_gen_setcond_tl(TCG_COND_EQ, cpu_R[dc->rd], 351 cpu_R[dc->ra], cpu_R[dc->rb]); 352 } 353 break; 354 case 3: 355 LOG_DIS("pcmpne r%d r%d r%d\n", dc->rd, dc->ra, dc->rb); 356 if (dc->rd) { 357 tcg_gen_setcond_tl(TCG_COND_NE, cpu_R[dc->rd], 358 cpu_R[dc->ra], cpu_R[dc->rb]); 359 } 360 break; 361 default: 362 cpu_abort(CPU(dc->cpu), 363 "unsupported pattern insn opcode=%x\n", dc->opcode); 364 break; 365 } 366 } 367 368 static void dec_and(DisasContext *dc) 369 { 370 unsigned int not; 371 372 if (!dc->type_b && (dc->imm & (1 << 10))) { 373 dec_pattern(dc); 374 return; 375 } 376 377 not = dc->opcode & (1 << 1); 378 LOG_DIS("and%s\n", not ? "n" : ""); 379 380 if (!dc->rd) 381 return; 382 383 if (not) { 384 tcg_gen_andc_tl(cpu_R[dc->rd], cpu_R[dc->ra], *(dec_alu_op_b(dc))); 385 } else 386 tcg_gen_and_tl(cpu_R[dc->rd], cpu_R[dc->ra], *(dec_alu_op_b(dc))); 387 } 388 389 static void dec_or(DisasContext *dc) 390 { 391 if (!dc->type_b && (dc->imm & (1 << 10))) { 392 dec_pattern(dc); 393 return; 394 } 395 396 LOG_DIS("or r%d r%d r%d imm=%x\n", dc->rd, dc->ra, dc->rb, dc->imm); 397 if (dc->rd) 398 tcg_gen_or_tl(cpu_R[dc->rd], cpu_R[dc->ra], *(dec_alu_op_b(dc))); 399 } 400 401 static void dec_xor(DisasContext *dc) 402 { 403 if (!dc->type_b && (dc->imm & (1 << 10))) { 404 dec_pattern(dc); 405 return; 406 } 407 408 LOG_DIS("xor r%d\n", dc->rd); 409 if (dc->rd) 410 tcg_gen_xor_tl(cpu_R[dc->rd], cpu_R[dc->ra], *(dec_alu_op_b(dc))); 411 } 412 413 static inline void msr_read(DisasContext *dc, TCGv d) 414 { 415 tcg_gen_mov_tl(d, cpu_SR[SR_MSR]); 416 } 417 418 static inline void msr_write(DisasContext *dc, TCGv v) 419 { 420 TCGv t; 421 422 t = tcg_temp_new(); 423 dc->cpustate_changed = 1; 424 /* PVR bit is not writable. */ 425 tcg_gen_andi_tl(t, v, ~MSR_PVR); 426 tcg_gen_andi_tl(cpu_SR[SR_MSR], cpu_SR[SR_MSR], MSR_PVR); 427 tcg_gen_or_tl(cpu_SR[SR_MSR], cpu_SR[SR_MSR], t); 428 tcg_temp_free(t); 429 } 430 431 static void dec_msr(DisasContext *dc) 432 { 433 CPUState *cs = CPU(dc->cpu); 434 TCGv t0, t1; 435 unsigned int sr, to, rn; 436 int mem_index = cpu_mmu_index(&dc->cpu->env, false); 437 438 sr = dc->imm & ((1 << 14) - 1); 439 to = dc->imm & (1 << 14); 440 dc->type_b = 1; 441 if (to) 442 dc->cpustate_changed = 1; 443 444 /* msrclr and msrset. */ 445 if (!(dc->imm & (1 << 15))) { 446 unsigned int clr = dc->ir & (1 << 16); 447 448 LOG_DIS("msr%s r%d imm=%x\n", clr ? "clr" : "set", 449 dc->rd, dc->imm); 450 451 if (!dc->cpu->cfg.use_msr_instr) { 452 /* nop??? */ 453 return; 454 } 455 456 if ((dc->tb_flags & MSR_EE_FLAG) 457 && mem_index == MMU_USER_IDX && (dc->imm != 4 && dc->imm != 0)) { 458 tcg_gen_movi_tl(cpu_SR[SR_ESR], ESR_EC_PRIVINSN); 459 t_gen_raise_exception(dc, EXCP_HW_EXCP); 460 return; 461 } 462 463 if (dc->rd) 464 msr_read(dc, cpu_R[dc->rd]); 465 466 t0 = tcg_temp_new(); 467 t1 = tcg_temp_new(); 468 msr_read(dc, t0); 469 tcg_gen_mov_tl(t1, *(dec_alu_op_b(dc))); 470 471 if (clr) { 472 tcg_gen_not_tl(t1, t1); 473 tcg_gen_and_tl(t0, t0, t1); 474 } else 475 tcg_gen_or_tl(t0, t0, t1); 476 msr_write(dc, t0); 477 tcg_temp_free(t0); 478 tcg_temp_free(t1); 479 tcg_gen_movi_tl(cpu_SR[SR_PC], dc->pc + 4); 480 dc->is_jmp = DISAS_UPDATE; 481 return; 482 } 483 484 if (to) { 485 if ((dc->tb_flags & MSR_EE_FLAG) 486 && mem_index == MMU_USER_IDX) { 487 tcg_gen_movi_tl(cpu_SR[SR_ESR], ESR_EC_PRIVINSN); 488 t_gen_raise_exception(dc, EXCP_HW_EXCP); 489 return; 490 } 491 } 492 493 #if !defined(CONFIG_USER_ONLY) 494 /* Catch read/writes to the mmu block. */ 495 if ((sr & ~0xff) == 0x1000) { 496 sr &= 7; 497 LOG_DIS("m%ss sr%d r%d imm=%x\n", to ? "t" : "f", sr, dc->ra, dc->imm); 498 if (to) 499 gen_helper_mmu_write(cpu_env, tcg_const_tl(sr), cpu_R[dc->ra]); 500 else 501 gen_helper_mmu_read(cpu_R[dc->rd], cpu_env, tcg_const_tl(sr)); 502 return; 503 } 504 #endif 505 506 if (to) { 507 LOG_DIS("m%ss sr%x r%d imm=%x\n", to ? "t" : "f", sr, dc->ra, dc->imm); 508 switch (sr) { 509 case 0: 510 break; 511 case 1: 512 msr_write(dc, cpu_R[dc->ra]); 513 break; 514 case 0x3: 515 tcg_gen_mov_tl(cpu_SR[SR_EAR], cpu_R[dc->ra]); 516 break; 517 case 0x5: 518 tcg_gen_mov_tl(cpu_SR[SR_ESR], cpu_R[dc->ra]); 519 break; 520 case 0x7: 521 tcg_gen_andi_tl(cpu_SR[SR_FSR], cpu_R[dc->ra], 31); 522 break; 523 case 0x800: 524 tcg_gen_st_tl(cpu_R[dc->ra], cpu_env, offsetof(CPUMBState, slr)); 525 break; 526 case 0x802: 527 tcg_gen_st_tl(cpu_R[dc->ra], cpu_env, offsetof(CPUMBState, shr)); 528 break; 529 default: 530 cpu_abort(CPU(dc->cpu), "unknown mts reg %x\n", sr); 531 break; 532 } 533 } else { 534 LOG_DIS("m%ss r%d sr%x imm=%x\n", to ? "t" : "f", dc->rd, sr, dc->imm); 535 536 switch (sr) { 537 case 0: 538 tcg_gen_movi_tl(cpu_R[dc->rd], dc->pc); 539 break; 540 case 1: 541 msr_read(dc, cpu_R[dc->rd]); 542 break; 543 case 0x3: 544 tcg_gen_mov_tl(cpu_R[dc->rd], cpu_SR[SR_EAR]); 545 break; 546 case 0x5: 547 tcg_gen_mov_tl(cpu_R[dc->rd], cpu_SR[SR_ESR]); 548 break; 549 case 0x7: 550 tcg_gen_mov_tl(cpu_R[dc->rd], cpu_SR[SR_FSR]); 551 break; 552 case 0xb: 553 tcg_gen_mov_tl(cpu_R[dc->rd], cpu_SR[SR_BTR]); 554 break; 555 case 0x800: 556 tcg_gen_ld_tl(cpu_R[dc->rd], cpu_env, offsetof(CPUMBState, slr)); 557 break; 558 case 0x802: 559 tcg_gen_ld_tl(cpu_R[dc->rd], cpu_env, offsetof(CPUMBState, shr)); 560 break; 561 case 0x2000: 562 case 0x2001: 563 case 0x2002: 564 case 0x2003: 565 case 0x2004: 566 case 0x2005: 567 case 0x2006: 568 case 0x2007: 569 case 0x2008: 570 case 0x2009: 571 case 0x200a: 572 case 0x200b: 573 case 0x200c: 574 rn = sr & 0xf; 575 tcg_gen_ld_tl(cpu_R[dc->rd], 576 cpu_env, offsetof(CPUMBState, pvr.regs[rn])); 577 break; 578 default: 579 cpu_abort(cs, "unknown mfs reg %x\n", sr); 580 break; 581 } 582 } 583 584 if (dc->rd == 0) { 585 tcg_gen_movi_tl(cpu_R[0], 0); 586 } 587 } 588 589 /* Multiplier unit. */ 590 static void dec_mul(DisasContext *dc) 591 { 592 TCGv tmp; 593 unsigned int subcode; 594 595 if ((dc->tb_flags & MSR_EE_FLAG) 596 && (dc->cpu->env.pvr.regs[2] & PVR2_ILL_OPCODE_EXC_MASK) 597 && !dc->cpu->cfg.use_hw_mul) { 598 tcg_gen_movi_tl(cpu_SR[SR_ESR], ESR_EC_ILLEGAL_OP); 599 t_gen_raise_exception(dc, EXCP_HW_EXCP); 600 return; 601 } 602 603 subcode = dc->imm & 3; 604 605 if (dc->type_b) { 606 LOG_DIS("muli r%d r%d %x\n", dc->rd, dc->ra, dc->imm); 607 tcg_gen_mul_tl(cpu_R[dc->rd], cpu_R[dc->ra], *(dec_alu_op_b(dc))); 608 return; 609 } 610 611 /* mulh, mulhsu and mulhu are not available if C_USE_HW_MUL is < 2. */ 612 if (subcode >= 1 && subcode <= 3 && dc->cpu->cfg.use_hw_mul < 2) { 613 /* nop??? */ 614 } 615 616 tmp = tcg_temp_new(); 617 switch (subcode) { 618 case 0: 619 LOG_DIS("mul r%d r%d r%d\n", dc->rd, dc->ra, dc->rb); 620 tcg_gen_mul_tl(cpu_R[dc->rd], cpu_R[dc->ra], cpu_R[dc->rb]); 621 break; 622 case 1: 623 LOG_DIS("mulh r%d r%d r%d\n", dc->rd, dc->ra, dc->rb); 624 tcg_gen_muls2_tl(tmp, cpu_R[dc->rd], cpu_R[dc->ra], cpu_R[dc->rb]); 625 break; 626 case 2: 627 LOG_DIS("mulhsu r%d r%d r%d\n", dc->rd, dc->ra, dc->rb); 628 tcg_gen_mulsu2_tl(tmp, cpu_R[dc->rd], cpu_R[dc->ra], cpu_R[dc->rb]); 629 break; 630 case 3: 631 LOG_DIS("mulhu r%d r%d r%d\n", dc->rd, dc->ra, dc->rb); 632 tcg_gen_mulu2_tl(tmp, cpu_R[dc->rd], cpu_R[dc->ra], cpu_R[dc->rb]); 633 break; 634 default: 635 cpu_abort(CPU(dc->cpu), "unknown MUL insn %x\n", subcode); 636 break; 637 } 638 tcg_temp_free(tmp); 639 } 640 641 /* Div unit. */ 642 static void dec_div(DisasContext *dc) 643 { 644 unsigned int u; 645 646 u = dc->imm & 2; 647 LOG_DIS("div\n"); 648 649 if ((dc->cpu->env.pvr.regs[2] & PVR2_ILL_OPCODE_EXC_MASK) 650 && !dc->cpu->cfg.use_div) { 651 tcg_gen_movi_tl(cpu_SR[SR_ESR], ESR_EC_ILLEGAL_OP); 652 t_gen_raise_exception(dc, EXCP_HW_EXCP); 653 } 654 655 if (u) 656 gen_helper_divu(cpu_R[dc->rd], cpu_env, *(dec_alu_op_b(dc)), 657 cpu_R[dc->ra]); 658 else 659 gen_helper_divs(cpu_R[dc->rd], cpu_env, *(dec_alu_op_b(dc)), 660 cpu_R[dc->ra]); 661 if (!dc->rd) 662 tcg_gen_movi_tl(cpu_R[dc->rd], 0); 663 } 664 665 static void dec_barrel(DisasContext *dc) 666 { 667 TCGv t0; 668 unsigned int imm_w, imm_s; 669 bool s, t, e = false, i = false; 670 671 if ((dc->tb_flags & MSR_EE_FLAG) 672 && (dc->cpu->env.pvr.regs[2] & PVR2_ILL_OPCODE_EXC_MASK) 673 && !dc->cpu->cfg.use_barrel) { 674 tcg_gen_movi_tl(cpu_SR[SR_ESR], ESR_EC_ILLEGAL_OP); 675 t_gen_raise_exception(dc, EXCP_HW_EXCP); 676 return; 677 } 678 679 if (dc->type_b) { 680 /* Insert and extract are only available in immediate mode. */ 681 i = extract32(dc->imm, 15, 1); 682 e = extract32(dc->imm, 14, 1); 683 } 684 s = extract32(dc->imm, 10, 1); 685 t = extract32(dc->imm, 9, 1); 686 imm_w = extract32(dc->imm, 6, 5); 687 imm_s = extract32(dc->imm, 0, 5); 688 689 LOG_DIS("bs%s%s%s r%d r%d r%d\n", 690 e ? "e" : "", 691 s ? "l" : "r", t ? "a" : "l", dc->rd, dc->ra, dc->rb); 692 693 if (e) { 694 if (imm_w + imm_s > 32 || imm_w == 0) { 695 /* These inputs have an undefined behavior. */ 696 qemu_log_mask(LOG_GUEST_ERROR, "bsefi: Bad input w=%d s=%d\n", 697 imm_w, imm_s); 698 } else { 699 tcg_gen_extract_i32(cpu_R[dc->rd], cpu_R[dc->ra], imm_s, imm_w); 700 } 701 } else if (i) { 702 int width = imm_w - imm_s + 1; 703 704 if (imm_w < imm_s) { 705 /* These inputs have an undefined behavior. */ 706 qemu_log_mask(LOG_GUEST_ERROR, "bsifi: Bad input w=%d s=%d\n", 707 imm_w, imm_s); 708 } else { 709 tcg_gen_deposit_i32(cpu_R[dc->rd], cpu_R[dc->rd], cpu_R[dc->ra], 710 imm_s, width); 711 } 712 } else { 713 t0 = tcg_temp_new(); 714 715 tcg_gen_mov_tl(t0, *(dec_alu_op_b(dc))); 716 tcg_gen_andi_tl(t0, t0, 31); 717 718 if (s) { 719 tcg_gen_shl_tl(cpu_R[dc->rd], cpu_R[dc->ra], t0); 720 } else { 721 if (t) { 722 tcg_gen_sar_tl(cpu_R[dc->rd], cpu_R[dc->ra], t0); 723 } else { 724 tcg_gen_shr_tl(cpu_R[dc->rd], cpu_R[dc->ra], t0); 725 } 726 } 727 tcg_temp_free(t0); 728 } 729 } 730 731 static void dec_bit(DisasContext *dc) 732 { 733 CPUState *cs = CPU(dc->cpu); 734 TCGv t0; 735 unsigned int op; 736 int mem_index = cpu_mmu_index(&dc->cpu->env, false); 737 738 op = dc->ir & ((1 << 9) - 1); 739 switch (op) { 740 case 0x21: 741 /* src. */ 742 t0 = tcg_temp_new(); 743 744 LOG_DIS("src r%d r%d\n", dc->rd, dc->ra); 745 tcg_gen_andi_tl(t0, cpu_SR[SR_MSR], MSR_CC); 746 write_carry(dc, cpu_R[dc->ra]); 747 if (dc->rd) { 748 tcg_gen_shri_tl(cpu_R[dc->rd], cpu_R[dc->ra], 1); 749 tcg_gen_or_tl(cpu_R[dc->rd], cpu_R[dc->rd], t0); 750 } 751 tcg_temp_free(t0); 752 break; 753 754 case 0x1: 755 case 0x41: 756 /* srl. */ 757 LOG_DIS("srl r%d r%d\n", dc->rd, dc->ra); 758 759 /* Update carry. Note that write carry only looks at the LSB. */ 760 write_carry(dc, cpu_R[dc->ra]); 761 if (dc->rd) { 762 if (op == 0x41) 763 tcg_gen_shri_tl(cpu_R[dc->rd], cpu_R[dc->ra], 1); 764 else 765 tcg_gen_sari_tl(cpu_R[dc->rd], cpu_R[dc->ra], 1); 766 } 767 break; 768 case 0x60: 769 LOG_DIS("ext8s r%d r%d\n", dc->rd, dc->ra); 770 tcg_gen_ext8s_i32(cpu_R[dc->rd], cpu_R[dc->ra]); 771 break; 772 case 0x61: 773 LOG_DIS("ext16s r%d r%d\n", dc->rd, dc->ra); 774 tcg_gen_ext16s_i32(cpu_R[dc->rd], cpu_R[dc->ra]); 775 break; 776 case 0x64: 777 case 0x66: 778 case 0x74: 779 case 0x76: 780 /* wdc. */ 781 LOG_DIS("wdc r%d\n", dc->ra); 782 if ((dc->tb_flags & MSR_EE_FLAG) 783 && mem_index == MMU_USER_IDX) { 784 tcg_gen_movi_tl(cpu_SR[SR_ESR], ESR_EC_PRIVINSN); 785 t_gen_raise_exception(dc, EXCP_HW_EXCP); 786 return; 787 } 788 break; 789 case 0x68: 790 /* wic. */ 791 LOG_DIS("wic r%d\n", dc->ra); 792 if ((dc->tb_flags & MSR_EE_FLAG) 793 && mem_index == MMU_USER_IDX) { 794 tcg_gen_movi_tl(cpu_SR[SR_ESR], ESR_EC_PRIVINSN); 795 t_gen_raise_exception(dc, EXCP_HW_EXCP); 796 return; 797 } 798 break; 799 case 0xe0: 800 if ((dc->tb_flags & MSR_EE_FLAG) 801 && (dc->cpu->env.pvr.regs[2] & PVR2_ILL_OPCODE_EXC_MASK) 802 && !dc->cpu->cfg.use_pcmp_instr) { 803 tcg_gen_movi_tl(cpu_SR[SR_ESR], ESR_EC_ILLEGAL_OP); 804 t_gen_raise_exception(dc, EXCP_HW_EXCP); 805 } 806 if (dc->cpu->cfg.use_pcmp_instr) { 807 tcg_gen_clzi_i32(cpu_R[dc->rd], cpu_R[dc->ra], 32); 808 } 809 break; 810 case 0x1e0: 811 /* swapb */ 812 LOG_DIS("swapb r%d r%d\n", dc->rd, dc->ra); 813 tcg_gen_bswap32_i32(cpu_R[dc->rd], cpu_R[dc->ra]); 814 break; 815 case 0x1e2: 816 /*swaph */ 817 LOG_DIS("swaph r%d r%d\n", dc->rd, dc->ra); 818 tcg_gen_rotri_i32(cpu_R[dc->rd], cpu_R[dc->ra], 16); 819 break; 820 default: 821 cpu_abort(cs, "unknown bit oc=%x op=%x rd=%d ra=%d rb=%d\n", 822 dc->pc, op, dc->rd, dc->ra, dc->rb); 823 break; 824 } 825 } 826 827 static inline void sync_jmpstate(DisasContext *dc) 828 { 829 if (dc->jmp == JMP_DIRECT || dc->jmp == JMP_DIRECT_CC) { 830 if (dc->jmp == JMP_DIRECT) { 831 tcg_gen_movi_tl(env_btaken, 1); 832 } 833 dc->jmp = JMP_INDIRECT; 834 tcg_gen_movi_tl(env_btarget, dc->jmp_pc); 835 } 836 } 837 838 static void dec_imm(DisasContext *dc) 839 { 840 LOG_DIS("imm %x\n", dc->imm << 16); 841 tcg_gen_movi_tl(env_imm, (dc->imm << 16)); 842 dc->tb_flags |= IMM_FLAG; 843 dc->clear_imm = 0; 844 } 845 846 static inline TCGv *compute_ldst_addr(DisasContext *dc, TCGv *t) 847 { 848 bool extimm = dc->tb_flags & IMM_FLAG; 849 /* Should be set to true if r1 is used by loadstores. */ 850 bool stackprot = false; 851 852 /* All load/stores use ra. */ 853 if (dc->ra == 1 && dc->cpu->cfg.stackprot) { 854 stackprot = true; 855 } 856 857 /* Treat the common cases first. */ 858 if (!dc->type_b) { 859 /* If any of the regs is r0, return a ptr to the other. */ 860 if (dc->ra == 0) { 861 return &cpu_R[dc->rb]; 862 } else if (dc->rb == 0) { 863 return &cpu_R[dc->ra]; 864 } 865 866 if (dc->rb == 1 && dc->cpu->cfg.stackprot) { 867 stackprot = true; 868 } 869 870 *t = tcg_temp_new(); 871 tcg_gen_add_tl(*t, cpu_R[dc->ra], cpu_R[dc->rb]); 872 873 if (stackprot) { 874 gen_helper_stackprot(cpu_env, *t); 875 } 876 return t; 877 } 878 /* Immediate. */ 879 if (!extimm) { 880 if (dc->imm == 0) { 881 return &cpu_R[dc->ra]; 882 } 883 *t = tcg_temp_new(); 884 tcg_gen_movi_tl(*t, (int32_t)((int16_t)dc->imm)); 885 tcg_gen_add_tl(*t, cpu_R[dc->ra], *t); 886 } else { 887 *t = tcg_temp_new(); 888 tcg_gen_add_tl(*t, cpu_R[dc->ra], *(dec_alu_op_b(dc))); 889 } 890 891 if (stackprot) { 892 gen_helper_stackprot(cpu_env, *t); 893 } 894 return t; 895 } 896 897 static void dec_load(DisasContext *dc) 898 { 899 TCGv t, v, *addr; 900 unsigned int size; 901 bool rev = false, ex = false; 902 TCGMemOp mop; 903 904 mop = dc->opcode & 3; 905 size = 1 << mop; 906 if (!dc->type_b) { 907 rev = extract32(dc->ir, 9, 1); 908 ex = extract32(dc->ir, 10, 1); 909 } 910 mop |= MO_TE; 911 if (rev) { 912 mop ^= MO_BSWAP; 913 } 914 915 if (size > 4 && (dc->tb_flags & MSR_EE_FLAG) 916 && (dc->cpu->env.pvr.regs[2] & PVR2_ILL_OPCODE_EXC_MASK)) { 917 tcg_gen_movi_tl(cpu_SR[SR_ESR], ESR_EC_ILLEGAL_OP); 918 t_gen_raise_exception(dc, EXCP_HW_EXCP); 919 return; 920 } 921 922 LOG_DIS("l%d%s%s%s\n", size, dc->type_b ? "i" : "", rev ? "r" : "", 923 ex ? "x" : ""); 924 925 t_sync_flags(dc); 926 addr = compute_ldst_addr(dc, &t); 927 928 /* 929 * When doing reverse accesses we need to do two things. 930 * 931 * 1. Reverse the address wrt endianness. 932 * 2. Byteswap the data lanes on the way back into the CPU core. 933 */ 934 if (rev && size != 4) { 935 /* Endian reverse the address. t is addr. */ 936 switch (size) { 937 case 1: 938 { 939 /* 00 -> 11 940 01 -> 10 941 10 -> 10 942 11 -> 00 */ 943 TCGv low = tcg_temp_new(); 944 945 /* Force addr into the temp. */ 946 if (addr != &t) { 947 t = tcg_temp_new(); 948 tcg_gen_mov_tl(t, *addr); 949 addr = &t; 950 } 951 952 tcg_gen_andi_tl(low, t, 3); 953 tcg_gen_sub_tl(low, tcg_const_tl(3), low); 954 tcg_gen_andi_tl(t, t, ~3); 955 tcg_gen_or_tl(t, t, low); 956 tcg_temp_free(low); 957 break; 958 } 959 960 case 2: 961 /* 00 -> 10 962 10 -> 00. */ 963 /* Force addr into the temp. */ 964 if (addr != &t) { 965 t = tcg_temp_new(); 966 tcg_gen_xori_tl(t, *addr, 2); 967 addr = &t; 968 } else { 969 tcg_gen_xori_tl(t, t, 2); 970 } 971 break; 972 default: 973 cpu_abort(CPU(dc->cpu), "Invalid reverse size\n"); 974 break; 975 } 976 } 977 978 /* lwx does not throw unaligned access errors, so force alignment */ 979 if (ex) { 980 /* Force addr into the temp. */ 981 if (addr != &t) { 982 t = tcg_temp_new(); 983 tcg_gen_mov_tl(t, *addr); 984 addr = &t; 985 } 986 tcg_gen_andi_tl(t, t, ~3); 987 } 988 989 /* If we get a fault on a dslot, the jmpstate better be in sync. */ 990 sync_jmpstate(dc); 991 992 /* Verify alignment if needed. */ 993 /* 994 * Microblaze gives MMU faults priority over faults due to 995 * unaligned addresses. That's why we speculatively do the load 996 * into v. If the load succeeds, we verify alignment of the 997 * address and if that succeeds we write into the destination reg. 998 */ 999 v = tcg_temp_new(); 1000 tcg_gen_qemu_ld_tl(v, *addr, cpu_mmu_index(&dc->cpu->env, false), mop); 1001 1002 if ((dc->cpu->env.pvr.regs[2] & PVR2_UNALIGNED_EXC_MASK) && size > 1) { 1003 tcg_gen_movi_tl(cpu_SR[SR_PC], dc->pc); 1004 gen_helper_memalign(cpu_env, *addr, tcg_const_tl(dc->rd), 1005 tcg_const_tl(0), tcg_const_tl(size - 1)); 1006 } 1007 1008 if (ex) { 1009 tcg_gen_mov_tl(env_res_addr, *addr); 1010 tcg_gen_mov_tl(env_res_val, v); 1011 } 1012 if (dc->rd) { 1013 tcg_gen_mov_tl(cpu_R[dc->rd], v); 1014 } 1015 tcg_temp_free(v); 1016 1017 if (ex) { /* lwx */ 1018 /* no support for AXI exclusive so always clear C */ 1019 write_carryi(dc, 0); 1020 } 1021 1022 if (addr == &t) 1023 tcg_temp_free(t); 1024 } 1025 1026 static void dec_store(DisasContext *dc) 1027 { 1028 TCGv t, *addr, swx_addr; 1029 TCGLabel *swx_skip = NULL; 1030 unsigned int size; 1031 bool rev = false, ex = false; 1032 TCGMemOp mop; 1033 1034 mop = dc->opcode & 3; 1035 size = 1 << mop; 1036 if (!dc->type_b) { 1037 rev = extract32(dc->ir, 9, 1); 1038 ex = extract32(dc->ir, 10, 1); 1039 } 1040 mop |= MO_TE; 1041 if (rev) { 1042 mop ^= MO_BSWAP; 1043 } 1044 1045 if (size > 4 && (dc->tb_flags & MSR_EE_FLAG) 1046 && (dc->cpu->env.pvr.regs[2] & PVR2_ILL_OPCODE_EXC_MASK)) { 1047 tcg_gen_movi_tl(cpu_SR[SR_ESR], ESR_EC_ILLEGAL_OP); 1048 t_gen_raise_exception(dc, EXCP_HW_EXCP); 1049 return; 1050 } 1051 1052 LOG_DIS("s%d%s%s%s\n", size, dc->type_b ? "i" : "", rev ? "r" : "", 1053 ex ? "x" : ""); 1054 t_sync_flags(dc); 1055 /* If we get a fault on a dslot, the jmpstate better be in sync. */ 1056 sync_jmpstate(dc); 1057 addr = compute_ldst_addr(dc, &t); 1058 1059 swx_addr = tcg_temp_local_new(); 1060 if (ex) { /* swx */ 1061 TCGv tval; 1062 1063 /* Force addr into the swx_addr. */ 1064 tcg_gen_mov_tl(swx_addr, *addr); 1065 addr = &swx_addr; 1066 /* swx does not throw unaligned access errors, so force alignment */ 1067 tcg_gen_andi_tl(swx_addr, swx_addr, ~3); 1068 1069 write_carryi(dc, 1); 1070 swx_skip = gen_new_label(); 1071 tcg_gen_brcond_tl(TCG_COND_NE, env_res_addr, swx_addr, swx_skip); 1072 1073 /* Compare the value loaded at lwx with current contents of 1074 the reserved location. 1075 FIXME: This only works for system emulation where we can expect 1076 this compare and the following write to be atomic. For user 1077 emulation we need to add atomicity between threads. */ 1078 tval = tcg_temp_new(); 1079 tcg_gen_qemu_ld_tl(tval, swx_addr, cpu_mmu_index(&dc->cpu->env, false), 1080 MO_TEUL); 1081 tcg_gen_brcond_tl(TCG_COND_NE, env_res_val, tval, swx_skip); 1082 write_carryi(dc, 0); 1083 tcg_temp_free(tval); 1084 } 1085 1086 if (rev && size != 4) { 1087 /* Endian reverse the address. t is addr. */ 1088 switch (size) { 1089 case 1: 1090 { 1091 /* 00 -> 11 1092 01 -> 10 1093 10 -> 10 1094 11 -> 00 */ 1095 TCGv low = tcg_temp_new(); 1096 1097 /* Force addr into the temp. */ 1098 if (addr != &t) { 1099 t = tcg_temp_new(); 1100 tcg_gen_mov_tl(t, *addr); 1101 addr = &t; 1102 } 1103 1104 tcg_gen_andi_tl(low, t, 3); 1105 tcg_gen_sub_tl(low, tcg_const_tl(3), low); 1106 tcg_gen_andi_tl(t, t, ~3); 1107 tcg_gen_or_tl(t, t, low); 1108 tcg_temp_free(low); 1109 break; 1110 } 1111 1112 case 2: 1113 /* 00 -> 10 1114 10 -> 00. */ 1115 /* Force addr into the temp. */ 1116 if (addr != &t) { 1117 t = tcg_temp_new(); 1118 tcg_gen_xori_tl(t, *addr, 2); 1119 addr = &t; 1120 } else { 1121 tcg_gen_xori_tl(t, t, 2); 1122 } 1123 break; 1124 default: 1125 cpu_abort(CPU(dc->cpu), "Invalid reverse size\n"); 1126 break; 1127 } 1128 } 1129 tcg_gen_qemu_st_tl(cpu_R[dc->rd], *addr, cpu_mmu_index(&dc->cpu->env, false), mop); 1130 1131 /* Verify alignment if needed. */ 1132 if ((dc->cpu->env.pvr.regs[2] & PVR2_UNALIGNED_EXC_MASK) && size > 1) { 1133 tcg_gen_movi_tl(cpu_SR[SR_PC], dc->pc); 1134 /* FIXME: if the alignment is wrong, we should restore the value 1135 * in memory. One possible way to achieve this is to probe 1136 * the MMU prior to the memaccess, thay way we could put 1137 * the alignment checks in between the probe and the mem 1138 * access. 1139 */ 1140 gen_helper_memalign(cpu_env, *addr, tcg_const_tl(dc->rd), 1141 tcg_const_tl(1), tcg_const_tl(size - 1)); 1142 } 1143 1144 if (ex) { 1145 gen_set_label(swx_skip); 1146 } 1147 tcg_temp_free(swx_addr); 1148 1149 if (addr == &t) 1150 tcg_temp_free(t); 1151 } 1152 1153 static inline void eval_cc(DisasContext *dc, unsigned int cc, 1154 TCGv d, TCGv a, TCGv b) 1155 { 1156 switch (cc) { 1157 case CC_EQ: 1158 tcg_gen_setcond_tl(TCG_COND_EQ, d, a, b); 1159 break; 1160 case CC_NE: 1161 tcg_gen_setcond_tl(TCG_COND_NE, d, a, b); 1162 break; 1163 case CC_LT: 1164 tcg_gen_setcond_tl(TCG_COND_LT, d, a, b); 1165 break; 1166 case CC_LE: 1167 tcg_gen_setcond_tl(TCG_COND_LE, d, a, b); 1168 break; 1169 case CC_GE: 1170 tcg_gen_setcond_tl(TCG_COND_GE, d, a, b); 1171 break; 1172 case CC_GT: 1173 tcg_gen_setcond_tl(TCG_COND_GT, d, a, b); 1174 break; 1175 default: 1176 cpu_abort(CPU(dc->cpu), "Unknown condition code %x.\n", cc); 1177 break; 1178 } 1179 } 1180 1181 static void eval_cond_jmp(DisasContext *dc, TCGv pc_true, TCGv pc_false) 1182 { 1183 TCGLabel *l1 = gen_new_label(); 1184 /* Conditional jmp. */ 1185 tcg_gen_mov_tl(cpu_SR[SR_PC], pc_false); 1186 tcg_gen_brcondi_tl(TCG_COND_EQ, env_btaken, 0, l1); 1187 tcg_gen_mov_tl(cpu_SR[SR_PC], pc_true); 1188 gen_set_label(l1); 1189 } 1190 1191 static void dec_bcc(DisasContext *dc) 1192 { 1193 unsigned int cc; 1194 unsigned int dslot; 1195 1196 cc = EXTRACT_FIELD(dc->ir, 21, 23); 1197 dslot = dc->ir & (1 << 25); 1198 LOG_DIS("bcc%s r%d %x\n", dslot ? "d" : "", dc->ra, dc->imm); 1199 1200 dc->delayed_branch = 1; 1201 if (dslot) { 1202 dc->delayed_branch = 2; 1203 dc->tb_flags |= D_FLAG; 1204 tcg_gen_st_tl(tcg_const_tl(dc->type_b && (dc->tb_flags & IMM_FLAG)), 1205 cpu_env, offsetof(CPUMBState, bimm)); 1206 } 1207 1208 if (dec_alu_op_b_is_small_imm(dc)) { 1209 int32_t offset = (int32_t)((int16_t)dc->imm); /* sign-extend. */ 1210 1211 tcg_gen_movi_tl(env_btarget, dc->pc + offset); 1212 dc->jmp = JMP_DIRECT_CC; 1213 dc->jmp_pc = dc->pc + offset; 1214 } else { 1215 dc->jmp = JMP_INDIRECT; 1216 tcg_gen_movi_tl(env_btarget, dc->pc); 1217 tcg_gen_add_tl(env_btarget, env_btarget, *(dec_alu_op_b(dc))); 1218 } 1219 eval_cc(dc, cc, env_btaken, cpu_R[dc->ra], tcg_const_tl(0)); 1220 } 1221 1222 static void dec_br(DisasContext *dc) 1223 { 1224 unsigned int dslot, link, abs, mbar; 1225 int mem_index = cpu_mmu_index(&dc->cpu->env, false); 1226 1227 dslot = dc->ir & (1 << 20); 1228 abs = dc->ir & (1 << 19); 1229 link = dc->ir & (1 << 18); 1230 1231 /* Memory barrier. */ 1232 mbar = (dc->ir >> 16) & 31; 1233 if (mbar == 2 && dc->imm == 4) { 1234 /* mbar IMM & 16 decodes to sleep. */ 1235 if (dc->rd & 16) { 1236 TCGv_i32 tmp_hlt = tcg_const_i32(EXCP_HLT); 1237 TCGv_i32 tmp_1 = tcg_const_i32(1); 1238 1239 LOG_DIS("sleep\n"); 1240 1241 t_sync_flags(dc); 1242 tcg_gen_st_i32(tmp_1, cpu_env, 1243 -offsetof(MicroBlazeCPU, env) 1244 +offsetof(CPUState, halted)); 1245 tcg_gen_movi_tl(cpu_SR[SR_PC], dc->pc + 4); 1246 gen_helper_raise_exception(cpu_env, tmp_hlt); 1247 tcg_temp_free_i32(tmp_hlt); 1248 tcg_temp_free_i32(tmp_1); 1249 return; 1250 } 1251 LOG_DIS("mbar %d\n", dc->rd); 1252 /* Break the TB. */ 1253 dc->cpustate_changed = 1; 1254 return; 1255 } 1256 1257 LOG_DIS("br%s%s%s%s imm=%x\n", 1258 abs ? "a" : "", link ? "l" : "", 1259 dc->type_b ? "i" : "", dslot ? "d" : "", 1260 dc->imm); 1261 1262 dc->delayed_branch = 1; 1263 if (dslot) { 1264 dc->delayed_branch = 2; 1265 dc->tb_flags |= D_FLAG; 1266 tcg_gen_st_tl(tcg_const_tl(dc->type_b && (dc->tb_flags & IMM_FLAG)), 1267 cpu_env, offsetof(CPUMBState, bimm)); 1268 } 1269 if (link && dc->rd) 1270 tcg_gen_movi_tl(cpu_R[dc->rd], dc->pc); 1271 1272 dc->jmp = JMP_INDIRECT; 1273 if (abs) { 1274 tcg_gen_movi_tl(env_btaken, 1); 1275 tcg_gen_mov_tl(env_btarget, *(dec_alu_op_b(dc))); 1276 if (link && !dslot) { 1277 if (!(dc->tb_flags & IMM_FLAG) && (dc->imm == 8 || dc->imm == 0x18)) 1278 t_gen_raise_exception(dc, EXCP_BREAK); 1279 if (dc->imm == 0) { 1280 if ((dc->tb_flags & MSR_EE_FLAG) && mem_index == MMU_USER_IDX) { 1281 tcg_gen_movi_tl(cpu_SR[SR_ESR], ESR_EC_PRIVINSN); 1282 t_gen_raise_exception(dc, EXCP_HW_EXCP); 1283 return; 1284 } 1285 1286 t_gen_raise_exception(dc, EXCP_DEBUG); 1287 } 1288 } 1289 } else { 1290 if (dec_alu_op_b_is_small_imm(dc)) { 1291 dc->jmp = JMP_DIRECT; 1292 dc->jmp_pc = dc->pc + (int32_t)((int16_t)dc->imm); 1293 } else { 1294 tcg_gen_movi_tl(env_btaken, 1); 1295 tcg_gen_movi_tl(env_btarget, dc->pc); 1296 tcg_gen_add_tl(env_btarget, env_btarget, *(dec_alu_op_b(dc))); 1297 } 1298 } 1299 } 1300 1301 static inline void do_rti(DisasContext *dc) 1302 { 1303 TCGv t0, t1; 1304 t0 = tcg_temp_new(); 1305 t1 = tcg_temp_new(); 1306 tcg_gen_shri_tl(t0, cpu_SR[SR_MSR], 1); 1307 tcg_gen_ori_tl(t1, cpu_SR[SR_MSR], MSR_IE); 1308 tcg_gen_andi_tl(t0, t0, (MSR_VM | MSR_UM)); 1309 1310 tcg_gen_andi_tl(t1, t1, ~(MSR_VM | MSR_UM)); 1311 tcg_gen_or_tl(t1, t1, t0); 1312 msr_write(dc, t1); 1313 tcg_temp_free(t1); 1314 tcg_temp_free(t0); 1315 dc->tb_flags &= ~DRTI_FLAG; 1316 } 1317 1318 static inline void do_rtb(DisasContext *dc) 1319 { 1320 TCGv t0, t1; 1321 t0 = tcg_temp_new(); 1322 t1 = tcg_temp_new(); 1323 tcg_gen_andi_tl(t1, cpu_SR[SR_MSR], ~MSR_BIP); 1324 tcg_gen_shri_tl(t0, t1, 1); 1325 tcg_gen_andi_tl(t0, t0, (MSR_VM | MSR_UM)); 1326 1327 tcg_gen_andi_tl(t1, t1, ~(MSR_VM | MSR_UM)); 1328 tcg_gen_or_tl(t1, t1, t0); 1329 msr_write(dc, t1); 1330 tcg_temp_free(t1); 1331 tcg_temp_free(t0); 1332 dc->tb_flags &= ~DRTB_FLAG; 1333 } 1334 1335 static inline void do_rte(DisasContext *dc) 1336 { 1337 TCGv t0, t1; 1338 t0 = tcg_temp_new(); 1339 t1 = tcg_temp_new(); 1340 1341 tcg_gen_ori_tl(t1, cpu_SR[SR_MSR], MSR_EE); 1342 tcg_gen_andi_tl(t1, t1, ~MSR_EIP); 1343 tcg_gen_shri_tl(t0, t1, 1); 1344 tcg_gen_andi_tl(t0, t0, (MSR_VM | MSR_UM)); 1345 1346 tcg_gen_andi_tl(t1, t1, ~(MSR_VM | MSR_UM)); 1347 tcg_gen_or_tl(t1, t1, t0); 1348 msr_write(dc, t1); 1349 tcg_temp_free(t1); 1350 tcg_temp_free(t0); 1351 dc->tb_flags &= ~DRTE_FLAG; 1352 } 1353 1354 static void dec_rts(DisasContext *dc) 1355 { 1356 unsigned int b_bit, i_bit, e_bit; 1357 int mem_index = cpu_mmu_index(&dc->cpu->env, false); 1358 1359 i_bit = dc->ir & (1 << 21); 1360 b_bit = dc->ir & (1 << 22); 1361 e_bit = dc->ir & (1 << 23); 1362 1363 dc->delayed_branch = 2; 1364 dc->tb_flags |= D_FLAG; 1365 tcg_gen_st_tl(tcg_const_tl(dc->type_b && (dc->tb_flags & IMM_FLAG)), 1366 cpu_env, offsetof(CPUMBState, bimm)); 1367 1368 if (i_bit) { 1369 LOG_DIS("rtid ir=%x\n", dc->ir); 1370 if ((dc->tb_flags & MSR_EE_FLAG) 1371 && mem_index == MMU_USER_IDX) { 1372 tcg_gen_movi_tl(cpu_SR[SR_ESR], ESR_EC_PRIVINSN); 1373 t_gen_raise_exception(dc, EXCP_HW_EXCP); 1374 } 1375 dc->tb_flags |= DRTI_FLAG; 1376 } else if (b_bit) { 1377 LOG_DIS("rtbd ir=%x\n", dc->ir); 1378 if ((dc->tb_flags & MSR_EE_FLAG) 1379 && mem_index == MMU_USER_IDX) { 1380 tcg_gen_movi_tl(cpu_SR[SR_ESR], ESR_EC_PRIVINSN); 1381 t_gen_raise_exception(dc, EXCP_HW_EXCP); 1382 } 1383 dc->tb_flags |= DRTB_FLAG; 1384 } else if (e_bit) { 1385 LOG_DIS("rted ir=%x\n", dc->ir); 1386 if ((dc->tb_flags & MSR_EE_FLAG) 1387 && mem_index == MMU_USER_IDX) { 1388 tcg_gen_movi_tl(cpu_SR[SR_ESR], ESR_EC_PRIVINSN); 1389 t_gen_raise_exception(dc, EXCP_HW_EXCP); 1390 } 1391 dc->tb_flags |= DRTE_FLAG; 1392 } else 1393 LOG_DIS("rts ir=%x\n", dc->ir); 1394 1395 dc->jmp = JMP_INDIRECT; 1396 tcg_gen_movi_tl(env_btaken, 1); 1397 tcg_gen_add_tl(env_btarget, cpu_R[dc->ra], *(dec_alu_op_b(dc))); 1398 } 1399 1400 static int dec_check_fpuv2(DisasContext *dc) 1401 { 1402 if ((dc->cpu->cfg.use_fpu != 2) && (dc->tb_flags & MSR_EE_FLAG)) { 1403 tcg_gen_movi_tl(cpu_SR[SR_ESR], ESR_EC_FPU); 1404 t_gen_raise_exception(dc, EXCP_HW_EXCP); 1405 } 1406 return (dc->cpu->cfg.use_fpu == 2) ? 0 : PVR2_USE_FPU2_MASK; 1407 } 1408 1409 static void dec_fpu(DisasContext *dc) 1410 { 1411 unsigned int fpu_insn; 1412 1413 if ((dc->tb_flags & MSR_EE_FLAG) 1414 && (dc->cpu->env.pvr.regs[2] & PVR2_ILL_OPCODE_EXC_MASK) 1415 && !dc->cpu->cfg.use_fpu) { 1416 tcg_gen_movi_tl(cpu_SR[SR_ESR], ESR_EC_ILLEGAL_OP); 1417 t_gen_raise_exception(dc, EXCP_HW_EXCP); 1418 return; 1419 } 1420 1421 fpu_insn = (dc->ir >> 7) & 7; 1422 1423 switch (fpu_insn) { 1424 case 0: 1425 gen_helper_fadd(cpu_R[dc->rd], cpu_env, cpu_R[dc->ra], 1426 cpu_R[dc->rb]); 1427 break; 1428 1429 case 1: 1430 gen_helper_frsub(cpu_R[dc->rd], cpu_env, cpu_R[dc->ra], 1431 cpu_R[dc->rb]); 1432 break; 1433 1434 case 2: 1435 gen_helper_fmul(cpu_R[dc->rd], cpu_env, cpu_R[dc->ra], 1436 cpu_R[dc->rb]); 1437 break; 1438 1439 case 3: 1440 gen_helper_fdiv(cpu_R[dc->rd], cpu_env, cpu_R[dc->ra], 1441 cpu_R[dc->rb]); 1442 break; 1443 1444 case 4: 1445 switch ((dc->ir >> 4) & 7) { 1446 case 0: 1447 gen_helper_fcmp_un(cpu_R[dc->rd], cpu_env, 1448 cpu_R[dc->ra], cpu_R[dc->rb]); 1449 break; 1450 case 1: 1451 gen_helper_fcmp_lt(cpu_R[dc->rd], cpu_env, 1452 cpu_R[dc->ra], cpu_R[dc->rb]); 1453 break; 1454 case 2: 1455 gen_helper_fcmp_eq(cpu_R[dc->rd], cpu_env, 1456 cpu_R[dc->ra], cpu_R[dc->rb]); 1457 break; 1458 case 3: 1459 gen_helper_fcmp_le(cpu_R[dc->rd], cpu_env, 1460 cpu_R[dc->ra], cpu_R[dc->rb]); 1461 break; 1462 case 4: 1463 gen_helper_fcmp_gt(cpu_R[dc->rd], cpu_env, 1464 cpu_R[dc->ra], cpu_R[dc->rb]); 1465 break; 1466 case 5: 1467 gen_helper_fcmp_ne(cpu_R[dc->rd], cpu_env, 1468 cpu_R[dc->ra], cpu_R[dc->rb]); 1469 break; 1470 case 6: 1471 gen_helper_fcmp_ge(cpu_R[dc->rd], cpu_env, 1472 cpu_R[dc->ra], cpu_R[dc->rb]); 1473 break; 1474 default: 1475 qemu_log_mask(LOG_UNIMP, 1476 "unimplemented fcmp fpu_insn=%x pc=%x" 1477 " opc=%x\n", 1478 fpu_insn, dc->pc, dc->opcode); 1479 dc->abort_at_next_insn = 1; 1480 break; 1481 } 1482 break; 1483 1484 case 5: 1485 if (!dec_check_fpuv2(dc)) { 1486 return; 1487 } 1488 gen_helper_flt(cpu_R[dc->rd], cpu_env, cpu_R[dc->ra]); 1489 break; 1490 1491 case 6: 1492 if (!dec_check_fpuv2(dc)) { 1493 return; 1494 } 1495 gen_helper_fint(cpu_R[dc->rd], cpu_env, cpu_R[dc->ra]); 1496 break; 1497 1498 case 7: 1499 if (!dec_check_fpuv2(dc)) { 1500 return; 1501 } 1502 gen_helper_fsqrt(cpu_R[dc->rd], cpu_env, cpu_R[dc->ra]); 1503 break; 1504 1505 default: 1506 qemu_log_mask(LOG_UNIMP, "unimplemented FPU insn fpu_insn=%x pc=%x" 1507 " opc=%x\n", 1508 fpu_insn, dc->pc, dc->opcode); 1509 dc->abort_at_next_insn = 1; 1510 break; 1511 } 1512 } 1513 1514 static void dec_null(DisasContext *dc) 1515 { 1516 if ((dc->tb_flags & MSR_EE_FLAG) 1517 && (dc->cpu->env.pvr.regs[2] & PVR2_ILL_OPCODE_EXC_MASK)) { 1518 tcg_gen_movi_tl(cpu_SR[SR_ESR], ESR_EC_ILLEGAL_OP); 1519 t_gen_raise_exception(dc, EXCP_HW_EXCP); 1520 return; 1521 } 1522 qemu_log_mask(LOG_GUEST_ERROR, "unknown insn pc=%x opc=%x\n", dc->pc, dc->opcode); 1523 dc->abort_at_next_insn = 1; 1524 } 1525 1526 /* Insns connected to FSL or AXI stream attached devices. */ 1527 static void dec_stream(DisasContext *dc) 1528 { 1529 int mem_index = cpu_mmu_index(&dc->cpu->env, false); 1530 TCGv_i32 t_id, t_ctrl; 1531 int ctrl; 1532 1533 LOG_DIS("%s%s imm=%x\n", dc->rd ? "get" : "put", 1534 dc->type_b ? "" : "d", dc->imm); 1535 1536 if ((dc->tb_flags & MSR_EE_FLAG) && (mem_index == MMU_USER_IDX)) { 1537 tcg_gen_movi_tl(cpu_SR[SR_ESR], ESR_EC_PRIVINSN); 1538 t_gen_raise_exception(dc, EXCP_HW_EXCP); 1539 return; 1540 } 1541 1542 t_id = tcg_temp_new(); 1543 if (dc->type_b) { 1544 tcg_gen_movi_tl(t_id, dc->imm & 0xf); 1545 ctrl = dc->imm >> 10; 1546 } else { 1547 tcg_gen_andi_tl(t_id, cpu_R[dc->rb], 0xf); 1548 ctrl = dc->imm >> 5; 1549 } 1550 1551 t_ctrl = tcg_const_tl(ctrl); 1552 1553 if (dc->rd == 0) { 1554 gen_helper_put(t_id, t_ctrl, cpu_R[dc->ra]); 1555 } else { 1556 gen_helper_get(cpu_R[dc->rd], t_id, t_ctrl); 1557 } 1558 tcg_temp_free(t_id); 1559 tcg_temp_free(t_ctrl); 1560 } 1561 1562 static struct decoder_info { 1563 struct { 1564 uint32_t bits; 1565 uint32_t mask; 1566 }; 1567 void (*dec)(DisasContext *dc); 1568 } decinfo[] = { 1569 {DEC_ADD, dec_add}, 1570 {DEC_SUB, dec_sub}, 1571 {DEC_AND, dec_and}, 1572 {DEC_XOR, dec_xor}, 1573 {DEC_OR, dec_or}, 1574 {DEC_BIT, dec_bit}, 1575 {DEC_BARREL, dec_barrel}, 1576 {DEC_LD, dec_load}, 1577 {DEC_ST, dec_store}, 1578 {DEC_IMM, dec_imm}, 1579 {DEC_BR, dec_br}, 1580 {DEC_BCC, dec_bcc}, 1581 {DEC_RTS, dec_rts}, 1582 {DEC_FPU, dec_fpu}, 1583 {DEC_MUL, dec_mul}, 1584 {DEC_DIV, dec_div}, 1585 {DEC_MSR, dec_msr}, 1586 {DEC_STREAM, dec_stream}, 1587 {{0, 0}, dec_null} 1588 }; 1589 1590 static inline void decode(DisasContext *dc, uint32_t ir) 1591 { 1592 int i; 1593 1594 dc->ir = ir; 1595 LOG_DIS("%8.8x\t", dc->ir); 1596 1597 if (dc->ir) 1598 dc->nr_nops = 0; 1599 else { 1600 if ((dc->tb_flags & MSR_EE_FLAG) 1601 && (dc->cpu->env.pvr.regs[2] & PVR2_ILL_OPCODE_EXC_MASK) 1602 && (dc->cpu->env.pvr.regs[2] & PVR2_OPCODE_0x0_ILL_MASK)) { 1603 tcg_gen_movi_tl(cpu_SR[SR_ESR], ESR_EC_ILLEGAL_OP); 1604 t_gen_raise_exception(dc, EXCP_HW_EXCP); 1605 return; 1606 } 1607 1608 LOG_DIS("nr_nops=%d\t", dc->nr_nops); 1609 dc->nr_nops++; 1610 if (dc->nr_nops > 4) { 1611 cpu_abort(CPU(dc->cpu), "fetching nop sequence\n"); 1612 } 1613 } 1614 /* bit 2 seems to indicate insn type. */ 1615 dc->type_b = ir & (1 << 29); 1616 1617 dc->opcode = EXTRACT_FIELD(ir, 26, 31); 1618 dc->rd = EXTRACT_FIELD(ir, 21, 25); 1619 dc->ra = EXTRACT_FIELD(ir, 16, 20); 1620 dc->rb = EXTRACT_FIELD(ir, 11, 15); 1621 dc->imm = EXTRACT_FIELD(ir, 0, 15); 1622 1623 /* Large switch for all insns. */ 1624 for (i = 0; i < ARRAY_SIZE(decinfo); i++) { 1625 if ((dc->opcode & decinfo[i].mask) == decinfo[i].bits) { 1626 decinfo[i].dec(dc); 1627 break; 1628 } 1629 } 1630 } 1631 1632 /* generate intermediate code for basic block 'tb'. */ 1633 void gen_intermediate_code(CPUState *cs, struct TranslationBlock *tb) 1634 { 1635 CPUMBState *env = cs->env_ptr; 1636 MicroBlazeCPU *cpu = mb_env_get_cpu(env); 1637 uint32_t pc_start; 1638 struct DisasContext ctx; 1639 struct DisasContext *dc = &ctx; 1640 uint32_t page_start, org_flags; 1641 target_ulong npc; 1642 int num_insns; 1643 int max_insns; 1644 1645 pc_start = tb->pc; 1646 dc->cpu = cpu; 1647 dc->tb = tb; 1648 org_flags = dc->synced_flags = dc->tb_flags = tb->flags; 1649 1650 dc->is_jmp = DISAS_NEXT; 1651 dc->jmp = 0; 1652 dc->delayed_branch = !!(dc->tb_flags & D_FLAG); 1653 if (dc->delayed_branch) { 1654 dc->jmp = JMP_INDIRECT; 1655 } 1656 dc->pc = pc_start; 1657 dc->singlestep_enabled = cs->singlestep_enabled; 1658 dc->cpustate_changed = 0; 1659 dc->abort_at_next_insn = 0; 1660 dc->nr_nops = 0; 1661 1662 if (pc_start & 3) { 1663 cpu_abort(cs, "Microblaze: unaligned PC=%x\n", pc_start); 1664 } 1665 1666 page_start = pc_start & TARGET_PAGE_MASK; 1667 num_insns = 0; 1668 max_insns = tb_cflags(tb) & CF_COUNT_MASK; 1669 if (max_insns == 0) { 1670 max_insns = CF_COUNT_MASK; 1671 } 1672 if (max_insns > TCG_MAX_INSNS) { 1673 max_insns = TCG_MAX_INSNS; 1674 } 1675 1676 gen_tb_start(tb); 1677 do 1678 { 1679 tcg_gen_insn_start(dc->pc); 1680 num_insns++; 1681 1682 #if SIM_COMPAT 1683 if (qemu_loglevel_mask(CPU_LOG_TB_IN_ASM)) { 1684 tcg_gen_movi_tl(cpu_SR[SR_PC], dc->pc); 1685 gen_helper_debug(); 1686 } 1687 #endif 1688 1689 if (unlikely(cpu_breakpoint_test(cs, dc->pc, BP_ANY))) { 1690 t_gen_raise_exception(dc, EXCP_DEBUG); 1691 dc->is_jmp = DISAS_UPDATE; 1692 /* The address covered by the breakpoint must be included in 1693 [tb->pc, tb->pc + tb->size) in order to for it to be 1694 properly cleared -- thus we increment the PC here so that 1695 the logic setting tb->size below does the right thing. */ 1696 dc->pc += 4; 1697 break; 1698 } 1699 1700 /* Pretty disas. */ 1701 LOG_DIS("%8.8x:\t", dc->pc); 1702 1703 if (num_insns == max_insns && (tb_cflags(tb) & CF_LAST_IO)) { 1704 gen_io_start(); 1705 } 1706 1707 dc->clear_imm = 1; 1708 decode(dc, cpu_ldl_code(env, dc->pc)); 1709 if (dc->clear_imm) 1710 dc->tb_flags &= ~IMM_FLAG; 1711 dc->pc += 4; 1712 1713 if (dc->delayed_branch) { 1714 dc->delayed_branch--; 1715 if (!dc->delayed_branch) { 1716 if (dc->tb_flags & DRTI_FLAG) 1717 do_rti(dc); 1718 if (dc->tb_flags & DRTB_FLAG) 1719 do_rtb(dc); 1720 if (dc->tb_flags & DRTE_FLAG) 1721 do_rte(dc); 1722 /* Clear the delay slot flag. */ 1723 dc->tb_flags &= ~D_FLAG; 1724 /* If it is a direct jump, try direct chaining. */ 1725 if (dc->jmp == JMP_INDIRECT) { 1726 eval_cond_jmp(dc, env_btarget, tcg_const_tl(dc->pc)); 1727 dc->is_jmp = DISAS_JUMP; 1728 } else if (dc->jmp == JMP_DIRECT) { 1729 t_sync_flags(dc); 1730 gen_goto_tb(dc, 0, dc->jmp_pc); 1731 dc->is_jmp = DISAS_TB_JUMP; 1732 } else if (dc->jmp == JMP_DIRECT_CC) { 1733 TCGLabel *l1 = gen_new_label(); 1734 t_sync_flags(dc); 1735 /* Conditional jmp. */ 1736 tcg_gen_brcondi_tl(TCG_COND_NE, env_btaken, 0, l1); 1737 gen_goto_tb(dc, 1, dc->pc); 1738 gen_set_label(l1); 1739 gen_goto_tb(dc, 0, dc->jmp_pc); 1740 1741 dc->is_jmp = DISAS_TB_JUMP; 1742 } 1743 break; 1744 } 1745 } 1746 if (cs->singlestep_enabled) { 1747 break; 1748 } 1749 } while (!dc->is_jmp && !dc->cpustate_changed 1750 && !tcg_op_buf_full() 1751 && !singlestep 1752 && (dc->pc - page_start < TARGET_PAGE_SIZE) 1753 && num_insns < max_insns); 1754 1755 npc = dc->pc; 1756 if (dc->jmp == JMP_DIRECT || dc->jmp == JMP_DIRECT_CC) { 1757 if (dc->tb_flags & D_FLAG) { 1758 dc->is_jmp = DISAS_UPDATE; 1759 tcg_gen_movi_tl(cpu_SR[SR_PC], npc); 1760 sync_jmpstate(dc); 1761 } else 1762 npc = dc->jmp_pc; 1763 } 1764 1765 if (tb_cflags(tb) & CF_LAST_IO) 1766 gen_io_end(); 1767 /* Force an update if the per-tb cpu state has changed. */ 1768 if (dc->is_jmp == DISAS_NEXT 1769 && (dc->cpustate_changed || org_flags != dc->tb_flags)) { 1770 dc->is_jmp = DISAS_UPDATE; 1771 tcg_gen_movi_tl(cpu_SR[SR_PC], npc); 1772 } 1773 t_sync_flags(dc); 1774 1775 if (unlikely(cs->singlestep_enabled)) { 1776 TCGv_i32 tmp = tcg_const_i32(EXCP_DEBUG); 1777 1778 if (dc->is_jmp != DISAS_JUMP) { 1779 tcg_gen_movi_tl(cpu_SR[SR_PC], npc); 1780 } 1781 gen_helper_raise_exception(cpu_env, tmp); 1782 tcg_temp_free_i32(tmp); 1783 } else { 1784 switch(dc->is_jmp) { 1785 case DISAS_NEXT: 1786 gen_goto_tb(dc, 1, npc); 1787 break; 1788 default: 1789 case DISAS_JUMP: 1790 case DISAS_UPDATE: 1791 /* indicate that the hash table must be used 1792 to find the next TB */ 1793 tcg_gen_exit_tb(0); 1794 break; 1795 case DISAS_TB_JUMP: 1796 /* nothing more to generate */ 1797 break; 1798 } 1799 } 1800 gen_tb_end(tb, num_insns); 1801 1802 tb->size = dc->pc - pc_start; 1803 tb->icount = num_insns; 1804 1805 #ifdef DEBUG_DISAS 1806 #if !SIM_COMPAT 1807 if (qemu_loglevel_mask(CPU_LOG_TB_IN_ASM) 1808 && qemu_log_in_addr_range(pc_start)) { 1809 qemu_log_lock(); 1810 qemu_log("--------------\n"); 1811 log_target_disas(cs, pc_start, dc->pc - pc_start); 1812 qemu_log_unlock(); 1813 } 1814 #endif 1815 #endif 1816 assert(!dc->abort_at_next_insn); 1817 } 1818 1819 void mb_cpu_dump_state(CPUState *cs, FILE *f, fprintf_function cpu_fprintf, 1820 int flags) 1821 { 1822 MicroBlazeCPU *cpu = MICROBLAZE_CPU(cs); 1823 CPUMBState *env = &cpu->env; 1824 int i; 1825 1826 if (!env || !f) 1827 return; 1828 1829 cpu_fprintf(f, "IN: PC=%x %s\n", 1830 env->sregs[SR_PC], lookup_symbol(env->sregs[SR_PC])); 1831 cpu_fprintf(f, "rmsr=%x resr=%x rear=%x debug=%x imm=%x iflags=%x fsr=%x\n", 1832 env->sregs[SR_MSR], env->sregs[SR_ESR], env->sregs[SR_EAR], 1833 env->debug, env->imm, env->iflags, env->sregs[SR_FSR]); 1834 cpu_fprintf(f, "btaken=%d btarget=%x mode=%s(saved=%s) eip=%d ie=%d\n", 1835 env->btaken, env->btarget, 1836 (env->sregs[SR_MSR] & MSR_UM) ? "user" : "kernel", 1837 (env->sregs[SR_MSR] & MSR_UMS) ? "user" : "kernel", 1838 (env->sregs[SR_MSR] & MSR_EIP), 1839 (env->sregs[SR_MSR] & MSR_IE)); 1840 1841 for (i = 0; i < 32; i++) { 1842 cpu_fprintf(f, "r%2.2d=%8.8x ", i, env->regs[i]); 1843 if ((i + 1) % 4 == 0) 1844 cpu_fprintf(f, "\n"); 1845 } 1846 cpu_fprintf(f, "\n\n"); 1847 } 1848 1849 void mb_tcg_init(void) 1850 { 1851 int i; 1852 1853 env_debug = tcg_global_mem_new(cpu_env, 1854 offsetof(CPUMBState, debug), 1855 "debug0"); 1856 env_iflags = tcg_global_mem_new(cpu_env, 1857 offsetof(CPUMBState, iflags), 1858 "iflags"); 1859 env_imm = tcg_global_mem_new(cpu_env, 1860 offsetof(CPUMBState, imm), 1861 "imm"); 1862 env_btarget = tcg_global_mem_new(cpu_env, 1863 offsetof(CPUMBState, btarget), 1864 "btarget"); 1865 env_btaken = tcg_global_mem_new(cpu_env, 1866 offsetof(CPUMBState, btaken), 1867 "btaken"); 1868 env_res_addr = tcg_global_mem_new(cpu_env, 1869 offsetof(CPUMBState, res_addr), 1870 "res_addr"); 1871 env_res_val = tcg_global_mem_new(cpu_env, 1872 offsetof(CPUMBState, res_val), 1873 "res_val"); 1874 for (i = 0; i < ARRAY_SIZE(cpu_R); i++) { 1875 cpu_R[i] = tcg_global_mem_new(cpu_env, 1876 offsetof(CPUMBState, regs[i]), 1877 regnames[i]); 1878 } 1879 for (i = 0; i < ARRAY_SIZE(cpu_SR); i++) { 1880 cpu_SR[i] = tcg_global_mem_new(cpu_env, 1881 offsetof(CPUMBState, sregs[i]), 1882 special_regnames[i]); 1883 } 1884 } 1885 1886 void restore_state_to_opc(CPUMBState *env, TranslationBlock *tb, 1887 target_ulong *data) 1888 { 1889 env->sregs[SR_PC] = data[0]; 1890 } 1891