Commit 5210977a8511fc0c4a8a1a68c01fa3b65e29edc0
Committed by
Blue Swirl
1 parent
49e66373
sparc64: trap handling corrections
On Sun, Jul 12, 2009 at 12:09 PM, Blue Swirl<blauwirbel@gmail.com> wrote: > On 7/12/09, Igor Kovalenko <igor.v.kovalenko@gmail.com> wrote: >> Good trap handling is required to process interrupts. >> This patch fixes the following: >> >> - sparc64 has no wim register >> - sparc64 has no psret register, use IE bit of pstate >> extract IE checking code to cpu_interrupts_enabled >> - alternate globals are not available if cpu has GL feature >> in this case bit AG of pstate is constant zero >> - write to pstate must actually write pstate >> even if cpu has GL feature >> >> Also timer interrupt is handled using do_interrupt. > > A bit too much for one patch. Please also remove the code instead of > commenting out. I now excluded timer interrupt related part. To my mind other changes are essentially tied together. > PUT_PSR for Sparc64 needs CC_OP = CC_OP_FLAGS; like Sparc32. Fixed, please find attached the updated version. -- Kind regards, Igor V. Kovalenko
Showing
5 changed files
with
77 additions
and
27 deletions
cpu-exec.c
| ... | ... | @@ -475,7 +475,7 @@ int cpu_exec(CPUState *env1) |
| 475 | 475 | } |
| 476 | 476 | #elif defined(TARGET_SPARC) |
| 477 | 477 | if ((interrupt_request & CPU_INTERRUPT_HARD) && |
| 478 | - (env->psret != 0)) { | |
| 478 | + cpu_interrupts_enabled(env)) { | |
| 479 | 479 | int pil = env->interrupt_index & 15; |
| 480 | 480 | int type = env->interrupt_index & 0xf0; |
| 481 | 481 | |
| ... | ... | @@ -486,7 +486,7 @@ int cpu_exec(CPUState *env1) |
| 486 | 486 | env->exception_index = env->interrupt_index; |
| 487 | 487 | do_interrupt(env); |
| 488 | 488 | env->interrupt_index = 0; |
| 489 | -#if !defined(TARGET_SPARC64) && !defined(CONFIG_USER_ONLY) | |
| 489 | +#if !defined(CONFIG_USER_ONLY) | |
| 490 | 490 | cpu_check_irqs(env); |
| 491 | 491 | #endif |
| 492 | 492 | next_tb = 0; | ... | ... |
target-sparc/cpu.h
| ... | ... | @@ -115,15 +115,18 @@ enum { |
| 115 | 115 | #define TBR_BASE_MASK 0xfffff000 |
| 116 | 116 | |
| 117 | 117 | #if defined(TARGET_SPARC64) |
| 118 | -#define PS_IG (1<<11) | |
| 119 | -#define PS_MG (1<<10) | |
| 118 | +#define PS_TCT (1<<12) /* UA2007, impl.dep. trap on control transfer */ | |
| 119 | +#define PS_IG (1<<11) /* v9, zero on UA2007 */ | |
| 120 | +#define PS_MG (1<<10) /* v9, zero on UA2007 */ | |
| 121 | +#define PS_CLE (1<<9) /* UA2007 */ | |
| 122 | +#define PS_TLE (1<<8) /* UA2007 */ | |
| 120 | 123 | #define PS_RMO (1<<7) |
| 121 | -#define PS_RED (1<<5) | |
| 122 | -#define PS_PEF (1<<4) | |
| 123 | -#define PS_AM (1<<3) | |
| 124 | +#define PS_RED (1<<5) /* v9, zero on UA2007 */ | |
| 125 | +#define PS_PEF (1<<4) /* enable fpu */ | |
| 126 | +#define PS_AM (1<<3) /* address mask */ | |
| 124 | 127 | #define PS_PRIV (1<<2) |
| 125 | 128 | #define PS_IE (1<<1) |
| 126 | -#define PS_AG (1<<0) | |
| 129 | +#define PS_AG (1<<0) /* v9, zero on UA2007 */ | |
| 127 | 130 | |
| 128 | 131 | #define FPRS_FEF (1<<2) |
| 129 | 132 | |
| ... | ... | @@ -291,11 +294,15 @@ typedef struct CPUSPARCState { |
| 291 | 294 | float32 fpr[TARGET_FPREGS]; /* floating point registers */ |
| 292 | 295 | uint32_t cwp; /* index of current register window (extracted |
| 293 | 296 | from PSR) */ |
| 297 | +#if !defined(TARGET_SPARC64) || defined(TARGET_ABI32) | |
| 294 | 298 | uint32_t wim; /* window invalid mask */ |
| 299 | +#endif | |
| 295 | 300 | target_ulong tbr; /* trap base register */ |
| 296 | 301 | int psrs; /* supervisor mode (extracted from PSR) */ |
| 297 | 302 | int psrps; /* previous supervisor mode */ |
| 303 | +#if !defined(TARGET_SPARC64) | |
| 298 | 304 | int psret; /* enable traps */ |
| 305 | +#endif | |
| 299 | 306 | uint32_t psrpil; /* interrupt blocking level */ |
| 300 | 307 | uint32_t pil_in; /* incoming interrupt level bitmap */ |
| 301 | 308 | int psref; /* enable fpu */ |
| ... | ... | @@ -378,12 +385,21 @@ void gen_intermediate_code_init(CPUSPARCState *env); |
| 378 | 385 | /* cpu-exec.c */ |
| 379 | 386 | int cpu_sparc_exec(CPUSPARCState *s); |
| 380 | 387 | |
| 388 | +#if !defined (TARGET_SPARC64) | |
| 381 | 389 | #define GET_PSR(env) (env->version | (env->psr & PSR_ICC) | \ |
| 382 | 390 | (env->psref? PSR_EF : 0) | \ |
| 383 | 391 | (env->psrpil << 8) | \ |
| 384 | 392 | (env->psrs? PSR_S : 0) | \ |
| 385 | 393 | (env->psrps? PSR_PS : 0) | \ |
| 386 | 394 | (env->psret? PSR_ET : 0) | env->cwp) |
| 395 | +#else | |
| 396 | +#define GET_PSR(env) (env->version | (env->psr & PSR_ICC) | \ | |
| 397 | + (env->psref? PSR_EF : 0) | \ | |
| 398 | + (env->psrpil << 8) | \ | |
| 399 | + (env->psrs? PSR_S : 0) | \ | |
| 400 | + (env->psrps? PSR_PS : 0) | \ | |
| 401 | + env->cwp) | |
| 402 | +#endif | |
| 387 | 403 | |
| 388 | 404 | #ifndef NO_CPU_IO_DEFS |
| 389 | 405 | static inline void memcpy32(target_ulong *dst, const target_ulong *src) |
| ... | ... | @@ -425,6 +441,7 @@ static inline int cpu_cwp_dec(CPUSPARCState *env1, int cwp) |
| 425 | 441 | } |
| 426 | 442 | #endif |
| 427 | 443 | |
| 444 | +#if !defined (TARGET_SPARC64) | |
| 428 | 445 | #define PUT_PSR(env, val) do { int _tmp = val; \ |
| 429 | 446 | env->psr = _tmp & PSR_ICC; \ |
| 430 | 447 | env->psref = (_tmp & PSR_EF)? 1 : 0; \ |
| ... | ... | @@ -435,6 +452,17 @@ static inline int cpu_cwp_dec(CPUSPARCState *env1, int cwp) |
| 435 | 452 | cpu_set_cwp(env, _tmp & PSR_CWP); \ |
| 436 | 453 | CC_OP = CC_OP_FLAGS; \ |
| 437 | 454 | } while (0) |
| 455 | +#else | |
| 456 | +#define PUT_PSR(env, val) do { int _tmp = val; \ | |
| 457 | + env->psr = _tmp & PSR_ICC; \ | |
| 458 | + env->psref = (_tmp & PSR_EF)? 1 : 0; \ | |
| 459 | + env->psrpil = (_tmp & PSR_PIL) >> 8; \ | |
| 460 | + env->psrs = (_tmp & PSR_S)? 1 : 0; \ | |
| 461 | + env->psrps = (_tmp & PSR_PS)? 1 : 0; \ | |
| 462 | + cpu_set_cwp(env, _tmp & PSR_CWP); \ | |
| 463 | + CC_OP = CC_OP_FLAGS; \ | |
| 464 | + } while (0) | |
| 465 | +#endif | |
| 438 | 466 | |
| 439 | 467 | #ifdef TARGET_SPARC64 |
| 440 | 468 | #define GET_CCR(env) (((env->xcc >> 20) << 4) | ((env->psr & PSR_ICC) >> 20)) | ... | ... |
target-sparc/exec.h
| ... | ... | @@ -24,10 +24,23 @@ static inline void regs_to_env(void) |
| 24 | 24 | /* op_helper.c */ |
| 25 | 25 | void do_interrupt(CPUState *env); |
| 26 | 26 | |
| 27 | +static inline int cpu_interrupts_enabled(CPUState *env1) | |
| 28 | +{ | |
| 29 | +#if !defined (TARGET_SPARC64) | |
| 30 | + if (env1->psret != 0) | |
| 31 | + return 1; | |
| 32 | +#else | |
| 33 | + if (env1->pstate & PS_IE) | |
| 34 | + return 1; | |
| 35 | +#endif | |
| 36 | + | |
| 37 | + return 0; | |
| 38 | +} | |
| 39 | + | |
| 27 | 40 | static inline int cpu_has_work(CPUState *env1) |
| 28 | 41 | { |
| 29 | 42 | return (env1->interrupt_request & CPU_INTERRUPT_HARD) && |
| 30 | - (env1->psret != 0); | |
| 43 | + cpu_interrupts_enabled(env1); | |
| 31 | 44 | } |
| 32 | 45 | |
| 33 | 46 | ... | ... |
target-sparc/helper.c
| ... | ... | @@ -667,7 +667,9 @@ void cpu_reset(CPUSPARCState *env) |
| 667 | 667 | |
| 668 | 668 | tlb_flush(env, 1); |
| 669 | 669 | env->cwp = 0; |
| 670 | +#ifndef TARGET_SPARC64 | |
| 670 | 671 | env->wim = 1; |
| 672 | +#endif | |
| 671 | 673 | env->regwptr = env->regbase + (env->cwp * 16); |
| 672 | 674 | #if defined(CONFIG_USER_ONLY) |
| 673 | 675 | #ifdef TARGET_SPARC64 |
| ... | ... | @@ -677,7 +679,9 @@ void cpu_reset(CPUSPARCState *env) |
| 677 | 679 | env->asi = 0x82; // Primary no-fault |
| 678 | 680 | #endif |
| 679 | 681 | #else |
| 682 | +#if !defined(TARGET_SPARC64) | |
| 680 | 683 | env->psret = 0; |
| 684 | +#endif | |
| 681 | 685 | env->psrs = 1; |
| 682 | 686 | env->psrps = 1; |
| 683 | 687 | CC_OP = CC_OP_FLAGS; | ... | ... |
target-sparc/op_helper.c
| ... | ... | @@ -3225,8 +3225,14 @@ static inline void change_pstate(uint64_t new_pstate) |
| 3225 | 3225 | uint64_t pstate_regs, new_pstate_regs; |
| 3226 | 3226 | uint64_t *src, *dst; |
| 3227 | 3227 | |
| 3228 | + if (env->def->features & CPU_FEATURE_GL) { | |
| 3229 | + // PS_AG is not implemented in this case | |
| 3230 | + new_pstate &= ~PS_AG; | |
| 3231 | + } | |
| 3232 | + | |
| 3228 | 3233 | pstate_regs = env->pstate & 0xc01; |
| 3229 | 3234 | new_pstate_regs = new_pstate & 0xc01; |
| 3235 | + | |
| 3230 | 3236 | if (new_pstate_regs != pstate_regs) { |
| 3231 | 3237 | // Switch global register bank |
| 3232 | 3238 | src = get_gregset(new_pstate_regs); |
| ... | ... | @@ -3239,8 +3245,7 @@ static inline void change_pstate(uint64_t new_pstate) |
| 3239 | 3245 | |
| 3240 | 3246 | void helper_wrpstate(target_ulong new_state) |
| 3241 | 3247 | { |
| 3242 | - if (!(env->def->features & CPU_FEATURE_GL)) | |
| 3243 | - change_pstate(new_state & 0xf3f); | |
| 3248 | + change_pstate(new_state & 0xf3f); | |
| 3244 | 3249 | } |
| 3245 | 3250 | |
| 3246 | 3251 | void helper_done(void) |
| ... | ... | @@ -3392,23 +3397,23 @@ void do_interrupt(CPUState *env) |
| 3392 | 3397 | env->tsptr->tpc = env->pc; |
| 3393 | 3398 | env->tsptr->tnpc = env->npc; |
| 3394 | 3399 | env->tsptr->tt = intno; |
| 3395 | - if (!(env->def->features & CPU_FEATURE_GL)) { | |
| 3396 | - switch (intno) { | |
| 3397 | - case TT_IVEC: | |
| 3398 | - change_pstate(PS_PEF | PS_PRIV | PS_IG); | |
| 3399 | - break; | |
| 3400 | - case TT_TFAULT: | |
| 3401 | - case TT_TMISS: | |
| 3402 | - case TT_DFAULT: | |
| 3403 | - case TT_DMISS: | |
| 3404 | - case TT_DPROT: | |
| 3405 | - change_pstate(PS_PEF | PS_PRIV | PS_MG); | |
| 3406 | - break; | |
| 3407 | - default: | |
| 3408 | - change_pstate(PS_PEF | PS_PRIV | PS_AG); | |
| 3409 | - break; | |
| 3410 | - } | |
| 3400 | + | |
| 3401 | + switch (intno) { | |
| 3402 | + case TT_IVEC: | |
| 3403 | + change_pstate(PS_PEF | PS_PRIV | PS_IG); | |
| 3404 | + break; | |
| 3405 | + case TT_TFAULT: | |
| 3406 | + case TT_TMISS: | |
| 3407 | + case TT_DFAULT: | |
| 3408 | + case TT_DMISS: | |
| 3409 | + case TT_DPROT: | |
| 3410 | + change_pstate(PS_PEF | PS_PRIV | PS_MG); | |
| 3411 | + break; | |
| 3412 | + default: | |
| 3413 | + change_pstate(PS_PEF | PS_PRIV | PS_AG); | |
| 3414 | + break; | |
| 3411 | 3415 | } |
| 3416 | + | |
| 3412 | 3417 | if (intno == TT_CLRWIN) |
| 3413 | 3418 | cpu_set_cwp(env, cpu_cwp_dec(env, env->cwp - 1)); |
| 3414 | 3419 | else if ((intno & 0x1c0) == TT_SPILL) | ... | ... |