mirror of
https://github.com/Motorhead1991/qemu.git
synced 2025-08-03 15:53:54 -06:00
target-sparc: Introduce cpu_raise_exception_ra
Several helpers call helper_raise_exception directly, which requires in turn that their callers have performed save_state. The new function allows a TCG return address to be passed in so that we can restore PC + NPC + flags data from that. This fixes a bug in the usage of helper_check_align, whose callers had not been calling save_state. It fixes another bug in which the divide helpers used GETPC at a level other than the direct callee from TCG. This allows the translator to avoid save_state prior to SAVE, RESTORE, and FLUSHW instructions. Tested-by: Mark Cave-Ayland <mark.cave-ayland@ilande.co.uk> Signed-off-by: Richard Henderson <rth@twiddle.net>
This commit is contained in:
parent
808832277a
commit
2f9d35fc40
5 changed files with 82 additions and 96 deletions
|
@ -19,6 +19,7 @@
|
|||
|
||||
#include "qemu/osdep.h"
|
||||
#include "cpu.h"
|
||||
#include "exec/exec-all.h"
|
||||
#include "exec/helper-proto.h"
|
||||
#include "trace.h"
|
||||
|
||||
|
@ -111,13 +112,13 @@ void helper_rett(CPUSPARCState *env)
|
|||
unsigned int cwp;
|
||||
|
||||
if (env->psret == 1) {
|
||||
helper_raise_exception(env, TT_ILL_INSN);
|
||||
cpu_raise_exception_ra(env, TT_ILL_INSN, GETPC());
|
||||
}
|
||||
|
||||
env->psret = 1;
|
||||
cwp = cpu_cwp_inc(env, env->cwp + 1) ;
|
||||
if (env->wim & (1 << cwp)) {
|
||||
helper_raise_exception(env, TT_WIN_UNF);
|
||||
cpu_raise_exception_ra(env, TT_WIN_UNF, GETPC());
|
||||
}
|
||||
cpu_set_cwp(env, cwp);
|
||||
env->psrs = env->psrps;
|
||||
|
@ -131,7 +132,7 @@ void helper_save(CPUSPARCState *env)
|
|||
|
||||
cwp = cpu_cwp_dec(env, env->cwp - 1);
|
||||
if (env->wim & (1 << cwp)) {
|
||||
helper_raise_exception(env, TT_WIN_OVF);
|
||||
cpu_raise_exception_ra(env, TT_WIN_OVF, GETPC());
|
||||
}
|
||||
cpu_set_cwp(env, cwp);
|
||||
}
|
||||
|
@ -142,7 +143,7 @@ void helper_restore(CPUSPARCState *env)
|
|||
|
||||
cwp = cpu_cwp_inc(env, env->cwp + 1);
|
||||
if (env->wim & (1 << cwp)) {
|
||||
helper_raise_exception(env, TT_WIN_UNF);
|
||||
cpu_raise_exception_ra(env, TT_WIN_UNF, GETPC());
|
||||
}
|
||||
cpu_set_cwp(env, cwp);
|
||||
}
|
||||
|
@ -150,7 +151,7 @@ void helper_restore(CPUSPARCState *env)
|
|||
void helper_wrpsr(CPUSPARCState *env, target_ulong new_psr)
|
||||
{
|
||||
if ((new_psr & PSR_CWP) >= env->nwindows) {
|
||||
helper_raise_exception(env, TT_ILL_INSN);
|
||||
cpu_raise_exception_ra(env, TT_ILL_INSN, GETPC());
|
||||
} else {
|
||||
cpu_put_psr(env, new_psr);
|
||||
}
|
||||
|
@ -170,14 +171,14 @@ void helper_save(CPUSPARCState *env)
|
|||
|
||||
cwp = cpu_cwp_dec(env, env->cwp - 1);
|
||||
if (env->cansave == 0) {
|
||||
helper_raise_exception(env, TT_SPILL | (env->otherwin != 0 ?
|
||||
(TT_WOTHER |
|
||||
((env->wstate & 0x38) >> 1)) :
|
||||
((env->wstate & 0x7) << 2)));
|
||||
int tt = TT_SPILL | (env->otherwin != 0
|
||||
? (TT_WOTHER | ((env->wstate & 0x38) >> 1))
|
||||
: ((env->wstate & 0x7) << 2));
|
||||
cpu_raise_exception_ra(env, tt, GETPC());
|
||||
} else {
|
||||
if (env->cleanwin - env->canrestore == 0) {
|
||||
/* XXX Clean windows without trap */
|
||||
helper_raise_exception(env, TT_CLRWIN);
|
||||
cpu_raise_exception_ra(env, TT_CLRWIN, GETPC());
|
||||
} else {
|
||||
env->cansave--;
|
||||
env->canrestore++;
|
||||
|
@ -192,10 +193,10 @@ void helper_restore(CPUSPARCState *env)
|
|||
|
||||
cwp = cpu_cwp_inc(env, env->cwp + 1);
|
||||
if (env->canrestore == 0) {
|
||||
helper_raise_exception(env, TT_FILL | (env->otherwin != 0 ?
|
||||
(TT_WOTHER |
|
||||
((env->wstate & 0x38) >> 1)) :
|
||||
((env->wstate & 0x7) << 2)));
|
||||
int tt = TT_FILL | (env->otherwin != 0
|
||||
? (TT_WOTHER | ((env->wstate & 0x38) >> 1))
|
||||
: ((env->wstate & 0x7) << 2));
|
||||
cpu_raise_exception_ra(env, tt, GETPC());
|
||||
} else {
|
||||
env->cansave++;
|
||||
env->canrestore--;
|
||||
|
@ -206,10 +207,10 @@ void helper_restore(CPUSPARCState *env)
|
|||
void helper_flushw(CPUSPARCState *env)
|
||||
{
|
||||
if (env->cansave != env->nwindows - 2) {
|
||||
helper_raise_exception(env, TT_SPILL | (env->otherwin != 0 ?
|
||||
(TT_WOTHER |
|
||||
((env->wstate & 0x38) >> 1)) :
|
||||
((env->wstate & 0x7) << 2)));
|
||||
int tt = TT_SPILL | (env->otherwin != 0
|
||||
? (TT_WOTHER | ((env->wstate & 0x38) >> 1))
|
||||
: ((env->wstate & 0x7) << 2));
|
||||
cpu_raise_exception_ra(env, tt, GETPC());
|
||||
}
|
||||
}
|
||||
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue