mirror of
https://github.com/qemu/qemu.git
synced 2024-12-13 14:33:31 +08:00
target/ppc: rework AIL logic in interrupt delivery
The AIL logic is becoming unmanageable spread all over powerpc_excp(), and it is slated to get even worse with POWER10 support. Move it all to a new helper function. Reviewed-by: Cédric Le Goater <clg@kaod.org> Tested-by: Cédric Le Goater <clg@kaod.org> Signed-off-by: Nicholas Piggin <npiggin@gmail.com> Message-Id: <20210501072436.145444-2-npiggin@gmail.com> [dwg: Corrected tab indenting] Signed-off-by: David Gibson <david@gibson.dropbear.id.au>
This commit is contained in:
parent
7468e2c842
commit
8b7e6b07a4
@ -1395,7 +1395,8 @@ static target_ulong h_set_mode_resource_addr_trans_mode(PowerPCCPU *cpu,
|
||||
return H_P4;
|
||||
}
|
||||
|
||||
if (mflags == AIL_RESERVED) {
|
||||
if (mflags == 1) {
|
||||
/* AIL=1 is reserved */
|
||||
return H_UNSUPPORTED_FLAG;
|
||||
}
|
||||
|
||||
|
@ -2405,14 +2405,6 @@ enum {
|
||||
HMER_XSCOM_STATUS_MASK = PPC_BITMASK(21, 23),
|
||||
};
|
||||
|
||||
/* Alternate Interrupt Location (AIL) */
|
||||
enum {
|
||||
AIL_NONE = 0,
|
||||
AIL_RESERVED = 1,
|
||||
AIL_0001_8000 = 2,
|
||||
AIL_C000_0000_0000_4000 = 3,
|
||||
};
|
||||
|
||||
/*****************************************************************************/
|
||||
|
||||
#define is_isa300(ctx) (!!(ctx->insns_flags2 & PPC2_ISA300))
|
||||
|
@ -136,25 +136,111 @@ static int powerpc_reset_wakeup(CPUState *cs, CPUPPCState *env, int excp,
|
||||
return POWERPC_EXCP_RESET;
|
||||
}
|
||||
|
||||
static uint64_t ppc_excp_vector_offset(CPUState *cs, int ail)
|
||||
/*
|
||||
* AIL - Alternate Interrupt Location, a mode that allows interrupts to be
|
||||
* taken with the MMU on, and which uses an alternate location (e.g., so the
|
||||
* kernel/hv can map the vectors there with an effective address).
|
||||
*
|
||||
* An interrupt is considered to be taken "with AIL" or "AIL applies" if they
|
||||
* are delivered in this way. AIL requires the LPCR to be set to enable this
|
||||
* mode, and then a number of conditions have to be true for AIL to apply.
|
||||
*
|
||||
* First of all, SRESET, MCE, and HMI are always delivered without AIL, because
|
||||
* they specifically want to be in real mode (e.g., the MCE might be signaling
|
||||
* a SLB multi-hit which requires SLB flush before the MMU can be enabled).
|
||||
*
|
||||
* After that, behaviour depends on the current MSR[IR], MSR[DR], MSR[HV],
|
||||
* whether or not the interrupt changes MSR[HV] from 0 to 1, and the current
|
||||
* radix mode (LPCR[HR]).
|
||||
*
|
||||
* POWER8, POWER9 with LPCR[HR]=0
|
||||
* | LPCR[AIL] | MSR[IR||DR] | MSR[HV] | new MSR[HV] | AIL |
|
||||
* +-----------+-------------+---------+-------------+-----+
|
||||
* | a | 00/01/10 | x | x | 0 |
|
||||
* | a | 11 | 0 | 1 | 0 |
|
||||
* | a | 11 | 1 | 1 | a |
|
||||
* | a | 11 | 0 | 0 | a |
|
||||
* +-------------------------------------------------------+
|
||||
*
|
||||
* POWER9 with LPCR[HR]=1
|
||||
* | LPCR[AIL] | MSR[IR||DR] | MSR[HV] | new MSR[HV] | AIL |
|
||||
* +-----------+-------------+---------+-------------+-----+
|
||||
* | a | 00/01/10 | x | x | 0 |
|
||||
* | a | 11 | x | x | a |
|
||||
* +-------------------------------------------------------+
|
||||
*
|
||||
* The difference with POWER9 being that MSR[HV] 0->1 interrupts can be sent to
|
||||
* the hypervisor in AIL mode if the guest is radix.
|
||||
*/
|
||||
static inline void ppc_excp_apply_ail(PowerPCCPU *cpu, int excp_model, int excp,
|
||||
target_ulong msr,
|
||||
target_ulong *new_msr,
|
||||
target_ulong *vector)
|
||||
{
|
||||
uint64_t offset = 0;
|
||||
#if defined(TARGET_PPC64)
|
||||
CPUPPCState *env = &cpu->env;
|
||||
bool mmu_all_on = ((msr >> MSR_IR) & 1) && ((msr >> MSR_DR) & 1);
|
||||
bool hv_escalation = !(msr & MSR_HVB) && (*new_msr & MSR_HVB);
|
||||
int ail = 0;
|
||||
|
||||
switch (ail) {
|
||||
case AIL_NONE:
|
||||
break;
|
||||
case AIL_0001_8000:
|
||||
offset = 0x18000;
|
||||
break;
|
||||
case AIL_C000_0000_0000_4000:
|
||||
offset = 0xc000000000004000ull;
|
||||
break;
|
||||
default:
|
||||
cpu_abort(cs, "Invalid AIL combination %d\n", ail);
|
||||
break;
|
||||
if (excp == POWERPC_EXCP_MCHECK ||
|
||||
excp == POWERPC_EXCP_RESET ||
|
||||
excp == POWERPC_EXCP_HV_MAINT) {
|
||||
/* SRESET, MCE, HMI never apply AIL */
|
||||
return;
|
||||
}
|
||||
|
||||
return offset;
|
||||
if (excp_model == POWERPC_EXCP_POWER8 ||
|
||||
excp_model == POWERPC_EXCP_POWER9) {
|
||||
if (!mmu_all_on) {
|
||||
/* AIL only works if MSR[IR] and MSR[DR] are both enabled. */
|
||||
return;
|
||||
}
|
||||
if (hv_escalation && !(env->spr[SPR_LPCR] & LPCR_HR)) {
|
||||
/*
|
||||
* AIL does not work if there is a MSR[HV] 0->1 transition and the
|
||||
* partition is in HPT mode. For radix guests, such interrupts are
|
||||
* allowed to be delivered to the hypervisor in ail mode.
|
||||
*/
|
||||
return;
|
||||
}
|
||||
|
||||
ail = (env->spr[SPR_LPCR] & LPCR_AIL) >> LPCR_AIL_SHIFT;
|
||||
if (ail == 0) {
|
||||
return;
|
||||
}
|
||||
if (ail == 1) {
|
||||
/* AIL=1 is reserved, treat it like AIL=0 */
|
||||
return;
|
||||
}
|
||||
} else {
|
||||
/* Other processors do not support AIL */
|
||||
return;
|
||||
}
|
||||
|
||||
/*
|
||||
* AIL applies, so the new MSR gets IR and DR set, and an offset applied
|
||||
* to the new IP.
|
||||
*/
|
||||
*new_msr |= (1 << MSR_IR) | (1 << MSR_DR);
|
||||
|
||||
if (excp != POWERPC_EXCP_SYSCALL_VECTORED) {
|
||||
if (ail == 2) {
|
||||
*vector |= 0x0000000000018000ull;
|
||||
} else if (ail == 3) {
|
||||
*vector |= 0xc000000000004000ull;
|
||||
}
|
||||
} else {
|
||||
/*
|
||||
* scv AIL is a little different. AIL=2 does not change the address,
|
||||
* only the MSR. AIL=3 replaces the 0x17000 base with 0xc...3000.
|
||||
*/
|
||||
if (ail == 3) {
|
||||
*vector &= ~0x0000000000017000ull; /* Un-apply the base offset */
|
||||
*vector |= 0xc000000000003000ull; /* Apply scv's AIL=3 offset */
|
||||
}
|
||||
}
|
||||
#endif
|
||||
}
|
||||
|
||||
static inline void powerpc_set_excp_state(PowerPCCPU *cpu,
|
||||
@ -197,7 +283,7 @@ static inline void powerpc_excp(PowerPCCPU *cpu, int excp_model, int excp)
|
||||
CPUState *cs = CPU(cpu);
|
||||
CPUPPCState *env = &cpu->env;
|
||||
target_ulong msr, new_msr, vector;
|
||||
int srr0, srr1, asrr0, asrr1, lev = -1, ail;
|
||||
int srr0, srr1, asrr0, asrr1, lev = -1;
|
||||
bool lpes0;
|
||||
|
||||
qemu_log_mask(CPU_LOG_INT, "Raise exception at " TARGET_FMT_lx
|
||||
@ -238,25 +324,16 @@ static inline void powerpc_excp(PowerPCCPU *cpu, int excp_model, int excp)
|
||||
*
|
||||
* On anything else, we behave as if LPES0 is 1
|
||||
* (externals don't alter MSR:HV)
|
||||
*
|
||||
* AIL is initialized here but can be cleared by
|
||||
* selected exceptions
|
||||
*/
|
||||
#if defined(TARGET_PPC64)
|
||||
if (excp_model == POWERPC_EXCP_POWER7 ||
|
||||
excp_model == POWERPC_EXCP_POWER8 ||
|
||||
excp_model == POWERPC_EXCP_POWER9) {
|
||||
lpes0 = !!(env->spr[SPR_LPCR] & LPCR_LPES0);
|
||||
if (excp_model != POWERPC_EXCP_POWER7) {
|
||||
ail = (env->spr[SPR_LPCR] & LPCR_AIL) >> LPCR_AIL_SHIFT;
|
||||
} else {
|
||||
ail = 0;
|
||||
}
|
||||
} else
|
||||
#endif /* defined(TARGET_PPC64) */
|
||||
{
|
||||
lpes0 = true;
|
||||
ail = 0;
|
||||
}
|
||||
|
||||
/*
|
||||
@ -315,7 +392,6 @@ static inline void powerpc_excp(PowerPCCPU *cpu, int excp_model, int excp)
|
||||
*/
|
||||
new_msr |= (target_ulong)MSR_HVB;
|
||||
}
|
||||
ail = 0;
|
||||
|
||||
/* machine check exceptions don't have ME set */
|
||||
new_msr &= ~((target_ulong)1 << MSR_ME);
|
||||
@ -519,7 +595,6 @@ static inline void powerpc_excp(PowerPCCPU *cpu, int excp_model, int excp)
|
||||
"exception %d with no HV support\n", excp);
|
||||
}
|
||||
}
|
||||
ail = 0;
|
||||
break;
|
||||
case POWERPC_EXCP_DSEG: /* Data segment exception */
|
||||
case POWERPC_EXCP_ISEG: /* Instruction segment exception */
|
||||
@ -790,24 +865,6 @@ static inline void powerpc_excp(PowerPCCPU *cpu, int excp_model, int excp)
|
||||
}
|
||||
#endif
|
||||
|
||||
/*
|
||||
* AIL only works if MSR[IR] and MSR[DR] are both enabled.
|
||||
*/
|
||||
if (!((msr >> MSR_IR) & 1) || !((msr >> MSR_DR) & 1)) {
|
||||
ail = 0;
|
||||
}
|
||||
|
||||
/*
|
||||
* AIL does not work if there is a MSR[HV] 0->1 transition and the
|
||||
* partition is in HPT mode. For radix guests, such interrupts are
|
||||
* allowed to be delivered to the hypervisor in ail mode.
|
||||
*/
|
||||
if ((new_msr & MSR_HVB) && !(msr & MSR_HVB)) {
|
||||
if (!(env->spr[SPR_LPCR] & LPCR_HR)) {
|
||||
ail = 0;
|
||||
}
|
||||
}
|
||||
|
||||
vector = env->excp_vectors[excp];
|
||||
if (vector == (target_ulong)-1ULL) {
|
||||
cpu_abort(cs, "Raised an exception without defined vector %d\n",
|
||||
@ -848,23 +905,8 @@ static inline void powerpc_excp(PowerPCCPU *cpu, int excp_model, int excp)
|
||||
/* Save MSR */
|
||||
env->spr[srr1] = msr;
|
||||
|
||||
/* Handle AIL */
|
||||
if (ail) {
|
||||
new_msr |= (1 << MSR_IR) | (1 << MSR_DR);
|
||||
vector |= ppc_excp_vector_offset(cs, ail);
|
||||
}
|
||||
|
||||
#if defined(TARGET_PPC64)
|
||||
} else {
|
||||
/* scv AIL is a little different */
|
||||
if (ail) {
|
||||
new_msr |= (1 << MSR_IR) | (1 << MSR_DR);
|
||||
}
|
||||
if (ail == AIL_C000_0000_0000_4000) {
|
||||
vector |= 0xc000000000003000ull;
|
||||
} else {
|
||||
vector |= 0x0000000000017000ull;
|
||||
}
|
||||
vector += lev * 0x20;
|
||||
|
||||
env->lr = env->nip;
|
||||
@ -872,6 +914,9 @@ static inline void powerpc_excp(PowerPCCPU *cpu, int excp_model, int excp)
|
||||
#endif
|
||||
}
|
||||
|
||||
/* This can update new_msr and vector if AIL applies */
|
||||
ppc_excp_apply_ail(cpu, excp_model, excp, msr, &new_msr, &vector);
|
||||
|
||||
powerpc_set_excp_state(cpu, vector, new_msr);
|
||||
}
|
||||
|
||||
|
@ -3456,7 +3456,7 @@ static void init_excp_POWER9(CPUPPCState *env)
|
||||
|
||||
#if !defined(CONFIG_USER_ONLY)
|
||||
env->excp_vectors[POWERPC_EXCP_HVIRT] = 0x00000EA0;
|
||||
env->excp_vectors[POWERPC_EXCP_SYSCALL_VECTORED] = 0x00000000;
|
||||
env->excp_vectors[POWERPC_EXCP_SYSCALL_VECTORED] = 0x00017000;
|
||||
#endif
|
||||
}
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user