提交 e22c357b 编写于 作者: D Doug Kwan 提交者: Alexander Graf

target-ppc: Allow little-endian user mode.

This allows running PPC64 little-endian in user mode if target is configured
that way.  In PPC64 LE user mode we set MSR.LE during initialization.
Signed-off-by: NDoug Kwan <dougkwan@google.com>
Signed-off-by: NTom Musta <tommusta@gmail.com>
Signed-off-by: NAlexander Graf <agraf@suse.de>
上级 d90b94cd
...@@ -1484,7 +1484,7 @@ static int do_store_exclusive(CPUPPCState *env) ...@@ -1484,7 +1484,7 @@ static int do_store_exclusive(CPUPPCState *env)
{ {
target_ulong addr; target_ulong addr;
target_ulong page_addr; target_ulong page_addr;
target_ulong val, val2 __attribute__((unused)); target_ulong val, val2 __attribute__((unused)) = 0;
int flags; int flags;
int segv = 0; int segv = 0;
...@@ -1527,6 +1527,12 @@ static int do_store_exclusive(CPUPPCState *env) ...@@ -1527,6 +1527,12 @@ static int do_store_exclusive(CPUPPCState *env)
case 8: segv = put_user_u64(val, addr); break; case 8: segv = put_user_u64(val, addr); break;
case 16: { case 16: {
if (val2 == env->reserve_val2) { if (val2 == env->reserve_val2) {
if (msr_le) {
val2 = val;
val = env->gpr[reg+1];
} else {
val2 = env->gpr[reg+1];
}
segv = put_user_u64(val, addr); segv = put_user_u64(val, addr);
if (!segv) { if (!segv) {
segv = put_user_u64(val2, addr + 8); segv = put_user_u64(val2, addr + 8);
......
...@@ -25,6 +25,15 @@ ...@@ -25,6 +25,15 @@
//#define DEBUG_OP //#define DEBUG_OP
static inline bool needs_byteswap(const CPUPPCState *env)
{
#if defined(TARGET_WORDS_BIGENDIAN)
return msr_le;
#else
return !msr_le;
#endif
}
/*****************************************************************************/ /*****************************************************************************/
/* Memory load and stores */ /* Memory load and stores */
...@@ -44,7 +53,7 @@ static inline target_ulong addr_add(CPUPPCState *env, target_ulong addr, ...@@ -44,7 +53,7 @@ static inline target_ulong addr_add(CPUPPCState *env, target_ulong addr,
void helper_lmw(CPUPPCState *env, target_ulong addr, uint32_t reg) void helper_lmw(CPUPPCState *env, target_ulong addr, uint32_t reg)
{ {
for (; reg < 32; reg++) { for (; reg < 32; reg++) {
if (msr_le) { if (needs_byteswap(env)) {
env->gpr[reg] = bswap32(cpu_ldl_data(env, addr)); env->gpr[reg] = bswap32(cpu_ldl_data(env, addr));
} else { } else {
env->gpr[reg] = cpu_ldl_data(env, addr); env->gpr[reg] = cpu_ldl_data(env, addr);
...@@ -56,7 +65,7 @@ void helper_lmw(CPUPPCState *env, target_ulong addr, uint32_t reg) ...@@ -56,7 +65,7 @@ void helper_lmw(CPUPPCState *env, target_ulong addr, uint32_t reg)
void helper_stmw(CPUPPCState *env, target_ulong addr, uint32_t reg) void helper_stmw(CPUPPCState *env, target_ulong addr, uint32_t reg)
{ {
for (; reg < 32; reg++) { for (; reg < 32; reg++) {
if (msr_le) { if (needs_byteswap(env)) {
cpu_stl_data(env, addr, bswap32((uint32_t)env->gpr[reg])); cpu_stl_data(env, addr, bswap32((uint32_t)env->gpr[reg]));
} else { } else {
cpu_stl_data(env, addr, (uint32_t)env->gpr[reg]); cpu_stl_data(env, addr, (uint32_t)env->gpr[reg]);
...@@ -199,6 +208,11 @@ target_ulong helper_lscbx(CPUPPCState *env, target_ulong addr, uint32_t reg, ...@@ -199,6 +208,11 @@ target_ulong helper_lscbx(CPUPPCState *env, target_ulong addr, uint32_t reg,
#define LO_IDX 0 #define LO_IDX 0
#endif #endif
/* We use msr_le to determine index ordering in a vector. However,
byteswapping is not simply controlled by msr_le. We also need to take
into account endianness of the target. This is done for the little-endian
PPC64 user-mode target. */
#define LVE(name, access, swap, element) \ #define LVE(name, access, swap, element) \
void helper_##name(CPUPPCState *env, ppc_avr_t *r, \ void helper_##name(CPUPPCState *env, ppc_avr_t *r, \
target_ulong addr) \ target_ulong addr) \
...@@ -207,9 +221,11 @@ target_ulong helper_lscbx(CPUPPCState *env, target_ulong addr, uint32_t reg, ...@@ -207,9 +221,11 @@ target_ulong helper_lscbx(CPUPPCState *env, target_ulong addr, uint32_t reg,
int adjust = HI_IDX*(n_elems - 1); \ int adjust = HI_IDX*(n_elems - 1); \
int sh = sizeof(r->element[0]) >> 1; \ int sh = sizeof(r->element[0]) >> 1; \
int index = (addr & 0xf) >> sh; \ int index = (addr & 0xf) >> sh; \
\
if (msr_le) { \ if (msr_le) { \
index = n_elems - index - 1; \ index = n_elems - index - 1; \
} \
\
if (needs_byteswap(env)) { \
r->element[LO_IDX ? index : (adjust - index)] = \ r->element[LO_IDX ? index : (adjust - index)] = \
swap(access(env, addr)); \ swap(access(env, addr)); \
} else { \ } else { \
...@@ -232,9 +248,11 @@ LVE(lvewx, cpu_ldl_data, bswap32, u32) ...@@ -232,9 +248,11 @@ LVE(lvewx, cpu_ldl_data, bswap32, u32)
int adjust = HI_IDX * (n_elems - 1); \ int adjust = HI_IDX * (n_elems - 1); \
int sh = sizeof(r->element[0]) >> 1; \ int sh = sizeof(r->element[0]) >> 1; \
int index = (addr & 0xf) >> sh; \ int index = (addr & 0xf) >> sh; \
\
if (msr_le) { \ if (msr_le) { \
index = n_elems - index - 1; \ index = n_elems - index - 1; \
} \
\
if (needs_byteswap(env)) { \
access(env, addr, swap(r->element[LO_IDX ? index : \ access(env, addr, swap(r->element[LO_IDX ? index : \
(adjust - index)])); \ (adjust - index)])); \
} else { \ } else { \
......
...@@ -196,6 +196,7 @@ typedef struct DisasContext { ...@@ -196,6 +196,7 @@ typedef struct DisasContext {
int access_type; int access_type;
/* Translation flags */ /* Translation flags */
int le_mode; int le_mode;
TCGMemOp default_tcg_memop_mask;
#if defined(TARGET_PPC64) #if defined(TARGET_PPC64)
int sf_mode; int sf_mode;
int has_cfar; int has_cfar;
...@@ -210,6 +211,16 @@ typedef struct DisasContext { ...@@ -210,6 +211,16 @@ typedef struct DisasContext {
uint64_t insns_flags2; uint64_t insns_flags2;
} DisasContext; } DisasContext;
/* Return true iff byteswap is needed in a scalar memop */
static inline bool need_byteswap(const DisasContext *ctx)
{
#if defined(TARGET_WORDS_BIGENDIAN)
return ctx->le_mode;
#else
return !ctx->le_mode;
#endif
}
/* True when active word size < size of target_long. */ /* True when active word size < size of target_long. */
#ifdef TARGET_PPC64 #ifdef TARGET_PPC64
# define NARROW_MODE(C) (!(C)->sf_mode) # define NARROW_MODE(C) (!(C)->sf_mode)
...@@ -2660,29 +2671,20 @@ static inline void gen_qemu_ld8s(DisasContext *ctx, TCGv arg1, TCGv arg2) ...@@ -2660,29 +2671,20 @@ static inline void gen_qemu_ld8s(DisasContext *ctx, TCGv arg1, TCGv arg2)
static inline void gen_qemu_ld16u(DisasContext *ctx, TCGv arg1, TCGv arg2) static inline void gen_qemu_ld16u(DisasContext *ctx, TCGv arg1, TCGv arg2)
{ {
tcg_gen_qemu_ld16u(arg1, arg2, ctx->mem_idx); TCGMemOp op = MO_UW | ctx->default_tcg_memop_mask;
if (unlikely(ctx->le_mode)) { tcg_gen_qemu_ld_tl(arg1, arg2, ctx->mem_idx, op);
tcg_gen_bswap16_tl(arg1, arg1);
}
} }
static inline void gen_qemu_ld16s(DisasContext *ctx, TCGv arg1, TCGv arg2) static inline void gen_qemu_ld16s(DisasContext *ctx, TCGv arg1, TCGv arg2)
{ {
if (unlikely(ctx->le_mode)) { TCGMemOp op = MO_SW | ctx->default_tcg_memop_mask;
tcg_gen_qemu_ld16u(arg1, arg2, ctx->mem_idx); tcg_gen_qemu_ld_tl(arg1, arg2, ctx->mem_idx, op);
tcg_gen_bswap16_tl(arg1, arg1);
tcg_gen_ext16s_tl(arg1, arg1);
} else {
tcg_gen_qemu_ld16s(arg1, arg2, ctx->mem_idx);
}
} }
static inline void gen_qemu_ld32u(DisasContext *ctx, TCGv arg1, TCGv arg2) static inline void gen_qemu_ld32u(DisasContext *ctx, TCGv arg1, TCGv arg2)
{ {
tcg_gen_qemu_ld32u(arg1, arg2, ctx->mem_idx); TCGMemOp op = MO_UL | ctx->default_tcg_memop_mask;
if (unlikely(ctx->le_mode)) { tcg_gen_qemu_ld_tl(arg1, arg2, ctx->mem_idx, op);
tcg_gen_bswap32_tl(arg1, arg1);
}
} }
static void gen_qemu_ld32u_i64(DisasContext *ctx, TCGv_i64 val, TCGv addr) static void gen_qemu_ld32u_i64(DisasContext *ctx, TCGv_i64 val, TCGv addr)
...@@ -2695,12 +2697,8 @@ static void gen_qemu_ld32u_i64(DisasContext *ctx, TCGv_i64 val, TCGv addr) ...@@ -2695,12 +2697,8 @@ static void gen_qemu_ld32u_i64(DisasContext *ctx, TCGv_i64 val, TCGv addr)
static inline void gen_qemu_ld32s(DisasContext *ctx, TCGv arg1, TCGv arg2) static inline void gen_qemu_ld32s(DisasContext *ctx, TCGv arg1, TCGv arg2)
{ {
if (unlikely(ctx->le_mode)) { TCGMemOp op = MO_SL | ctx->default_tcg_memop_mask;
tcg_gen_qemu_ld32u(arg1, arg2, ctx->mem_idx); tcg_gen_qemu_ld_tl(arg1, arg2, ctx->mem_idx, op);
tcg_gen_bswap32_tl(arg1, arg1);
tcg_gen_ext32s_tl(arg1, arg1);
} else
tcg_gen_qemu_ld32s(arg1, arg2, ctx->mem_idx);
} }
static void gen_qemu_ld32s_i64(DisasContext *ctx, TCGv_i64 val, TCGv addr) static void gen_qemu_ld32s_i64(DisasContext *ctx, TCGv_i64 val, TCGv addr)
...@@ -2713,10 +2711,8 @@ static void gen_qemu_ld32s_i64(DisasContext *ctx, TCGv_i64 val, TCGv addr) ...@@ -2713,10 +2711,8 @@ static void gen_qemu_ld32s_i64(DisasContext *ctx, TCGv_i64 val, TCGv addr)
static inline void gen_qemu_ld64(DisasContext *ctx, TCGv_i64 arg1, TCGv arg2) static inline void gen_qemu_ld64(DisasContext *ctx, TCGv_i64 arg1, TCGv arg2)
{ {
tcg_gen_qemu_ld64(arg1, arg2, ctx->mem_idx); TCGMemOp op = MO_Q | ctx->default_tcg_memop_mask;
if (unlikely(ctx->le_mode)) { tcg_gen_qemu_ld_i64(arg1, arg2, ctx->mem_idx, op);
tcg_gen_bswap64_i64(arg1, arg1);
}
} }
static inline void gen_qemu_st8(DisasContext *ctx, TCGv arg1, TCGv arg2) static inline void gen_qemu_st8(DisasContext *ctx, TCGv arg1, TCGv arg2)
...@@ -2726,28 +2722,14 @@ static inline void gen_qemu_st8(DisasContext *ctx, TCGv arg1, TCGv arg2) ...@@ -2726,28 +2722,14 @@ static inline void gen_qemu_st8(DisasContext *ctx, TCGv arg1, TCGv arg2)
static inline void gen_qemu_st16(DisasContext *ctx, TCGv arg1, TCGv arg2) static inline void gen_qemu_st16(DisasContext *ctx, TCGv arg1, TCGv arg2)
{ {
if (unlikely(ctx->le_mode)) { TCGMemOp op = MO_UW | ctx->default_tcg_memop_mask;
TCGv t0 = tcg_temp_new(); tcg_gen_qemu_st_tl(arg1, arg2, ctx->mem_idx, op);
tcg_gen_ext16u_tl(t0, arg1);
tcg_gen_bswap16_tl(t0, t0);
tcg_gen_qemu_st16(t0, arg2, ctx->mem_idx);
tcg_temp_free(t0);
} else {
tcg_gen_qemu_st16(arg1, arg2, ctx->mem_idx);
}
} }
static inline void gen_qemu_st32(DisasContext *ctx, TCGv arg1, TCGv arg2) static inline void gen_qemu_st32(DisasContext *ctx, TCGv arg1, TCGv arg2)
{ {
if (unlikely(ctx->le_mode)) { TCGMemOp op = MO_UL | ctx->default_tcg_memop_mask;
TCGv t0 = tcg_temp_new(); tcg_gen_qemu_st_tl(arg1, arg2, ctx->mem_idx, op);
tcg_gen_ext32u_tl(t0, arg1);
tcg_gen_bswap32_tl(t0, t0);
tcg_gen_qemu_st32(t0, arg2, ctx->mem_idx);
tcg_temp_free(t0);
} else {
tcg_gen_qemu_st32(arg1, arg2, ctx->mem_idx);
}
} }
static void gen_qemu_st32_i64(DisasContext *ctx, TCGv_i64 val, TCGv addr) static void gen_qemu_st32_i64(DisasContext *ctx, TCGv_i64 val, TCGv addr)
...@@ -2760,13 +2742,8 @@ static void gen_qemu_st32_i64(DisasContext *ctx, TCGv_i64 val, TCGv addr) ...@@ -2760,13 +2742,8 @@ static void gen_qemu_st32_i64(DisasContext *ctx, TCGv_i64 val, TCGv addr)
static inline void gen_qemu_st64(DisasContext *ctx, TCGv_i64 arg1, TCGv arg2) static inline void gen_qemu_st64(DisasContext *ctx, TCGv_i64 arg1, TCGv arg2)
{ {
if (unlikely(ctx->le_mode)) { TCGMemOp op = MO_Q | ctx->default_tcg_memop_mask;
TCGv_i64 t0 = tcg_temp_new_i64(); tcg_gen_qemu_st_i64(arg1, arg2, ctx->mem_idx, op);
tcg_gen_bswap64_i64(t0, arg1);
tcg_gen_qemu_st64(t0, arg2, ctx->mem_idx);
tcg_temp_free_i64(t0);
} else
tcg_gen_qemu_st64(arg1, arg2, ctx->mem_idx);
} }
#define GEN_LD(name, ldop, opc, type) \ #define GEN_LD(name, ldop, opc, type) \
...@@ -2910,6 +2887,8 @@ static void gen_lq(DisasContext *ctx) ...@@ -2910,6 +2887,8 @@ static void gen_lq(DisasContext *ctx)
EA = tcg_temp_new(); EA = tcg_temp_new();
gen_addr_imm_index(ctx, EA, 0x0F); gen_addr_imm_index(ctx, EA, 0x0F);
/* We only need to swap high and low halves. gen_qemu_ld64 does necessary
64-bit byteswap already. */
if (unlikely(ctx->le_mode)) { if (unlikely(ctx->le_mode)) {
gen_qemu_ld64(ctx, cpu_gpr[rd+1], EA); gen_qemu_ld64(ctx, cpu_gpr[rd+1], EA);
gen_addr_add(ctx, EA, EA, 8); gen_addr_add(ctx, EA, EA, 8);
...@@ -3028,6 +3007,8 @@ static void gen_std(DisasContext *ctx) ...@@ -3028,6 +3007,8 @@ static void gen_std(DisasContext *ctx)
EA = tcg_temp_new(); EA = tcg_temp_new();
gen_addr_imm_index(ctx, EA, 0x03); gen_addr_imm_index(ctx, EA, 0x03);
/* We only need to swap high and low halves. gen_qemu_st64 does
necessary 64-bit byteswap already. */
if (unlikely(ctx->le_mode)) { if (unlikely(ctx->le_mode)) {
gen_qemu_st64(ctx, cpu_gpr[rs+1], EA); gen_qemu_st64(ctx, cpu_gpr[rs+1], EA);
gen_addr_add(ctx, EA, EA, 8); gen_addr_add(ctx, EA, EA, 8);
...@@ -3057,23 +3038,20 @@ static void gen_std(DisasContext *ctx) ...@@ -3057,23 +3038,20 @@ static void gen_std(DisasContext *ctx)
} }
#endif #endif
/*** Integer load and store with byte reverse ***/ /*** Integer load and store with byte reverse ***/
/* lhbrx */ /* lhbrx */
static inline void gen_qemu_ld16ur(DisasContext *ctx, TCGv arg1, TCGv arg2) static inline void gen_qemu_ld16ur(DisasContext *ctx, TCGv arg1, TCGv arg2)
{ {
tcg_gen_qemu_ld16u(arg1, arg2, ctx->mem_idx); TCGMemOp op = MO_UW | (ctx->default_tcg_memop_mask ^ MO_BSWAP);
if (likely(!ctx->le_mode)) { tcg_gen_qemu_ld_tl(arg1, arg2, ctx->mem_idx, op);
tcg_gen_bswap16_tl(arg1, arg1);
}
} }
GEN_LDX(lhbr, ld16ur, 0x16, 0x18, PPC_INTEGER); GEN_LDX(lhbr, ld16ur, 0x16, 0x18, PPC_INTEGER);
/* lwbrx */ /* lwbrx */
static inline void gen_qemu_ld32ur(DisasContext *ctx, TCGv arg1, TCGv arg2) static inline void gen_qemu_ld32ur(DisasContext *ctx, TCGv arg1, TCGv arg2)
{ {
tcg_gen_qemu_ld32u(arg1, arg2, ctx->mem_idx); TCGMemOp op = MO_UL | (ctx->default_tcg_memop_mask ^ MO_BSWAP);
if (likely(!ctx->le_mode)) { tcg_gen_qemu_ld_tl(arg1, arg2, ctx->mem_idx, op);
tcg_gen_bswap32_tl(arg1, arg1);
}
} }
GEN_LDX(lwbr, ld32ur, 0x16, 0x10, PPC_INTEGER); GEN_LDX(lwbr, ld32ur, 0x16, 0x10, PPC_INTEGER);
...@@ -3081,10 +3059,8 @@ GEN_LDX(lwbr, ld32ur, 0x16, 0x10, PPC_INTEGER); ...@@ -3081,10 +3059,8 @@ GEN_LDX(lwbr, ld32ur, 0x16, 0x10, PPC_INTEGER);
/* ldbrx */ /* ldbrx */
static inline void gen_qemu_ld64ur(DisasContext *ctx, TCGv arg1, TCGv arg2) static inline void gen_qemu_ld64ur(DisasContext *ctx, TCGv arg1, TCGv arg2)
{ {
tcg_gen_qemu_ld64(arg1, arg2, ctx->mem_idx); TCGMemOp op = MO_Q | (ctx->default_tcg_memop_mask ^ MO_BSWAP);
if (likely(!ctx->le_mode)) { tcg_gen_qemu_ld_i64(arg1, arg2, ctx->mem_idx, op);
tcg_gen_bswap64_tl(arg1, arg1);
}
} }
GEN_LDX_E(ldbr, ld64ur, 0x14, 0x10, PPC_NONE, PPC2_DBRX); GEN_LDX_E(ldbr, ld64ur, 0x14, 0x10, PPC_NONE, PPC2_DBRX);
#endif /* TARGET_PPC64 */ #endif /* TARGET_PPC64 */
...@@ -3092,30 +3068,16 @@ GEN_LDX_E(ldbr, ld64ur, 0x14, 0x10, PPC_NONE, PPC2_DBRX); ...@@ -3092,30 +3068,16 @@ GEN_LDX_E(ldbr, ld64ur, 0x14, 0x10, PPC_NONE, PPC2_DBRX);
/* sthbrx */ /* sthbrx */
static inline void gen_qemu_st16r(DisasContext *ctx, TCGv arg1, TCGv arg2) static inline void gen_qemu_st16r(DisasContext *ctx, TCGv arg1, TCGv arg2)
{ {
if (likely(!ctx->le_mode)) { TCGMemOp op = MO_UW | (ctx->default_tcg_memop_mask ^ MO_BSWAP);
TCGv t0 = tcg_temp_new(); tcg_gen_qemu_st_tl(arg1, arg2, ctx->mem_idx, op);
tcg_gen_ext16u_tl(t0, arg1);
tcg_gen_bswap16_tl(t0, t0);
tcg_gen_qemu_st16(t0, arg2, ctx->mem_idx);
tcg_temp_free(t0);
} else {
tcg_gen_qemu_st16(arg1, arg2, ctx->mem_idx);
}
} }
GEN_STX(sthbr, st16r, 0x16, 0x1C, PPC_INTEGER); GEN_STX(sthbr, st16r, 0x16, 0x1C, PPC_INTEGER);
/* stwbrx */ /* stwbrx */
static inline void gen_qemu_st32r(DisasContext *ctx, TCGv arg1, TCGv arg2) static inline void gen_qemu_st32r(DisasContext *ctx, TCGv arg1, TCGv arg2)
{ {
if (likely(!ctx->le_mode)) { TCGMemOp op = MO_UL | (ctx->default_tcg_memop_mask ^ MO_BSWAP);
TCGv t0 = tcg_temp_new(); tcg_gen_qemu_st_tl(arg1, arg2, ctx->mem_idx, op);
tcg_gen_ext32u_tl(t0, arg1);
tcg_gen_bswap32_tl(t0, t0);
tcg_gen_qemu_st32(t0, arg2, ctx->mem_idx);
tcg_temp_free(t0);
} else {
tcg_gen_qemu_st32(arg1, arg2, ctx->mem_idx);
}
} }
GEN_STX(stwbr, st32r, 0x16, 0x14, PPC_INTEGER); GEN_STX(stwbr, st32r, 0x16, 0x14, PPC_INTEGER);
...@@ -3123,14 +3085,8 @@ GEN_STX(stwbr, st32r, 0x16, 0x14, PPC_INTEGER); ...@@ -3123,14 +3085,8 @@ GEN_STX(stwbr, st32r, 0x16, 0x14, PPC_INTEGER);
/* stdbrx */ /* stdbrx */
static inline void gen_qemu_st64r(DisasContext *ctx, TCGv arg1, TCGv arg2) static inline void gen_qemu_st64r(DisasContext *ctx, TCGv arg1, TCGv arg2)
{ {
if (likely(!ctx->le_mode)) { TCGMemOp op = MO_Q | (ctx->default_tcg_memop_mask ^ MO_BSWAP);
TCGv t0 = tcg_temp_new(); tcg_gen_qemu_st_i64(arg1, arg2, ctx->mem_idx, op);
tcg_gen_bswap64_tl(t0, arg1);
tcg_gen_qemu_st64(t0, arg2, ctx->mem_idx);
tcg_temp_free(t0);
} else {
tcg_gen_qemu_st64(arg1, arg2, ctx->mem_idx);
}
} }
GEN_STX_E(stdbr, st64r, 0x14, 0x14, PPC_NONE, PPC2_DBRX); GEN_STX_E(stdbr, st64r, 0x14, 0x14, PPC_NONE, PPC2_DBRX);
#endif /* TARGET_PPC64 */ #endif /* TARGET_PPC64 */
...@@ -3550,7 +3506,9 @@ static void gen_lfdp(DisasContext *ctx) ...@@ -3550,7 +3506,9 @@ static void gen_lfdp(DisasContext *ctx)
} }
gen_set_access_type(ctx, ACCESS_FLOAT); gen_set_access_type(ctx, ACCESS_FLOAT);
EA = tcg_temp_new(); EA = tcg_temp_new();
gen_addr_imm_index(ctx, EA, 0); \ gen_addr_imm_index(ctx, EA, 0);
/* We only need to swap high and low halves. gen_qemu_ld64 does necessary
64-bit byteswap already. */
if (unlikely(ctx->le_mode)) { if (unlikely(ctx->le_mode)) {
gen_qemu_ld64(ctx, cpu_fpr[rD(ctx->opcode) + 1], EA); gen_qemu_ld64(ctx, cpu_fpr[rD(ctx->opcode) + 1], EA);
tcg_gen_addi_tl(EA, EA, 8); tcg_gen_addi_tl(EA, EA, 8);
...@@ -3574,6 +3532,8 @@ static void gen_lfdpx(DisasContext *ctx) ...@@ -3574,6 +3532,8 @@ static void gen_lfdpx(DisasContext *ctx)
gen_set_access_type(ctx, ACCESS_FLOAT); gen_set_access_type(ctx, ACCESS_FLOAT);
EA = tcg_temp_new(); EA = tcg_temp_new();
gen_addr_reg_index(ctx, EA); gen_addr_reg_index(ctx, EA);
/* We only need to swap high and low halves. gen_qemu_ld64 does necessary
64-bit byteswap already. */
if (unlikely(ctx->le_mode)) { if (unlikely(ctx->le_mode)) {
gen_qemu_ld64(ctx, cpu_fpr[rD(ctx->opcode) + 1], EA); gen_qemu_ld64(ctx, cpu_fpr[rD(ctx->opcode) + 1], EA);
tcg_gen_addi_tl(EA, EA, 8); tcg_gen_addi_tl(EA, EA, 8);
...@@ -3722,7 +3682,9 @@ static void gen_stfdp(DisasContext *ctx) ...@@ -3722,7 +3682,9 @@ static void gen_stfdp(DisasContext *ctx)
} }
gen_set_access_type(ctx, ACCESS_FLOAT); gen_set_access_type(ctx, ACCESS_FLOAT);
EA = tcg_temp_new(); EA = tcg_temp_new();
gen_addr_imm_index(ctx, EA, 0); \ gen_addr_imm_index(ctx, EA, 0);
/* We only need to swap high and low halves. gen_qemu_st64 does necessary
64-bit byteswap already. */
if (unlikely(ctx->le_mode)) { if (unlikely(ctx->le_mode)) {
gen_qemu_st64(ctx, cpu_fpr[rD(ctx->opcode) + 1], EA); gen_qemu_st64(ctx, cpu_fpr[rD(ctx->opcode) + 1], EA);
tcg_gen_addi_tl(EA, EA, 8); tcg_gen_addi_tl(EA, EA, 8);
...@@ -3746,6 +3708,8 @@ static void gen_stfdpx(DisasContext *ctx) ...@@ -3746,6 +3708,8 @@ static void gen_stfdpx(DisasContext *ctx)
gen_set_access_type(ctx, ACCESS_FLOAT); gen_set_access_type(ctx, ACCESS_FLOAT);
EA = tcg_temp_new(); EA = tcg_temp_new();
gen_addr_reg_index(ctx, EA); gen_addr_reg_index(ctx, EA);
/* We only need to swap high and low halves. gen_qemu_st64 does necessary
64-bit byteswap already. */
if (unlikely(ctx->le_mode)) { if (unlikely(ctx->le_mode)) {
gen_qemu_st64(ctx, cpu_fpr[rD(ctx->opcode) + 1], EA); gen_qemu_st64(ctx, cpu_fpr[rD(ctx->opcode) + 1], EA);
tcg_gen_addi_tl(EA, EA, 8); tcg_gen_addi_tl(EA, EA, 8);
...@@ -6716,6 +6680,8 @@ static void glue(gen_, name)(DisasContext *ctx) ...@@ -6716,6 +6680,8 @@ static void glue(gen_, name)(DisasContext *ctx)
EA = tcg_temp_new(); \ EA = tcg_temp_new(); \
gen_addr_reg_index(ctx, EA); \ gen_addr_reg_index(ctx, EA); \
tcg_gen_andi_tl(EA, EA, ~0xf); \ tcg_gen_andi_tl(EA, EA, ~0xf); \
/* We only need to swap high and low halves. gen_qemu_ld64 does necessary \
64-bit byteswap already. */ \
if (ctx->le_mode) { \ if (ctx->le_mode) { \
gen_qemu_ld64(ctx, cpu_avrl[rD(ctx->opcode)], EA); \ gen_qemu_ld64(ctx, cpu_avrl[rD(ctx->opcode)], EA); \
tcg_gen_addi_tl(EA, EA, 8); \ tcg_gen_addi_tl(EA, EA, 8); \
...@@ -6740,6 +6706,8 @@ static void gen_st##name(DisasContext *ctx) \ ...@@ -6740,6 +6706,8 @@ static void gen_st##name(DisasContext *ctx) \
EA = tcg_temp_new(); \ EA = tcg_temp_new(); \
gen_addr_reg_index(ctx, EA); \ gen_addr_reg_index(ctx, EA); \
tcg_gen_andi_tl(EA, EA, ~0xf); \ tcg_gen_andi_tl(EA, EA, ~0xf); \
/* We only need to swap high and low halves. gen_qemu_st64 does necessary \
64-bit byteswap already. */ \
if (ctx->le_mode) { \ if (ctx->le_mode) { \
gen_qemu_st64(ctx, cpu_avrl[rD(ctx->opcode)], EA); \ gen_qemu_st64(ctx, cpu_avrl[rD(ctx->opcode)], EA); \
tcg_gen_addi_tl(EA, EA, 8); \ tcg_gen_addi_tl(EA, EA, 8); \
...@@ -11742,6 +11710,7 @@ static inline void gen_intermediate_code_internal(PowerPCCPU *cpu, ...@@ -11742,6 +11710,7 @@ static inline void gen_intermediate_code_internal(PowerPCCPU *cpu,
ctx.insns_flags2 = env->insns_flags2; ctx.insns_flags2 = env->insns_flags2;
ctx.access_type = -1; ctx.access_type = -1;
ctx.le_mode = env->hflags & (1 << MSR_LE) ? 1 : 0; ctx.le_mode = env->hflags & (1 << MSR_LE) ? 1 : 0;
ctx.default_tcg_memop_mask = ctx.le_mode ? MO_LE : MO_BE;
#if defined(TARGET_PPC64) #if defined(TARGET_PPC64)
ctx.sf_mode = msr_is_64bit(env, env->msr); ctx.sf_mode = msr_is_64bit(env, env->msr);
ctx.has_cfar = !!(env->flags & POWERPC_FLAG_CFAR); ctx.has_cfar = !!(env->flags & POWERPC_FLAG_CFAR);
...@@ -11807,7 +11776,7 @@ static inline void gen_intermediate_code_internal(PowerPCCPU *cpu, ...@@ -11807,7 +11776,7 @@ static inline void gen_intermediate_code_internal(PowerPCCPU *cpu,
ctx.nip, ctx.mem_idx, (int)msr_ir); ctx.nip, ctx.mem_idx, (int)msr_ir);
if (num_insns + 1 == max_insns && (tb->cflags & CF_LAST_IO)) if (num_insns + 1 == max_insns && (tb->cflags & CF_LAST_IO))
gen_io_start(); gen_io_start();
if (unlikely(ctx.le_mode)) { if (unlikely(need_byteswap(&ctx))) {
ctx.opcode = bswap32(cpu_ldl_code(env, ctx.nip)); ctx.opcode = bswap32(cpu_ldl_code(env, ctx.nip));
} else { } else {
ctx.opcode = cpu_ldl_code(env, ctx.nip); ctx.opcode = cpu_ldl_code(env, ctx.nip);
......
...@@ -9315,6 +9315,9 @@ static void ppc_cpu_reset(CPUState *s) ...@@ -9315,6 +9315,9 @@ static void ppc_cpu_reset(CPUState *s)
msr |= (target_ulong)1 << MSR_VR; /* Allow altivec usage */ msr |= (target_ulong)1 << MSR_VR; /* Allow altivec usage */
msr |= (target_ulong)1 << MSR_SPE; /* Allow SPE usage */ msr |= (target_ulong)1 << MSR_SPE; /* Allow SPE usage */
msr |= (target_ulong)1 << MSR_PR; msr |= (target_ulong)1 << MSR_PR;
#if !defined(TARGET_WORDS_BIGENDIAN)
msr |= (target_ulong)1 << MSR_LE; /* Little-endian user mode */
#endif
#endif #endif
#if defined(TARGET_PPC64) #if defined(TARGET_PPC64)
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册