Re: [PATCH v9 4/5] riscv: Introduce satp mode hw capabilities
On Thu, Feb 2, 2023 at 9:01 PM Alexandre Ghiti wrote: > Hi Frank, > > On Wed, Feb 1, 2023 at 4:49 PM Frank Chang wrote: > > > > On Tue, Jan 31, 2023 at 10:36 PM Alexandre Ghiti > wrote: > >> > >> Currently, the max satp mode is set with the only constraint that it > must be > >> implemented in QEMU, i.e. set in valid_vm_1_10_[32|64]. > >> > >> But we actually need to add another level of constraint: what the hw is > >> actually capable of, because currently, a linux booting on a sifive-u54 > >> boots in sv57 mode which is incompatible with the cpu's sv39 max > >> capability. > >> > >> So add a new bitmap to RISCVSATPMap which contains this capability and > >> initialize it in every XXX_cpu_init. > >> > >> Finally: > >> - valid_vm_1_10_[32|64] constrains which satp mode the CPU can use > >> - the CPU hw capabilities constrains what the user may select > >> - the user's selection then constrains what's available to the guest > >> OS. > >> > >> Signed-off-by: Alexandre Ghiti > >> Reviewed-by: Andrew Jones > >> --- > >> target/riscv/cpu.c | 79 +++--- > >> target/riscv/cpu.h | 8 +++-- > >> 2 files changed, 60 insertions(+), 27 deletions(-) > >> > >> diff --git a/target/riscv/cpu.c b/target/riscv/cpu.c > >> index 3a7a1746aa..6dd76355ec 100644 > >> --- a/target/riscv/cpu.c > >> +++ b/target/riscv/cpu.c > >> @@ -292,26 +292,36 @@ const char *satp_mode_str(uint8_t satp_mode, bool > is_32_bit) > >> g_assert_not_reached(); > >> } > >> > >> -/* Sets the satp mode to the max supported */ > >> -static void set_satp_mode_default_map(RISCVCPU *cpu) > >> +static void set_satp_mode_max_supported(RISCVCPU *cpu, > >> +uint8_t satp_mode) > >> { > >> bool rv32 = riscv_cpu_mxl(>env) == MXL_RV32; > >> +const bool *valid_vm = rv32 ? valid_vm_1_10_32 : valid_vm_1_10_64; > > > > > > This will break user-mode QEMU. > > valid_vm_1_10_32 and valid_vm_1_10_64 are defined in !CONFIG_USER_ONLY > section. > > This issue also exists in patch 3. > > You have to move valid_vm_1_10_32 and valid_vm_1_10_64 out from > !CONFIG_USER_ONLY. > > Indeed, good catch, thanks! Rather than moving valid_vm_1_10_[32|64], > I'm going to try to surround all the satp handling inside #ifdef > CONFIG_USER_ONLY. But if it's too cumbersome, I'll do as you suggest. > Hi Alex, I think surrounding all the satp handling inside #ifdef is okay, since satp is not been used in user-mode QEMU. Regards, Frank Chang > > Thanks again, > > Alex > > > > > Regards, > > Frank Chang > > > >> > >> -if (riscv_feature(>env, RISCV_FEATURE_MMU)) { > >> -cpu->cfg.satp_mode.map |= > >> -(1 << satp_mode_from_str(rv32 ? "sv32" : > "sv57")); > >> -} else { > >> -cpu->cfg.satp_mode.map |= (1 << satp_mode_from_str("mbare")); > >> +for (int i = 0; i <= satp_mode; ++i) { > >> +if (valid_vm[i]) { > >> +cpu->cfg.satp_mode.supported |= (1 << i); > >> +} > >> } > >> } > >> > >> +/* Set the satp mode to the max supported */ > >> +static void set_satp_mode_default_map(RISCVCPU *cpu) > >> +{ > >> +cpu->cfg.satp_mode.map = cpu->cfg.satp_mode.supported; > >> +} > >> + > >> static void riscv_any_cpu_init(Object *obj) > >> { > >> CPURISCVState *env = _CPU(obj)->env; > >> +RISCVCPU *cpu = RISCV_CPU(obj); > >> + > >> #if defined(TARGET_RISCV32) > >> set_misa(env, MXL_RV32, RVI | RVM | RVA | RVF | RVD | RVC | RVU); > >> +set_satp_mode_max_supported(cpu, VM_1_10_SV32); > >> #elif defined(TARGET_RISCV64) > >> set_misa(env, MXL_RV64, RVI | RVM | RVA | RVF | RVD | RVC | RVU); > >> +set_satp_mode_max_supported(cpu, VM_1_10_SV57); > >> #endif > >> set_priv_version(env, PRIV_VERSION_1_12_0); > >> register_cpu_props(obj); > >> @@ -321,18 +331,24 @@ static void riscv_any_cpu_init(Object *obj) > >> static void rv64_base_cpu_init(Object *obj) > >> { > >> CPURISCVState *env = _CPU(obj)->env; > >> +RISCVCPU *cpu = RISCV_CPU(obj); > >> + > >> /* We set this in the realise function */ > >> set_misa(env, MXL_RV64, 0); > >> register_cpu_props(obj); > >> /* Set latest version of privileged specification */ > >> set_priv_version(env, PRIV_VERSION_1_12_0); > >> +set_satp_mode_max_supported(cpu, VM_1_10_SV57); > >> } > >> > >> static void rv64_sifive_u_cpu_init(Object *obj) > >> { > >> CPURISCVState *env = _CPU(obj)->env; > >> +RISCVCPU *cpu = RISCV_CPU(obj); > >> + > >> set_misa(env, MXL_RV64, RVI | RVM | RVA | RVF | RVD | RVC | RVS | > RVU); > >> set_priv_version(env, PRIV_VERSION_1_10_0); > >> +set_satp_mode_max_supported(cpu, VM_1_10_SV39); > >> } > >> > >> static void rv64_sifive_e_cpu_init(Object *obj) > >> @@ -343,6 +359,7 @@ static void rv64_sifive_e_cpu_init(Object *obj) > >> set_misa(env, MXL_RV64, RVI | RVM | RVA | RVC | RVU); > >> set_priv_version(env, PRIV_VERSION_1_10_0); > >> cpu->cfg.mmu =
Re: [PATCH v9 4/5] riscv: Introduce satp mode hw capabilities
Hi Frank, On Wed, Feb 1, 2023 at 4:49 PM Frank Chang wrote: > > On Tue, Jan 31, 2023 at 10:36 PM Alexandre Ghiti > wrote: >> >> Currently, the max satp mode is set with the only constraint that it must be >> implemented in QEMU, i.e. set in valid_vm_1_10_[32|64]. >> >> But we actually need to add another level of constraint: what the hw is >> actually capable of, because currently, a linux booting on a sifive-u54 >> boots in sv57 mode which is incompatible with the cpu's sv39 max >> capability. >> >> So add a new bitmap to RISCVSATPMap which contains this capability and >> initialize it in every XXX_cpu_init. >> >> Finally: >> - valid_vm_1_10_[32|64] constrains which satp mode the CPU can use >> - the CPU hw capabilities constrains what the user may select >> - the user's selection then constrains what's available to the guest >> OS. >> >> Signed-off-by: Alexandre Ghiti >> Reviewed-by: Andrew Jones >> --- >> target/riscv/cpu.c | 79 +++--- >> target/riscv/cpu.h | 8 +++-- >> 2 files changed, 60 insertions(+), 27 deletions(-) >> >> diff --git a/target/riscv/cpu.c b/target/riscv/cpu.c >> index 3a7a1746aa..6dd76355ec 100644 >> --- a/target/riscv/cpu.c >> +++ b/target/riscv/cpu.c >> @@ -292,26 +292,36 @@ const char *satp_mode_str(uint8_t satp_mode, bool >> is_32_bit) >> g_assert_not_reached(); >> } >> >> -/* Sets the satp mode to the max supported */ >> -static void set_satp_mode_default_map(RISCVCPU *cpu) >> +static void set_satp_mode_max_supported(RISCVCPU *cpu, >> +uint8_t satp_mode) >> { >> bool rv32 = riscv_cpu_mxl(>env) == MXL_RV32; >> +const bool *valid_vm = rv32 ? valid_vm_1_10_32 : valid_vm_1_10_64; > > > This will break user-mode QEMU. > valid_vm_1_10_32 and valid_vm_1_10_64 are defined in !CONFIG_USER_ONLY > section. > This issue also exists in patch 3. > You have to move valid_vm_1_10_32 and valid_vm_1_10_64 out from > !CONFIG_USER_ONLY. Indeed, good catch, thanks! Rather than moving valid_vm_1_10_[32|64], I'm going to try to surround all the satp handling inside #ifdef CONFIG_USER_ONLY. But if it's too cumbersome, I'll do as you suggest. Thanks again, Alex > > Regards, > Frank Chang > >> >> -if (riscv_feature(>env, RISCV_FEATURE_MMU)) { >> -cpu->cfg.satp_mode.map |= >> -(1 << satp_mode_from_str(rv32 ? "sv32" : "sv57")); >> -} else { >> -cpu->cfg.satp_mode.map |= (1 << satp_mode_from_str("mbare")); >> +for (int i = 0; i <= satp_mode; ++i) { >> +if (valid_vm[i]) { >> +cpu->cfg.satp_mode.supported |= (1 << i); >> +} >> } >> } >> >> +/* Set the satp mode to the max supported */ >> +static void set_satp_mode_default_map(RISCVCPU *cpu) >> +{ >> +cpu->cfg.satp_mode.map = cpu->cfg.satp_mode.supported; >> +} >> + >> static void riscv_any_cpu_init(Object *obj) >> { >> CPURISCVState *env = _CPU(obj)->env; >> +RISCVCPU *cpu = RISCV_CPU(obj); >> + >> #if defined(TARGET_RISCV32) >> set_misa(env, MXL_RV32, RVI | RVM | RVA | RVF | RVD | RVC | RVU); >> +set_satp_mode_max_supported(cpu, VM_1_10_SV32); >> #elif defined(TARGET_RISCV64) >> set_misa(env, MXL_RV64, RVI | RVM | RVA | RVF | RVD | RVC | RVU); >> +set_satp_mode_max_supported(cpu, VM_1_10_SV57); >> #endif >> set_priv_version(env, PRIV_VERSION_1_12_0); >> register_cpu_props(obj); >> @@ -321,18 +331,24 @@ static void riscv_any_cpu_init(Object *obj) >> static void rv64_base_cpu_init(Object *obj) >> { >> CPURISCVState *env = _CPU(obj)->env; >> +RISCVCPU *cpu = RISCV_CPU(obj); >> + >> /* We set this in the realise function */ >> set_misa(env, MXL_RV64, 0); >> register_cpu_props(obj); >> /* Set latest version of privileged specification */ >> set_priv_version(env, PRIV_VERSION_1_12_0); >> +set_satp_mode_max_supported(cpu, VM_1_10_SV57); >> } >> >> static void rv64_sifive_u_cpu_init(Object *obj) >> { >> CPURISCVState *env = _CPU(obj)->env; >> +RISCVCPU *cpu = RISCV_CPU(obj); >> + >> set_misa(env, MXL_RV64, RVI | RVM | RVA | RVF | RVD | RVC | RVS | RVU); >> set_priv_version(env, PRIV_VERSION_1_10_0); >> +set_satp_mode_max_supported(cpu, VM_1_10_SV39); >> } >> >> static void rv64_sifive_e_cpu_init(Object *obj) >> @@ -343,6 +359,7 @@ static void rv64_sifive_e_cpu_init(Object *obj) >> set_misa(env, MXL_RV64, RVI | RVM | RVA | RVC | RVU); >> set_priv_version(env, PRIV_VERSION_1_10_0); >> cpu->cfg.mmu = false; >> +set_satp_mode_max_supported(cpu, VM_1_10_MBARE); >> } >> >> static void rv128_base_cpu_init(Object *obj) >> @@ -354,28 +371,36 @@ static void rv128_base_cpu_init(Object *obj) >> exit(EXIT_FAILURE); >> } >> CPURISCVState *env = _CPU(obj)->env; >> +RISCVCPU *cpu = RISCV_CPU(obj); >> /* We set this in the realise function */ >> set_misa(env, MXL_RV128, 0); >> register_cpu_props(obj); >> /* Set
Re: [PATCH v9 4/5] riscv: Introduce satp mode hw capabilities
On Tue, Jan 31, 2023 at 10:36 PM Alexandre Ghiti wrote: > Currently, the max satp mode is set with the only constraint that it must > be > implemented in QEMU, i.e. set in valid_vm_1_10_[32|64]. > > But we actually need to add another level of constraint: what the hw is > actually capable of, because currently, a linux booting on a sifive-u54 > boots in sv57 mode which is incompatible with the cpu's sv39 max > capability. > > So add a new bitmap to RISCVSATPMap which contains this capability and > initialize it in every XXX_cpu_init. > > Finally: > - valid_vm_1_10_[32|64] constrains which satp mode the CPU can use > - the CPU hw capabilities constrains what the user may select > - the user's selection then constrains what's available to the guest > OS. > > Signed-off-by: Alexandre Ghiti > Reviewed-by: Andrew Jones > --- > target/riscv/cpu.c | 79 +++--- > target/riscv/cpu.h | 8 +++-- > 2 files changed, 60 insertions(+), 27 deletions(-) > > diff --git a/target/riscv/cpu.c b/target/riscv/cpu.c > index 3a7a1746aa..6dd76355ec 100644 > --- a/target/riscv/cpu.c > +++ b/target/riscv/cpu.c > @@ -292,26 +292,36 @@ const char *satp_mode_str(uint8_t satp_mode, bool > is_32_bit) > g_assert_not_reached(); > } > > -/* Sets the satp mode to the max supported */ > -static void set_satp_mode_default_map(RISCVCPU *cpu) > +static void set_satp_mode_max_supported(RISCVCPU *cpu, > +uint8_t satp_mode) > { > bool rv32 = riscv_cpu_mxl(>env) == MXL_RV32; > +const bool *valid_vm = rv32 ? valid_vm_1_10_32 : valid_vm_1_10_64; > This will break user-mode QEMU. valid_vm_1_10_32 and valid_vm_1_10_64 are defined in !CONFIG_USER_ONLY section. This issue also exists in patch 3. You have to move valid_vm_1_10_32 and valid_vm_1_10_64 out from !CONFIG_USER_ONLY. Regards, Frank Chang > -if (riscv_feature(>env, RISCV_FEATURE_MMU)) { > -cpu->cfg.satp_mode.map |= > -(1 << satp_mode_from_str(rv32 ? "sv32" : "sv57")); > -} else { > -cpu->cfg.satp_mode.map |= (1 << satp_mode_from_str("mbare")); > +for (int i = 0; i <= satp_mode; ++i) { > +if (valid_vm[i]) { > +cpu->cfg.satp_mode.supported |= (1 << i); > +} > } > } > > +/* Set the satp mode to the max supported */ > +static void set_satp_mode_default_map(RISCVCPU *cpu) > +{ > +cpu->cfg.satp_mode.map = cpu->cfg.satp_mode.supported; > +} > + > static void riscv_any_cpu_init(Object *obj) > { > CPURISCVState *env = _CPU(obj)->env; > +RISCVCPU *cpu = RISCV_CPU(obj); > + > #if defined(TARGET_RISCV32) > set_misa(env, MXL_RV32, RVI | RVM | RVA | RVF | RVD | RVC | RVU); > +set_satp_mode_max_supported(cpu, VM_1_10_SV32); > #elif defined(TARGET_RISCV64) > set_misa(env, MXL_RV64, RVI | RVM | RVA | RVF | RVD | RVC | RVU); > +set_satp_mode_max_supported(cpu, VM_1_10_SV57); > #endif > set_priv_version(env, PRIV_VERSION_1_12_0); > register_cpu_props(obj); > @@ -321,18 +331,24 @@ static void riscv_any_cpu_init(Object *obj) > static void rv64_base_cpu_init(Object *obj) > { > CPURISCVState *env = _CPU(obj)->env; > +RISCVCPU *cpu = RISCV_CPU(obj); > + > /* We set this in the realise function */ > set_misa(env, MXL_RV64, 0); > register_cpu_props(obj); > /* Set latest version of privileged specification */ > set_priv_version(env, PRIV_VERSION_1_12_0); > +set_satp_mode_max_supported(cpu, VM_1_10_SV57); > } > > static void rv64_sifive_u_cpu_init(Object *obj) > { > CPURISCVState *env = _CPU(obj)->env; > +RISCVCPU *cpu = RISCV_CPU(obj); > + > set_misa(env, MXL_RV64, RVI | RVM | RVA | RVF | RVD | RVC | RVS | > RVU); > set_priv_version(env, PRIV_VERSION_1_10_0); > +set_satp_mode_max_supported(cpu, VM_1_10_SV39); > } > > static void rv64_sifive_e_cpu_init(Object *obj) > @@ -343,6 +359,7 @@ static void rv64_sifive_e_cpu_init(Object *obj) > set_misa(env, MXL_RV64, RVI | RVM | RVA | RVC | RVU); > set_priv_version(env, PRIV_VERSION_1_10_0); > cpu->cfg.mmu = false; > +set_satp_mode_max_supported(cpu, VM_1_10_MBARE); > } > > static void rv128_base_cpu_init(Object *obj) > @@ -354,28 +371,36 @@ static void rv128_base_cpu_init(Object *obj) > exit(EXIT_FAILURE); > } > CPURISCVState *env = _CPU(obj)->env; > +RISCVCPU *cpu = RISCV_CPU(obj); > /* We set this in the realise function */ > set_misa(env, MXL_RV128, 0); > register_cpu_props(obj); > /* Set latest version of privileged specification */ > set_priv_version(env, PRIV_VERSION_1_12_0); > +set_satp_mode_max_supported(cpu, VM_1_10_SV57); > } > #else > static void rv32_base_cpu_init(Object *obj) > { > CPURISCVState *env = _CPU(obj)->env; > +RISCVCPU *cpu = RISCV_CPU(obj); > + > /* We set this in the realise function */ > set_misa(env, MXL_RV32, 0); > register_cpu_props(obj); > /*
Re: [PATCH v9 4/5] riscv: Introduce satp mode hw capabilities
On Tue, Jan 31, 2023 at 10:41 PM Alexandre Ghiti wrote: > > Currently, the max satp mode is set with the only constraint that it must be > implemented in QEMU, i.e. set in valid_vm_1_10_[32|64]. > > But we actually need to add another level of constraint: what the hw is > actually capable of, because currently, a linux booting on a sifive-u54 > boots in sv57 mode which is incompatible with the cpu's sv39 max > capability. > > So add a new bitmap to RISCVSATPMap which contains this capability and > initialize it in every XXX_cpu_init. > > Finally: > - valid_vm_1_10_[32|64] constrains which satp mode the CPU can use > - the CPU hw capabilities constrains what the user may select > - the user's selection then constrains what's available to the guest > OS. > > Signed-off-by: Alexandre Ghiti > Reviewed-by: Andrew Jones > --- > target/riscv/cpu.c | 79 +++--- > target/riscv/cpu.h | 8 +++-- > 2 files changed, 60 insertions(+), 27 deletions(-) > > diff --git a/target/riscv/cpu.c b/target/riscv/cpu.c > index 3a7a1746aa..6dd76355ec 100644 > --- a/target/riscv/cpu.c > +++ b/target/riscv/cpu.c > @@ -292,26 +292,36 @@ const char *satp_mode_str(uint8_t satp_mode, bool > is_32_bit) > g_assert_not_reached(); > } > > -/* Sets the satp mode to the max supported */ > -static void set_satp_mode_default_map(RISCVCPU *cpu) > +static void set_satp_mode_max_supported(RISCVCPU *cpu, > +uint8_t satp_mode) > { > bool rv32 = riscv_cpu_mxl(>env) == MXL_RV32; > +const bool *valid_vm = rv32 ? valid_vm_1_10_32 : valid_vm_1_10_64; > > -if (riscv_feature(>env, RISCV_FEATURE_MMU)) { > -cpu->cfg.satp_mode.map |= > -(1 << satp_mode_from_str(rv32 ? "sv32" : "sv57")); > -} else { > -cpu->cfg.satp_mode.map |= (1 << satp_mode_from_str("mbare")); > +for (int i = 0; i <= satp_mode; ++i) { > +if (valid_vm[i]) { > +cpu->cfg.satp_mode.supported |= (1 << i); > +} > } > } > > +/* Set the satp mode to the max supported */ > +static void set_satp_mode_default_map(RISCVCPU *cpu) > +{ > +cpu->cfg.satp_mode.map = cpu->cfg.satp_mode.supported; > +} > + > static void riscv_any_cpu_init(Object *obj) > { > CPURISCVState *env = _CPU(obj)->env; > +RISCVCPU *cpu = RISCV_CPU(obj); > + > #if defined(TARGET_RISCV32) > set_misa(env, MXL_RV32, RVI | RVM | RVA | RVF | RVD | RVC | RVU); > +set_satp_mode_max_supported(cpu, VM_1_10_SV32); > #elif defined(TARGET_RISCV64) > set_misa(env, MXL_RV64, RVI | RVM | RVA | RVF | RVD | RVC | RVU); > +set_satp_mode_max_supported(cpu, VM_1_10_SV57); > #endif > set_priv_version(env, PRIV_VERSION_1_12_0); > register_cpu_props(obj); > @@ -321,18 +331,24 @@ static void riscv_any_cpu_init(Object *obj) > static void rv64_base_cpu_init(Object *obj) > { > CPURISCVState *env = _CPU(obj)->env; > +RISCVCPU *cpu = RISCV_CPU(obj); > + > /* We set this in the realise function */ > set_misa(env, MXL_RV64, 0); > register_cpu_props(obj); > /* Set latest version of privileged specification */ > set_priv_version(env, PRIV_VERSION_1_12_0); > +set_satp_mode_max_supported(cpu, VM_1_10_SV57); > } > > static void rv64_sifive_u_cpu_init(Object *obj) > { > CPURISCVState *env = _CPU(obj)->env; > +RISCVCPU *cpu = RISCV_CPU(obj); > + > set_misa(env, MXL_RV64, RVI | RVM | RVA | RVF | RVD | RVC | RVS | RVU); > set_priv_version(env, PRIV_VERSION_1_10_0); > +set_satp_mode_max_supported(cpu, VM_1_10_SV39); > } > > static void rv64_sifive_e_cpu_init(Object *obj) > @@ -343,6 +359,7 @@ static void rv64_sifive_e_cpu_init(Object *obj) > set_misa(env, MXL_RV64, RVI | RVM | RVA | RVC | RVU); > set_priv_version(env, PRIV_VERSION_1_10_0); > cpu->cfg.mmu = false; > +set_satp_mode_max_supported(cpu, VM_1_10_MBARE); > } > > static void rv128_base_cpu_init(Object *obj) > @@ -354,28 +371,36 @@ static void rv128_base_cpu_init(Object *obj) > exit(EXIT_FAILURE); > } > CPURISCVState *env = _CPU(obj)->env; > +RISCVCPU *cpu = RISCV_CPU(obj); nits: for consistency with other cpu_init, needs a blank line after this > /* We set this in the realise function */ > set_misa(env, MXL_RV128, 0); > register_cpu_props(obj); > /* Set latest version of privileged specification */ > set_priv_version(env, PRIV_VERSION_1_12_0); > +set_satp_mode_max_supported(cpu, VM_1_10_SV57); > } > #else > static void rv32_base_cpu_init(Object *obj) > { > CPURISCVState *env = _CPU(obj)->env; > +RISCVCPU *cpu = RISCV_CPU(obj); > + > /* We set this in the realise function */ > set_misa(env, MXL_RV32, 0); > register_cpu_props(obj); > /* Set latest version of privileged specification */ > set_priv_version(env, PRIV_VERSION_1_12_0); > +set_satp_mode_max_supported(cpu, VM_1_10_SV32); > } > > static void
[PATCH v9 4/5] riscv: Introduce satp mode hw capabilities
Currently, the max satp mode is set with the only constraint that it must be implemented in QEMU, i.e. set in valid_vm_1_10_[32|64]. But we actually need to add another level of constraint: what the hw is actually capable of, because currently, a linux booting on a sifive-u54 boots in sv57 mode which is incompatible with the cpu's sv39 max capability. So add a new bitmap to RISCVSATPMap which contains this capability and initialize it in every XXX_cpu_init. Finally: - valid_vm_1_10_[32|64] constrains which satp mode the CPU can use - the CPU hw capabilities constrains what the user may select - the user's selection then constrains what's available to the guest OS. Signed-off-by: Alexandre Ghiti Reviewed-by: Andrew Jones --- target/riscv/cpu.c | 79 +++--- target/riscv/cpu.h | 8 +++-- 2 files changed, 60 insertions(+), 27 deletions(-) diff --git a/target/riscv/cpu.c b/target/riscv/cpu.c index 3a7a1746aa..6dd76355ec 100644 --- a/target/riscv/cpu.c +++ b/target/riscv/cpu.c @@ -292,26 +292,36 @@ const char *satp_mode_str(uint8_t satp_mode, bool is_32_bit) g_assert_not_reached(); } -/* Sets the satp mode to the max supported */ -static void set_satp_mode_default_map(RISCVCPU *cpu) +static void set_satp_mode_max_supported(RISCVCPU *cpu, +uint8_t satp_mode) { bool rv32 = riscv_cpu_mxl(>env) == MXL_RV32; +const bool *valid_vm = rv32 ? valid_vm_1_10_32 : valid_vm_1_10_64; -if (riscv_feature(>env, RISCV_FEATURE_MMU)) { -cpu->cfg.satp_mode.map |= -(1 << satp_mode_from_str(rv32 ? "sv32" : "sv57")); -} else { -cpu->cfg.satp_mode.map |= (1 << satp_mode_from_str("mbare")); +for (int i = 0; i <= satp_mode; ++i) { +if (valid_vm[i]) { +cpu->cfg.satp_mode.supported |= (1 << i); +} } } +/* Set the satp mode to the max supported */ +static void set_satp_mode_default_map(RISCVCPU *cpu) +{ +cpu->cfg.satp_mode.map = cpu->cfg.satp_mode.supported; +} + static void riscv_any_cpu_init(Object *obj) { CPURISCVState *env = _CPU(obj)->env; +RISCVCPU *cpu = RISCV_CPU(obj); + #if defined(TARGET_RISCV32) set_misa(env, MXL_RV32, RVI | RVM | RVA | RVF | RVD | RVC | RVU); +set_satp_mode_max_supported(cpu, VM_1_10_SV32); #elif defined(TARGET_RISCV64) set_misa(env, MXL_RV64, RVI | RVM | RVA | RVF | RVD | RVC | RVU); +set_satp_mode_max_supported(cpu, VM_1_10_SV57); #endif set_priv_version(env, PRIV_VERSION_1_12_0); register_cpu_props(obj); @@ -321,18 +331,24 @@ static void riscv_any_cpu_init(Object *obj) static void rv64_base_cpu_init(Object *obj) { CPURISCVState *env = _CPU(obj)->env; +RISCVCPU *cpu = RISCV_CPU(obj); + /* We set this in the realise function */ set_misa(env, MXL_RV64, 0); register_cpu_props(obj); /* Set latest version of privileged specification */ set_priv_version(env, PRIV_VERSION_1_12_0); +set_satp_mode_max_supported(cpu, VM_1_10_SV57); } static void rv64_sifive_u_cpu_init(Object *obj) { CPURISCVState *env = _CPU(obj)->env; +RISCVCPU *cpu = RISCV_CPU(obj); + set_misa(env, MXL_RV64, RVI | RVM | RVA | RVF | RVD | RVC | RVS | RVU); set_priv_version(env, PRIV_VERSION_1_10_0); +set_satp_mode_max_supported(cpu, VM_1_10_SV39); } static void rv64_sifive_e_cpu_init(Object *obj) @@ -343,6 +359,7 @@ static void rv64_sifive_e_cpu_init(Object *obj) set_misa(env, MXL_RV64, RVI | RVM | RVA | RVC | RVU); set_priv_version(env, PRIV_VERSION_1_10_0); cpu->cfg.mmu = false; +set_satp_mode_max_supported(cpu, VM_1_10_MBARE); } static void rv128_base_cpu_init(Object *obj) @@ -354,28 +371,36 @@ static void rv128_base_cpu_init(Object *obj) exit(EXIT_FAILURE); } CPURISCVState *env = _CPU(obj)->env; +RISCVCPU *cpu = RISCV_CPU(obj); /* We set this in the realise function */ set_misa(env, MXL_RV128, 0); register_cpu_props(obj); /* Set latest version of privileged specification */ set_priv_version(env, PRIV_VERSION_1_12_0); +set_satp_mode_max_supported(cpu, VM_1_10_SV57); } #else static void rv32_base_cpu_init(Object *obj) { CPURISCVState *env = _CPU(obj)->env; +RISCVCPU *cpu = RISCV_CPU(obj); + /* We set this in the realise function */ set_misa(env, MXL_RV32, 0); register_cpu_props(obj); /* Set latest version of privileged specification */ set_priv_version(env, PRIV_VERSION_1_12_0); +set_satp_mode_max_supported(cpu, VM_1_10_SV32); } static void rv32_sifive_u_cpu_init(Object *obj) { CPURISCVState *env = _CPU(obj)->env; +RISCVCPU *cpu = RISCV_CPU(obj); + set_misa(env, MXL_RV32, RVI | RVM | RVA | RVF | RVD | RVC | RVS | RVU); set_priv_version(env, PRIV_VERSION_1_10_0); +set_satp_mode_max_supported(cpu, VM_1_10_SV32); } static void rv32_sifive_e_cpu_init(Object *obj) @@ -386,6 +411,7 @@ static void