Commit 99c475abf16b10923baac09682a9d801ae421ac6
1 parent
dfe86665
armv5te support (Paul Brook)
git-svn-id: svn://svn.savannah.nongnu.org/qemu/trunk@1258 c046a42c-6fe2-441c-8c8c-71466251a162
Showing
9 changed files
with
909 additions
and
32 deletions
Changelog
cpu-exec.c
| ... | ... | @@ -160,7 +160,8 @@ int cpu_exec(CPUState *env1) |
| 160 | 160 | env->CF = (psr >> 29) & 1; |
| 161 | 161 | env->NZF = (psr & 0xc0000000) ^ 0x40000000; |
| 162 | 162 | env->VF = (psr << 3) & 0x80000000; |
| 163 | - env->cpsr = psr & ~0xf0000000; | |
| 163 | + env->QF = (psr >> 27) & 1; | |
| 164 | + env->cpsr = psr & ~CACHED_CPSR_BITS; | |
| 164 | 165 | } |
| 165 | 166 | #elif defined(TARGET_SPARC) |
| 166 | 167 | #elif defined(TARGET_PPC) |
| ... | ... | @@ -303,7 +304,7 @@ int cpu_exec(CPUState *env1) |
| 303 | 304 | #elif defined(TARGET_ARM) |
| 304 | 305 | env->cpsr = compute_cpsr(); |
| 305 | 306 | cpu_dump_state(env, logfile, fprintf, 0); |
| 306 | - env->cpsr &= ~0xf0000000; | |
| 307 | + env->cpsr &= ~CACHED_CPSR_BITS; | |
| 307 | 308 | #elif defined(TARGET_SPARC) |
| 308 | 309 | cpu_dump_state (env, logfile, fprintf, 0); |
| 309 | 310 | #elif defined(TARGET_PPC) |
| ... | ... | @@ -322,7 +323,7 @@ int cpu_exec(CPUState *env1) |
| 322 | 323 | cs_base = env->segs[R_CS].base; |
| 323 | 324 | pc = cs_base + env->eip; |
| 324 | 325 | #elif defined(TARGET_ARM) |
| 325 | - flags = 0; | |
| 326 | + flags = env->thumb; | |
| 326 | 327 | cs_base = 0; |
| 327 | 328 | pc = env->regs[15]; |
| 328 | 329 | #elif defined(TARGET_SPARC) | ... | ... |
linux-user/arm/syscall.h
| ... | ... | @@ -30,7 +30,7 @@ struct target_pt_regs { |
| 30 | 30 | #define ARM_NR_cacheflush (ARM_SYSCALL_BASE + 0xf0000 + 2) |
| 31 | 31 | |
| 32 | 32 | #if defined(TARGET_WORDS_BIGENDIAN) |
| 33 | -#define UNAME_MACHINE "armv4b" | |
| 33 | +#define UNAME_MACHINE "armv5teb" | |
| 34 | 34 | #else |
| 35 | -#define UNAME_MACHINE "armv4l" | |
| 35 | +#define UNAME_MACHINE "armv5tel" | |
| 36 | 36 | #endif | ... | ... |
linux-user/syscall.c
| ... | ... | @@ -2944,11 +2944,35 @@ long do_syscall(void *cpu_env, int num, long arg1, long arg2, long arg3, |
| 2944 | 2944 | #endif |
| 2945 | 2945 | #ifdef TARGET_NR_getgroups32 |
| 2946 | 2946 | case TARGET_NR_getgroups32: |
| 2947 | - goto unimplemented; | |
| 2947 | + { | |
| 2948 | + int gidsetsize = arg1; | |
| 2949 | + uint32_t *target_grouplist = (void *)arg2; | |
| 2950 | + gid_t *grouplist; | |
| 2951 | + int i; | |
| 2952 | + | |
| 2953 | + grouplist = alloca(gidsetsize * sizeof(gid_t)); | |
| 2954 | + ret = get_errno(getgroups(gidsetsize, grouplist)); | |
| 2955 | + if (!is_error(ret)) { | |
| 2956 | + for(i = 0;i < gidsetsize; i++) | |
| 2957 | + put_user(grouplist[i], &target_grouplist[i]); | |
| 2958 | + } | |
| 2959 | + } | |
| 2960 | + break; | |
| 2948 | 2961 | #endif |
| 2949 | 2962 | #ifdef TARGET_NR_setgroups32 |
| 2950 | 2963 | case TARGET_NR_setgroups32: |
| 2951 | - goto unimplemented; | |
| 2964 | + { | |
| 2965 | + int gidsetsize = arg1; | |
| 2966 | + uint32_t *target_grouplist = (void *)arg2; | |
| 2967 | + gid_t *grouplist; | |
| 2968 | + int i; | |
| 2969 | + | |
| 2970 | + grouplist = alloca(gidsetsize * sizeof(gid_t)); | |
| 2971 | + for(i = 0;i < gidsetsize; i++) | |
| 2972 | + get_user(grouplist[i], &target_grouplist[i]); | |
| 2973 | + ret = get_errno(setgroups(gidsetsize, grouplist)); | |
| 2974 | + } | |
| 2975 | + break; | |
| 2952 | 2976 | #endif |
| 2953 | 2977 | #ifdef TARGET_NR_fchown32 |
| 2954 | 2978 | case TARGET_NR_fchown32: | ... | ... |
target-arm/cpu.h
| ... | ... | @@ -35,6 +35,9 @@ typedef struct CPUARMState { |
| 35 | 35 | uint32_t CF; /* 0 or 1 */ |
| 36 | 36 | uint32_t VF; /* V is the bit 31. All other bits are undefined */ |
| 37 | 37 | uint32_t NZF; /* N is bit 31. Z is computed from NZF */ |
| 38 | + uint32_t QF; /* 0 or 1 */ | |
| 39 | + | |
| 40 | + int thumb; /* 0 = arm mode, 1 = thumb mode */ | |
| 38 | 41 | |
| 39 | 42 | /* exception/interrupt handling */ |
| 40 | 43 | jmp_buf jmp_env; | ... | ... |
target-arm/exec.h
| ... | ... | @@ -31,12 +31,14 @@ void cpu_lock(void); |
| 31 | 31 | void cpu_unlock(void); |
| 32 | 32 | void cpu_loop_exit(void); |
| 33 | 33 | |
| 34 | +/* Implemented CPSR bits. */ | |
| 35 | +#define CACHED_CPSR_BITS 0xf8000000 | |
| 34 | 36 | static inline int compute_cpsr(void) |
| 35 | 37 | { |
| 36 | 38 | int ZF; |
| 37 | 39 | ZF = (env->NZF == 0); |
| 38 | 40 | return env->cpsr | (env->NZF & 0x80000000) | (ZF << 30) | |
| 39 | - (env->CF << 29) | ((env->VF & 0x80000000) >> 3); | |
| 41 | + (env->CF << 29) | ((env->VF & 0x80000000) >> 3) | (env->QF << 27); | |
| 40 | 42 | } |
| 41 | 43 | |
| 42 | 44 | static inline void env_to_regs(void) | ... | ... |
target-arm/op.c
| ... | ... | @@ -81,8 +81,15 @@ |
| 81 | 81 | |
| 82 | 82 | #define REGNAME r15 |
| 83 | 83 | #define REG (env->regs[15]) |
| 84 | +#define SET_REG(x) REG = x & ~(uint32_t)1 | |
| 84 | 85 | #include "op_template.h" |
| 85 | 86 | |
| 87 | +void OPPROTO op_bx_T0(void) | |
| 88 | +{ | |
| 89 | + env->regs[15] = T0 & ~(uint32_t)1; | |
| 90 | + env->thumb = (T0 & 1) != 0; | |
| 91 | +} | |
| 92 | + | |
| 86 | 93 | void OPPROTO op_movl_T0_0(void) |
| 87 | 94 | { |
| 88 | 95 | T0 = 0; |
| ... | ... | @@ -382,6 +389,14 @@ void OPPROTO op_imull_T0_T1(void) |
| 382 | 389 | T0 = res; |
| 383 | 390 | } |
| 384 | 391 | |
| 392 | +/* 48 bit signed mul, top 32 bits */ | |
| 393 | +void OPPROTO op_imulw_T0_T1(void) | |
| 394 | +{ | |
| 395 | + uint64_t res; | |
| 396 | + res = (int64_t)((int32_t)T0) * (int64_t)((int32_t)T1); | |
| 397 | + T0 = res >> 16; | |
| 398 | +} | |
| 399 | + | |
| 385 | 400 | void OPPROTO op_addq_T0_T1(void) |
| 386 | 401 | { |
| 387 | 402 | uint64_t res; |
| ... | ... | @@ -391,6 +406,15 @@ void OPPROTO op_addq_T0_T1(void) |
| 391 | 406 | T0 = res; |
| 392 | 407 | } |
| 393 | 408 | |
| 409 | +void OPPROTO op_addq_lo_T0_T1(void) | |
| 410 | +{ | |
| 411 | + uint64_t res; | |
| 412 | + res = ((uint64_t)T1 << 32) | T0; | |
| 413 | + res += (uint64_t)(env->regs[PARAM1]); | |
| 414 | + T1 = res >> 32; | |
| 415 | + T0 = res; | |
| 416 | +} | |
| 417 | + | |
| 394 | 418 | void OPPROTO op_logicq_cc(void) |
| 395 | 419 | { |
| 396 | 420 | env->NZF = (T1 & 0x80000000) | ((T0 | T1) != 0); |
| ... | ... | @@ -694,6 +718,126 @@ void OPPROTO op_rorl_T1_T0_cc(void) |
| 694 | 718 | FORCE_RET(); |
| 695 | 719 | } |
| 696 | 720 | |
| 721 | +/* misc */ | |
| 722 | +void OPPROTO op_clz_T0(void) | |
| 723 | +{ | |
| 724 | + int count; | |
| 725 | + for (count = 32; T0 > 0; count--) | |
| 726 | + T0 = T0 >> 1; | |
| 727 | + T0 = count; | |
| 728 | + FORCE_RET(); | |
| 729 | +} | |
| 730 | + | |
| 731 | +void OPPROTO op_sarl_T0_im(void) | |
| 732 | +{ | |
| 733 | + T0 = (int32_t)T0 >> PARAM1; | |
| 734 | +} | |
| 735 | + | |
| 736 | +/* 16->32 Sign extend */ | |
| 737 | +void OPPROTO op_sxl_T0(void) | |
| 738 | +{ | |
| 739 | + T0 = (int16_t)T0; | |
| 740 | +} | |
| 741 | + | |
| 742 | +void OPPROTO op_sxl_T1(void) | |
| 743 | +{ | |
| 744 | + T1 = (int16_t)T1; | |
| 745 | +} | |
| 746 | + | |
| 747 | +#define SIGNBIT (uint32_t)0x80000000 | |
| 748 | +/* saturating arithmetic */ | |
| 749 | +void OPPROTO op_addl_T0_T1_setq(void) | |
| 750 | +{ | |
| 751 | + uint32_t res; | |
| 752 | + | |
| 753 | + res = T0 + T1; | |
| 754 | + if (((res ^ T0) & SIGNBIT) && !((T0 ^ T1) & SIGNBIT)) | |
| 755 | + env->QF = 1; | |
| 756 | + | |
| 757 | + T0 = res; | |
| 758 | + FORCE_RET(); | |
| 759 | +} | |
| 760 | + | |
| 761 | +void OPPROTO op_addl_T0_T1_saturate(void) | |
| 762 | +{ | |
| 763 | + uint32_t res; | |
| 764 | + | |
| 765 | + res = T0 + T1; | |
| 766 | + if (((res ^ T0) & SIGNBIT) && !((T0 ^ T1) & SIGNBIT)) { | |
| 767 | + env->QF = 1; | |
| 768 | + if (T0 & SIGNBIT) | |
| 769 | + T0 = 0x80000000; | |
| 770 | + else | |
| 771 | + T0 = 0x7fffffff; | |
| 772 | + } | |
| 773 | + else | |
| 774 | + T0 = res; | |
| 775 | + | |
| 776 | + FORCE_RET(); | |
| 777 | +} | |
| 778 | + | |
| 779 | +void OPPROTO op_subl_T0_T1_saturate(void) | |
| 780 | +{ | |
| 781 | + uint32_t res; | |
| 782 | + | |
| 783 | + res = T0 - T1; | |
| 784 | + if (((res ^ T0) & SIGNBIT) && ((T0 ^ T1) & SIGNBIT)) { | |
| 785 | + env->QF = 1; | |
| 786 | + if (T0 & SIGNBIT) | |
| 787 | + T0 = 0x8000000; | |
| 788 | + else | |
| 789 | + T0 = 0x7fffffff; | |
| 790 | + } | |
| 791 | + else | |
| 792 | + T0 = res; | |
| 793 | + | |
| 794 | + FORCE_RET(); | |
| 795 | +} | |
| 796 | + | |
| 797 | +/* thumb shift by immediate */ | |
| 798 | +void OPPROTO op_shll_T0_im_thumb(void) | |
| 799 | +{ | |
| 800 | + int shift; | |
| 801 | + shift = PARAM1; | |
| 802 | + if (shift != 0) { | |
| 803 | + env->CF = (T1 >> (32 - shift)) & 1; | |
| 804 | + T0 = T0 << shift; | |
| 805 | + } | |
| 806 | + env->NZF = T0; | |
| 807 | + FORCE_RET(); | |
| 808 | +} | |
| 809 | + | |
| 810 | +void OPPROTO op_shrl_T0_im_thumb(void) | |
| 811 | +{ | |
| 812 | + int shift; | |
| 813 | + | |
| 814 | + shift = PARAM1; | |
| 815 | + if (shift == 0) { | |
| 816 | + env->CF = 0; | |
| 817 | + T0 = 0; | |
| 818 | + } else { | |
| 819 | + env->CF = (T0 >> (shift - 1)) & 1; | |
| 820 | + T0 = T0 >> shift; | |
| 821 | + } | |
| 822 | + FORCE_RET(); | |
| 823 | +} | |
| 824 | + | |
| 825 | +void OPPROTO op_sarl_T0_im_thumb(void) | |
| 826 | +{ | |
| 827 | + int shift; | |
| 828 | + | |
| 829 | + shift = PARAM1; | |
| 830 | + if (shift == 0) { | |
| 831 | + T0 = ((int32_t)T0) >> 31; | |
| 832 | + env->CF = T0 & 1; | |
| 833 | + } else { | |
| 834 | + env->CF = (T0 >> (shift - 1)) & 1; | |
| 835 | + T0 = ((int32_t)T0) >> shift; | |
| 836 | + } | |
| 837 | + env->NZF = T0; | |
| 838 | + FORCE_RET(); | |
| 839 | +} | |
| 840 | + | |
| 697 | 841 | /* exceptions */ |
| 698 | 842 | |
| 699 | 843 | void OPPROTO op_swi(void) | ... | ... |
target-arm/op_template.h
| ... | ... | @@ -19,6 +19,10 @@ |
| 19 | 19 | * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA |
| 20 | 20 | */ |
| 21 | 21 | |
| 22 | +#ifndef SET_REG | |
| 23 | +#define SET_REG(x) REG = x | |
| 24 | +#endif | |
| 25 | + | |
| 22 | 26 | void OPPROTO glue(op_movl_T0_, REGNAME)(void) |
| 23 | 27 | { |
| 24 | 28 | T0 = REG; |
| ... | ... | @@ -36,13 +40,14 @@ void OPPROTO glue(op_movl_T2_, REGNAME)(void) |
| 36 | 40 | |
| 37 | 41 | void OPPROTO glue(glue(op_movl_, REGNAME), _T0)(void) |
| 38 | 42 | { |
| 39 | - REG = T0; | |
| 43 | + SET_REG (T0); | |
| 40 | 44 | } |
| 41 | 45 | |
| 42 | 46 | void OPPROTO glue(glue(op_movl_, REGNAME), _T1)(void) |
| 43 | 47 | { |
| 44 | - REG = T1; | |
| 48 | + SET_REG (T1); | |
| 45 | 49 | } |
| 46 | 50 | |
| 47 | 51 | #undef REG |
| 48 | 52 | #undef REGNAME |
| 53 | +#undef SET_REG | ... | ... |
target-arm/translate.c
| ... | ... | @@ -245,6 +245,18 @@ static GenOpFunc1 *gen_op_movl_TN_im[3] = { |
| 245 | 245 | gen_op_movl_T2_im, |
| 246 | 246 | }; |
| 247 | 247 | |
| 248 | +static GenOpFunc1 *gen_shift_T0_im_thumb[3] = { | |
| 249 | + gen_op_shll_T0_im_thumb, | |
| 250 | + gen_op_shrl_T0_im_thumb, | |
| 251 | + gen_op_sarl_T0_im_thumb, | |
| 252 | +}; | |
| 253 | + | |
| 254 | +static inline void gen_bx(DisasContext *s) | |
| 255 | +{ | |
| 256 | + s->is_jmp = DISAS_UPDATE; | |
| 257 | + gen_op_bx_T0(); | |
| 258 | +} | |
| 259 | + | |
| 248 | 260 | static inline void gen_movl_TN_reg(DisasContext *s, int reg, int t) |
| 249 | 261 | { |
| 250 | 262 | int val; |
| ... | ... | @@ -350,17 +362,166 @@ static void disas_arm_insn(DisasContext *s) |
| 350 | 362 | s->pc += 4; |
| 351 | 363 | |
| 352 | 364 | cond = insn >> 28; |
| 353 | - if (cond == 0xf) | |
| 365 | + if (cond == 0xf){ | |
| 366 | + if ((insn & 0x0d70f000) == 0x0550f000) | |
| 367 | + return; /* PLD */ | |
| 368 | + else if ((insn & 0x0e000000) == 0x0a000000) { | |
| 369 | + /* branch link and change to thumb (blx <offset>) */ | |
| 370 | + int32_t offset; | |
| 371 | + | |
| 372 | + val = (uint32_t)s->pc; | |
| 373 | + gen_op_movl_T0_im(val); | |
| 374 | + gen_movl_reg_T0(s, 14); | |
| 375 | + /* Sign-extend the 24-bit offset */ | |
| 376 | + offset = (((int32_t)insn) << 8) >> 8; | |
| 377 | + /* offset * 4 + bit24 * 2 + (thumb bit) */ | |
| 378 | + val += (offset << 2) | ((insn >> 23) & 2) | 1; | |
| 379 | + /* pipeline offset */ | |
| 380 | + val += 4; | |
| 381 | + gen_op_movl_T0_im(val); | |
| 382 | + gen_bx(s); | |
| 383 | + return; | |
| 384 | + } | |
| 354 | 385 | goto illegal_op; |
| 386 | + } | |
| 355 | 387 | if (cond != 0xe) { |
| 356 | 388 | /* if not always execute, we generate a conditional jump to |
| 357 | 389 | next instruction */ |
| 358 | 390 | gen_test_cc[cond ^ 1]((long)s->tb, (long)s->pc); |
| 359 | 391 | s->is_jmp = DISAS_JUMP_NEXT; |
| 360 | 392 | } |
| 361 | - if (((insn & 0x0e000000) == 0 && | |
| 362 | - (insn & 0x00000090) != 0x90) || | |
| 363 | - ((insn & 0x0e000000) == (1 << 25))) { | |
| 393 | + if ((insn & 0x0f900000) == 0x03000000) { | |
| 394 | + if ((insn & 0x0ff0f000) != 0x0360f000) | |
| 395 | + goto illegal_op; | |
| 396 | + /* CPSR = immediate */ | |
| 397 | + val = insn & 0xff; | |
| 398 | + shift = ((insn >> 8) & 0xf) * 2; | |
| 399 | + if (shift) | |
| 400 | + val = (val >> shift) | (val << (32 - shift)); | |
| 401 | + gen_op_movl_T0_im(val); | |
| 402 | + if (insn & (1 << 19)) | |
| 403 | + gen_op_movl_psr_T0(); | |
| 404 | + } else if ((insn & 0x0f900000) == 0x01000000 | |
| 405 | + && (insn & 0x00000090) != 0x00000090) { | |
| 406 | + /* miscellaneous instructions */ | |
| 407 | + op1 = (insn >> 21) & 3; | |
| 408 | + sh = (insn >> 4) & 0xf; | |
| 409 | + rm = insn & 0xf; | |
| 410 | + switch (sh) { | |
| 411 | + case 0x0: /* move program status register */ | |
| 412 | + if (op1 & 2) { | |
| 413 | + /* SPSR not accessible in user mode */ | |
| 414 | + goto illegal_op; | |
| 415 | + } | |
| 416 | + if (op1 & 1) { | |
| 417 | + /* CPSR = reg */ | |
| 418 | + gen_movl_T0_reg(s, rm); | |
| 419 | + if (insn & (1 << 19)) | |
| 420 | + gen_op_movl_psr_T0(); | |
| 421 | + } else { | |
| 422 | + /* reg = CPSR */ | |
| 423 | + rd = (insn >> 12) & 0xf; | |
| 424 | + gen_op_movl_T0_psr(); | |
| 425 | + gen_movl_reg_T0(s, rd); | |
| 426 | + } | |
| 427 | + case 0x1: | |
| 428 | + if (op1 == 1) { | |
| 429 | + /* branch/exchange thumb (bx). */ | |
| 430 | + gen_movl_T0_reg(s, rm); | |
| 431 | + gen_bx(s); | |
| 432 | + } else if (op1 == 3) { | |
| 433 | + /* clz */ | |
| 434 | + rd = (insn >> 12) & 0xf; | |
| 435 | + gen_movl_T0_reg(s, rm); | |
| 436 | + gen_op_clz_T0(); | |
| 437 | + gen_movl_reg_T0(s, rd); | |
| 438 | + } else { | |
| 439 | + goto illegal_op; | |
| 440 | + } | |
| 441 | + break; | |
| 442 | + case 0x3: | |
| 443 | + if (op1 != 1) | |
| 444 | + goto illegal_op; | |
| 445 | + | |
| 446 | + /* branch link/exchange thumb (blx) */ | |
| 447 | + val = (uint32_t)s->pc; | |
| 448 | + gen_op_movl_T0_im(val); | |
| 449 | + gen_movl_reg_T0(s, 14); | |
| 450 | + gen_movl_T0_reg(s, rm); | |
| 451 | + gen_bx(s); | |
| 452 | + break; | |
| 453 | + case 0x5: /* saturating add/subtract */ | |
| 454 | + rd = (insn >> 12) & 0xf; | |
| 455 | + rn = (insn >> 16) & 0xf; | |
| 456 | + gen_movl_T0_reg(s, rn); | |
| 457 | + if (op1 & 2) { | |
| 458 | + gen_movl_T1_reg(s, rn); | |
| 459 | + if (op1 & 1) | |
| 460 | + gen_op_subl_T0_T1_saturate(); | |
| 461 | + else | |
| 462 | + gen_op_addl_T0_T1_saturate(); | |
| 463 | + } | |
| 464 | + gen_movl_T1_reg(s, rm); | |
| 465 | + if (op1 & 1) | |
| 466 | + gen_op_subl_T0_T1_saturate(); | |
| 467 | + else | |
| 468 | + gen_op_addl_T0_T1_saturate(); | |
| 469 | + gen_movl_reg_T0(s, rn); | |
| 470 | + break; | |
| 471 | + case 0x8: /* signed multiply */ | |
| 472 | + case 0xa: | |
| 473 | + case 0xc: | |
| 474 | + case 0xe: | |
| 475 | + rs = (insn >> 8) & 0xf; | |
| 476 | + rn = (insn >> 12) & 0xf; | |
| 477 | + rd = (insn >> 16) & 0xf; | |
| 478 | + if (op1 == 1) { | |
| 479 | + /* (32 * 16) >> 16 */ | |
| 480 | + gen_movl_T0_reg(s, rm); | |
| 481 | + gen_movl_T1_reg(s, rs); | |
| 482 | + if (sh & 4) | |
| 483 | + gen_op_sarl_T1_im(16); | |
| 484 | + else | |
| 485 | + gen_op_sxl_T1(); | |
| 486 | + gen_op_imulw_T0_T1(); | |
| 487 | + if ((sh & 2) == 0) { | |
| 488 | + gen_movl_T1_reg(s, rn); | |
| 489 | + gen_op_addl_T0_T1_setq(); | |
| 490 | + } | |
| 491 | + gen_movl_reg_T0(s, rd); | |
| 492 | + } else { | |
| 493 | + /* 16 * 16 */ | |
| 494 | + gen_movl_T0_reg(s, rm); | |
| 495 | + if (sh & 2) | |
| 496 | + gen_op_sarl_T0_im(16); | |
| 497 | + else | |
| 498 | + gen_op_sxl_T0(); | |
| 499 | + gen_movl_T1_reg(s, rs); | |
| 500 | + if (sh & 4) | |
| 501 | + gen_op_sarl_T1_im(16); | |
| 502 | + else | |
| 503 | + gen_op_sxl_T1(); | |
| 504 | + if (op1 == 2) { | |
| 505 | + gen_op_imull_T0_T1(); | |
| 506 | + gen_op_addq_T0_T1(rn, rd); | |
| 507 | + gen_movl_reg_T0(s, rn); | |
| 508 | + gen_movl_reg_T1(s, rd); | |
| 509 | + } else { | |
| 510 | + gen_op_mul_T0_T1(); | |
| 511 | + if (op1 == 0) { | |
| 512 | + gen_movl_T1_reg(s, rn); | |
| 513 | + gen_op_addl_T0_T1_setq(); | |
| 514 | + } | |
| 515 | + gen_movl_reg_T0(s, rd); | |
| 516 | + } | |
| 517 | + } | |
| 518 | + break; | |
| 519 | + default: | |
| 520 | + goto illegal_op; | |
| 521 | + } | |
| 522 | + } else if (((insn & 0x0e000000) == 0 && | |
| 523 | + (insn & 0x00000090) != 0x90) || | |
| 524 | + ((insn & 0x0e000000) == (1 << 25))) { | |
| 364 | 525 | int set_cc, logic_cc, shiftop; |
| 365 | 526 | |
| 366 | 527 | op1 = (insn >> 21) & 0xf; |
| ... | ... | @@ -519,6 +680,7 @@ static void disas_arm_insn(DisasContext *s) |
| 519 | 680 | switch(op1) { |
| 520 | 681 | case 0x0: |
| 521 | 682 | case 0x1: |
| 683 | + /* multiplies, extra load/stores */ | |
| 522 | 684 | sh = (insn >> 5) & 3; |
| 523 | 685 | if (sh == 0) { |
| 524 | 686 | if (op1 == 0x0) { |
| ... | ... | @@ -526,7 +688,7 @@ static void disas_arm_insn(DisasContext *s) |
| 526 | 688 | rn = (insn >> 12) & 0xf; |
| 527 | 689 | rs = (insn >> 8) & 0xf; |
| 528 | 690 | rm = (insn) & 0xf; |
| 529 | - if (!(insn & (1 << 23))) { | |
| 691 | + if (((insn >> 22) & 3) == 0) { | |
| 530 | 692 | /* 32 bit mul */ |
| 531 | 693 | gen_movl_T0_reg(s, rs); |
| 532 | 694 | gen_movl_T1_reg(s, rm); |
| ... | ... | @@ -546,30 +708,39 @@ static void disas_arm_insn(DisasContext *s) |
| 546 | 708 | gen_op_imull_T0_T1(); |
| 547 | 709 | else |
| 548 | 710 | gen_op_mull_T0_T1(); |
| 549 | - if (insn & (1 << 21)) | |
| 711 | + if (insn & (1 << 21)) /* mult accumulate */ | |
| 550 | 712 | gen_op_addq_T0_T1(rn, rd); |
| 713 | + if (!(insn & (1 << 23))) { /* double accumulate */ | |
| 714 | + gen_op_addq_lo_T0_T1(rn); | |
| 715 | + gen_op_addq_lo_T0_T1(rd); | |
| 716 | + } | |
| 551 | 717 | if (insn & (1 << 20)) |
| 552 | 718 | gen_op_logicq_cc(); |
| 553 | 719 | gen_movl_reg_T0(s, rn); |
| 554 | 720 | gen_movl_reg_T1(s, rd); |
| 555 | 721 | } |
| 556 | 722 | } else { |
| 557 | - /* SWP instruction */ | |
| 558 | 723 | rn = (insn >> 16) & 0xf; |
| 559 | 724 | rd = (insn >> 12) & 0xf; |
| 560 | - rm = (insn) & 0xf; | |
| 561 | - | |
| 562 | - gen_movl_T0_reg(s, rm); | |
| 563 | - gen_movl_T1_reg(s, rn); | |
| 564 | - if (insn & (1 << 22)) { | |
| 565 | - gen_op_swpb_T0_T1(); | |
| 725 | + if (insn & (1 << 23)) { | |
| 726 | + /* load/store exclusive */ | |
| 727 | + goto illegal_op; | |
| 566 | 728 | } else { |
| 567 | - gen_op_swpl_T0_T1(); | |
| 729 | + /* SWP instruction */ | |
| 730 | + rm = (insn) & 0xf; | |
| 731 | + | |
| 732 | + gen_movl_T0_reg(s, rm); | |
| 733 | + gen_movl_T1_reg(s, rn); | |
| 734 | + if (insn & (1 << 22)) { | |
| 735 | + gen_op_swpb_T0_T1(); | |
| 736 | + } else { | |
| 737 | + gen_op_swpl_T0_T1(); | |
| 738 | + } | |
| 739 | + gen_movl_reg_T0(s, rd); | |
| 568 | 740 | } |
| 569 | - gen_movl_reg_T0(s, rd); | |
| 570 | 741 | } |
| 571 | 742 | } else { |
| 572 | - /* load/store half word */ | |
| 743 | + /* Misc load/store */ | |
| 573 | 744 | rn = (insn >> 16) & 0xf; |
| 574 | 745 | rd = (insn >> 12) & 0xf; |
| 575 | 746 | gen_movl_T1_reg(s, rn); |
| ... | ... | @@ -590,6 +761,27 @@ static void disas_arm_insn(DisasContext *s) |
| 590 | 761 | break; |
| 591 | 762 | } |
| 592 | 763 | gen_movl_reg_T0(s, rd); |
| 764 | + } else if (sh & 2) { | |
| 765 | + /* doubleword */ | |
| 766 | + if (sh & 1) { | |
| 767 | + /* store */ | |
| 768 | + gen_movl_T0_reg(s, rd); | |
| 769 | + gen_op_stl_T0_T1(); | |
| 770 | + gen_op_addl_T1_im(4); | |
| 771 | + gen_movl_T0_reg(s, rd + 1); | |
| 772 | + gen_op_stl_T0_T1(); | |
| 773 | + if ((insn & (1 << 24)) || (insn & (1 << 20))) | |
| 774 | + gen_op_addl_T1_im(-4); | |
| 775 | + } else { | |
| 776 | + /* load */ | |
| 777 | + gen_op_ldl_T0_T1(); | |
| 778 | + gen_movl_reg_T0(s, rd); | |
| 779 | + gen_op_addl_T1_im(4); | |
| 780 | + gen_op_ldl_T0_T1(); | |
| 781 | + gen_movl_reg_T0(s, rd + 1); | |
| 782 | + if ((insn & (1 << 24)) || (insn & (1 << 20))) | |
| 783 | + gen_op_addl_T1_im(-4); | |
| 784 | + } | |
| 593 | 785 | } else { |
| 594 | 786 | /* store */ |
| 595 | 787 | gen_movl_T0_reg(s, rd); |
| ... | ... | @@ -619,7 +811,10 @@ static void disas_arm_insn(DisasContext *s) |
| 619 | 811 | gen_op_ldub_T0_T1(); |
| 620 | 812 | else |
| 621 | 813 | gen_op_ldl_T0_T1(); |
| 622 | - gen_movl_reg_T0(s, rd); | |
| 814 | + if (rd == 15) | |
| 815 | + gen_bx(s); | |
| 816 | + else | |
| 817 | + gen_movl_reg_T0(s, rd); | |
| 623 | 818 | } else { |
| 624 | 819 | /* store */ |
| 625 | 820 | gen_movl_T0_reg(s, rd); |
| ... | ... | @@ -676,7 +871,10 @@ static void disas_arm_insn(DisasContext *s) |
| 676 | 871 | if (insn & (1 << 20)) { |
| 677 | 872 | /* load */ |
| 678 | 873 | gen_op_ldl_T0_T1(); |
| 679 | - gen_movl_reg_T0(s, i); | |
| 874 | + if (i == 15) | |
| 875 | + gen_bx(s); | |
| 876 | + else | |
| 877 | + gen_movl_reg_T0(s, i); | |
| 680 | 878 | } else { |
| 681 | 879 | /* store */ |
| 682 | 880 | if (i == 15) { |
| ... | ... | @@ -720,15 +918,15 @@ static void disas_arm_insn(DisasContext *s) |
| 720 | 918 | case 0xa: |
| 721 | 919 | case 0xb: |
| 722 | 920 | { |
| 723 | - int offset; | |
| 921 | + int32_t offset; | |
| 724 | 922 | |
| 725 | 923 | /* branch (and link) */ |
| 726 | - val = (int)s->pc; | |
| 924 | + val = (int32_t)s->pc; | |
| 727 | 925 | if (insn & (1 << 24)) { |
| 728 | 926 | gen_op_movl_T0_im(val); |
| 729 | 927 | gen_op_movl_reg_TN[0][14](); |
| 730 | 928 | } |
| 731 | - offset = (((int)insn << 8) >> 8); | |
| 929 | + offset = (((int32_t)insn << 8) >> 8); | |
| 732 | 930 | val += (offset << 2) + 4; |
| 733 | 931 | gen_op_jmp((long)s->tb, val); |
| 734 | 932 | s->is_jmp = DISAS_TB_JUMP; |
| ... | ... | @@ -752,6 +950,500 @@ static void disas_arm_insn(DisasContext *s) |
| 752 | 950 | } |
| 753 | 951 | } |
| 754 | 952 | |
| 953 | +static void disas_thumb_insn(DisasContext *s) | |
| 954 | +{ | |
| 955 | + uint32_t val, insn, op, rm, rn, rd, shift, cond; | |
| 956 | + int32_t offset; | |
| 957 | + int i; | |
| 958 | + | |
| 959 | + insn = lduw(s->pc); | |
| 960 | + s->pc += 2; | |
| 961 | + | |
| 962 | + switch (insn >> 12) { | |
| 963 | + case 0: case 1: | |
| 964 | + rd = insn & 7; | |
| 965 | + op = (insn >> 11) & 3; | |
| 966 | + if (op == 3) { | |
| 967 | + /* add/subtract */ | |
| 968 | + rn = (insn >> 3) & 7; | |
| 969 | + gen_movl_T0_reg(s, rn); | |
| 970 | + if (insn & (1 << 10)) { | |
| 971 | + /* immediate */ | |
| 972 | + gen_op_movl_T1_im((insn >> 6) & 7); | |
| 973 | + } else { | |
| 974 | + /* reg */ | |
| 975 | + rm = (insn >> 6) & 7; | |
| 976 | + gen_movl_T1_reg(s, rm); | |
| 977 | + } | |
| 978 | + if (insn & (1 << 9)) | |
| 979 | + gen_op_addl_T0_T1_cc(); | |
| 980 | + else | |
| 981 | + gen_op_addl_T0_T1_cc(); | |
| 982 | + gen_movl_reg_T0(s, rd); | |
| 983 | + } else { | |
| 984 | + /* shift immediate */ | |
| 985 | + rm = (insn >> 3) & 7; | |
| 986 | + shift = (insn >> 6) & 0x1f; | |
| 987 | + gen_movl_T0_reg(s, rm); | |
| 988 | + gen_shift_T0_im_thumb[op](shift); | |
| 989 | + gen_movl_reg_T0(s, rd); | |
| 990 | + } | |
| 991 | + break; | |
| 992 | + case 2: case 3: | |
| 993 | + /* arithmetic large immediate */ | |
| 994 | + op = (insn >> 11) & 3; | |
| 995 | + rd = (insn >> 8) & 0x7; | |
| 996 | + if (op == 0) { | |
| 997 | + gen_op_movl_T0_im(insn & 0xff); | |
| 998 | + } else { | |
| 999 | + gen_movl_T0_reg(s, rd); | |
| 1000 | + gen_op_movl_T1_im(insn & 0xff); | |
| 1001 | + } | |
| 1002 | + switch (op) { | |
| 1003 | + case 0: /* mov */ | |
| 1004 | + gen_op_logic_T0_cc(); | |
| 1005 | + break; | |
| 1006 | + case 1: /* cmp */ | |
| 1007 | + gen_op_subl_T0_T1_cc(); | |
| 1008 | + break; | |
| 1009 | + case 2: /* add */ | |
| 1010 | + gen_op_addl_T0_T1_cc(); | |
| 1011 | + break; | |
| 1012 | + case 3: /* sub */ | |
| 1013 | + gen_op_subl_T0_T1_cc(); | |
| 1014 | + break; | |
| 1015 | + } | |
| 1016 | + if (op != 1) | |
| 1017 | + gen_movl_reg_T0(s, rd); | |
| 1018 | + break; | |
| 1019 | + case 4: | |
| 1020 | + if (insn & (1 << 11)) { | |
| 1021 | + rd = (insn >> 8) & 7; | |
| 1022 | + /* load pc-relative */ | |
| 1023 | + val = (insn & 0xff) * 4; | |
| 1024 | + gen_op_movl_T1_im(val); | |
| 1025 | + gen_movl_T2_reg(s, 15); | |
| 1026 | + gen_op_addl_T1_T2(); | |
| 1027 | + gen_op_ldl_T0_T1(); | |
| 1028 | + gen_movl_reg_T0(s, rd); | |
| 1029 | + break; | |
| 1030 | + } | |
| 1031 | + if (insn & (1 << 10)) { | |
| 1032 | + /* data processing extended or blx */ | |
| 1033 | + rd = (insn & 7) | ((insn >> 4) & 8); | |
| 1034 | + rm = (insn >> 3) & 0xf; | |
| 1035 | + op = (insn >> 8) & 3; | |
| 1036 | + switch (op) { | |
| 1037 | + case 0: /* add */ | |
| 1038 | + gen_movl_T0_reg(s, rd); | |
| 1039 | + gen_movl_T1_reg(s, rm); | |
| 1040 | + gen_op_addl_T0_T1(); | |
| 1041 | + gen_movl_reg_T0(s, rd); | |
| 1042 | + break; | |
| 1043 | + case 1: /* cmp */ | |
| 1044 | + gen_movl_T0_reg(s, rd); | |
| 1045 | + gen_movl_T1_reg(s, rm); | |
| 1046 | + gen_op_subl_T0_T1_cc(); | |
| 1047 | + break; | |
| 1048 | + case 2: /* mov/cpy */ | |
| 1049 | + gen_movl_T0_reg(s, rm); | |
| 1050 | + gen_movl_reg_T0(s, rd); | |
| 1051 | + break; | |
| 1052 | + case 3:/* branch [and link] exchange thumb register */ | |
| 1053 | + if (insn & (1 << 7)) { | |
| 1054 | + val = (uint32_t)s->pc | 1; | |
| 1055 | + gen_op_movl_T1_im(val); | |
| 1056 | + gen_movl_reg_T1(s, 14); | |
| 1057 | + } | |
| 1058 | + gen_movl_T0_reg(s, rm); | |
| 1059 | + gen_bx(s); | |
| 1060 | + break; | |
| 1061 | + } | |
| 1062 | + break; | |
| 1063 | + } | |
| 1064 | + | |
| 1065 | + /* data processing register */ | |
| 1066 | + rd = insn & 7; | |
| 1067 | + rm = (insn >> 3) & 7; | |
| 1068 | + op = (insn >> 6) & 0xf; | |
| 1069 | + if (op == 2 || op == 3 || op == 4 || op == 7) { | |
| 1070 | + /* the shift/rotate ops want the operands backwards */ | |
| 1071 | + val = rm; | |
| 1072 | + rm = rd; | |
| 1073 | + rd = val; | |
| 1074 | + val = 1; | |
| 1075 | + } else { | |
| 1076 | + val = 0; | |
| 1077 | + } | |
| 1078 | + | |
| 1079 | + if (op == 9) /* neg */ | |
| 1080 | + gen_op_movl_T0_im(0); | |
| 1081 | + else if (op != 0xf) /* mvn doesn't read its first operand */ | |
| 1082 | + gen_movl_T0_reg(s, rd); | |
| 1083 | + | |
| 1084 | + gen_movl_T1_reg(s, rm); | |
| 1085 | + switch (insn >> 6) { | |
| 1086 | + case 0x0: /* and */ | |
| 1087 | + gen_op_andl_T0_T1(); | |
| 1088 | + gen_op_logic_T0_cc(); | |
| 1089 | + break; | |
| 1090 | + case 0x1: /* eor */ | |
| 1091 | + gen_op_xorl_T0_T1(); | |
| 1092 | + gen_op_logic_T0_cc(); | |
| 1093 | + break; | |
| 1094 | + case 0x2: /* lsl */ | |
| 1095 | + gen_op_shll_T1_T0_cc(); | |
| 1096 | + break; | |
| 1097 | + case 0x3: /* lsr */ | |
| 1098 | + gen_op_shrl_T1_T0_cc(); | |
| 1099 | + break; | |
| 1100 | + case 0x4: /* asr */ | |
| 1101 | + gen_op_sarl_T1_T0_cc(); | |
| 1102 | + break; | |
| 1103 | + case 0x5: /* adc */ | |
| 1104 | + gen_op_adcl_T0_T1_cc(); | |
| 1105 | + break; | |
| 1106 | + case 0x6: /* sbc */ | |
| 1107 | + gen_op_sbcl_T0_T1_cc(); | |
| 1108 | + break; | |
| 1109 | + case 0x7: /* ror */ | |
| 1110 | + gen_op_rorl_T1_T0_cc(); | |
| 1111 | + break; | |
| 1112 | + case 0x8: /* tst */ | |
| 1113 | + gen_op_andl_T0_T1(); | |
| 1114 | + gen_op_logic_T0_cc(); | |
| 1115 | + rd = 16; | |
| 1116 | + case 0x9: /* neg */ | |
| 1117 | + gen_op_rsbl_T0_T1_cc(); | |
| 1118 | + break; | |
| 1119 | + case 0xa: /* cmp */ | |
| 1120 | + gen_op_subl_T0_T1_cc(); | |
| 1121 | + rd = 16; | |
| 1122 | + break; | |
| 1123 | + case 0xb: /* cmn */ | |
| 1124 | + gen_op_addl_T0_T1_cc(); | |
| 1125 | + rd = 16; | |
| 1126 | + break; | |
| 1127 | + case 0xc: /* orr */ | |
| 1128 | + gen_op_orl_T0_T1(); | |
| 1129 | + gen_op_logic_T0_cc(); | |
| 1130 | + break; | |
| 1131 | + case 0xd: /* mul */ | |
| 1132 | + gen_op_mull_T0_T1(); | |
| 1133 | + gen_op_logic_T0_cc(); | |
| 1134 | + break; | |
| 1135 | + case 0xe: /* bic */ | |
| 1136 | + gen_op_bicl_T0_T1(); | |
| 1137 | + gen_op_logic_T0_cc(); | |
| 1138 | + break; | |
| 1139 | + case 0xf: /* mvn */ | |
| 1140 | + gen_op_notl_T1(); | |
| 1141 | + gen_op_logic_T1_cc(); | |
| 1142 | + val = 1; | |
| 1143 | + break; | |
| 1144 | + } | |
| 1145 | + if (rd != 16) { | |
| 1146 | + if (val) | |
| 1147 | + gen_movl_reg_T1(s, rd); | |
| 1148 | + else | |
| 1149 | + gen_movl_reg_T0(s, rd); | |
| 1150 | + } | |
| 1151 | + break; | |
| 1152 | + | |
| 1153 | + case 5: | |
| 1154 | + /* load/store register offset. */ | |
| 1155 | + rd = insn & 7; | |
| 1156 | + rn = (insn >> 3) & 7; | |
| 1157 | + rm = (insn >> 6) & 7; | |
| 1158 | + op = (insn >> 9) & 7; | |
| 1159 | + gen_movl_T1_reg(s, rn); | |
| 1160 | + gen_movl_T2_reg(s, rm); | |
| 1161 | + gen_op_addl_T1_T2(); | |
| 1162 | + | |
| 1163 | + if (op < 3) /* store */ | |
| 1164 | + gen_movl_T0_reg(s, rd); | |
| 1165 | + | |
| 1166 | + switch (op) { | |
| 1167 | + case 0: /* str */ | |
| 1168 | + gen_op_stl_T0_T1(); | |
| 1169 | + break; | |
| 1170 | + case 1: /* strh */ | |
| 1171 | + gen_op_stw_T0_T1(); | |
| 1172 | + break; | |
| 1173 | + case 2: /* strb */ | |
| 1174 | + gen_op_stb_T0_T1(); | |
| 1175 | + break; | |
| 1176 | + case 3: /* ldrsb */ | |
| 1177 | + gen_op_ldsb_T0_T1(); | |
| 1178 | + break; | |
| 1179 | + case 4: /* ldr */ | |
| 1180 | + gen_op_ldl_T0_T1(); | |
| 1181 | + break; | |
| 1182 | + case 5: /* ldrh */ | |
| 1183 | + gen_op_ldsw_T0_T1(); | |
| 1184 | + break; | |
| 1185 | + case 6: /* ldrb */ | |
| 1186 | + gen_op_ldub_T0_T1(); | |
| 1187 | + break; | |
| 1188 | + case 7: /* ldrsh */ | |
| 1189 | + gen_op_ldsw_T0_T1(); | |
| 1190 | + break; | |
| 1191 | + } | |
| 1192 | + if (op >= 3) /* load */ | |
| 1193 | + gen_movl_reg_T0(s, rd); | |
| 1194 | + break; | |
| 1195 | + | |
| 1196 | + case 6: | |
| 1197 | + /* load/store word immediate offset */ | |
| 1198 | + rd = insn & 7; | |
| 1199 | + rn = (insn >> 3) & 7; | |
| 1200 | + gen_movl_T1_reg(s, rn); | |
| 1201 | + val = (insn >> 4) & 0x7c; | |
| 1202 | + gen_op_movl_T2_im(val); | |
| 1203 | + gen_op_addl_T1_T2(); | |
| 1204 | + | |
| 1205 | + if (insn & (1 << 11)) { | |
| 1206 | + /* load */ | |
| 1207 | + gen_op_ldl_T0_T1(); | |
| 1208 | + gen_movl_reg_T0(s, rd); | |
| 1209 | + } else { | |
| 1210 | + /* store */ | |
| 1211 | + gen_movl_T0_reg(s, rd); | |
| 1212 | + gen_op_stl_T0_T1(); | |
| 1213 | + } | |
| 1214 | + break; | |
| 1215 | + | |
| 1216 | + case 7: | |
| 1217 | + /* load/store byte immediate offset */ | |
| 1218 | + rd = insn & 7; | |
| 1219 | + rn = (insn >> 3) & 7; | |
| 1220 | + gen_movl_T1_reg(s, rn); | |
| 1221 | + val = (insn >> 6) & 0x1f; | |
| 1222 | + gen_op_movl_T2_im(val); | |
| 1223 | + gen_op_addl_T1_T2(); | |
| 1224 | + | |
| 1225 | + if (insn & (1 << 11)) { | |
| 1226 | + /* load */ | |
| 1227 | + gen_op_ldub_T0_T1(); | |
| 1228 | + gen_movl_reg_T0(s, rd); | |
| 1229 | + } else { | |
| 1230 | + /* store */ | |
| 1231 | + gen_movl_T0_reg(s, rd); | |
| 1232 | + gen_op_stb_T0_T1(); | |
| 1233 | + } | |
| 1234 | + break; | |
| 1235 | + | |
| 1236 | + case 8: | |
| 1237 | + /* load/store halfword immediate offset */ | |
| 1238 | + rd = insn & 7; | |
| 1239 | + rn = (insn >> 3) & 7; | |
| 1240 | + gen_movl_T1_reg(s, rn); | |
| 1241 | + val = (insn >> 5) & 0x3e; | |
| 1242 | + gen_op_movl_T2_im(val); | |
| 1243 | + gen_op_addl_T1_T2(); | |
| 1244 | + | |
| 1245 | + if (insn & (1 << 11)) { | |
| 1246 | + /* load */ | |
| 1247 | + gen_op_lduw_T0_T1(); | |
| 1248 | + gen_movl_reg_T0(s, rd); | |
| 1249 | + } else { | |
| 1250 | + /* store */ | |
| 1251 | + gen_movl_T0_reg(s, rd); | |
| 1252 | + gen_op_stw_T0_T1(); | |
| 1253 | + } | |
| 1254 | + break; | |
| 1255 | + | |
| 1256 | + case 9: | |
| 1257 | + /* load/store from stack */ | |
| 1258 | + rd = (insn >> 8) & 7; | |
| 1259 | + gen_movl_T1_reg(s, 13); | |
| 1260 | + val = (insn & 0xff) * 4; | |
| 1261 | + gen_op_movl_T2_im(val); | |
| 1262 | + gen_op_addl_T1_T2(); | |
| 1263 | + | |
| 1264 | + if (insn & (1 << 11)) { | |
| 1265 | + /* load */ | |
| 1266 | + gen_op_ldl_T0_T1(); | |
| 1267 | + gen_movl_reg_T0(s, rd); | |
| 1268 | + } else { | |
| 1269 | + /* store */ | |
| 1270 | + gen_movl_T0_reg(s, rd); | |
| 1271 | + gen_op_stl_T0_T1(); | |
| 1272 | + } | |
| 1273 | + break; | |
| 1274 | + | |
| 1275 | + case 10: | |
| 1276 | + /* add to high reg */ | |
| 1277 | + rd = (insn >> 8) & 7; | |
| 1278 | + if (insn & (1 << 11)) | |
| 1279 | + rm = 13; /* sp */ | |
| 1280 | + else | |
| 1281 | + rm = 15; /* pc */ | |
| 1282 | + gen_movl_T0_reg(s, rm); | |
| 1283 | + val = (insn & 0xff) * 4; | |
| 1284 | + gen_op_movl_T1_im(val); | |
| 1285 | + gen_op_addl_T0_T1(); | |
| 1286 | + gen_movl_reg_T0(s, rd); | |
| 1287 | + break; | |
| 1288 | + | |
| 1289 | + case 11: | |
| 1290 | + /* misc */ | |
| 1291 | + op = (insn >> 8) & 0xf; | |
| 1292 | + switch (op) { | |
| 1293 | + case 0: | |
| 1294 | + /* adjust stack pointer */ | |
| 1295 | + gen_movl_T1_reg(s, 13); | |
| 1296 | + val = (insn & 0x7f) * 4; | |
| 1297 | + if (insn & (1 << 7)) | |
| 1298 | + val = -(int32_t)val; | |
| 1299 | + gen_op_movl_T2_im(val); | |
| 1300 | + gen_op_addl_T1_T2(); | |
| 1301 | + gen_movl_reg_T1(s, 13); | |
| 1302 | + break; | |
| 1303 | + | |
| 1304 | + case 4: case 5: case 0xc: case 0xd: | |
| 1305 | + /* push/pop */ | |
| 1306 | + gen_movl_T1_reg(s, 13); | |
| 1307 | + if (insn & (1 << 11)) | |
| 1308 | + val = 4; | |
| 1309 | + else | |
| 1310 | + val = -4; | |
| 1311 | + gen_op_movl_T2_im(val); | |
| 1312 | + for (i = 0; i < 8; i++) { | |
| 1313 | + if (insn & (1 << i)) { | |
| 1314 | + if (insn & (1 << 11)) { | |
| 1315 | + /* pop */ | |
| 1316 | + gen_op_ldl_T0_T1(); | |
| 1317 | + gen_movl_reg_T0(s, i); | |
| 1318 | + } else { | |
| 1319 | + /* push */ | |
| 1320 | + gen_movl_T0_reg(s, i); | |
| 1321 | + gen_op_stl_T0_T1(); | |
| 1322 | + } | |
| 1323 | + /* move to the next address */ | |
| 1324 | + gen_op_addl_T1_T2(); | |
| 1325 | + } | |
| 1326 | + } | |
| 1327 | + if (insn & (1 << 8)) { | |
| 1328 | + if (insn & (1 << 11)) { | |
| 1329 | + /* pop pc */ | |
| 1330 | + gen_op_ldl_T0_T1(); | |
| 1331 | + /* don't set the pc until the rest of the instruction | |
| 1332 | + has completed */ | |
| 1333 | + } else { | |
| 1334 | + /* push lr */ | |
| 1335 | + gen_movl_T0_reg(s, 14); | |
| 1336 | + gen_op_stl_T0_T1(); | |
| 1337 | + } | |
| 1338 | + gen_op_addl_T1_T2(); | |
| 1339 | + } | |
| 1340 | + | |
| 1341 | + /* write back the new stack pointer */ | |
| 1342 | + gen_movl_reg_T1(s, 13); | |
| 1343 | + /* set the new PC value */ | |
| 1344 | + if ((insn & 0x0900) == 0x0900) | |
| 1345 | + gen_bx(s); | |
| 1346 | + break; | |
| 1347 | + | |
| 1348 | + default: | |
| 1349 | + goto undef; | |
| 1350 | + } | |
| 1351 | + break; | |
| 1352 | + | |
| 1353 | + case 12: | |
| 1354 | + /* load/store multiple */ | |
| 1355 | + rn = (insn >> 8) & 0x7; | |
| 1356 | + gen_movl_T1_reg(s, rn); | |
| 1357 | + gen_op_movl_T2_im(4); | |
| 1358 | + val = 0; | |
| 1359 | + for (i = 0; i < 8; i++) { | |
| 1360 | + if (insn & (1 << i)) { | |
| 1361 | + /* advance to the next address */ | |
| 1362 | + if (val) | |
| 1363 | + gen_op_addl_T1_T2(); | |
| 1364 | + else | |
| 1365 | + val = 1; | |
| 1366 | + if (insn & (1 << 11)) { | |
| 1367 | + /* load */ | |
| 1368 | + gen_op_ldl_T0_T1(); | |
| 1369 | + gen_movl_reg_T0(s, i); | |
| 1370 | + } else { | |
| 1371 | + /* store */ | |
| 1372 | + gen_movl_T0_reg(s, i); | |
| 1373 | + gen_op_stl_T0_T1(); | |
| 1374 | + } | |
| 1375 | + } | |
| 1376 | + } | |
| 1377 | + break; | |
| 1378 | + | |
| 1379 | + case 13: | |
| 1380 | + /* conditional branch or swi */ | |
| 1381 | + cond = (insn >> 8) & 0xf; | |
| 1382 | + if (cond == 0xe) | |
| 1383 | + goto undef; | |
| 1384 | + | |
| 1385 | + if (cond == 0xf) { | |
| 1386 | + /* swi */ | |
| 1387 | + gen_op_movl_T0_im((long)s->pc | 1); | |
| 1388 | + /* Don't set r15. */ | |
| 1389 | + gen_op_movl_reg_TN[0][15](); | |
| 1390 | + gen_op_swi(); | |
| 1391 | + s->is_jmp = DISAS_JUMP; | |
| 1392 | + break; | |
| 1393 | + } | |
| 1394 | + /* generate a conditional jump to next instruction */ | |
| 1395 | + gen_test_cc[cond ^ 1]((long)s->tb, (long)s->pc); | |
| 1396 | + s->is_jmp = DISAS_JUMP_NEXT; | |
| 1397 | + gen_movl_T1_reg(s, 15); | |
| 1398 | + | |
| 1399 | + /* jump to the offset */ | |
| 1400 | + val = (uint32_t)s->pc; | |
| 1401 | + offset = ((int32_t)insn << 24) >> 24; | |
| 1402 | + val += (offset << 1) + 2; | |
| 1403 | + gen_op_jmp((long)s->tb, val); | |
| 1404 | + s->is_jmp = DISAS_TB_JUMP; | |
| 1405 | + break; | |
| 1406 | + | |
| 1407 | + case 14: | |
| 1408 | + /* unconditional branch */ | |
| 1409 | + if (insn & (1 << 11)) | |
| 1410 | + goto undef; /* Second half of a blx */ | |
| 1411 | + val = (uint32_t)s->pc; | |
| 1412 | + offset = ((int32_t)insn << 21) >> 21; | |
| 1413 | + val += (offset << 1) + 2; | |
| 1414 | + gen_op_jmp((long)s->tb, val); | |
| 1415 | + s->is_jmp = DISAS_TB_JUMP; | |
| 1416 | + break; | |
| 1417 | + | |
| 1418 | + case 15: | |
| 1419 | + /* branch and link [and switch to arm] */ | |
| 1420 | + offset = ((int32_t)insn << 21) >> 10; | |
| 1421 | + insn = lduw(s->pc); | |
| 1422 | + offset |= insn & 0x7ff; | |
| 1423 | + | |
| 1424 | + val = (uint32_t)s->pc + 2; | |
| 1425 | + gen_op_movl_T1_im(val | 1); | |
| 1426 | + gen_movl_reg_T1(s, 14); | |
| 1427 | + | |
| 1428 | + val += offset; | |
| 1429 | + if (insn & (1 << 11)) { | |
| 1430 | + /* bl */ | |
| 1431 | + gen_op_jmp((long)s->tb, val); | |
| 1432 | + s->is_jmp = DISAS_TB_JUMP; | |
| 1433 | + } else { | |
| 1434 | + /* blx */ | |
| 1435 | + gen_op_movl_T0_im(val); | |
| 1436 | + gen_bx(s); | |
| 1437 | + } | |
| 1438 | + } | |
| 1439 | + return; | |
| 1440 | +undef: | |
| 1441 | + gen_op_movl_T0_im((long)s->pc - 4); | |
| 1442 | + gen_op_movl_reg_TN[0][15](); | |
| 1443 | + gen_op_undef_insn(); | |
| 1444 | + s->is_jmp = DISAS_JUMP; | |
| 1445 | +} | |
| 1446 | + | |
| 755 | 1447 | /* generate intermediate code in gen_opc_buf and gen_opparam_buf for |
| 756 | 1448 | basic block 'tb'. If search_pc is TRUE, also generate PC |
| 757 | 1449 | information for each intermediate instruction. */ |
| ... | ... | @@ -787,7 +1479,10 @@ static inline int gen_intermediate_code_internal(CPUState *env, |
| 787 | 1479 | gen_opc_pc[lj] = dc->pc; |
| 788 | 1480 | gen_opc_instr_start[lj] = 1; |
| 789 | 1481 | } |
| 790 | - disas_arm_insn(dc); | |
| 1482 | + if (env->thumb) | |
| 1483 | + disas_thumb_insn(dc); | |
| 1484 | + else | |
| 1485 | + disas_arm_insn(dc); | |
| 791 | 1486 | } while (!dc->is_jmp && gen_opc_ptr < gen_opc_end && |
| 792 | 1487 | (dc->pc - pc_start) < (TARGET_PAGE_SIZE - 32)); |
| 793 | 1488 | switch(dc->is_jmp) { |
| ... | ... | @@ -797,6 +1492,7 @@ static inline int gen_intermediate_code_internal(CPUState *env, |
| 797 | 1492 | break; |
| 798 | 1493 | default: |
| 799 | 1494 | case DISAS_JUMP: |
| 1495 | + case DISAS_UPDATE: | |
| 800 | 1496 | /* indicate that the hash table must be used to find the next TB */ |
| 801 | 1497 | gen_op_movl_T0_0(); |
| 802 | 1498 | gen_op_exit_tb(); | ... | ... |