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" 14*a2ce7dbdSPaolo 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 34d93bb9d5SMarkus Armbruster #ifndef CONFIG_VNC 35d93bb9d5SMarkus Armbruster { "query-vnc", ERROR_CLASS_GENERIC_ERROR }, 36d93bb9d5SMarkus Armbruster { "query-vnc-servers", ERROR_CLASS_GENERIC_ERROR }, 37d93bb9d5SMarkus Armbruster #endif 38d93bb9d5SMarkus Armbruster #ifndef CONFIG_REPLICATION 39d93bb9d5SMarkus Armbruster { "query-xen-replication-status", ERROR_CLASS_COMMAND_NOT_FOUND }, 40d93bb9d5SMarkus Armbruster #endif 41d93bb9d5SMarkus Armbruster /* Likewise, and require special QEMU command-line arguments: */ 42d93bb9d5SMarkus Armbruster { "query-acpi-ospm-status", ERROR_CLASS_GENERIC_ERROR }, 43d93bb9d5SMarkus Armbruster { "query-balloon", ERROR_CLASS_DEVICE_NOT_ACTIVE }, 44d93bb9d5SMarkus Armbruster { "query-hotpluggable-cpus", ERROR_CLASS_GENERIC_ERROR }, 45d93bb9d5SMarkus Armbruster { "query-vm-generation-id", ERROR_CLASS_GENERIC_ERROR }, 46d93bb9d5SMarkus Armbruster { NULL, -1 } 47d93bb9d5SMarkus Armbruster }; 48d93bb9d5SMarkus Armbruster int i; 49d93bb9d5SMarkus Armbruster 50d93bb9d5SMarkus Armbruster for (i = 0; fails[i].cmd; i++) { 51d93bb9d5SMarkus Armbruster if (!strcmp(cmd, fails[i].cmd)) { 52d93bb9d5SMarkus Armbruster return fails[i].err_class; 53d93bb9d5SMarkus Armbruster } 54d93bb9d5SMarkus Armbruster } 55d93bb9d5SMarkus Armbruster return -1; 56d93bb9d5SMarkus Armbruster } 57d93bb9d5SMarkus Armbruster 58d93bb9d5SMarkus Armbruster static void test_query(const void *data) 59d93bb9d5SMarkus Armbruster { 60d93bb9d5SMarkus Armbruster const char *cmd = data; 61d93bb9d5SMarkus Armbruster int expected_error_class = query_error_class(cmd); 62d93bb9d5SMarkus Armbruster QDict *resp, *error; 63d93bb9d5SMarkus Armbruster const char *error_class; 64da9cd2d0SThomas Huth QTestState *qts; 65d93bb9d5SMarkus Armbruster 66da9cd2d0SThomas Huth qts = qtest_init(common_args); 67d93bb9d5SMarkus Armbruster 68da9cd2d0SThomas Huth resp = qtest_qmp(qts, "{ 'execute': %s }", cmd); 69d93bb9d5SMarkus Armbruster error = qdict_get_qdict(resp, "error"); 70d93bb9d5SMarkus Armbruster error_class = error ? qdict_get_str(error, "class") : NULL; 71d93bb9d5SMarkus Armbruster 72d93bb9d5SMarkus Armbruster if (expected_error_class < 0) { 73d93bb9d5SMarkus Armbruster g_assert(qdict_haskey(resp, "return")); 74d93bb9d5SMarkus Armbruster } else { 75d93bb9d5SMarkus Armbruster g_assert(error); 76d93bb9d5SMarkus Armbruster g_assert_cmpint(qapi_enum_parse(&QapiErrorClass_lookup, error_class, 77d93bb9d5SMarkus Armbruster -1, &error_abort), 78d93bb9d5SMarkus Armbruster ==, expected_error_class); 79d93bb9d5SMarkus Armbruster } 80d93bb9d5SMarkus Armbruster qobject_unref(resp); 81d93bb9d5SMarkus Armbruster 82da9cd2d0SThomas Huth qtest_quit(qts); 83d93bb9d5SMarkus Armbruster } 84d93bb9d5SMarkus Armbruster 85d93bb9d5SMarkus Armbruster static bool query_is_blacklisted(const char *cmd) 86d93bb9d5SMarkus Armbruster { 87d93bb9d5SMarkus Armbruster const char *blacklist[] = { 88d93bb9d5SMarkus Armbruster /* Not actually queries: */ 89d93bb9d5SMarkus Armbruster "add-fd", 90d93bb9d5SMarkus Armbruster /* Success depends on target arch: */ 91d93bb9d5SMarkus Armbruster "query-cpu-definitions", /* arm, i386, ppc, s390x */ 92d93bb9d5SMarkus Armbruster "query-gic-capabilities", /* arm */ 93d93bb9d5SMarkus Armbruster /* Success depends on target-specific build configuration: */ 94d93bb9d5SMarkus Armbruster "query-pci", /* CONFIG_PCI */ 95d93bb9d5SMarkus Armbruster /* Success depends on launching SEV guest */ 96d93bb9d5SMarkus Armbruster "query-sev-launch-measure", 97d93bb9d5SMarkus Armbruster /* Success depends on Host or Hypervisor SEV support */ 98d93bb9d5SMarkus Armbruster "query-sev", 99d93bb9d5SMarkus Armbruster "query-sev-capabilities", 100d93bb9d5SMarkus Armbruster NULL 101d93bb9d5SMarkus Armbruster }; 102d93bb9d5SMarkus Armbruster int i; 103d93bb9d5SMarkus Armbruster 104d93bb9d5SMarkus Armbruster for (i = 0; blacklist[i]; i++) { 105d93bb9d5SMarkus Armbruster if (!strcmp(cmd, blacklist[i])) { 106d93bb9d5SMarkus Armbruster return true; 107d93bb9d5SMarkus Armbruster } 108d93bb9d5SMarkus Armbruster } 109d93bb9d5SMarkus Armbruster return false; 110d93bb9d5SMarkus Armbruster } 111d93bb9d5SMarkus Armbruster 112d93bb9d5SMarkus Armbruster typedef struct { 113d93bb9d5SMarkus Armbruster SchemaInfoList *list; 114d93bb9d5SMarkus Armbruster GHashTable *hash; 115d93bb9d5SMarkus Armbruster } QmpSchema; 116d93bb9d5SMarkus Armbruster 117d93bb9d5SMarkus Armbruster static void qmp_schema_init(QmpSchema *schema) 118d93bb9d5SMarkus Armbruster { 119d93bb9d5SMarkus Armbruster QDict *resp; 120d93bb9d5SMarkus Armbruster Visitor *qiv; 121d93bb9d5SMarkus Armbruster SchemaInfoList *tail; 122da9cd2d0SThomas Huth QTestState *qts; 123d93bb9d5SMarkus Armbruster 124da9cd2d0SThomas Huth qts = qtest_init(common_args); 125da9cd2d0SThomas Huth 126da9cd2d0SThomas Huth resp = qtest_qmp(qts, "{ 'execute': 'query-qmp-schema' }"); 127d93bb9d5SMarkus Armbruster 128d93bb9d5SMarkus Armbruster qiv = qobject_input_visitor_new(qdict_get(resp, "return")); 129d93bb9d5SMarkus Armbruster visit_type_SchemaInfoList(qiv, NULL, &schema->list, &error_abort); 130d93bb9d5SMarkus Armbruster visit_free(qiv); 131d93bb9d5SMarkus Armbruster 132d93bb9d5SMarkus Armbruster qobject_unref(resp); 133da9cd2d0SThomas Huth qtest_quit(qts); 134d93bb9d5SMarkus Armbruster 135d93bb9d5SMarkus Armbruster schema->hash = g_hash_table_new(g_str_hash, g_str_equal); 136d93bb9d5SMarkus Armbruster 137d93bb9d5SMarkus Armbruster /* Build @schema: hash table mapping entity name to SchemaInfo */ 138d93bb9d5SMarkus Armbruster for (tail = schema->list; tail; tail = tail->next) { 139d93bb9d5SMarkus Armbruster g_hash_table_insert(schema->hash, tail->value->name, tail->value); 140d93bb9d5SMarkus Armbruster } 141d93bb9d5SMarkus Armbruster } 142d93bb9d5SMarkus Armbruster 143d93bb9d5SMarkus Armbruster static SchemaInfo *qmp_schema_lookup(QmpSchema *schema, const char *name) 144d93bb9d5SMarkus Armbruster { 145d93bb9d5SMarkus Armbruster return g_hash_table_lookup(schema->hash, name); 146d93bb9d5SMarkus Armbruster } 147d93bb9d5SMarkus Armbruster 148d93bb9d5SMarkus Armbruster static void qmp_schema_cleanup(QmpSchema *schema) 149d93bb9d5SMarkus Armbruster { 150d93bb9d5SMarkus Armbruster qapi_free_SchemaInfoList(schema->list); 151d93bb9d5SMarkus Armbruster g_hash_table_destroy(schema->hash); 152d93bb9d5SMarkus Armbruster } 153d93bb9d5SMarkus Armbruster 154d93bb9d5SMarkus Armbruster static bool object_type_has_mandatory_members(SchemaInfo *type) 155d93bb9d5SMarkus Armbruster { 156d93bb9d5SMarkus Armbruster SchemaInfoObjectMemberList *tail; 157d93bb9d5SMarkus Armbruster 158d93bb9d5SMarkus Armbruster g_assert(type->meta_type == SCHEMA_META_TYPE_OBJECT); 159d93bb9d5SMarkus Armbruster 160d93bb9d5SMarkus Armbruster for (tail = type->u.object.members; tail; tail = tail->next) { 161d93bb9d5SMarkus Armbruster if (!tail->value->has_q_default) { 162d93bb9d5SMarkus Armbruster return true; 163d93bb9d5SMarkus Armbruster } 164d93bb9d5SMarkus Armbruster } 165d93bb9d5SMarkus Armbruster 166d93bb9d5SMarkus Armbruster return false; 167d93bb9d5SMarkus Armbruster } 168d93bb9d5SMarkus Armbruster 169d93bb9d5SMarkus Armbruster static void add_query_tests(QmpSchema *schema) 170d93bb9d5SMarkus Armbruster { 171d93bb9d5SMarkus Armbruster SchemaInfoList *tail; 172d93bb9d5SMarkus Armbruster SchemaInfo *si, *arg_type, *ret_type; 173d93bb9d5SMarkus Armbruster char *test_name; 174d93bb9d5SMarkus Armbruster 175d93bb9d5SMarkus Armbruster /* Test the query-like commands */ 176d93bb9d5SMarkus Armbruster for (tail = schema->list; tail; tail = tail->next) { 177d93bb9d5SMarkus Armbruster si = tail->value; 178d93bb9d5SMarkus Armbruster if (si->meta_type != SCHEMA_META_TYPE_COMMAND) { 179d93bb9d5SMarkus Armbruster continue; 180d93bb9d5SMarkus Armbruster } 181d93bb9d5SMarkus Armbruster 182d93bb9d5SMarkus Armbruster if (query_is_blacklisted(si->name)) { 183d93bb9d5SMarkus Armbruster continue; 184d93bb9d5SMarkus Armbruster } 185d93bb9d5SMarkus Armbruster 186d93bb9d5SMarkus Armbruster arg_type = qmp_schema_lookup(schema, si->u.command.arg_type); 187d93bb9d5SMarkus Armbruster if (object_type_has_mandatory_members(arg_type)) { 188d93bb9d5SMarkus Armbruster continue; 189d93bb9d5SMarkus Armbruster } 190d93bb9d5SMarkus Armbruster 191d93bb9d5SMarkus Armbruster ret_type = qmp_schema_lookup(schema, si->u.command.ret_type); 192d93bb9d5SMarkus Armbruster if (ret_type->meta_type == SCHEMA_META_TYPE_OBJECT 193d93bb9d5SMarkus Armbruster && !ret_type->u.object.members) { 194d93bb9d5SMarkus Armbruster continue; 195d93bb9d5SMarkus Armbruster } 196d93bb9d5SMarkus Armbruster 197d93bb9d5SMarkus Armbruster test_name = g_strdup_printf("qmp/%s", si->name); 198d93bb9d5SMarkus Armbruster qtest_add_data_func(test_name, si->name, test_query); 199d93bb9d5SMarkus Armbruster g_free(test_name); 200d93bb9d5SMarkus Armbruster } 201d93bb9d5SMarkus Armbruster } 202d93bb9d5SMarkus Armbruster 2035b88849eSEric Auger static void test_object_add_failure_modes(void) 204442b09b8SMarc-André Lureau { 205442b09b8SMarc-André Lureau QTestState *qts; 206442b09b8SMarc-André Lureau QDict *resp; 207442b09b8SMarc-André Lureau 2085b88849eSEric Auger /* attempt to create an object without props */ 209442b09b8SMarc-André Lureau qts = qtest_init(common_args); 210442b09b8SMarc-André Lureau resp = qtest_qmp(qts, "{'execute': 'object-add', 'arguments':" 211442b09b8SMarc-André Lureau " {'qom-type': 'memory-backend-ram', 'id': 'ram1' } }"); 212442b09b8SMarc-André Lureau g_assert_nonnull(resp); 213442b09b8SMarc-André Lureau qmp_assert_error_class(resp, "GenericError"); 214442b09b8SMarc-André Lureau 2155b88849eSEric Auger /* attempt to create an object without qom-type */ 2169fc719b8SEric Auger resp = qtest_qmp(qts, "{'execute': 'object-add', 'arguments':" 2175b88849eSEric Auger " {'id': 'ram1' } }"); 2185b88849eSEric Auger g_assert_nonnull(resp); 2195b88849eSEric Auger qmp_assert_error_class(resp, "GenericError"); 2205b88849eSEric Auger 2215b88849eSEric Auger /* attempt to delete an object that does not exist */ 2225b88849eSEric Auger resp = qtest_qmp(qts, "{'execute': 'object-del', 'arguments':" 2235b88849eSEric Auger " {'id': 'ram1' } }"); 2245b88849eSEric Auger g_assert_nonnull(resp); 2255b88849eSEric Auger qmp_assert_error_class(resp, "GenericError"); 2265b88849eSEric Auger 2275b88849eSEric Auger /* attempt to create 2 objects with duplicate id */ 2285b88849eSEric Auger resp = qtest_qmp(qts, "{'execute': 'object-add', 'arguments':" 2295b88849eSEric Auger " {'qom-type': 'memory-backend-ram', 'id': 'ram1'," 2305b88849eSEric Auger " 'props': {'size': 1048576 } } }"); 2319fc719b8SEric Auger g_assert_nonnull(resp); 2329fc719b8SEric Auger g_assert(qdict_haskey(resp, "return")); 23315c51f72SLi Qiang qobject_unref(resp); 23415c51f72SLi Qiang 2359fc719b8SEric Auger resp = qtest_qmp(qts, "{'execute': 'object-add', 'arguments':" 2365b88849eSEric Auger " {'qom-type': 'memory-backend-ram', 'id': 'ram1'," 2375b88849eSEric Auger " 'props': {'size': 1048576 } } }"); 2389fc719b8SEric Auger g_assert_nonnull(resp); 2399fc719b8SEric Auger qmp_assert_error_class(resp, "GenericError"); 2405b88849eSEric Auger 2415b88849eSEric Auger /* delete ram1 object */ 2425b88849eSEric Auger resp = qtest_qmp(qts, "{'execute': 'object-del', 'arguments':" 2435b88849eSEric Auger " {'id': 'ram1' } }"); 2445b88849eSEric Auger g_assert_nonnull(resp); 2455b88849eSEric Auger g_assert(qdict_haskey(resp, "return")); 24615c51f72SLi Qiang qobject_unref(resp); 2475b88849eSEric Auger 2485b88849eSEric Auger /* attempt to create an object with a property of a wrong type */ 2495b88849eSEric Auger resp = qtest_qmp(qts, "{'execute': 'object-add', 'arguments':" 2505b88849eSEric Auger " {'qom-type': 'memory-backend-ram', 'id': 'ram1'," 2515b88849eSEric Auger " 'props': {'size': '1048576' } } }"); 2525b88849eSEric Auger g_assert_nonnull(resp); 2535b88849eSEric Auger /* now do it right */ 2545b88849eSEric Auger qmp_assert_error_class(resp, "GenericError"); 25515c51f72SLi Qiang 2565b88849eSEric Auger resp = qtest_qmp(qts, "{'execute': 'object-add', 'arguments':" 2575b88849eSEric Auger " {'qom-type': 'memory-backend-ram', 'id': 'ram1'," 2585b88849eSEric Auger " 'props': {'size': 1048576 } } }"); 2595b88849eSEric Auger g_assert_nonnull(resp); 2605b88849eSEric Auger g_assert(qdict_haskey(resp, "return")); 26115c51f72SLi Qiang qobject_unref(resp); 2625b88849eSEric Auger 2635b88849eSEric Auger /* delete ram1 object */ 2645b88849eSEric Auger resp = qtest_qmp(qts, "{'execute': 'object-del', 'arguments':" 2655b88849eSEric Auger " {'id': 'ram1' } }"); 2665b88849eSEric Auger g_assert_nonnull(resp); 2675b88849eSEric Auger g_assert(qdict_haskey(resp, "return")); 26815c51f72SLi Qiang qobject_unref(resp); 2695b88849eSEric Auger 2705b88849eSEric Auger /* attempt to create an object without the id */ 2715b88849eSEric Auger resp = qtest_qmp(qts, "{'execute': 'object-add', 'arguments':" 2725b88849eSEric Auger " {'qom-type': 'memory-backend-ram'," 2735b88849eSEric Auger " 'props': {'size': 1048576 } } }"); 2745b88849eSEric Auger g_assert_nonnull(resp); 2755b88849eSEric Auger qmp_assert_error_class(resp, "GenericError"); 27615c51f72SLi Qiang 2775b88849eSEric Auger /* now do it right */ 2785b88849eSEric Auger resp = qtest_qmp(qts, "{'execute': 'object-add', 'arguments':" 2795b88849eSEric Auger " {'qom-type': 'memory-backend-ram', 'id': 'ram1'," 2805b88849eSEric Auger " 'props': {'size': 1048576 } } }"); 2815b88849eSEric Auger g_assert_nonnull(resp); 2825b88849eSEric Auger g_assert(qdict_haskey(resp, "return")); 28315c51f72SLi Qiang qobject_unref(resp); 2845b88849eSEric Auger 2855b88849eSEric Auger /* delete ram1 object */ 2865b88849eSEric Auger resp = qtest_qmp(qts, "{'execute': 'object-del', 'arguments':" 2875b88849eSEric Auger " {'id': 'ram1' } }"); 2885b88849eSEric Auger g_assert_nonnull(resp); 2895b88849eSEric Auger g_assert(qdict_haskey(resp, "return")); 29015c51f72SLi Qiang qobject_unref(resp); 2915b88849eSEric Auger 2925b88849eSEric Auger /* attempt to set a non existing property */ 2935b88849eSEric Auger resp = qtest_qmp(qts, "{'execute': 'object-add', 'arguments':" 2945b88849eSEric Auger " {'qom-type': 'memory-backend-ram', 'id': 'ram1'," 2955b88849eSEric Auger " 'props': {'sized': 1048576 } } }"); 2965b88849eSEric Auger g_assert_nonnull(resp); 2975b88849eSEric Auger qmp_assert_error_class(resp, "GenericError"); 29815c51f72SLi Qiang 2995b88849eSEric Auger /* now do it right */ 3005b88849eSEric Auger resp = qtest_qmp(qts, "{'execute': 'object-add', 'arguments':" 3015b88849eSEric Auger " {'qom-type': 'memory-backend-ram', 'id': 'ram1'," 3025b88849eSEric Auger " 'props': {'size': 1048576 } } }"); 3035b88849eSEric Auger g_assert_nonnull(resp); 3045b88849eSEric Auger g_assert(qdict_haskey(resp, "return")); 30515c51f72SLi Qiang qobject_unref(resp); 3065b88849eSEric Auger 3075b88849eSEric Auger /* delete ram1 object without id */ 3085b88849eSEric Auger resp = qtest_qmp(qts, "{'execute': 'object-del', 'arguments':" 3095b88849eSEric Auger " {'ida': 'ram1' } }"); 3105b88849eSEric Auger g_assert_nonnull(resp); 31115c51f72SLi Qiang qobject_unref(resp); 3125b88849eSEric Auger 3135b88849eSEric Auger /* delete ram1 object */ 3145b88849eSEric Auger resp = qtest_qmp(qts, "{'execute': 'object-del', 'arguments':" 3155b88849eSEric Auger " {'id': 'ram1' } }"); 3165b88849eSEric Auger g_assert_nonnull(resp); 3175b88849eSEric Auger g_assert(qdict_haskey(resp, "return")); 31815c51f72SLi Qiang qobject_unref(resp); 3195b88849eSEric Auger 3205b88849eSEric Auger /* delete ram1 object that does not exist anymore*/ 3215b88849eSEric Auger resp = qtest_qmp(qts, "{'execute': 'object-del', 'arguments':" 3225b88849eSEric Auger " {'id': 'ram1' } }"); 3235b88849eSEric Auger g_assert_nonnull(resp); 3245b88849eSEric Auger qmp_assert_error_class(resp, "GenericError"); 3255b88849eSEric Auger 3269fc719b8SEric Auger qtest_quit(qts); 3279fc719b8SEric Auger } 3289fc719b8SEric Auger 329d93bb9d5SMarkus Armbruster int main(int argc, char *argv[]) 330d93bb9d5SMarkus Armbruster { 331d93bb9d5SMarkus Armbruster QmpSchema schema; 332d93bb9d5SMarkus Armbruster int ret; 333d93bb9d5SMarkus Armbruster 334d93bb9d5SMarkus Armbruster g_test_init(&argc, &argv, NULL); 335d93bb9d5SMarkus Armbruster 336d93bb9d5SMarkus Armbruster qmp_schema_init(&schema); 337d93bb9d5SMarkus Armbruster add_query_tests(&schema); 338442b09b8SMarc-André Lureau 3395b88849eSEric Auger qtest_add_func("qmp/object-add-failure-modes", 3405b88849eSEric Auger test_object_add_failure_modes); 341442b09b8SMarc-André Lureau 342d93bb9d5SMarkus Armbruster ret = g_test_run(); 343d93bb9d5SMarkus Armbruster 344d93bb9d5SMarkus Armbruster qmp_schema_cleanup(&schema); 345d93bb9d5SMarkus Armbruster return ret; 346d93bb9d5SMarkus Armbruster } 347