1d93bb9d5SMarkus Armbruster /* 2d93bb9d5SMarkus Armbruster * QMP command test cases 3d93bb9d5SMarkus Armbruster * 4d93bb9d5SMarkus Armbruster * Copyright (c) 2017 Red Hat Inc. 5d93bb9d5SMarkus Armbruster * 6d93bb9d5SMarkus Armbruster * Authors: 7d93bb9d5SMarkus Armbruster * Markus Armbruster <armbru@redhat.com> 8d93bb9d5SMarkus Armbruster * 9d93bb9d5SMarkus Armbruster * This work is licensed under the terms of the GNU GPL, version 2 or later. 10d93bb9d5SMarkus Armbruster * See the COPYING file in the top-level directory. 11d93bb9d5SMarkus Armbruster */ 12d93bb9d5SMarkus Armbruster 13d93bb9d5SMarkus Armbruster #include "qemu/osdep.h" 14a2ce7dbdSPaolo Bonzini #include "libqos/libqtest.h" 15d93bb9d5SMarkus Armbruster #include "qapi/error.h" 16d93bb9d5SMarkus Armbruster #include "qapi/qapi-visit-introspect.h" 17d93bb9d5SMarkus Armbruster #include "qapi/qmp/qdict.h" 18d93bb9d5SMarkus Armbruster #include "qapi/qobject-input-visitor.h" 19d93bb9d5SMarkus Armbruster 20d93bb9d5SMarkus Armbruster const char common_args[] = "-nodefaults -machine none"; 21d93bb9d5SMarkus Armbruster 22d93bb9d5SMarkus Armbruster /* Query smoke tests */ 23d93bb9d5SMarkus Armbruster 24d93bb9d5SMarkus Armbruster static int query_error_class(const char *cmd) 25d93bb9d5SMarkus Armbruster { 26d93bb9d5SMarkus Armbruster static struct { 27d93bb9d5SMarkus Armbruster const char *cmd; 28d93bb9d5SMarkus Armbruster int err_class; 29d93bb9d5SMarkus Armbruster } fails[] = { 30d93bb9d5SMarkus Armbruster /* Success depends on build configuration: */ 31d93bb9d5SMarkus Armbruster #ifndef CONFIG_SPICE 32d93bb9d5SMarkus Armbruster { "query-spice", ERROR_CLASS_COMMAND_NOT_FOUND }, 33d93bb9d5SMarkus Armbruster #endif 34*9b1c9116SClaudio Fontana #ifndef CONFIG_TCG 35*9b1c9116SClaudio Fontana { "query-replay", ERROR_CLASS_COMMAND_NOT_FOUND }, 36*9b1c9116SClaudio Fontana #endif 37d93bb9d5SMarkus Armbruster #ifndef CONFIG_VNC 38d93bb9d5SMarkus Armbruster { "query-vnc", ERROR_CLASS_GENERIC_ERROR }, 39d93bb9d5SMarkus Armbruster { "query-vnc-servers", ERROR_CLASS_GENERIC_ERROR }, 40d93bb9d5SMarkus Armbruster #endif 41d93bb9d5SMarkus Armbruster #ifndef CONFIG_REPLICATION 42d93bb9d5SMarkus Armbruster { "query-xen-replication-status", ERROR_CLASS_COMMAND_NOT_FOUND }, 43d93bb9d5SMarkus Armbruster #endif 44d93bb9d5SMarkus Armbruster /* Likewise, and require special QEMU command-line arguments: */ 45d93bb9d5SMarkus Armbruster { "query-acpi-ospm-status", ERROR_CLASS_GENERIC_ERROR }, 46d93bb9d5SMarkus Armbruster { "query-balloon", ERROR_CLASS_DEVICE_NOT_ACTIVE }, 47d93bb9d5SMarkus Armbruster { "query-hotpluggable-cpus", ERROR_CLASS_GENERIC_ERROR }, 48d93bb9d5SMarkus Armbruster { "query-vm-generation-id", ERROR_CLASS_GENERIC_ERROR }, 49d93bb9d5SMarkus Armbruster { NULL, -1 } 50d93bb9d5SMarkus Armbruster }; 51d93bb9d5SMarkus Armbruster int i; 52d93bb9d5SMarkus Armbruster 53d93bb9d5SMarkus Armbruster for (i = 0; fails[i].cmd; i++) { 54d93bb9d5SMarkus Armbruster if (!strcmp(cmd, fails[i].cmd)) { 55d93bb9d5SMarkus Armbruster return fails[i].err_class; 56d93bb9d5SMarkus Armbruster } 57d93bb9d5SMarkus Armbruster } 58d93bb9d5SMarkus Armbruster return -1; 59d93bb9d5SMarkus Armbruster } 60d93bb9d5SMarkus Armbruster 61d93bb9d5SMarkus Armbruster static void test_query(const void *data) 62d93bb9d5SMarkus Armbruster { 63d93bb9d5SMarkus Armbruster const char *cmd = data; 64d93bb9d5SMarkus Armbruster int expected_error_class = query_error_class(cmd); 65d93bb9d5SMarkus Armbruster QDict *resp, *error; 66d93bb9d5SMarkus Armbruster const char *error_class; 67da9cd2d0SThomas Huth QTestState *qts; 68d93bb9d5SMarkus Armbruster 69da9cd2d0SThomas Huth qts = qtest_init(common_args); 70d93bb9d5SMarkus Armbruster 71da9cd2d0SThomas Huth resp = qtest_qmp(qts, "{ 'execute': %s }", cmd); 72d93bb9d5SMarkus Armbruster error = qdict_get_qdict(resp, "error"); 73d93bb9d5SMarkus Armbruster error_class = error ? qdict_get_str(error, "class") : NULL; 74d93bb9d5SMarkus Armbruster 75d93bb9d5SMarkus Armbruster if (expected_error_class < 0) { 76d93bb9d5SMarkus Armbruster g_assert(qdict_haskey(resp, "return")); 77d93bb9d5SMarkus Armbruster } else { 78d93bb9d5SMarkus Armbruster g_assert(error); 79d93bb9d5SMarkus Armbruster g_assert_cmpint(qapi_enum_parse(&QapiErrorClass_lookup, error_class, 80d93bb9d5SMarkus Armbruster -1, &error_abort), 81d93bb9d5SMarkus Armbruster ==, expected_error_class); 82d93bb9d5SMarkus Armbruster } 83d93bb9d5SMarkus Armbruster qobject_unref(resp); 84d93bb9d5SMarkus Armbruster 85da9cd2d0SThomas Huth qtest_quit(qts); 86d93bb9d5SMarkus Armbruster } 87d93bb9d5SMarkus Armbruster 88d97b4b0dSThomas Huth static bool query_is_ignored(const char *cmd) 89d93bb9d5SMarkus Armbruster { 90d97b4b0dSThomas Huth const char *ignored[] = { 91d93bb9d5SMarkus Armbruster /* Not actually queries: */ 92d93bb9d5SMarkus Armbruster "add-fd", 93d93bb9d5SMarkus Armbruster /* Success depends on target arch: */ 94d93bb9d5SMarkus Armbruster "query-cpu-definitions", /* arm, i386, ppc, s390x */ 95d93bb9d5SMarkus Armbruster "query-gic-capabilities", /* arm */ 96d93bb9d5SMarkus Armbruster /* Success depends on target-specific build configuration: */ 97d93bb9d5SMarkus Armbruster "query-pci", /* CONFIG_PCI */ 98d93bb9d5SMarkus Armbruster /* Success depends on launching SEV guest */ 99d93bb9d5SMarkus Armbruster "query-sev-launch-measure", 100d93bb9d5SMarkus Armbruster /* Success depends on Host or Hypervisor SEV support */ 101d93bb9d5SMarkus Armbruster "query-sev", 102d93bb9d5SMarkus Armbruster "query-sev-capabilities", 103d93bb9d5SMarkus Armbruster NULL 104d93bb9d5SMarkus Armbruster }; 105d93bb9d5SMarkus Armbruster int i; 106d93bb9d5SMarkus Armbruster 107d97b4b0dSThomas Huth for (i = 0; ignored[i]; i++) { 108d97b4b0dSThomas Huth if (!strcmp(cmd, ignored[i])) { 109d93bb9d5SMarkus Armbruster return true; 110d93bb9d5SMarkus Armbruster } 111d93bb9d5SMarkus Armbruster } 112d93bb9d5SMarkus Armbruster return false; 113d93bb9d5SMarkus Armbruster } 114d93bb9d5SMarkus Armbruster 115d93bb9d5SMarkus Armbruster typedef struct { 116d93bb9d5SMarkus Armbruster SchemaInfoList *list; 117d93bb9d5SMarkus Armbruster GHashTable *hash; 118d93bb9d5SMarkus Armbruster } QmpSchema; 119d93bb9d5SMarkus Armbruster 120d93bb9d5SMarkus Armbruster static void qmp_schema_init(QmpSchema *schema) 121d93bb9d5SMarkus Armbruster { 122d93bb9d5SMarkus Armbruster QDict *resp; 123d93bb9d5SMarkus Armbruster Visitor *qiv; 124d93bb9d5SMarkus Armbruster SchemaInfoList *tail; 125da9cd2d0SThomas Huth QTestState *qts; 126d93bb9d5SMarkus Armbruster 127da9cd2d0SThomas Huth qts = qtest_init(common_args); 128da9cd2d0SThomas Huth 129da9cd2d0SThomas Huth resp = qtest_qmp(qts, "{ 'execute': 'query-qmp-schema' }"); 130d93bb9d5SMarkus Armbruster 131d93bb9d5SMarkus Armbruster qiv = qobject_input_visitor_new(qdict_get(resp, "return")); 132d93bb9d5SMarkus Armbruster visit_type_SchemaInfoList(qiv, NULL, &schema->list, &error_abort); 133d93bb9d5SMarkus Armbruster visit_free(qiv); 134d93bb9d5SMarkus Armbruster 135d93bb9d5SMarkus Armbruster qobject_unref(resp); 136da9cd2d0SThomas Huth qtest_quit(qts); 137d93bb9d5SMarkus Armbruster 138d93bb9d5SMarkus Armbruster schema->hash = g_hash_table_new(g_str_hash, g_str_equal); 139d93bb9d5SMarkus Armbruster 140d93bb9d5SMarkus Armbruster /* Build @schema: hash table mapping entity name to SchemaInfo */ 141d93bb9d5SMarkus Armbruster for (tail = schema->list; tail; tail = tail->next) { 142d93bb9d5SMarkus Armbruster g_hash_table_insert(schema->hash, tail->value->name, tail->value); 143d93bb9d5SMarkus Armbruster } 144d93bb9d5SMarkus Armbruster } 145d93bb9d5SMarkus Armbruster 146d93bb9d5SMarkus Armbruster static SchemaInfo *qmp_schema_lookup(QmpSchema *schema, const char *name) 147d93bb9d5SMarkus Armbruster { 148d93bb9d5SMarkus Armbruster return g_hash_table_lookup(schema->hash, name); 149d93bb9d5SMarkus Armbruster } 150d93bb9d5SMarkus Armbruster 151d93bb9d5SMarkus Armbruster static void qmp_schema_cleanup(QmpSchema *schema) 152d93bb9d5SMarkus Armbruster { 153d93bb9d5SMarkus Armbruster qapi_free_SchemaInfoList(schema->list); 154d93bb9d5SMarkus Armbruster g_hash_table_destroy(schema->hash); 155d93bb9d5SMarkus Armbruster } 156d93bb9d5SMarkus Armbruster 157d93bb9d5SMarkus Armbruster static bool object_type_has_mandatory_members(SchemaInfo *type) 158d93bb9d5SMarkus Armbruster { 159d93bb9d5SMarkus Armbruster SchemaInfoObjectMemberList *tail; 160d93bb9d5SMarkus Armbruster 161d93bb9d5SMarkus Armbruster g_assert(type->meta_type == SCHEMA_META_TYPE_OBJECT); 162d93bb9d5SMarkus Armbruster 163d93bb9d5SMarkus Armbruster for (tail = type->u.object.members; tail; tail = tail->next) { 164d93bb9d5SMarkus Armbruster if (!tail->value->has_q_default) { 165d93bb9d5SMarkus Armbruster return true; 166d93bb9d5SMarkus Armbruster } 167d93bb9d5SMarkus Armbruster } 168d93bb9d5SMarkus Armbruster 169d93bb9d5SMarkus Armbruster return false; 170d93bb9d5SMarkus Armbruster } 171d93bb9d5SMarkus Armbruster 172d93bb9d5SMarkus Armbruster static void add_query_tests(QmpSchema *schema) 173d93bb9d5SMarkus Armbruster { 174d93bb9d5SMarkus Armbruster SchemaInfoList *tail; 175d93bb9d5SMarkus Armbruster SchemaInfo *si, *arg_type, *ret_type; 176d93bb9d5SMarkus Armbruster char *test_name; 177d93bb9d5SMarkus Armbruster 178d93bb9d5SMarkus Armbruster /* Test the query-like commands */ 179d93bb9d5SMarkus Armbruster for (tail = schema->list; tail; tail = tail->next) { 180d93bb9d5SMarkus Armbruster si = tail->value; 181d93bb9d5SMarkus Armbruster if (si->meta_type != SCHEMA_META_TYPE_COMMAND) { 182d93bb9d5SMarkus Armbruster continue; 183d93bb9d5SMarkus Armbruster } 184d93bb9d5SMarkus Armbruster 185d97b4b0dSThomas Huth if (query_is_ignored(si->name)) { 186d93bb9d5SMarkus Armbruster continue; 187d93bb9d5SMarkus Armbruster } 188d93bb9d5SMarkus Armbruster 189d93bb9d5SMarkus Armbruster arg_type = qmp_schema_lookup(schema, si->u.command.arg_type); 190d93bb9d5SMarkus Armbruster if (object_type_has_mandatory_members(arg_type)) { 191d93bb9d5SMarkus Armbruster continue; 192d93bb9d5SMarkus Armbruster } 193d93bb9d5SMarkus Armbruster 194d93bb9d5SMarkus Armbruster ret_type = qmp_schema_lookup(schema, si->u.command.ret_type); 195d93bb9d5SMarkus Armbruster if (ret_type->meta_type == SCHEMA_META_TYPE_OBJECT 196d93bb9d5SMarkus Armbruster && !ret_type->u.object.members) { 197d93bb9d5SMarkus Armbruster continue; 198d93bb9d5SMarkus Armbruster } 199d93bb9d5SMarkus Armbruster 200d93bb9d5SMarkus Armbruster test_name = g_strdup_printf("qmp/%s", si->name); 201d93bb9d5SMarkus Armbruster qtest_add_data_func(test_name, si->name, test_query); 202d93bb9d5SMarkus Armbruster g_free(test_name); 203d93bb9d5SMarkus Armbruster } 204d93bb9d5SMarkus Armbruster } 205d93bb9d5SMarkus Armbruster 2065b88849eSEric Auger static void test_object_add_failure_modes(void) 207442b09b8SMarc-André Lureau { 208442b09b8SMarc-André Lureau QTestState *qts; 209442b09b8SMarc-André Lureau QDict *resp; 210442b09b8SMarc-André Lureau 2115b88849eSEric Auger /* attempt to create an object without props */ 212442b09b8SMarc-André Lureau qts = qtest_init(common_args); 213442b09b8SMarc-André Lureau resp = qtest_qmp(qts, "{'execute': 'object-add', 'arguments':" 214442b09b8SMarc-André Lureau " {'qom-type': 'memory-backend-ram', 'id': 'ram1' } }"); 215442b09b8SMarc-André Lureau g_assert_nonnull(resp); 2163bc1b8eeSMarkus Armbruster qmp_expect_error_and_unref(resp, "GenericError"); 217442b09b8SMarc-André Lureau 2185b88849eSEric Auger /* attempt to create an object without qom-type */ 2199fc719b8SEric Auger resp = qtest_qmp(qts, "{'execute': 'object-add', 'arguments':" 2205b88849eSEric Auger " {'id': 'ram1' } }"); 2215b88849eSEric Auger g_assert_nonnull(resp); 2223bc1b8eeSMarkus Armbruster qmp_expect_error_and_unref(resp, "GenericError"); 2235b88849eSEric Auger 2245b88849eSEric Auger /* attempt to delete an object that does not exist */ 2255b88849eSEric Auger resp = qtest_qmp(qts, "{'execute': 'object-del', 'arguments':" 2265b88849eSEric Auger " {'id': 'ram1' } }"); 2275b88849eSEric Auger g_assert_nonnull(resp); 2283bc1b8eeSMarkus Armbruster qmp_expect_error_and_unref(resp, "GenericError"); 2295b88849eSEric Auger 2305b88849eSEric Auger /* attempt to create 2 objects with duplicate id */ 2315b88849eSEric Auger resp = qtest_qmp(qts, "{'execute': 'object-add', 'arguments':" 2325b88849eSEric Auger " {'qom-type': 'memory-backend-ram', 'id': 'ram1'," 2335b88849eSEric Auger " 'props': {'size': 1048576 } } }"); 2349fc719b8SEric Auger g_assert_nonnull(resp); 2359fc719b8SEric Auger g_assert(qdict_haskey(resp, "return")); 23615c51f72SLi Qiang qobject_unref(resp); 23715c51f72SLi Qiang 2389fc719b8SEric Auger resp = qtest_qmp(qts, "{'execute': 'object-add', 'arguments':" 2395b88849eSEric Auger " {'qom-type': 'memory-backend-ram', 'id': 'ram1'," 2405b88849eSEric Auger " 'props': {'size': 1048576 } } }"); 2419fc719b8SEric Auger g_assert_nonnull(resp); 2423bc1b8eeSMarkus Armbruster qmp_expect_error_and_unref(resp, "GenericError"); 2435b88849eSEric Auger 2445b88849eSEric Auger /* delete ram1 object */ 2455b88849eSEric Auger resp = qtest_qmp(qts, "{'execute': 'object-del', 'arguments':" 2465b88849eSEric Auger " {'id': 'ram1' } }"); 2475b88849eSEric Auger g_assert_nonnull(resp); 2485b88849eSEric Auger g_assert(qdict_haskey(resp, "return")); 24915c51f72SLi Qiang qobject_unref(resp); 2505b88849eSEric Auger 2515b88849eSEric Auger /* attempt to create an object with a property of a wrong type */ 2525b88849eSEric Auger resp = qtest_qmp(qts, "{'execute': 'object-add', 'arguments':" 2535b88849eSEric Auger " {'qom-type': 'memory-backend-ram', 'id': 'ram1'," 2545b88849eSEric Auger " 'props': {'size': '1048576' } } }"); 2555b88849eSEric Auger g_assert_nonnull(resp); 2565b88849eSEric Auger /* now do it right */ 2573bc1b8eeSMarkus Armbruster qmp_expect_error_and_unref(resp, "GenericError"); 25815c51f72SLi Qiang 2595b88849eSEric Auger resp = qtest_qmp(qts, "{'execute': 'object-add', 'arguments':" 2605b88849eSEric Auger " {'qom-type': 'memory-backend-ram', 'id': 'ram1'," 2615b88849eSEric Auger " 'props': {'size': 1048576 } } }"); 2625b88849eSEric Auger g_assert_nonnull(resp); 2635b88849eSEric Auger g_assert(qdict_haskey(resp, "return")); 26415c51f72SLi Qiang qobject_unref(resp); 2655b88849eSEric Auger 2665b88849eSEric Auger /* delete ram1 object */ 2675b88849eSEric Auger resp = qtest_qmp(qts, "{'execute': 'object-del', 'arguments':" 2685b88849eSEric Auger " {'id': 'ram1' } }"); 2695b88849eSEric Auger g_assert_nonnull(resp); 2705b88849eSEric Auger g_assert(qdict_haskey(resp, "return")); 27115c51f72SLi Qiang qobject_unref(resp); 2725b88849eSEric Auger 2735b88849eSEric Auger /* attempt to create an object without the id */ 2745b88849eSEric Auger resp = qtest_qmp(qts, "{'execute': 'object-add', 'arguments':" 2755b88849eSEric Auger " {'qom-type': 'memory-backend-ram'," 2765b88849eSEric Auger " 'props': {'size': 1048576 } } }"); 2775b88849eSEric Auger g_assert_nonnull(resp); 2783bc1b8eeSMarkus Armbruster qmp_expect_error_and_unref(resp, "GenericError"); 27915c51f72SLi Qiang 2805b88849eSEric Auger /* now do it right */ 2815b88849eSEric Auger resp = qtest_qmp(qts, "{'execute': 'object-add', 'arguments':" 2825b88849eSEric Auger " {'qom-type': 'memory-backend-ram', 'id': 'ram1'," 2835b88849eSEric Auger " 'props': {'size': 1048576 } } }"); 2845b88849eSEric Auger g_assert_nonnull(resp); 2855b88849eSEric Auger g_assert(qdict_haskey(resp, "return")); 28615c51f72SLi Qiang qobject_unref(resp); 2875b88849eSEric Auger 2885b88849eSEric Auger /* delete ram1 object */ 2895b88849eSEric Auger resp = qtest_qmp(qts, "{'execute': 'object-del', 'arguments':" 2905b88849eSEric Auger " {'id': 'ram1' } }"); 2915b88849eSEric Auger g_assert_nonnull(resp); 2925b88849eSEric Auger g_assert(qdict_haskey(resp, "return")); 29315c51f72SLi Qiang qobject_unref(resp); 2945b88849eSEric Auger 2955b88849eSEric Auger /* attempt to set a non existing property */ 2965b88849eSEric Auger resp = qtest_qmp(qts, "{'execute': 'object-add', 'arguments':" 2975b88849eSEric Auger " {'qom-type': 'memory-backend-ram', 'id': 'ram1'," 2985b88849eSEric Auger " 'props': {'sized': 1048576 } } }"); 2995b88849eSEric Auger g_assert_nonnull(resp); 3003bc1b8eeSMarkus Armbruster qmp_expect_error_and_unref(resp, "GenericError"); 30115c51f72SLi Qiang 3025b88849eSEric Auger /* now do it right */ 3035b88849eSEric Auger resp = qtest_qmp(qts, "{'execute': 'object-add', 'arguments':" 3045b88849eSEric Auger " {'qom-type': 'memory-backend-ram', 'id': 'ram1'," 3055b88849eSEric Auger " 'props': {'size': 1048576 } } }"); 3065b88849eSEric Auger g_assert_nonnull(resp); 3075b88849eSEric Auger g_assert(qdict_haskey(resp, "return")); 30815c51f72SLi Qiang qobject_unref(resp); 3095b88849eSEric Auger 3105b88849eSEric Auger /* delete ram1 object without id */ 3115b88849eSEric Auger resp = qtest_qmp(qts, "{'execute': 'object-del', 'arguments':" 3125b88849eSEric Auger " {'ida': 'ram1' } }"); 3135b88849eSEric Auger g_assert_nonnull(resp); 31415c51f72SLi Qiang qobject_unref(resp); 3155b88849eSEric Auger 3165b88849eSEric Auger /* delete ram1 object */ 3175b88849eSEric Auger resp = qtest_qmp(qts, "{'execute': 'object-del', 'arguments':" 3185b88849eSEric Auger " {'id': 'ram1' } }"); 3195b88849eSEric Auger g_assert_nonnull(resp); 3205b88849eSEric Auger g_assert(qdict_haskey(resp, "return")); 32115c51f72SLi Qiang qobject_unref(resp); 3225b88849eSEric Auger 3235b88849eSEric Auger /* delete ram1 object that does not exist anymore*/ 3245b88849eSEric Auger resp = qtest_qmp(qts, "{'execute': 'object-del', 'arguments':" 3255b88849eSEric Auger " {'id': 'ram1' } }"); 3265b88849eSEric Auger g_assert_nonnull(resp); 3273bc1b8eeSMarkus Armbruster qmp_expect_error_and_unref(resp, "GenericError"); 3285b88849eSEric Auger 3299fc719b8SEric Auger qtest_quit(qts); 3309fc719b8SEric Auger } 3319fc719b8SEric Auger 332d93bb9d5SMarkus Armbruster int main(int argc, char *argv[]) 333d93bb9d5SMarkus Armbruster { 334d93bb9d5SMarkus Armbruster QmpSchema schema; 335d93bb9d5SMarkus Armbruster int ret; 336d93bb9d5SMarkus Armbruster 337d93bb9d5SMarkus Armbruster g_test_init(&argc, &argv, NULL); 338d93bb9d5SMarkus Armbruster 339d93bb9d5SMarkus Armbruster qmp_schema_init(&schema); 340d93bb9d5SMarkus Armbruster add_query_tests(&schema); 341442b09b8SMarc-André Lureau 3425b88849eSEric Auger qtest_add_func("qmp/object-add-failure-modes", 3435b88849eSEric Auger test_object_add_failure_modes); 344442b09b8SMarc-André Lureau 345d93bb9d5SMarkus Armbruster ret = g_test_run(); 346d93bb9d5SMarkus Armbruster 347d93bb9d5SMarkus Armbruster qmp_schema_cleanup(&schema); 348d93bb9d5SMarkus Armbruster return ret; 349d93bb9d5SMarkus Armbruster } 350