ppc/booke: fix crit/mcheck/debug exceptions
Book E does not play games with certain bits of xSRR1 being MSR save bits and others being error status. xSRR1 is the old MSR, period. This was causing things like MSR[CE] to be lost, even in the saved version, as soon as you take an exception. rfci/rfdi/rfmci are fixed to pass the actual xSRR1 register contents, rather than the register number. Put FIXME comments on the hack that is "asrr0/1". The whole point of separate exception levels is so that you can, for example, take a machine check or debug interrupt without corrupting critical-level operations. The right xSRR0/1 set needs to be chosen based on CPU type flags. Signed-off-by: Scott Wood <scottwood@freescale.com> Signed-off-by: Alexander Graf <agraf@suse.de>
This commit is contained in:
		
							parent
							
								
									e99fd8af63
								
							
						
					
					
						commit
						a1bb73849f
					
				@ -84,7 +84,11 @@ static inline void powerpc_excp(PowerPCCPU *cpu, int excp_model, int excp)
 | 
				
			|||||||
                  " => %08x (%02x)\n", env->nip, excp, env->error_code);
 | 
					                  " => %08x (%02x)\n", env->nip, excp, env->error_code);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    /* new srr1 value excluding must-be-zero bits */
 | 
					    /* new srr1 value excluding must-be-zero bits */
 | 
				
			||||||
    msr = env->msr & ~0x783f0000ULL;
 | 
					    if (excp_model == POWERPC_EXCP_BOOKE) {
 | 
				
			||||||
 | 
					        msr = env->msr;
 | 
				
			||||||
 | 
					    } else {
 | 
				
			||||||
 | 
					        msr = env->msr & ~0x783f0000ULL;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    /* new interrupt handler msr */
 | 
					    /* new interrupt handler msr */
 | 
				
			||||||
    new_msr = env->msr & ((target_ulong)1 << MSR_ME);
 | 
					    new_msr = env->msr & ((target_ulong)1 << MSR_ME);
 | 
				
			||||||
@ -145,6 +149,7 @@ static inline void powerpc_excp(PowerPCCPU *cpu, int excp_model, int excp)
 | 
				
			|||||||
            srr1 = SPR_40x_SRR3;
 | 
					            srr1 = SPR_40x_SRR3;
 | 
				
			||||||
            break;
 | 
					            break;
 | 
				
			||||||
        case POWERPC_EXCP_BOOKE:
 | 
					        case POWERPC_EXCP_BOOKE:
 | 
				
			||||||
 | 
					            /* FIXME: choose one or the other based on CPU type */
 | 
				
			||||||
            srr0 = SPR_BOOKE_MCSRR0;
 | 
					            srr0 = SPR_BOOKE_MCSRR0;
 | 
				
			||||||
            srr1 = SPR_BOOKE_MCSRR1;
 | 
					            srr1 = SPR_BOOKE_MCSRR1;
 | 
				
			||||||
            asrr0 = SPR_BOOKE_CSRR0;
 | 
					            asrr0 = SPR_BOOKE_CSRR0;
 | 
				
			||||||
@ -275,6 +280,7 @@ static inline void powerpc_excp(PowerPCCPU *cpu, int excp_model, int excp)
 | 
				
			|||||||
    case POWERPC_EXCP_DEBUG:     /* Debug interrupt                          */
 | 
					    case POWERPC_EXCP_DEBUG:     /* Debug interrupt                          */
 | 
				
			||||||
        switch (excp_model) {
 | 
					        switch (excp_model) {
 | 
				
			||||||
        case POWERPC_EXCP_BOOKE:
 | 
					        case POWERPC_EXCP_BOOKE:
 | 
				
			||||||
 | 
					            /* FIXME: choose one or the other based on CPU type */
 | 
				
			||||||
            srr0 = SPR_BOOKE_DSRR0;
 | 
					            srr0 = SPR_BOOKE_DSRR0;
 | 
				
			||||||
            srr1 = SPR_BOOKE_DSRR1;
 | 
					            srr1 = SPR_BOOKE_DSRR1;
 | 
				
			||||||
            asrr0 = SPR_BOOKE_CSRR0;
 | 
					            asrr0 = SPR_BOOKE_CSRR0;
 | 
				
			||||||
@ -836,8 +842,13 @@ static inline void do_rfi(CPUPPCState *env, target_ulong nip, target_ulong msr,
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
void helper_rfi(CPUPPCState *env)
 | 
					void helper_rfi(CPUPPCState *env)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
    do_rfi(env, env->spr[SPR_SRR0], env->spr[SPR_SRR1],
 | 
					    if (env->excp_model == POWERPC_EXCP_BOOKE) {
 | 
				
			||||||
           ~((target_ulong)0x783F0000), 1);
 | 
					        do_rfi(env, env->spr[SPR_SRR0], env->spr[SPR_SRR1],
 | 
				
			||||||
 | 
					               ~((target_ulong)0), 0);
 | 
				
			||||||
 | 
					    } else {
 | 
				
			||||||
 | 
					        do_rfi(env, env->spr[SPR_SRR0], env->spr[SPR_SRR1],
 | 
				
			||||||
 | 
					               ~((target_ulong)0x783F0000), 1);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#if defined(TARGET_PPC64)
 | 
					#if defined(TARGET_PPC64)
 | 
				
			||||||
@ -864,20 +875,22 @@ void helper_40x_rfci(CPUPPCState *env)
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
void helper_rfci(CPUPPCState *env)
 | 
					void helper_rfci(CPUPPCState *env)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
    do_rfi(env, env->spr[SPR_BOOKE_CSRR0], SPR_BOOKE_CSRR1,
 | 
					    do_rfi(env, env->spr[SPR_BOOKE_CSRR0], env->spr[SPR_BOOKE_CSRR1],
 | 
				
			||||||
           ~((target_ulong)0x3FFF0000), 0);
 | 
					           ~((target_ulong)0), 0);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void helper_rfdi(CPUPPCState *env)
 | 
					void helper_rfdi(CPUPPCState *env)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
    do_rfi(env, env->spr[SPR_BOOKE_DSRR0], SPR_BOOKE_DSRR1,
 | 
					    /* FIXME: choose CSRR1 or DSRR1 based on cpu type */
 | 
				
			||||||
           ~((target_ulong)0x3FFF0000), 0);
 | 
					    do_rfi(env, env->spr[SPR_BOOKE_DSRR0], env->spr[SPR_BOOKE_DSRR1],
 | 
				
			||||||
 | 
					           ~((target_ulong)0), 0);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void helper_rfmci(CPUPPCState *env)
 | 
					void helper_rfmci(CPUPPCState *env)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
    do_rfi(env, env->spr[SPR_BOOKE_MCSRR0], SPR_BOOKE_MCSRR1,
 | 
					    /* FIXME: choose CSRR1 or MCSRR1 based on cpu type */
 | 
				
			||||||
           ~((target_ulong)0x3FFF0000), 0);
 | 
					    do_rfi(env, env->spr[SPR_BOOKE_MCSRR0], env->spr[SPR_BOOKE_MCSRR1],
 | 
				
			||||||
 | 
					           ~((target_ulong)0), 0);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#endif
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
				
			|||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user