xref: /qemu/tests/unit/test-qmp-cmds.c (revision 05ebf841efac494d8bd1f6d74642c3e9a3df4c19)
1681c28a3SPeter Maydell #include "qemu/osdep.h"
291fa93e5SMarkus Armbruster #include "qapi/compat-policy.h"
36b673957SMarkus Armbruster #include "qapi/qmp/qdict.h"
43d16042cSMarkus Armbruster #include "qapi/qmp/qjson.h"
515280c36SMarkus Armbruster #include "qapi/qmp/qnum.h"
66b673957SMarkus Armbruster #include "qapi/qmp/qstring.h"
7e688df6bSMarkus Armbruster #include "qapi/error.h"
8b3db211fSDaniel P. Berrange #include "qapi/qobject-input-visitor.h"
9d98150f0SLaszlo Ersek #include "tests/test-qapi-types.h"
10d98150f0SLaszlo Ersek #include "tests/test-qapi-visit.h"
11eb815e24SMarkus Armbruster #include "test-qapi-commands.h"
1200ca24ffSMarkus Armbruster #include "test-qapi-init-commands.h"
1369ed8366SMichael Roth 
141527badbSMarkus Armbruster static QmpCommandList qmp_commands;
151527badbSMarkus Armbruster 
16*05ebf841SMarkus Armbruster UserDefThree *qmp_test_cmd_return_def_three(Error **errp)
17967c8851SMarc-André Lureau {
18967c8851SMarc-André Lureau     return NULL;
19967c8851SMarc-André Lureau }
20967c8851SMarc-André Lureau 
2169ed8366SMichael Roth void qmp_user_def_cmd(Error **errp)
2269ed8366SMichael Roth {
2369ed8366SMichael Roth }
2469ed8366SMichael Roth 
257b13f2c2SIgor Mammedov void qmp_test_flags_command(Error **errp)
261a1b11dcSPeter Xu {
271a1b11dcSPeter Xu }
281a1b11dcSPeter Xu 
29ae6bf766SMarc-André Lureau void qmp_cmd_success_response(Error **errp)
30ae6bf766SMarc-André Lureau {
31ae6bf766SMarc-André Lureau }
32ae6bf766SMarc-André Lureau 
3304f22362SKevin Wolf void qmp_coroutine_cmd(Error **errp)
3404f22362SKevin Wolf {
3504f22362SKevin Wolf }
3604f22362SKevin Wolf 
37972a1101SEric Blake Empty2 *qmp_user_def_cmd0(Error **errp)
38972a1101SEric Blake {
39972a1101SEric Blake     return g_new0(Empty2, 1);
40972a1101SEric Blake }
41972a1101SEric Blake 
4269ed8366SMichael Roth void qmp_user_def_cmd1(UserDefOne * ud1, Error **errp)
4369ed8366SMichael Roth {
4469ed8366SMichael Roth }
4569ed8366SMichael Roth 
4691fa93e5SMarkus Armbruster FeatureStruct1 *qmp_test_features0(bool has_fs0, FeatureStruct0 *fs0,
4791fa93e5SMarkus Armbruster                                    bool has_fs1, FeatureStruct1 *fs1,
4891fa93e5SMarkus Armbruster                                    bool has_fs2, FeatureStruct2 *fs2,
4991fa93e5SMarkus Armbruster                                    bool has_fs3, FeatureStruct3 *fs3,
5091fa93e5SMarkus Armbruster                                    bool has_fs4, FeatureStruct4 *fs4,
5191fa93e5SMarkus Armbruster                                    bool has_cfs1, CondFeatureStruct1 *cfs1,
5291fa93e5SMarkus Armbruster                                    bool has_cfs2, CondFeatureStruct2 *cfs2,
5391fa93e5SMarkus Armbruster                                    bool has_cfs3, CondFeatureStruct3 *cfs3,
548aa3a33eSKevin Wolf                                    Error **errp)
558aa3a33eSKevin Wolf {
5691fa93e5SMarkus Armbruster     return g_new0(FeatureStruct1, 1);
578aa3a33eSKevin Wolf }
588aa3a33eSKevin Wolf 
592e2e0df2SPeter Krempa void qmp_test_command_features1(Error **errp)
602e2e0df2SPeter Krempa {
612e2e0df2SPeter Krempa }
622e2e0df2SPeter Krempa 
632e2e0df2SPeter Krempa void qmp_test_command_features3(Error **errp)
642e2e0df2SPeter Krempa {
652e2e0df2SPeter Krempa }
662e2e0df2SPeter Krempa 
672e2e0df2SPeter Krempa void qmp_test_command_cond_features1(Error **errp)
682e2e0df2SPeter Krempa {
692e2e0df2SPeter Krempa }
702e2e0df2SPeter Krempa 
712e2e0df2SPeter Krempa void qmp_test_command_cond_features2(Error **errp)
722e2e0df2SPeter Krempa {
732e2e0df2SPeter Krempa }
742e2e0df2SPeter Krempa 
752e2e0df2SPeter Krempa void qmp_test_command_cond_features3(Error **errp)
762e2e0df2SPeter Krempa {
772e2e0df2SPeter Krempa }
782e2e0df2SPeter Krempa 
79ab22ad96SMarkus Armbruster UserDefTwo *qmp_user_def_cmd2(UserDefOne *ud1a,
80ab22ad96SMarkus Armbruster                               bool has_udb1, UserDefOne *ud1b,
81ab22ad96SMarkus Armbruster                               Error **errp)
8269ed8366SMichael Roth {
8369ed8366SMichael Roth     UserDefTwo *ret;
847267c094SAnthony Liguori     UserDefOne *ud1c = g_malloc0(sizeof(UserDefOne));
857267c094SAnthony Liguori     UserDefOne *ud1d = g_malloc0(sizeof(UserDefOne));
8669ed8366SMichael Roth 
8769ed8366SMichael Roth     ud1c->string = strdup(ud1a->string);
88ddf21908SEric Blake     ud1c->integer = ud1a->integer;
89ab22ad96SMarkus Armbruster     ud1d->string = strdup(has_udb1 ? ud1b->string : "blah0");
90ddf21908SEric Blake     ud1d->integer = has_udb1 ? ud1b->integer : 0;
9169ed8366SMichael Roth 
92b6fcf32dSEric Blake     ret = g_new0(UserDefTwo, 1);
93b6fcf32dSEric Blake     ret->string0 = strdup("blah1");
946446a592SEric Blake     ret->dict1 = g_new0(UserDefTwoDict, 1);
956446a592SEric Blake     ret->dict1->string1 = strdup("blah2");
966446a592SEric Blake     ret->dict1->dict2 = g_new0(UserDefTwoDictDict, 1);
976446a592SEric Blake     ret->dict1->dict2->userdef = ud1c;
986446a592SEric Blake     ret->dict1->dict2->string = strdup("blah3");
996446a592SEric Blake     ret->dict1->dict3 = g_new0(UserDefTwoDictDict, 1);
1006446a592SEric Blake     ret->dict1->has_dict3 = true;
1016446a592SEric Blake     ret->dict1->dict3->userdef = ud1d;
1026446a592SEric Blake     ret->dict1->dict3->string = strdup("blah4");
10369ed8366SMichael Roth 
10469ed8366SMichael Roth     return ret;
10569ed8366SMichael Roth }
10669ed8366SMichael Roth 
107cae95eaeSEric Blake int64_t qmp_guest_get_time(int64_t a, bool has_b, int64_t b, Error **errp)
108c2216a8aSMarkus Armbruster {
109c2216a8aSMarkus Armbruster     return a + (has_b ? b : 0);
110c2216a8aSMarkus Armbruster }
111c2216a8aSMarkus Armbruster 
11228770e05SMarkus Armbruster QObject *qmp_guest_sync(QObject *arg, Error **errp)
11328770e05SMarkus Armbruster {
11428770e05SMarkus Armbruster     return arg;
11528770e05SMarkus Armbruster }
11628770e05SMarkus Armbruster 
117c818408eSEric Blake void qmp_boxed_struct(UserDefZero *arg, Error **errp)
118c818408eSEric Blake {
119c818408eSEric Blake }
120c818408eSEric Blake 
121b359f4b2SMarkus Armbruster void qmp_boxed_union(UserDefListUnion *arg, Error **errp)
122c818408eSEric Blake {
123c818408eSEric Blake }
124c818408eSEric Blake 
125675b214bSMarkus Armbruster void qmp_boxed_empty(Empty1 *arg, Error **errp)
126675b214bSMarkus Armbruster {
127675b214bSMarkus Armbruster }
128675b214bSMarkus Armbruster 
129e3c4c3d7SEric Blake __org_qemu_x_Union1 *qmp___org_qemu_x_command(__org_qemu_x_EnumList *a,
130e3c4c3d7SEric Blake                                               __org_qemu_x_StructList *b,
131e3c4c3d7SEric Blake                                               __org_qemu_x_Union2 *c,
132e3c4c3d7SEric Blake                                               __org_qemu_x_Alt *d,
133e3c4c3d7SEric Blake                                               Error **errp)
134e3c4c3d7SEric Blake {
135e3c4c3d7SEric Blake     __org_qemu_x_Union1 *ret = g_new0(__org_qemu_x_Union1, 1);
136e3c4c3d7SEric Blake 
137c363acefSEric Blake     ret->type = ORG_QEMU_X_UNION1_KIND___ORG_QEMU_X_BRANCH;
13832bafa8fSEric Blake     ret->u.__org_qemu_x_branch.data = strdup("blah1");
139e3c4c3d7SEric Blake 
140c43567c1SEric Blake     /* Also test that 'wchar-t' was munged to 'q_wchar_t' */
141c43567c1SEric Blake     if (b && b->value && !b->value->has_q_wchar_t) {
142c43567c1SEric Blake         b->value->q_wchar_t = 1;
143c43567c1SEric Blake     }
144e3c4c3d7SEric Blake     return ret;
145e3c4c3d7SEric Blake }
146e3c4c3d7SEric Blake 
147e3c4c3d7SEric Blake 
1483d16042cSMarkus Armbruster static QObject *do_qmp_dispatch(bool allow_oob, const char *template, ...)
1493306459aSMarkus Armbruster {
1503d16042cSMarkus Armbruster     va_list ap;
1513d16042cSMarkus Armbruster     QDict *req, *resp;
1523306459aSMarkus Armbruster     QObject *ret;
1533306459aSMarkus Armbruster 
1543d16042cSMarkus Armbruster     va_start(ap, template);
1553d16042cSMarkus Armbruster     req = qdict_from_vjsonf_nofail(template, ap);
1563d16042cSMarkus Armbruster     va_end(ap);
1573d16042cSMarkus Armbruster 
15841725fa7SKevin Wolf     resp = qmp_dispatch(&qmp_commands, QOBJECT(req), allow_oob, NULL);
159ef9f5f0dSMarkus Armbruster     g_assert(resp);
1603306459aSMarkus Armbruster     ret = qdict_get(resp, "return");
1613306459aSMarkus Armbruster     g_assert(ret);
162ef9f5f0dSMarkus Armbruster     g_assert(qdict_size(resp) == 1);
1633306459aSMarkus Armbruster 
1643306459aSMarkus Armbruster     qobject_ref(ret);
1653306459aSMarkus Armbruster     qobject_unref(resp);
1663d16042cSMarkus Armbruster     qobject_unref(req);
1673306459aSMarkus Armbruster     return ret;
1683306459aSMarkus Armbruster }
1693306459aSMarkus Armbruster 
1703d16042cSMarkus Armbruster static void do_qmp_dispatch_error(bool allow_oob, ErrorClass cls,
1713d16042cSMarkus Armbruster                                   const char *template, ...)
1723306459aSMarkus Armbruster {
1733d16042cSMarkus Armbruster     va_list ap;
1743d16042cSMarkus Armbruster     QDict *req, *resp;
175ef9f5f0dSMarkus Armbruster     QDict *error;
1763306459aSMarkus Armbruster 
1773d16042cSMarkus Armbruster     va_start(ap, template);
1783d16042cSMarkus Armbruster     req = qdict_from_vjsonf_nofail(template, ap);
1793d16042cSMarkus Armbruster     va_end(ap);
1803d16042cSMarkus Armbruster 
18141725fa7SKevin Wolf     resp = qmp_dispatch(&qmp_commands, QOBJECT(req), allow_oob, NULL);
182ef9f5f0dSMarkus Armbruster     g_assert(resp);
183ef9f5f0dSMarkus Armbruster     error = qdict_get_qdict(resp, "error");
184ef9f5f0dSMarkus Armbruster     g_assert(error);
185ef9f5f0dSMarkus Armbruster     g_assert_cmpstr(qdict_get_try_str(error, "class"),
186ef9f5f0dSMarkus Armbruster                     ==, QapiErrorClass_str(cls));
187ef9f5f0dSMarkus Armbruster     g_assert(qdict_get_try_str(error, "desc"));
188ef9f5f0dSMarkus Armbruster     g_assert(qdict_size(error) == 2);
189ef9f5f0dSMarkus Armbruster     g_assert(qdict_size(resp) == 1);
1903306459aSMarkus Armbruster 
1913306459aSMarkus Armbruster     qobject_unref(resp);
1923d16042cSMarkus Armbruster     qobject_unref(req);
1933306459aSMarkus Armbruster }
1943306459aSMarkus Armbruster 
19569ed8366SMichael Roth /* test commands with no input and no return value */
19669ed8366SMichael Roth static void test_dispatch_cmd(void)
19769ed8366SMichael Roth {
198ef9f5f0dSMarkus Armbruster     QDict *ret;
19969ed8366SMichael Roth 
2003d16042cSMarkus Armbruster     ret = qobject_to(QDict,
2013d16042cSMarkus Armbruster                      do_qmp_dispatch(false,
202*05ebf841SMarkus Armbruster                                      "{ 'execute': 'user-def-cmd' }"));
203ef9f5f0dSMarkus Armbruster     assert(ret && qdict_size(ret) == 0);
2043306459aSMarkus Armbruster     qobject_unref(ret);
20569ed8366SMichael Roth }
20669ed8366SMichael Roth 
207c55f070bSPeter Xu static void test_dispatch_cmd_oob(void)
208c55f070bSPeter Xu {
209ef9f5f0dSMarkus Armbruster     QDict *ret;
210c55f070bSPeter Xu 
2113d16042cSMarkus Armbruster     ret = qobject_to(QDict,
2123d16042cSMarkus Armbruster                      do_qmp_dispatch(true,
2133d16042cSMarkus Armbruster                                      "{ 'exec-oob': 'test-flags-command' }"));
214ef9f5f0dSMarkus Armbruster     assert(ret && qdict_size(ret) == 0);
2153306459aSMarkus Armbruster     qobject_unref(ret);
216c55f070bSPeter Xu }
217c55f070bSPeter Xu 
21869ed8366SMichael Roth /* test commands that return an error due to invalid parameters */
21951009170SPeter Maydell static void test_dispatch_cmd_failure(void)
22069ed8366SMichael Roth {
2213d16042cSMarkus Armbruster     /* missing arguments */
2223d16042cSMarkus Armbruster     do_qmp_dispatch_error(false, ERROR_CLASS_GENERIC_ERROR,
223*05ebf841SMarkus Armbruster                           "{ 'execute': 'user-def-cmd2' }");
22469ed8366SMichael Roth 
2253d16042cSMarkus Armbruster     /* extra arguments */
2263d16042cSMarkus Armbruster     do_qmp_dispatch_error(false, ERROR_CLASS_GENERIC_ERROR,
227*05ebf841SMarkus Armbruster                           "{ 'execute': 'user-def-cmd',"
2283d16042cSMarkus Armbruster                           " 'arguments': { 'a': 66 } }");
22969ed8366SMichael Roth }
23069ed8366SMichael Roth 
231ae6bf766SMarc-André Lureau static void test_dispatch_cmd_success_response(void)
232ae6bf766SMarc-André Lureau {
233ae6bf766SMarc-André Lureau     QDict *req = qdict_new();
234ae6bf766SMarc-André Lureau     QDict *resp;
235ae6bf766SMarc-André Lureau 
236ae6bf766SMarc-André Lureau     qdict_put_str(req, "execute", "cmd-success-response");
23741725fa7SKevin Wolf     resp = qmp_dispatch(&qmp_commands, QOBJECT(req), false, NULL);
238ae6bf766SMarc-André Lureau     g_assert_null(resp);
239ae6bf766SMarc-André Lureau     qobject_unref(req);
240ae6bf766SMarc-André Lureau }
241ae6bf766SMarc-André Lureau 
24269ed8366SMichael Roth /* test commands that involve both input parameters and return values */
24369ed8366SMichael Roth static void test_dispatch_cmd_io(void)
24469ed8366SMichael Roth {
245357765feSMarkus Armbruster     QDict *ret, *ret_dict, *ret_dict_dict, *ret_dict_dict_userdef;
246357765feSMarkus Armbruster     QDict *ret_dict_dict2, *ret_dict_dict2_userdef;
24701b2ffceSMarc-André Lureau     QNum *ret3;
24801b2ffceSMarc-André Lureau     int64_t val;
24969ed8366SMichael Roth 
2503d16042cSMarkus Armbruster     ret = qobject_to(QDict, do_qmp_dispatch(false,
251*05ebf841SMarkus Armbruster         "{ 'execute': 'user-def-cmd2', 'arguments': {"
2523d16042cSMarkus Armbruster         " 'ud1a': { 'integer': 42, 'string': 'hello' },"
2533d16042cSMarkus Armbruster         " 'ud1b': { 'integer': 422, 'string': 'hello2' } } }"));
25469ed8366SMichael Roth 
255b6fcf32dSEric Blake     assert(!strcmp(qdict_get_str(ret, "string0"), "blah1"));
256b6fcf32dSEric Blake     ret_dict = qdict_get_qdict(ret, "dict1");
257b6fcf32dSEric Blake     assert(!strcmp(qdict_get_str(ret_dict, "string1"), "blah2"));
258b6fcf32dSEric Blake     ret_dict_dict = qdict_get_qdict(ret_dict, "dict2");
259357765feSMarkus Armbruster     ret_dict_dict_userdef = qdict_get_qdict(ret_dict_dict, "userdef");
260357765feSMarkus Armbruster     assert(qdict_get_int(ret_dict_dict_userdef, "integer") == 42);
261357765feSMarkus Armbruster     assert(!strcmp(qdict_get_str(ret_dict_dict_userdef, "string"), "hello"));
262357765feSMarkus Armbruster     assert(!strcmp(qdict_get_str(ret_dict_dict, "string"), "blah3"));
263b6fcf32dSEric Blake     ret_dict_dict2 = qdict_get_qdict(ret_dict, "dict3");
264357765feSMarkus Armbruster     ret_dict_dict2_userdef = qdict_get_qdict(ret_dict_dict2, "userdef");
265357765feSMarkus Armbruster     assert(qdict_get_int(ret_dict_dict2_userdef, "integer") == 422);
266357765feSMarkus Armbruster     assert(!strcmp(qdict_get_str(ret_dict_dict2_userdef, "string"), "hello2"));
267357765feSMarkus Armbruster     assert(!strcmp(qdict_get_str(ret_dict_dict2, "string"), "blah4"));
268cb3e7f08SMarc-André Lureau     qobject_unref(ret);
269c2216a8aSMarkus Armbruster 
2703d16042cSMarkus Armbruster     ret3 = qobject_to(QNum, do_qmp_dispatch(false,
2713d16042cSMarkus Armbruster         "{ 'execute': 'guest-get-time', 'arguments': { 'a': 66 } }"));
27201b2ffceSMarc-André Lureau     g_assert(qnum_get_try_int(ret3, &val));
27301b2ffceSMarc-André Lureau     g_assert_cmpint(val, ==, 66);
274cb3e7f08SMarc-André Lureau     qobject_unref(ret3);
27569ed8366SMichael Roth }
27669ed8366SMichael Roth 
277d2032598SMarkus Armbruster static void test_dispatch_cmd_deprecated(void)
278d2032598SMarkus Armbruster {
279d2032598SMarkus Armbruster     const char *cmd = "{ 'execute': 'test-command-features1' }";
280d2032598SMarkus Armbruster     QDict *ret;
281d2032598SMarkus Armbruster 
282d2032598SMarkus Armbruster     memset(&compat_policy, 0, sizeof(compat_policy));
283d2032598SMarkus Armbruster 
284d2032598SMarkus Armbruster     /* accept */
285d2032598SMarkus Armbruster     ret = qobject_to(QDict, do_qmp_dispatch(false, cmd));
286d2032598SMarkus Armbruster     assert(ret && qdict_size(ret) == 0);
287d2032598SMarkus Armbruster     qobject_unref(ret);
288d2032598SMarkus Armbruster 
289d2032598SMarkus Armbruster     compat_policy.has_deprecated_input = true;
290d2032598SMarkus Armbruster     compat_policy.deprecated_input = COMPAT_POLICY_INPUT_ACCEPT;
291d2032598SMarkus Armbruster     ret = qobject_to(QDict, do_qmp_dispatch(false, cmd));
292d2032598SMarkus Armbruster     assert(ret && qdict_size(ret) == 0);
293d2032598SMarkus Armbruster     qobject_unref(ret);
294d2032598SMarkus Armbruster 
295d2032598SMarkus Armbruster     compat_policy.deprecated_input = COMPAT_POLICY_INPUT_REJECT;
296d2032598SMarkus Armbruster     do_qmp_dispatch_error(false, ERROR_CLASS_COMMAND_NOT_FOUND, cmd);
297d2032598SMarkus Armbruster }
298d2032598SMarkus Armbruster 
299db291641SMarkus Armbruster static void test_dispatch_cmd_arg_deprecated(void)
300db291641SMarkus Armbruster {
301db291641SMarkus Armbruster     const char *cmd = "{ 'execute': 'test-features0',"
302db291641SMarkus Armbruster         " 'arguments': { 'fs1': { 'foo': 42 } } }";
303db291641SMarkus Armbruster     QDict *ret;
304db291641SMarkus Armbruster 
305db291641SMarkus Armbruster     memset(&compat_policy, 0, sizeof(compat_policy));
306db291641SMarkus Armbruster 
307db291641SMarkus Armbruster     /* accept */
308db291641SMarkus Armbruster     ret = qobject_to(QDict, do_qmp_dispatch(false, cmd));
309db291641SMarkus Armbruster     assert(ret && qdict_size(ret) == 1);
310db291641SMarkus Armbruster     qobject_unref(ret);
311db291641SMarkus Armbruster 
312db291641SMarkus Armbruster     compat_policy.has_deprecated_input = true;
313db291641SMarkus Armbruster     compat_policy.deprecated_input = COMPAT_POLICY_INPUT_ACCEPT;
314db291641SMarkus Armbruster     ret = qobject_to(QDict, do_qmp_dispatch(false, cmd));
315db291641SMarkus Armbruster     assert(ret && qdict_size(ret) == 1);
316db291641SMarkus Armbruster     qobject_unref(ret);
317db291641SMarkus Armbruster 
318db291641SMarkus Armbruster     compat_policy.deprecated_input = COMPAT_POLICY_INPUT_REJECT;
319db291641SMarkus Armbruster     do_qmp_dispatch_error(false, ERROR_CLASS_GENERIC_ERROR, cmd);
320db291641SMarkus Armbruster }
321db291641SMarkus Armbruster 
32291fa93e5SMarkus Armbruster static void test_dispatch_cmd_ret_deprecated(void)
32391fa93e5SMarkus Armbruster {
32491fa93e5SMarkus Armbruster     const char *cmd = "{ 'execute': 'test-features0' }";
32591fa93e5SMarkus Armbruster     QDict *ret;
32691fa93e5SMarkus Armbruster 
32791fa93e5SMarkus Armbruster     memset(&compat_policy, 0, sizeof(compat_policy));
32891fa93e5SMarkus Armbruster 
32991fa93e5SMarkus Armbruster     /* default accept */
33091fa93e5SMarkus Armbruster     ret = qobject_to(QDict, do_qmp_dispatch(false, cmd));
33191fa93e5SMarkus Armbruster     assert(ret && qdict_size(ret) == 1);
33291fa93e5SMarkus Armbruster     qobject_unref(ret);
33391fa93e5SMarkus Armbruster 
33491fa93e5SMarkus Armbruster     compat_policy.has_deprecated_output = true;
33591fa93e5SMarkus Armbruster     compat_policy.deprecated_output = COMPAT_POLICY_OUTPUT_ACCEPT;
33691fa93e5SMarkus Armbruster     ret = qobject_to(QDict, do_qmp_dispatch(false, cmd));
33791fa93e5SMarkus Armbruster     assert(ret && qdict_size(ret) == 1);
33891fa93e5SMarkus Armbruster     qobject_unref(ret);
33991fa93e5SMarkus Armbruster 
34091fa93e5SMarkus Armbruster     compat_policy.deprecated_output = COMPAT_POLICY_OUTPUT_HIDE;
34191fa93e5SMarkus Armbruster     ret = qobject_to(QDict, do_qmp_dispatch(false, cmd));
34291fa93e5SMarkus Armbruster     assert(ret && qdict_size(ret) == 0);
34391fa93e5SMarkus Armbruster     qobject_unref(ret);
34491fa93e5SMarkus Armbruster }
34591fa93e5SMarkus Armbruster 
3465cd5f0d0SMichael Roth /* test generated dealloc functions for generated types */
3475cd5f0d0SMichael Roth static void test_dealloc_types(void)
3485cd5f0d0SMichael Roth {
3495cd5f0d0SMichael Roth     UserDefOne *ud1test, *ud1a, *ud1b;
3505cd5f0d0SMichael Roth     UserDefOneList *ud1list;
3515cd5f0d0SMichael Roth 
3525cd5f0d0SMichael Roth     ud1test = g_malloc0(sizeof(UserDefOne));
353ddf21908SEric Blake     ud1test->integer = 42;
3545cd5f0d0SMichael Roth     ud1test->string = g_strdup("hi there 42");
3555cd5f0d0SMichael Roth 
3565cd5f0d0SMichael Roth     qapi_free_UserDefOne(ud1test);
3575cd5f0d0SMichael Roth 
3585cd5f0d0SMichael Roth     ud1a = g_malloc0(sizeof(UserDefOne));
359ddf21908SEric Blake     ud1a->integer = 43;
3605cd5f0d0SMichael Roth     ud1a->string = g_strdup("hi there 43");
3615cd5f0d0SMichael Roth 
3625cd5f0d0SMichael Roth     ud1b = g_malloc0(sizeof(UserDefOne));
363ddf21908SEric Blake     ud1b->integer = 44;
3645cd5f0d0SMichael Roth     ud1b->string = g_strdup("hi there 44");
3655cd5f0d0SMichael Roth 
3665cd5f0d0SMichael Roth     ud1list = g_malloc0(sizeof(UserDefOneList));
3675cd5f0d0SMichael Roth     ud1list->value = ud1a;
3685cd5f0d0SMichael Roth     ud1list->next = g_malloc0(sizeof(UserDefOneList));
3695cd5f0d0SMichael Roth     ud1list->next->value = ud1b;
3705cd5f0d0SMichael Roth 
3715cd5f0d0SMichael Roth     qapi_free_UserDefOneList(ud1list);
3725cd5f0d0SMichael Roth }
3735cd5f0d0SMichael Roth 
374d98150f0SLaszlo Ersek /* test generated deallocation on an object whose construction was prematurely
375d98150f0SLaszlo Ersek  * terminated due to an error */
376d98150f0SLaszlo Ersek static void test_dealloc_partial(void)
377d98150f0SLaszlo Ersek {
378d98150f0SLaszlo Ersek     static const char text[] = "don't leak me";
379d98150f0SLaszlo Ersek 
380d98150f0SLaszlo Ersek     UserDefTwo *ud2 = NULL;
381d98150f0SLaszlo Ersek     Error *err = NULL;
382d98150f0SLaszlo Ersek 
383d98150f0SLaszlo Ersek     /* create partial object */
384d98150f0SLaszlo Ersek     {
385d98150f0SLaszlo Ersek         QDict *ud2_dict;
386b70ce101SEric Blake         Visitor *v;
387d98150f0SLaszlo Ersek 
388d98150f0SLaszlo Ersek         ud2_dict = qdict_new();
38946f5ac20SEric Blake         qdict_put_str(ud2_dict, "string0", text);
390d98150f0SLaszlo Ersek 
391048abb7bSMarkus Armbruster         v = qobject_input_visitor_new(QOBJECT(ud2_dict));
392b70ce101SEric Blake         visit_type_UserDefTwo(v, NULL, &ud2, &err);
393b70ce101SEric Blake         visit_free(v);
394cb3e7f08SMarc-André Lureau         qobject_unref(ud2_dict);
395d98150f0SLaszlo Ersek     }
396d98150f0SLaszlo Ersek 
39768ab47e4SEric Blake     /* verify that visit_type_XXX() cleans up properly on error */
398a12a5a1aSEric Blake     error_free_or_abort(&err);
39968ab47e4SEric Blake     assert(!ud2);
40068ab47e4SEric Blake 
40168ab47e4SEric Blake     /* Manually create a partial object, leaving ud2->dict1 at NULL */
40268ab47e4SEric Blake     ud2 = g_new0(UserDefTwo, 1);
40368ab47e4SEric Blake     ud2->string0 = g_strdup(text);
404d98150f0SLaszlo Ersek 
405d98150f0SLaszlo Ersek     /* tear down partial object */
406d98150f0SLaszlo Ersek     qapi_free_UserDefTwo(ud2);
407d98150f0SLaszlo Ersek }
408d98150f0SLaszlo Ersek 
409d98150f0SLaszlo Ersek 
41069ed8366SMichael Roth int main(int argc, char **argv)
41169ed8366SMichael Roth {
41269ed8366SMichael Roth     g_test_init(&argc, &argv, NULL);
41369ed8366SMichael Roth 
414214e4a5bSMarc-André Lureau     g_test_add_func("/qmp/dispatch_cmd", test_dispatch_cmd);
415c55f070bSPeter Xu     g_test_add_func("/qmp/dispatch_cmd_oob", test_dispatch_cmd_oob);
416214e4a5bSMarc-André Lureau     g_test_add_func("/qmp/dispatch_cmd_failure", test_dispatch_cmd_failure);
417214e4a5bSMarc-André Lureau     g_test_add_func("/qmp/dispatch_cmd_io", test_dispatch_cmd_io);
418ae6bf766SMarc-André Lureau     g_test_add_func("/qmp/dispatch_cmd_success_response",
419ae6bf766SMarc-André Lureau                     test_dispatch_cmd_success_response);
420d2032598SMarkus Armbruster     g_test_add_func("/qmp/dispatch_cmd_deprecated",
421d2032598SMarkus Armbruster                     test_dispatch_cmd_deprecated);
422db291641SMarkus Armbruster     g_test_add_func("/qmp/dispatch_cmd_arg_deprecated",
423db291641SMarkus Armbruster                     test_dispatch_cmd_arg_deprecated);
42491fa93e5SMarkus Armbruster     g_test_add_func("/qmp/dispatch_cmd_ret_deprecated",
42591fa93e5SMarkus Armbruster                     test_dispatch_cmd_ret_deprecated);
426214e4a5bSMarc-André Lureau     g_test_add_func("/qmp/dealloc_types", test_dealloc_types);
427214e4a5bSMarc-André Lureau     g_test_add_func("/qmp/dealloc_partial", test_dealloc_partial);
42869ed8366SMichael Roth 
4291527badbSMarkus Armbruster     test_qmp_init_marshal(&qmp_commands);
43069ed8366SMichael Roth     g_test_run();
43169ed8366SMichael Roth 
43269ed8366SMichael Roth     return 0;
43369ed8366SMichael Roth }
434