hw/arm/smmuv3: Support and advertise nesting

Everything is in place, consolidate parsing of STE cfg and setting
translation stage.

Advertise nesting if stage requested is "nested".

Reviewed-by: Jean-Philippe Brucker <jean-philippe@linaro.org>
Reviewed-by: Eric Auger <eric.auger@redhat.com>
Signed-off-by: Mostafa Saleh <smostafa@google.com>
Reviewed-by: Alex Bennée <alex.bennee@linaro.org>
Message-id: 20240715084519.1189624-18-smostafa@google.com
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
This commit is contained in:
Mostafa Saleh 2024-07-15 08:45:17 +00:00 committed by Peter Maydell
parent f913118593
commit 58377c3632

View File

@ -261,6 +261,9 @@ static void smmuv3_init_regs(SMMUv3State *s)
/* Based on sys property, the stages supported in smmu will be advertised.*/ /* Based on sys property, the stages supported in smmu will be advertised.*/
if (s->stage && !strcmp("2", s->stage)) { if (s->stage && !strcmp("2", s->stage)) {
s->idr[0] = FIELD_DP32(s->idr[0], IDR0, S2P, 1); s->idr[0] = FIELD_DP32(s->idr[0], IDR0, S2P, 1);
} else if (s->stage && !strcmp("nested", s->stage)) {
s->idr[0] = FIELD_DP32(s->idr[0], IDR0, S1P, 1);
s->idr[0] = FIELD_DP32(s->idr[0], IDR0, S2P, 1);
} else { } else {
s->idr[0] = FIELD_DP32(s->idr[0], IDR0, S1P, 1); s->idr[0] = FIELD_DP32(s->idr[0], IDR0, S1P, 1);
} }
@ -425,8 +428,6 @@ static bool s2_pgtable_config_valid(uint8_t sl0, uint8_t t0sz, uint8_t gran)
static int decode_ste_s2_cfg(SMMUTransCfg *cfg, STE *ste) static int decode_ste_s2_cfg(SMMUTransCfg *cfg, STE *ste)
{ {
cfg->stage = SMMU_STAGE_2;
if (STE_S2AA64(ste) == 0x0) { if (STE_S2AA64(ste) == 0x0) {
qemu_log_mask(LOG_UNIMP, qemu_log_mask(LOG_UNIMP,
"SMMUv3 AArch32 tables not supported\n"); "SMMUv3 AArch32 tables not supported\n");
@ -509,6 +510,27 @@ bad_ste:
return -EINVAL; return -EINVAL;
} }
static void decode_ste_config(SMMUTransCfg *cfg, uint32_t config)
{
if (STE_CFG_ABORT(config)) {
cfg->aborted = true;
return;
}
if (STE_CFG_BYPASS(config)) {
cfg->bypassed = true;
return;
}
if (STE_CFG_S1_ENABLED(config)) {
cfg->stage = SMMU_STAGE_1;
}
if (STE_CFG_S2_ENABLED(config)) {
cfg->stage |= SMMU_STAGE_2;
}
}
/* Returns < 0 in case of invalid STE, 0 otherwise */ /* Returns < 0 in case of invalid STE, 0 otherwise */
static int decode_ste(SMMUv3State *s, SMMUTransCfg *cfg, static int decode_ste(SMMUv3State *s, SMMUTransCfg *cfg,
STE *ste, SMMUEventInfo *event) STE *ste, SMMUEventInfo *event)
@ -525,13 +547,9 @@ static int decode_ste(SMMUv3State *s, SMMUTransCfg *cfg,
config = STE_CONFIG(ste); config = STE_CONFIG(ste);
if (STE_CFG_ABORT(config)) { decode_ste_config(cfg, config);
cfg->aborted = true;
return 0;
}
if (STE_CFG_BYPASS(config)) { if (cfg->aborted || cfg->bypassed) {
cfg->bypassed = true;
return 0; return 0;
} }
@ -704,7 +722,6 @@ static int decode_cd(SMMUv3State *s, SMMUTransCfg *cfg,
/* we support only those at the moment */ /* we support only those at the moment */
cfg->aa64 = true; cfg->aa64 = true;
cfg->stage = SMMU_STAGE_1;
cfg->oas = oas2bits(CD_IPS(cd)); cfg->oas = oas2bits(CD_IPS(cd));
cfg->oas = MIN(oas2bits(SMMU_IDR5_OAS), cfg->oas); cfg->oas = MIN(oas2bits(SMMU_IDR5_OAS), cfg->oas);