diff options
Diffstat (limited to 'target-arm/translate.c')
-rw-r--r-- | target-arm/translate.c | 86 |
1 files changed, 53 insertions, 33 deletions
diff --git a/target-arm/translate.c b/target-arm/translate.c index c9005c4..2965741 100644 --- a/target-arm/translate.c +++ b/target-arm/translate.c @@ -38,6 +38,7 @@ typedef struct DisasContext { int condlabel; struct TranslationBlock *tb; int singlestep_enabled; + int thumb; } DisasContext; #define DISAS_JUMP_NEXT 4 @@ -268,8 +269,11 @@ static inline void gen_movl_TN_reg(DisasContext *s, int reg, int t) int val; if (reg == 15) { - /* normaly, since we updated PC, we need only to add 4 */ - val = (long)s->pc + 4; + /* normaly, since we updated PC, we need only to add one insn */ + if (s->thumb) + val = (long)s->pc + 2; + else + val = (long)s->pc + 4; gen_op_movl_TN_im[t](val); } else { gen_op_movl_TN_reg[t][reg](); @@ -897,6 +901,8 @@ static inline void gen_jmp (DisasContext *s, uint32_t dest) { if (__builtin_expect(s->singlestep_enabled, 0)) { /* An indirect jump so that we still trigger the debug exception. */ + if (s->thumb) + dest |= 1; gen_op_movl_T0_im(dest); gen_bx(s); } else { @@ -1552,7 +1558,7 @@ static void disas_thumb_insn(DisasContext *s) gen_movl_T1_reg(s, rm); } if (insn & (1 << 9)) - gen_op_addl_T0_T1_cc(); + gen_op_subl_T0_T1_cc(); else gen_op_addl_T0_T1_cc(); gen_movl_reg_T0(s, rd); @@ -1595,11 +1601,10 @@ static void disas_thumb_insn(DisasContext *s) case 4: if (insn & (1 << 11)) { rd = (insn >> 8) & 7; - /* load pc-relative */ - val = (insn & 0xff) * 4; + /* load pc-relative. Bit 1 of PC is ignored. */ + val = s->pc + 2 + ((insn & 0xff) * 4); + val &= ~(uint32_t)2; gen_op_movl_T1_im(val); - gen_movl_T2_reg(s, 15); - gen_op_addl_T1_T2(); gen_op_ldl_T0_T1(); gen_movl_reg_T0(s, rd); break; @@ -1658,7 +1663,7 @@ static void disas_thumb_insn(DisasContext *s) gen_movl_T0_reg(s, rd); gen_movl_T1_reg(s, rm); - switch (insn >> 6) { + switch (op) { case 0x0: /* and */ gen_op_andl_T0_T1(); gen_op_logic_T0_cc(); @@ -1689,8 +1694,9 @@ static void disas_thumb_insn(DisasContext *s) gen_op_andl_T0_T1(); gen_op_logic_T0_cc(); rd = 16; + break; case 0x9: /* neg */ - gen_op_rsbl_T0_T1_cc(); + gen_op_subl_T0_T1_cc(); break; case 0xa: /* cmp */ gen_op_subl_T0_T1_cc(); @@ -1716,11 +1722,12 @@ static void disas_thumb_insn(DisasContext *s) gen_op_notl_T1(); gen_op_logic_T1_cc(); val = 1; + rm = rd; break; } if (rd != 16) { if (val) - gen_movl_reg_T1(s, rd); + gen_movl_reg_T1(s, rm); else gen_movl_reg_T0(s, rd); } @@ -1756,7 +1763,7 @@ static void disas_thumb_insn(DisasContext *s) gen_op_ldl_T0_T1(); break; case 5: /* ldrh */ - gen_op_ldsw_T0_T1(); + gen_op_lduw_T0_T1(); break; case 6: /* ldrb */ gen_op_ldub_T0_T1(); @@ -1851,11 +1858,13 @@ static void disas_thumb_insn(DisasContext *s) case 10: /* add to high reg */ rd = (insn >> 8) & 7; - if (insn & (1 << 11)) - rm = 13; /* sp */ - else - rm = 15; /* pc */ - gen_movl_T0_reg(s, rm); + if (insn & (1 << 11)) { + /* SP */ + gen_movl_T0_reg(s, 13); + } else { + /* PC. bit 1 is ignored. */ + gen_op_movl_T0_im((s->pc + 2) & ~(uint32_t)2); + } val = (insn & 0xff) * 4; gen_op_movl_T1_im(val); gen_op_addl_T0_T1(); @@ -1880,11 +1889,19 @@ static void disas_thumb_insn(DisasContext *s) case 4: case 5: case 0xc: case 0xd: /* push/pop */ gen_movl_T1_reg(s, 13); - if (insn & (1 << 11)) - val = 4; + if (insn & (1 << 8)) + offset = 4; else - val = -4; - gen_op_movl_T2_im(val); + offset = 0; + for (i = 0; i < 8; i++) { + if (insn & (1 << i)) + offset += 4; + } + if ((insn & (1 << 11)) == 0) { + gen_op_movl_T2_im(-offset); + gen_op_addl_T1_T2(); + } + gen_op_movl_T2_im(4); for (i = 0; i < 8; i++) { if (insn & (1 << i)) { if (insn & (1 << 11)) { @@ -1896,7 +1913,7 @@ static void disas_thumb_insn(DisasContext *s) gen_movl_T0_reg(s, i); gen_op_stl_T0_T1(); } - /* move to the next address */ + /* advance to the next address. */ gen_op_addl_T1_T2(); } } @@ -1913,7 +1930,10 @@ static void disas_thumb_insn(DisasContext *s) } gen_op_addl_T1_T2(); } - + if ((insn & (1 << 11)) == 0) { + gen_op_movl_T2_im(-offset); + gen_op_addl_T1_T2(); + } /* write back the new stack pointer */ gen_movl_reg_T1(s, 13); /* set the new PC value */ @@ -1931,14 +1951,8 @@ static void disas_thumb_insn(DisasContext *s) rn = (insn >> 8) & 0x7; gen_movl_T1_reg(s, rn); gen_op_movl_T2_im(4); - val = 0; for (i = 0; i < 8; i++) { if (insn & (1 << i)) { - /* advance to the next address */ - if (val) - gen_op_addl_T1_T2(); - else - val = 1; if (insn & (1 << 11)) { /* load */ gen_op_ldl_T0_T1(); @@ -1948,8 +1962,12 @@ static void disas_thumb_insn(DisasContext *s) gen_movl_T0_reg(s, i); gen_op_stl_T0_T1(); } + /* advance to the next address */ + gen_op_addl_T1_T2(); } } + /* Base register writeback. */ + gen_movl_reg_T1(s, rn); break; case 13: @@ -1976,9 +1994,9 @@ static void disas_thumb_insn(DisasContext *s) gen_movl_T1_reg(s, 15); /* jump to the offset */ - val = (uint32_t)s->pc; + val = (uint32_t)s->pc + 2; offset = ((int32_t)insn << 24) >> 24; - val += (offset << 1) + 2; + val += offset << 1; gen_jmp(s, val); break; @@ -2002,19 +2020,20 @@ static void disas_thumb_insn(DisasContext *s) gen_op_movl_T1_im(val | 1); gen_movl_reg_T1(s, 14); - val += offset; + val += offset << 1; if (insn & (1 << 11)) { /* bl */ gen_jmp(s, val); } else { /* blx */ + val &= ~(uint32_t)2; gen_op_movl_T0_im(val); gen_bx(s); } } return; undef: - gen_op_movl_T0_im((long)s->pc - 4); + gen_op_movl_T0_im((long)s->pc - 2); gen_op_movl_reg_TN[0][15](); gen_op_undef_insn(); s->is_jmp = DISAS_JUMP; @@ -2045,6 +2064,7 @@ static inline int gen_intermediate_code_internal(CPUState *env, dc->pc = pc_start; dc->singlestep_enabled = env->singlestep_enabled; dc->condjmp = 0; + dc->thumb = env->thumb; nb_gen_labels = 0; lj = -1; do { @@ -2128,7 +2148,7 @@ static inline int gen_intermediate_code_internal(CPUState *env, if (loglevel & CPU_LOG_TB_IN_ASM) { fprintf(logfile, "----------------\n"); fprintf(logfile, "IN: %s\n", lookup_symbol(pc_start)); - target_disas(logfile, pc_start, dc->pc - pc_start, 0); + target_disas(logfile, pc_start, dc->pc - pc_start, env->thumb); fprintf(logfile, "\n"); if (loglevel & (CPU_LOG_TB_OP)) { fprintf(logfile, "OP:\n"); |