[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index] [Xen-devel] [RFC 21/22] arm: Allow the user to specify the GIC version
A platform may have a GIC compatible with previous version of the device. This is allow to virtualize an unmodified OS on new hardware if the GIC is compatible with older version. When a guest is created, the vGIC will emulate same version as the hardware. Although, the user can specify in the configuration file the preferred version (currently on GICv2 and GICv3 are supported). Signed-off-by: Julien Grall <julien.grall@xxxxxxxxxx> Cc: Ian Jackson <ian.jackson@xxxxxxxxxxxxx> Cc: Wei Liu <wei.liu2@xxxxxxxxxx> --- The hypervisor will check if the GIC is able to virtualize the version specified by the user (via the DOMCTL createdomain). If it's not compatible an error will be send on the Xen console which will make the error not obvious for user. I'm wondering if we should expose to the toolstack the vGIC versions supported via a mecanism similar to xen_get_caps? I didn't add the documention because I wasn't sure in which section I should put it in the xl man. --- tools/libxl/libxl.h | 6 ++++++ tools/libxl/libxl_arm.c | 16 +++++++++++++++- tools/libxl/libxl_types.idl | 10 +++++++++- tools/libxl/xl_cmdimpl.c | 12 ++++++++++++ xen/arch/arm/domain.c | 45 ++++++++++++++++++++++++++------------------ xen/arch/arm/vgic.c | 9 ++++++++- xen/include/asm-arm/domain.h | 2 ++ 7 files changed, 79 insertions(+), 21 deletions(-) diff --git a/tools/libxl/libxl.h b/tools/libxl/libxl.h index 44bd8e2..3595c0c 100644 --- a/tools/libxl/libxl.h +++ b/tools/libxl/libxl.h @@ -192,6 +192,12 @@ * is not present, instead of ERROR_INVAL. */ #define LIBXL_HAVE_ERROR_DOMAIN_NOTFOUND 1 + +/* + * libxl_domain_build_info has the gic_version field. + */ +#define LIBXL_HAVE_BUILDINFO_GIC_VERSION 1 + /* * libxl ABI compatibility * diff --git a/tools/libxl/libxl_arm.c b/tools/libxl/libxl_arm.c index 946618c..c88fc94 100644 --- a/tools/libxl/libxl_arm.c +++ b/tools/libxl/libxl_arm.c @@ -39,7 +39,21 @@ int libxl__arch_domain_prepare_config(libxl__gc *gc, libxl_domain_config *d_config, xc_domain_configuration_t *xc_config) { - xc_config->gic_version = XEN_DOMCTL_CONFIG_GIC_DEFAULT; + switch (d_config->b_info.gic_version) { + case LIBXL_GIC_VERSION_DEFAULT: + xc_config->gic_version = XEN_DOMCTL_CONFIG_GIC_DEFAULT; + break; + case LIBXL_GIC_VERSION_2: + xc_config->gic_version = XEN_DOMCTL_CONFIG_GIC_V2; + break; + case LIBXL_GIC_VERSION_3: + xc_config->gic_version = XEN_DOMCTL_CONFIG_GIC_V3; + break; + default: + LOG(ERROR, "Unkwnon GIC version %s\n", + libxl_gic_version_to_string(d_config->b_info.gic_version)); + break; + } return 0; } diff --git a/tools/libxl/libxl_types.idl b/tools/libxl/libxl_types.idl index 117b61d..6dbcae5 100644 --- a/tools/libxl/libxl_types.idl +++ b/tools/libxl/libxl_types.idl @@ -365,6 +365,12 @@ libxl_vnode_info = Struct("vnode_info", [ ("vcpus", libxl_bitmap), # vcpus in this node ]) +libxl_gic_version = Enumeration("gic_version", [ + (0, "DEFAULT"), + (1, "2"), + (2, "3") + ], init_val = "LIBXL_GIC_VERSION_DEFAULT") + libxl_domain_build_info = Struct("domain_build_info",[ ("max_vcpus", integer), ("avail_vcpus", libxl_bitmap), @@ -387,7 +393,9 @@ libxl_domain_build_info = Struct("domain_build_info",[ ("blkdev_start", string), ("vnuma_nodes", Array(libxl_vnode_info, "num_vnuma_nodes")), - + + ("gic_version", libxl_gic_version), + ("device_model_version", libxl_device_model_version), ("device_model_stubdomain", libxl_defbool), # if you set device_model you must set device_model_version too diff --git a/tools/libxl/xl_cmdimpl.c b/tools/libxl/xl_cmdimpl.c index 648ca08..b033c0b 100644 --- a/tools/libxl/xl_cmdimpl.c +++ b/tools/libxl/xl_cmdimpl.c @@ -1298,6 +1298,18 @@ static void parse_config_data(const char *config_source, !xlu_cfg_get_string (config, "cpus_soft", &buf, 0)) parse_vcpu_affinity(b_info, cpus, buf, num_cpus, false); + if (!xlu_cfg_get_string (config, "gic_version", &buf, 1)) { + if (!strcmp(buf, "v2")) + b_info->gic_version = LIBXL_GIC_VERSION_2; + else if (!strcmp(buf, "v3")) + b_info->gic_version = LIBXL_GIC_VERSION_3; + else { + fprintf(stderr, + "Uknown gic_version \"%s\" specified\n", buf); + exit(1); + } + } + libxl_defbool_set(&b_info->claim_mode, claim_mode); if (xlu_cfg_get_string (config, "on_poweroff", &buf, 0)) diff --git a/xen/arch/arm/domain.c b/xen/arch/arm/domain.c index 9b113eb..a27e0f0 100644 --- a/xen/arch/arm/domain.c +++ b/xen/arch/arm/domain.c @@ -535,7 +535,6 @@ int arch_domain_create(struct domain *d, unsigned int domcr_flags, struct xen_arch_domainconfig *config) { int rc; - uint8_t gic_version; d->arch.relmem = RELMEM_not_started; @@ -564,28 +563,38 @@ int arch_domain_create(struct domain *d, unsigned int domcr_flags, if ( (rc = p2m_alloc_table(d)) != 0 ) goto fail; - /* - * Currently the vGIC is emulating the same version of the - * hardware GIC. Only the value XEN_DOMCTL_CONFIG_GIC_DEFAULT - * is allowed. The DOMCTL will return the actual version of the - * GIC. - */ - rc = -EOPNOTSUPP; - if ( config->gic_version != XEN_DOMCTL_CONFIG_GIC_DEFAULT ) - goto fail; - - switch ( gic_hw_version() ) + switch ( config->gic_version ) { - case GIC_V3: - gic_version = XEN_DOMCTL_CONFIG_GIC_V3; + case XEN_DOMCTL_CONFIG_GIC_DEFAULT: + switch ( gic_hw_version () ) + { + case GIC_V2: + config->gic_version = XEN_DOMCTL_CONFIG_GIC_V2; + d->arch.vgic.version = GIC_V2; + break; + + case GIC_V3: + config->gic_version = XEN_DOMCTL_CONFIG_GIC_V3; + d->arch.vgic.version = GIC_V3; + break; + + default: + BUG(); + } + break; + + case XEN_DOMCTL_CONFIG_GIC_V2: + d->arch.vgic.version = GIC_V2; break; - case GIC_V2: - gic_version = XEN_DOMCTL_CONFIG_GIC_V2; + + case XEN_DOMCTL_CONFIG_GIC_V3: + d->arch.vgic.version = GIC_V3; break; + default: - BUG(); + rc = -EOPNOTSUPP; + goto fail; } - config->gic_version = gic_version; if ( (rc = domain_vgic_init(d)) != 0 ) goto fail; diff --git a/xen/arch/arm/vgic.c b/xen/arch/arm/vgic.c index 8393ad4..ba4bc88 100644 --- a/xen/arch/arm/vgic.c +++ b/xen/arch/arm/vgic.c @@ -80,7 +80,14 @@ int domain_vgic_init(struct domain *d) else d->arch.vgic.nr_spis = 0; /* We don't need SPIs for the guest */ - switch ( gic_hw_version() ) + if ( !(d->arch.vgic.version & gic_info()->vgic_versions) ) + { + printk(XENLOG_G_ERR "domain %d: vGIC requested is not supported\n", + d->domain_id); + return -ENODEV; + } + + switch ( d->arch.vgic.version ) { #ifdef CONFIG_ARM_64 case GIC_V3: diff --git a/xen/include/asm-arm/domain.h b/xen/include/asm-arm/domain.h index c2a0aab..75b17af 100644 --- a/xen/include/asm-arm/domain.h +++ b/xen/include/asm-arm/domain.h @@ -77,6 +77,8 @@ struct arch_domain } virt_timer_base; struct { + /* Version of the vGIC */ + enum gic_version version; /* GIC HW version specific vGIC driver handler */ const struct vgic_ops *handler; /* -- 2.1.4 _______________________________________________ Xen-devel mailing list Xen-devel@xxxxxxxxxxxxx http://lists.xen.org/xen-devel
|
Lists.xenproject.org is hosted with RackSpace, monitoring our |