target-m68k: add 680x0 divu/divs variants

Update helper to set the throwing location in case of div-by-0.
Cleanup divX.w and add quad word variants of divX.l.

Backports commit 0ccb9c1d8128a020720d5c6abf99a470742a1b94 from qemu
This commit is contained in:
Laurent Vivier 2018-03-01 11:37:55 -05:00 committed by Lioncash
parent f3990e8f87
commit 77b8b2f3b8
No known key found for this signature in database
GPG key ID: 4E3C3CC1031BA9C7
5 changed files with 164 additions and 35 deletions

View file

@ -97,10 +97,6 @@ typedef struct CPUM68KState {
uint32_t macsr; uint32_t macsr;
uint32_t mac_mask; uint32_t mac_mask;
/* Temporary storage for DIV helpers. */
uint32_t div1;
uint32_t div2;
/* MMU status. */ /* MMU status. */
struct { struct {
uint32_t ar; uint32_t ar;

View file

@ -3,8 +3,12 @@ DEF_HELPER_4(uc_tracecode, void, i32, i32, ptr, i64)
DEF_HELPER_1(bitrev, i32, i32) DEF_HELPER_1(bitrev, i32, i32)
DEF_HELPER_1(ff1, i32, i32) DEF_HELPER_1(ff1, i32, i32)
DEF_HELPER_FLAGS_2(sats, TCG_CALL_NO_RWG_SE, i32, i32, i32) DEF_HELPER_FLAGS_2(sats, TCG_CALL_NO_RWG_SE, i32, i32, i32)
DEF_HELPER_2(divu, void, env, i32) DEF_HELPER_3(divuw, void, env, int, i32)
DEF_HELPER_2(divs, void, env, i32) DEF_HELPER_3(divsw, void, env, int, s32)
DEF_HELPER_4(divul, void, env, int, int, i32)
DEF_HELPER_4(divsl, void, env, int, int, s32)
DEF_HELPER_4(divull, void, env, int, int, i32)
DEF_HELPER_4(divsll, void, env, int, int, s32)
DEF_HELPER_3(shl_cc, i32, env, i32, i32) DEF_HELPER_3(shl_cc, i32, env, i32, i32)
DEF_HELPER_3(shr_cc, i32, env, i32, i32) DEF_HELPER_3(shr_cc, i32, env, i32, i32)
DEF_HELPER_3(sar_cc, i32, env, i32, i32) DEF_HELPER_3(sar_cc, i32, env, i32, i32)

View file

@ -157,12 +157,17 @@ bool m68k_cpu_exec_interrupt(CPUState *cs, int interrupt_request)
return false; return false;
} }
static void raise_exception(CPUM68KState *env, int tt) static void raise_exception_ra(CPUM68KState *env, int tt, uintptr_t raddr)
{ {
CPUState *cs = CPU(m68k_env_get_cpu(env)); CPUState *cs = CPU(m68k_env_get_cpu(env));
cs->exception_index = tt; cs->exception_index = tt;
cpu_loop_exit(cs); cpu_loop_exit_restore(cs, raddr);
}
static void raise_exception(CPUM68KState *env, int tt)
{
raise_exception_ra(env, tt, 0);
} }
void HELPER(raise_exception)(CPUM68KState *env, uint32_t tt) void HELPER(raise_exception)(CPUM68KState *env, uint32_t tt)
@ -170,51 +175,179 @@ void HELPER(raise_exception)(CPUM68KState *env, uint32_t tt)
raise_exception(env, tt); raise_exception(env, tt);
} }
void HELPER(divu)(CPUM68KState *env, uint32_t word) void HELPER(divuw)(CPUM68KState *env, int destr, uint32_t den)
{ {
uint32_t num; uint32_t num = env->dregs[destr];
uint32_t den; uint32_t quot, rem;
uint32_t quot;
if (den == 0) {
raise_exception_ra(env, EXCP_DIV0, GETPC());
}
quot = num / den;
rem = num % den;
env->cc_c = 0; /* always cleared, even if overflow */
if (quot > 0xffff) {
env->cc_v = -1;
/* real 68040 keeps N and unset Z on overflow,
* whereas documentation says "undefined"
*/
env->cc_z = 1;
return;
}
env->dregs[destr] = deposit32(quot, 16, 16, rem);
env->cc_z = (int16_t)quot;
env->cc_n = (int16_t)quot;
env->cc_v = 0;
}
void HELPER(divsw)(CPUM68KState *env, int destr, int32_t den)
{
int32_t num = env->dregs[destr];
uint32_t quot, rem;
if (den == 0) {
raise_exception_ra(env, EXCP_DIV0, GETPC());
}
quot = num / den;
rem = num % den;
env->cc_c = 0; /* always cleared, even if overflow */
if (quot != (int16_t)quot) {
env->cc_v = -1;
/* nothing else is modified */
/* real 68040 keeps N and unset Z on overflow,
* whereas documentation says "undefined"
*/
env->cc_z = 1;
return;
}
env->dregs[destr] = deposit32(quot, 16, 16, rem);
env->cc_z = (int16_t)quot;
env->cc_n = (int16_t)quot;
env->cc_v = 0;
}
void HELPER(divul)(CPUM68KState *env, int numr, int regr, uint32_t den)
{
uint32_t num = env->dregs[numr];
uint32_t quot, rem;
if (den == 0) {
raise_exception_ra(env, EXCP_DIV0, GETPC());
}
quot = num / den;
rem = num % den;
env->cc_c = 0;
env->cc_z = quot;
env->cc_n = quot;
env->cc_v = 0;
if (m68k_feature(env, M68K_FEATURE_CF_ISA_A)) {
if (numr == regr) {
env->dregs[numr] = quot;
} else {
env->dregs[regr] = rem;
}
} else {
env->dregs[regr] = rem;
env->dregs[numr] = quot;
}
}
void HELPER(divsl)(CPUM68KState *env, int numr, int regr, int32_t den)
{
int32_t num = env->dregs[numr];
int32_t quot, rem;
if (den == 0) {
raise_exception_ra(env, EXCP_DIV0, GETPC());
}
quot = num / den;
rem = num % den;
env->cc_c = 0;
env->cc_z = quot;
env->cc_n = quot;
env->cc_v = 0;
if (m68k_feature(env, M68K_FEATURE_CF_ISA_A)) {
if (numr == regr) {
env->dregs[numr] = quot;
} else {
env->dregs[regr] = rem;
}
} else {
env->dregs[regr] = rem;
env->dregs[numr] = quot;
}
}
void HELPER(divull)(CPUM68KState *env, int numr, int regr, uint32_t den)
{
uint64_t num = deposit64(env->dregs[numr], 32, 32, env->dregs[regr]);
uint64_t quot;
uint32_t rem; uint32_t rem;
num = env->div1;
den = env->div2;
/* ??? This needs to make sure the throwing location is accurate. */
if (den == 0) { if (den == 0) {
raise_exception(env, EXCP_DIV0); raise_exception_ra(env, EXCP_DIV0, GETPC());
} }
quot = num / den; quot = num / den;
rem = num % den; rem = num % den;
env->cc_v = (word && quot > 0xffff ? -1 : 0); env->cc_c = 0; /* always cleared, even if overflow */
if (quot > 0xffffffffULL) {
env->cc_v = -1;
/* real 68040 keeps N and unset Z on overflow,
* whereas documentation says "undefined"
*/
env->cc_z = 1;
return;
}
env->cc_z = quot; env->cc_z = quot;
env->cc_n = quot; env->cc_n = quot;
env->cc_c = 0; env->cc_v = 0;
env->div1 = quot; /*
env->div2 = rem; * If Dq and Dr are the same, the quotient is returned.
* therefore we set Dq last.
*/
env->dregs[regr] = rem;
env->dregs[numr] = quot;
} }
void HELPER(divs)(CPUM68KState *env, uint32_t word) void HELPER(divsll)(CPUM68KState *env, int numr, int regr, int32_t den)
{ {
int32_t num; int64_t num = deposit64(env->dregs[numr], 32, 32, env->dregs[regr]);
int32_t den; int64_t quot;
int32_t quot;
int32_t rem; int32_t rem;
num = env->div1;
den = env->div2;
if (den == 0) { if (den == 0) {
raise_exception(env, EXCP_DIV0); raise_exception_ra(env, EXCP_DIV0, GETPC());
} }
quot = num / den; quot = num / den;
rem = num % den; rem = num % den;
env->cc_v = (word && quot != (int16_t)quot ? -1 : 0); env->cc_c = 0; /* always cleared, even if overflow */
if (quot != (int32_t)quot) {
env->cc_v = -1;
/* real 68040 keeps N and unset Z on overflow,
* whereas documentation says "undefined"
*/
env->cc_z = 1;
return;
}
env->cc_z = quot; env->cc_z = quot;
env->cc_n = quot; env->cc_n = quot;
env->cc_c = 0; env->cc_v = 0;
env->div1 = quot; /*
env->div2 = rem; * If Dq and Dr are the same, the quotient is returned.
* therefore we set Dq last.
*/
env->dregs[regr] = rem;
env->dregs[numr] = quot;
} }

View file

@ -7,7 +7,5 @@ DEFO32(CC_N, cc_n)
DEFO32(CC_V, cc_v) DEFO32(CC_V, cc_v)
DEFO32(CC_Z, cc_z) DEFO32(CC_Z, cc_z)
DEFO32(CC_X, cc_x) DEFO32(CC_X, cc_x)
DEFO32(DIV1, div1)
DEFO32(DIV2, div2)
DEFO32(MACSR, macsr) DEFO32(MACSR, macsr)
DEFO32(MAC_MASK, mac_mask) DEFO32(MAC_MASK, mac_mask)

View file

@ -919,8 +919,6 @@ struct TCGContext {
TCGv QREG_CC_N; TCGv QREG_CC_N;
TCGv QREG_CC_V; TCGv QREG_CC_V;
TCGv QREG_CC_Z; TCGv QREG_CC_Z;
TCGv QREG_DIV1;
TCGv QREG_DIV2;
TCGv QREG_MACSR; TCGv QREG_MACSR;
TCGv QREG_MAC_MASK; TCGv QREG_MAC_MASK;
TCGv NULL_QREG; TCGv NULL_QREG;