mirror of https://github.com/xqemu/xqemu.git
target/mips: Implement emulation of nanoMIPS ROTX instruction
Added a helper for ROTX based on the pseudocode from the architecture spec. This instraction was not present in previous MIPS instruction sets. Signed-off-by: Yongbok Kim <yongbok.kim@mips.com> Signed-off-by: Aleksandar Markovic <amarkovic@wavecomp.com> Signed-off-by: Stefan Markovic <smarkovic@wavecomp.com> Reviewed-by: Aleksandar Markovic <amarkovic@wavecomp.com> Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
This commit is contained in:
parent
eac5266459
commit
e222f50672
|
@ -40,6 +40,8 @@ DEF_HELPER_FLAGS_1(bitswap, TCG_CALL_NO_RWG_SE, tl, tl)
|
||||||
DEF_HELPER_FLAGS_1(dbitswap, TCG_CALL_NO_RWG_SE, tl, tl)
|
DEF_HELPER_FLAGS_1(dbitswap, TCG_CALL_NO_RWG_SE, tl, tl)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
DEF_HELPER_FLAGS_4(rotx, TCG_CALL_NO_RWG_SE, tl, tl, i32, i32, i32)
|
||||||
|
|
||||||
#ifndef CONFIG_USER_ONLY
|
#ifndef CONFIG_USER_ONLY
|
||||||
/* CP0 helpers */
|
/* CP0 helpers */
|
||||||
DEF_HELPER_1(mfc0_mvpcontrol, tl, env)
|
DEF_HELPER_1(mfc0_mvpcontrol, tl, env)
|
||||||
|
|
|
@ -249,6 +249,100 @@ target_ulong helper_bitswap(target_ulong rt)
|
||||||
return (int32_t)bitswap(rt);
|
return (int32_t)bitswap(rt);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
target_ulong helper_rotx(target_ulong rs, uint32_t shift, uint32_t shiftx,
|
||||||
|
uint32_t stripe)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
uint64_t tmp0 = ((uint64_t)rs) << 32 | ((uint64_t)rs & 0xffffffff);
|
||||||
|
uint64_t tmp1 = tmp0;
|
||||||
|
for (i = 0; i <= 46; i++) {
|
||||||
|
int s;
|
||||||
|
if (i & 0x8) {
|
||||||
|
s = shift;
|
||||||
|
} else {
|
||||||
|
s = shiftx;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (stripe != 0 && !(i & 0x4)) {
|
||||||
|
s = ~s;
|
||||||
|
}
|
||||||
|
if (s & 0x10) {
|
||||||
|
if (tmp0 & (1LL << (i + 16))) {
|
||||||
|
tmp1 |= 1LL << i;
|
||||||
|
} else {
|
||||||
|
tmp1 &= ~(1LL << i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
uint64_t tmp2 = tmp1;
|
||||||
|
for (i = 0; i <= 38; i++) {
|
||||||
|
int s;
|
||||||
|
if (i & 0x4) {
|
||||||
|
s = shift;
|
||||||
|
} else {
|
||||||
|
s = shiftx;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (s & 0x8) {
|
||||||
|
if (tmp1 & (1LL << (i + 8))) {
|
||||||
|
tmp2 |= 1LL << i;
|
||||||
|
} else {
|
||||||
|
tmp2 &= ~(1LL << i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
uint64_t tmp3 = tmp2;
|
||||||
|
for (i = 0; i <= 34; i++) {
|
||||||
|
int s;
|
||||||
|
if (i & 0x2) {
|
||||||
|
s = shift;
|
||||||
|
} else {
|
||||||
|
s = shiftx;
|
||||||
|
}
|
||||||
|
if (s & 0x4) {
|
||||||
|
if (tmp2 & (1LL << (i + 4))) {
|
||||||
|
tmp3 |= 1LL << i;
|
||||||
|
} else {
|
||||||
|
tmp3 &= ~(1LL << i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
uint64_t tmp4 = tmp3;
|
||||||
|
for (i = 0; i <= 32; i++) {
|
||||||
|
int s;
|
||||||
|
if (i & 0x1) {
|
||||||
|
s = shift;
|
||||||
|
} else {
|
||||||
|
s = shiftx;
|
||||||
|
}
|
||||||
|
if (s & 0x2) {
|
||||||
|
if (tmp3 & (1LL << (i + 2))) {
|
||||||
|
tmp4 |= 1LL << i;
|
||||||
|
} else {
|
||||||
|
tmp4 &= ~(1LL << i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
uint64_t tmp5 = tmp4;
|
||||||
|
for (i = 0; i <= 31; i++) {
|
||||||
|
int s;
|
||||||
|
s = shift;
|
||||||
|
if (s & 0x1) {
|
||||||
|
if (tmp4 & (1LL << (i + 1))) {
|
||||||
|
tmp5 |= 1LL << i;
|
||||||
|
} else {
|
||||||
|
tmp5 &= ~(1LL << i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (int64_t)(int32_t)(uint32_t)tmp5;
|
||||||
|
}
|
||||||
|
|
||||||
#ifndef CONFIG_USER_ONLY
|
#ifndef CONFIG_USER_ONLY
|
||||||
|
|
||||||
static inline hwaddr do_translate_address(CPUMIPSState *env,
|
static inline hwaddr do_translate_address(CPUMIPSState *env,
|
||||||
|
|
|
@ -17813,6 +17813,21 @@ static int decode_nanomips_32_48_opc(CPUMIPSState *env, DisasContext *ctx)
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case NM_P_ROTX:
|
case NM_P_ROTX:
|
||||||
|
if (rt != 0) {
|
||||||
|
TCGv t0 = tcg_temp_new();
|
||||||
|
TCGv_i32 shift = tcg_const_i32(extract32(ctx->opcode, 0, 5));
|
||||||
|
TCGv_i32 shiftx = tcg_const_i32(extract32(ctx->opcode, 7, 4)
|
||||||
|
<< 1);
|
||||||
|
TCGv_i32 stripe = tcg_const_i32(extract32(ctx->opcode, 6, 1));
|
||||||
|
|
||||||
|
gen_load_gpr(t0, rs);
|
||||||
|
gen_helper_rotx(cpu_gpr[rt], t0, shift, shiftx, stripe);
|
||||||
|
tcg_temp_free(t0);
|
||||||
|
|
||||||
|
tcg_temp_free_i32(shift);
|
||||||
|
tcg_temp_free_i32(shiftx);
|
||||||
|
tcg_temp_free_i32(stripe);
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case NM_P_INS:
|
case NM_P_INS:
|
||||||
switch (((ctx->opcode >> 10) & 2) |
|
switch (((ctx->opcode >> 10) & 2) |
|
||||||
|
|
Loading…
Reference in New Issue