Add TCG native negation op.
git-svn-id: svn://svn.savannah.nongnu.org/qemu/trunk@4426 c046a42c-6fe2-441c-8c8c-71466251a162
This commit is contained in:
parent
44cd42ee40
commit
390efc54fb
@ -457,12 +457,6 @@ static inline void tcg_gen_not_i32(TCGv t0, TCGv t1)
|
||||
tcg_gen_xori_i32(t0, t1, ~0);
|
||||
}
|
||||
|
||||
/* FIXME: Implement this natively. */
|
||||
static inline void tcg_gen_neg_i64(TCGv dest, TCGv src)
|
||||
{
|
||||
tcg_gen_sub_i64(dest, tcg_const_i64(0), src);
|
||||
}
|
||||
|
||||
/* T0 &= ~T1. Clobbers T1. */
|
||||
/* FIXME: Implement bic natively. */
|
||||
static inline void tcg_gen_bic_i32(TCGv dest, TCGv t0, TCGv t1)
|
||||
@ -8111,7 +8105,7 @@ static void disas_thumb_insn(CPUState *env, DisasContext *s)
|
||||
break;
|
||||
case 0x9: /* neg */
|
||||
if (s->condexec_mask)
|
||||
gen_op_subl_T0_T1();
|
||||
tcg_gen_neg_i32(cpu_T[0], cpu_T[1]);
|
||||
else
|
||||
gen_op_subl_T0_T1_cc();
|
||||
break;
|
||||
|
@ -318,7 +318,7 @@ static void t_gen_lz_i32(TCGv d, TCGv x)
|
||||
|
||||
/* y = -(x >> 16) */
|
||||
tcg_gen_shri_i32(y, x, 16);
|
||||
tcg_gen_sub_i32(y, tcg_const_i32(0), y);
|
||||
tcg_gen_neg_i32(y, y);
|
||||
|
||||
/* m = (y >> 16) & 16 */
|
||||
tcg_gen_sari_i32(m, y, 16);
|
||||
@ -753,9 +753,9 @@ static void crisv32_alu_op(DisasContext *dc, int op, int rd, int size)
|
||||
t_gen_add_flag(cpu_T[0], 8); /* R_FLAG. */
|
||||
break;
|
||||
case CC_OP_SUB:
|
||||
tcg_gen_sub_tl(cpu_T[1], tcg_const_tl(0), cpu_T[1]);
|
||||
tcg_gen_neg_tl(cpu_T[1], cpu_T[1]);
|
||||
tcg_gen_add_tl(cpu_T[0], cpu_T[0], cpu_T[1]);
|
||||
tcg_gen_sub_tl(cpu_T[1], tcg_const_tl(0), cpu_T[1]);
|
||||
tcg_gen_neg_tl(cpu_T[1], cpu_T[1]);
|
||||
/* CRIS flag evaluation needs ~src. */
|
||||
tcg_gen_xori_tl(cpu_T[1], cpu_T[1], -1);
|
||||
|
||||
@ -784,9 +784,7 @@ static void crisv32_alu_op(DisasContext *dc, int op, int rd, int size)
|
||||
t_gen_asr(cpu_T[0], cpu_T[0], cpu_T[1]);
|
||||
break;
|
||||
case CC_OP_NEG:
|
||||
/* Hopefully the TCG backend recognizes this pattern
|
||||
and makes a real neg out of it. */
|
||||
tcg_gen_sub_tl(cpu_T[0], tcg_const_tl(0), cpu_T[1]);
|
||||
tcg_gen_neg_tl(cpu_T[0], cpu_T[1]);
|
||||
/* Extended arithmetics. */
|
||||
t_gen_subx_carry(cpu_T[0]);
|
||||
break;
|
||||
@ -829,10 +827,10 @@ static void crisv32_alu_op(DisasContext *dc, int op, int rd, int size)
|
||||
}
|
||||
break;
|
||||
case CC_OP_CMP:
|
||||
tcg_gen_sub_tl(cpu_T[1], tcg_const_tl(0), cpu_T[1]);
|
||||
tcg_gen_neg_tl(cpu_T[1], cpu_T[1]);
|
||||
tcg_gen_add_tl(cpu_T[0], cpu_T[0], cpu_T[1]);
|
||||
/* CRIS flag evaluation needs ~src. */
|
||||
tcg_gen_sub_tl(cpu_T[1], tcg_const_tl(0), cpu_T[1]);
|
||||
tcg_gen_neg_tl(cpu_T[1], cpu_T[1]);
|
||||
/* CRIS flag evaluation needs ~src. */
|
||||
tcg_gen_xori_tl(cpu_T[1], cpu_T[1], -1);
|
||||
|
||||
@ -1642,7 +1640,7 @@ static unsigned int dec_abs_r(DisasContext *dc)
|
||||
/* TODO: consider a branch free approach. */
|
||||
l1 = gen_new_label();
|
||||
tcg_gen_brcond_tl(TCG_COND_GE, cpu_T[1], tcg_const_tl(0), l1);
|
||||
tcg_gen_sub_tl(cpu_T[1], tcg_const_tl(0), cpu_T[1]);
|
||||
tcg_gen_neg_tl(cpu_T[1], cpu_T[1]);
|
||||
gen_set_label(l1);
|
||||
crisv32_alu_op(dc, CC_OP_MOVE, dc->op2, 4);
|
||||
return 2;
|
||||
|
@ -203,6 +203,10 @@ t0=t1+t2
|
||||
|
||||
t0=t1-t2
|
||||
|
||||
* neg_i32/i64 t0, t1
|
||||
|
||||
t0=-t1 (two's complement)
|
||||
|
||||
* mul_i32/i64 t0, t1, t2
|
||||
|
||||
t0=t1*t2
|
||||
|
20
tcg/tcg-op.h
20
tcg/tcg-op.h
@ -1208,6 +1208,24 @@ static inline void tcg_gen_bswap_i64(TCGv ret, TCGv arg)
|
||||
|
||||
#endif
|
||||
|
||||
static inline void tcg_gen_neg_i32(TCGv ret, TCGv arg)
|
||||
{
|
||||
#ifdef TCG_TARGET_HAS_neg_i32
|
||||
tcg_gen_op2(INDEX_op_neg_i32, ret, arg);
|
||||
#else
|
||||
tcg_gen_sub_i32(ret, tcg_const_i32(0), arg);
|
||||
#endif
|
||||
}
|
||||
|
||||
static inline void tcg_gen_neg_i64(TCGv ret, TCGv arg)
|
||||
{
|
||||
#ifdef TCG_TARGET_HAS_neg_i64
|
||||
tcg_gen_op2(INDEX_op_neg_i64, ret, arg);
|
||||
#else
|
||||
tcg_gen_sub_i64(ret, tcg_const_i64(0), arg);
|
||||
#endif
|
||||
}
|
||||
|
||||
|
||||
static inline void tcg_gen_discard_i32(TCGv arg)
|
||||
{
|
||||
@ -1441,6 +1459,7 @@ static inline void tcg_gen_qemu_st64(TCGv arg, TCGv addr, int mem_index)
|
||||
#define tcg_gen_add_tl tcg_gen_add_i64
|
||||
#define tcg_gen_addi_tl tcg_gen_addi_i64
|
||||
#define tcg_gen_sub_tl tcg_gen_sub_i64
|
||||
#define tcg_gen_neg_tl tcg_gen_neg_i64
|
||||
#define tcg_gen_subi_tl tcg_gen_subi_i64
|
||||
#define tcg_gen_and_tl tcg_gen_and_i64
|
||||
#define tcg_gen_andi_tl tcg_gen_andi_i64
|
||||
@ -1483,6 +1502,7 @@ static inline void tcg_gen_qemu_st64(TCGv arg, TCGv addr, int mem_index)
|
||||
#define tcg_gen_add_tl tcg_gen_add_i32
|
||||
#define tcg_gen_addi_tl tcg_gen_addi_i32
|
||||
#define tcg_gen_sub_tl tcg_gen_sub_i32
|
||||
#define tcg_gen_neg_tl tcg_gen_neg_i32
|
||||
#define tcg_gen_subi_tl tcg_gen_subi_i32
|
||||
#define tcg_gen_and_tl tcg_gen_and_i32
|
||||
#define tcg_gen_andi_tl tcg_gen_andi_i32
|
||||
|
@ -148,6 +148,12 @@ DEF2(ext32s_i64, 1, 1, 0, 0)
|
||||
DEF2(bswap_i64, 1, 1, 0, 0)
|
||||
#endif
|
||||
#endif
|
||||
#ifdef TCG_TARGET_HAS_neg_i32
|
||||
DEF2(neg_i32, 1, 1, 0, 0)
|
||||
#endif
|
||||
#ifdef TCG_TARGET_HAS_neg_i64
|
||||
DEF2(neg_i64, 1, 1, 0, 0)
|
||||
#endif
|
||||
|
||||
/* QEMU specific */
|
||||
DEF2(exit_tb, 0, 0, 1, TCG_OPF_BB_END | TCG_OPF_SIDE_EFFECTS)
|
||||
|
@ -1092,6 +1092,13 @@ static inline void tcg_out_op(TCGContext *s, int opc, const TCGArg *args,
|
||||
tcg_out_opc(s, (0xc8 + (args[0] & 7)) | P_EXT | P_REXW, 0, args[0], 0);
|
||||
break;
|
||||
|
||||
case INDEX_op_neg_i32:
|
||||
tcg_out_modrm(s, 0xf7, 3, args[0]);
|
||||
break;
|
||||
case INDEX_op_neg_i64:
|
||||
tcg_out_modrm(s, 0xf7 | P_REXW, 3, args[0]);
|
||||
break;
|
||||
|
||||
case INDEX_op_qemu_ld8u:
|
||||
tcg_out_qemu_ld(s, args, 0);
|
||||
break;
|
||||
@ -1247,6 +1254,9 @@ static const TCGTargetOpDef x86_64_op_defs[] = {
|
||||
{ INDEX_op_bswap_i32, { "r", "0" } },
|
||||
{ INDEX_op_bswap_i64, { "r", "0" } },
|
||||
|
||||
{ INDEX_op_neg_i32, { "r", "0" } },
|
||||
{ INDEX_op_neg_i64, { "r", "0" } },
|
||||
|
||||
{ INDEX_op_qemu_ld8u, { "r", "L" } },
|
||||
{ INDEX_op_qemu_ld8s, { "r", "L" } },
|
||||
{ INDEX_op_qemu_ld16u, { "r", "L" } },
|
||||
|
@ -57,6 +57,8 @@ enum {
|
||||
/* optional instructions */
|
||||
#define TCG_TARGET_HAS_bswap_i32
|
||||
#define TCG_TARGET_HAS_bswap_i64
|
||||
#define TCG_TARGET_HAS_neg_i32
|
||||
#define TCG_TARGET_HAS_neg_i64
|
||||
|
||||
/* Note: must be synced with dyngen-exec.h */
|
||||
#define TCG_AREG0 TCG_REG_R14
|
||||
|
Loading…
Reference in New Issue
Block a user