From: Bruce Rogers Date: Thu, 21 Jan 2021 16:34:32 -0700 Subject: module: for virtio-gpu, pre-load module to avoid abort on missing module If the hw-display-virtio-gpu module is not loadable when the virtio-gpu device is referenced either on the command line or the monitor, qemu will call abort. We can fail gracefully by moving the attempted module load to a context better situated to handle errors properly. (bsc#1181103) Signed-off-by: Bruce Rogers --- include/qemu/module.h | 1 + qom/object.c | 12 ++++++++++++ qom/qom-qmp-cmds.c | 17 +++++++++++++++++ softmmu/qdev-monitor.c | 15 +++++++++++++++ 4 files changed, 45 insertions(+) diff --git a/include/qemu/module.h b/include/qemu/module.h index 944d403cbd1535cc121af76a94f2..4b42dd285eeac1ba12e5c9e18ac0 100644 --- a/include/qemu/module.h +++ b/include/qemu/module.h @@ -72,5 +72,6 @@ void module_call_init(module_init_type type); bool module_load_one(const char *prefix, const char *lib_name, bool mayfail); void module_load_qom_one(const char *type); void module_load_qom_all(void); +int module_load_check(const char *name); #endif diff --git a/qom/object.c b/qom/object.c index 8b8cd1bdc9d176af921315b9cc2f..588f09ae22ba33bd6c3298995fc6 100644 --- a/qom/object.c +++ b/qom/object.c @@ -566,6 +566,18 @@ static void object_initialize_with_type(Object *obj, size_t size, TypeImpl *type object_post_init_with_type(obj, type); } +#ifdef CONFIG_MODULES +int module_load_check(const char *name) +{ + TypeImpl *type = type_get_by_name(name); + if (!type) { + module_load_qom_one(name); + type = type_get_by_name(name); + } + return type == NULL; +} +#endif + void object_initialize(void *data, size_t size, const char *typename) { TypeImpl *type = type_get_by_name(typename); diff --git a/qom/qom-qmp-cmds.c b/qom/qom-qmp-cmds.c index 310ab2d0481d1517b0432be23256..cf130dc875bb6a7921fd7fb5af26 100644 --- a/qom/qom-qmp-cmds.c +++ b/qom/qom-qmp-cmds.c @@ -131,6 +131,23 @@ ObjectPropertyInfoList *qmp_device_list_properties(const char *typename, ObjectPropertyIterator iter; ObjectPropertyInfoList *prop_list = NULL; +#ifdef CONFIG_MODULES + if (!strcmp(typename, "virtio-gpu-pci") || !strcmp(typename, "virtio-gpu-ccw")) { + if (module_load_check("virtio-gpu-device")) { + ObjectPropertyInfo *info; + info = g_new0(ObjectPropertyInfo, 1); + info->name = g_strdup("dummy"); + info->type = g_strdup("dummy"); + info->has_description = false; + info->description = NULL; + info->default_value = 0; + info->has_default_value = 0; + QAPI_LIST_PREPEND(prop_list, info); + return prop_list; + } + } +#endif + klass = module_object_class_by_name(typename); if (klass == NULL) { error_set(errp, ERROR_CLASS_DEVICE_NOT_FOUND, diff --git a/softmmu/qdev-monitor.c b/softmmu/qdev-monitor.c index bf79d0bbcd986320eb609f37253e..564dfaeeda9c3ae9dbf1afa97825 100644 --- a/softmmu/qdev-monitor.c +++ b/softmmu/qdev-monitor.c @@ -263,6 +263,13 @@ int qdev_device_help(QemuOpts *opts) int i; driver = qemu_opt_get(opts, "driver"); +#ifdef CONFIG_MODULES + if (driver && !strcmp(driver, "virtio-gpu")) { + if (module_load_check("virtio-gpu-device")) { + return 0; + } + } +#endif if (driver && is_help_option(driver)) { qdev_print_devinfos(false); return 1; @@ -650,6 +657,14 @@ DeviceState *qdev_device_add(QemuOpts *opts, Error **errp) return NULL; } +#ifdef CONFIG_MODULES + if (!strcmp(driver, "virtio-gpu-pci") || !strcmp(driver, "virtio-gpu-ccw")) { + if (module_load_check("virtio-gpu-device")) { + error_setg(errp, "loadable module for %s not available!", driver); + return NULL; + } + } +#endif /* create device */ dev = qdev_new(driver);