@@ -137,6 +137,11 @@ static void set_vext_version(CPURISCVState *env, int vext_ver)
env->vext_ver = vext_ver;
}
+static void set_pext_version(CPURISCVState *env, int pext_ver)
+{
+ env->pext_ver = pext_ver;
+}
+
static void set_feature(CPURISCVState *env, int feature)
{
env->features |= (1ULL << feature);
@@ -395,6 +400,7 @@ static void riscv_cpu_realize(DeviceState *dev, Error **errp)
int priv_version = PRIV_VERSION_1_11_0;
int bext_version = BEXT_VERSION_0_93_0;
int vext_version = VEXT_VERSION_0_07_1;
+ int pext_version = PEXT_VERSION_0_09_4;
target_ulong target_misa = env->misa;
Error *local_err = NULL;
@@ -420,6 +426,7 @@ static void riscv_cpu_realize(DeviceState *dev, Error **errp)
set_priv_version(env, priv_version);
set_bext_version(env, bext_version);
set_vext_version(env, vext_version);
+ set_pext_version(env, pext_version);
if (cpu->cfg.mmu) {
set_feature(env, RISCV_FEATURE_MMU);
@@ -553,6 +560,30 @@ static void riscv_cpu_realize(DeviceState *dev, Error **errp)
}
set_vext_version(env, vext_version);
}
+ if (cpu->cfg.ext_p) {
+ target_misa |= RVP;
+ if (cpu->cfg.pext_spec) {
+ if (!g_strcmp0(cpu->cfg.pext_spec, "v0.9.4")) {
+ pext_version = PEXT_VERSION_0_09_4;
+ } else {
+ error_setg(errp,
+ "Unsupported packed spec version '%s'",
+ cpu->cfg.pext_spec);
+ return;
+ }
+ } else {
+ qemu_log("packed verison is not specified, "
+ "use the default value v0.9.4\n");
+ }
+ if (env->misa == RV64) {
+ if (!cpu->cfg.ext_psfoperand) {
+ error_setg(errp, "The Zpsfoperand"
+ "sub-extensions is required for RV64P.");
+ return;
+ }
+ }
+ set_pext_version(env, pext_version);
+ }
set_misa(env, target_misa);
}
@@ -63,6 +63,7 @@
#define RVF RV('F')
#define RVD RV('D')
#define RVV RV('V')
+#define RVP RV('P')
#define RVC RV('C')
#define RVS RV('S')
#define RVU RV('U')
@@ -85,6 +86,7 @@ enum {
#define BEXT_VERSION_0_93_0 0x00009300
#define VEXT_VERSION_0_07_1 0x00000701
+#define PEXT_VERSION_0_09_4 0x00000904
enum {
TRANSLATE_SUCCESS,
@@ -135,6 +137,7 @@ struct CPURISCVState {
target_ulong priv_ver;
target_ulong bext_ver;
target_ulong vext_ver;
+ target_ulong pext_ver;
target_ulong misa;
target_ulong misa_mask;
@@ -293,14 +296,17 @@ struct RISCVCPU {
bool ext_u;
bool ext_h;
bool ext_v;
+ bool ext_p;
bool ext_counters;
bool ext_ifencei;
bool ext_icsr;
+ bool ext_psfoperand;
char *priv_spec;
char *user_spec;
char *bext_spec;
char *vext_spec;
+ char *pext_spec;
uint16_t vlen;
uint16_t elen;
bool mmu;
@@ -56,6 +56,7 @@ typedef struct DisasContext {
to reset this known value. */
int frm;
bool ext_ifencei;
+ bool ext_psfoperand;
bool hlsx;
/* vector extension */
bool vill;
@@ -965,6 +966,7 @@ static void riscv_tr_init_disas_context(DisasContextBase *dcbase, CPUState *cs)
ctx->lmul = FIELD_EX32(tb_flags, TB_FLAGS, LMUL);
ctx->mlen = 1 << (ctx->sew + 3 - ctx->lmul);
ctx->vl_eq_vlmax = FIELD_EX32(tb_flags, TB_FLAGS, VL_EQ_VLMAX);
+ ctx->ext_psfoperand = cpu->cfg.ext_psfoperand;
ctx->cs = cs;
}