xref: /qemu/tests/unit/test-qmp-cmds.c (revision 4b2fc7dbc4203c52b7726249328fcde49626f565)
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 
1605ebf841SMarkus 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 
46*4b2fc7dbSMarkus Armbruster FeatureStruct1 *qmp_test_features0(FeatureStruct0 *fs0,
47*4b2fc7dbSMarkus Armbruster                                    FeatureStruct1 *fs1,
48*4b2fc7dbSMarkus Armbruster                                    FeatureStruct2 *fs2,
49*4b2fc7dbSMarkus Armbruster                                    FeatureStruct3 *fs3,
50*4b2fc7dbSMarkus Armbruster                                    FeatureStruct4 *fs4,
51*4b2fc7dbSMarkus Armbruster                                    CondFeatureStruct1 *cfs1,
52*4b2fc7dbSMarkus Armbruster                                    CondFeatureStruct2 *cfs2,
53*4b2fc7dbSMarkus Armbruster                                    CondFeatureStruct3 *cfs3,
54*4b2fc7dbSMarkus Armbruster                                    CondFeatureStruct4 *cfs4,
558aa3a33eSKevin Wolf                                    Error **errp)
568aa3a33eSKevin Wolf {
5791fa93e5SMarkus Armbruster     return g_new0(FeatureStruct1, 1);
588aa3a33eSKevin Wolf }
598aa3a33eSKevin Wolf 
602e2e0df2SPeter Krempa void qmp_test_command_features1(Error **errp)
612e2e0df2SPeter Krempa {
622e2e0df2SPeter Krempa }
632e2e0df2SPeter Krempa 
642e2e0df2SPeter Krempa void qmp_test_command_features3(Error **errp)
652e2e0df2SPeter Krempa {
662e2e0df2SPeter Krempa }
672e2e0df2SPeter Krempa 
682e2e0df2SPeter Krempa void qmp_test_command_cond_features1(Error **errp)
692e2e0df2SPeter Krempa {
702e2e0df2SPeter Krempa }
712e2e0df2SPeter Krempa 
722e2e0df2SPeter Krempa void qmp_test_command_cond_features2(Error **errp)
732e2e0df2SPeter Krempa {
742e2e0df2SPeter Krempa }
752e2e0df2SPeter Krempa 
762e2e0df2SPeter Krempa void qmp_test_command_cond_features3(Error **errp)
772e2e0df2SPeter Krempa {
782e2e0df2SPeter Krempa }
792e2e0df2SPeter Krempa 
80*4b2fc7dbSMarkus Armbruster UserDefTwo *qmp_user_def_cmd2(UserDefOne *ud1a, UserDefOne *ud1b,
81ab22ad96SMarkus Armbruster                               Error **errp)
8269ed8366SMichael Roth {
8369ed8366SMichael Roth     UserDefTwo *ret;
84b21e2380SMarkus Armbruster     UserDefOne *ud1c = g_new0(UserDefOne, 1);
85b21e2380SMarkus Armbruster     UserDefOne *ud1d = g_new0(UserDefOne, 1);
8669ed8366SMichael Roth 
8769ed8366SMichael Roth     ud1c->string = strdup(ud1a->string);
88ddf21908SEric Blake     ud1c->integer = ud1a->integer;
89*4b2fc7dbSMarkus Armbruster     ud1d->string = strdup(ud1b ? ud1b->string : "blah0");
90*4b2fc7dbSMarkus Armbruster     ud1d->integer = ud1b ? 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->dict3->userdef = ud1d;
1016446a592SEric Blake     ret->dict1->dict3->string = strdup("blah4");
10269ed8366SMichael Roth 
10369ed8366SMichael Roth     return ret;
10469ed8366SMichael Roth }
10569ed8366SMichael Roth 
106cae95eaeSEric Blake int64_t qmp_guest_get_time(int64_t a, bool has_b, int64_t b, Error **errp)
107c2216a8aSMarkus Armbruster {
108c2216a8aSMarkus Armbruster     return a + (has_b ? b : 0);
109c2216a8aSMarkus Armbruster }
110c2216a8aSMarkus Armbruster 
11128770e05SMarkus Armbruster QObject *qmp_guest_sync(QObject *arg, Error **errp)
11228770e05SMarkus Armbruster {
11328770e05SMarkus Armbruster     return arg;
11428770e05SMarkus Armbruster }
11528770e05SMarkus Armbruster 
116c818408eSEric Blake void qmp_boxed_struct(UserDefZero *arg, Error **errp)
117c818408eSEric Blake {
118c818408eSEric Blake }
119c818408eSEric Blake 
1201e65e16cSMarkus Armbruster void qmp_boxed_union(UserDefFlatUnion *arg, Error **errp)
121c818408eSEric Blake {
122c818408eSEric Blake }
123c818408eSEric Blake 
124675b214bSMarkus Armbruster void qmp_boxed_empty(Empty1 *arg, Error **errp)
125675b214bSMarkus Armbruster {
126675b214bSMarkus Armbruster }
127675b214bSMarkus Armbruster 
128bb5821ddSMarkus Armbruster void qmp___org_qemu_x_command(__org_qemu_x_EnumList *a,
129e3c4c3d7SEric Blake                               __org_qemu_x_StructList *b,
130bb5821ddSMarkus Armbruster                               __org_qemu_x_Union *c,
131e3c4c3d7SEric Blake                               __org_qemu_x_Alt *d,
132e3c4c3d7SEric Blake                               Error **errp)
133e3c4c3d7SEric Blake {
134c43567c1SEric Blake     /* Also test that 'wchar-t' was munged to 'q_wchar_t' */
135c43567c1SEric Blake     if (b && b->value && !b->value->has_q_wchar_t) {
136c43567c1SEric Blake         b->value->q_wchar_t = 1;
137c43567c1SEric Blake     }
138e3c4c3d7SEric Blake }
139e3c4c3d7SEric Blake 
140e3c4c3d7SEric Blake 
1413d16042cSMarkus Armbruster static QObject *do_qmp_dispatch(bool allow_oob, const char *template, ...)
1423306459aSMarkus Armbruster {
1433d16042cSMarkus Armbruster     va_list ap;
1443d16042cSMarkus Armbruster     QDict *req, *resp;
1453306459aSMarkus Armbruster     QObject *ret;
1463306459aSMarkus Armbruster 
1473d16042cSMarkus Armbruster     va_start(ap, template);
1483d16042cSMarkus Armbruster     req = qdict_from_vjsonf_nofail(template, ap);
1493d16042cSMarkus Armbruster     va_end(ap);
1503d16042cSMarkus Armbruster 
15141725fa7SKevin Wolf     resp = qmp_dispatch(&qmp_commands, QOBJECT(req), allow_oob, NULL);
152ef9f5f0dSMarkus Armbruster     g_assert(resp);
1533306459aSMarkus Armbruster     ret = qdict_get(resp, "return");
1543306459aSMarkus Armbruster     g_assert(ret);
155ef9f5f0dSMarkus Armbruster     g_assert(qdict_size(resp) == 1);
1563306459aSMarkus Armbruster 
1573306459aSMarkus Armbruster     qobject_ref(ret);
1583306459aSMarkus Armbruster     qobject_unref(resp);
1593d16042cSMarkus Armbruster     qobject_unref(req);
1603306459aSMarkus Armbruster     return ret;
1613306459aSMarkus Armbruster }
1623306459aSMarkus Armbruster 
1633d16042cSMarkus Armbruster static void do_qmp_dispatch_error(bool allow_oob, ErrorClass cls,
1643d16042cSMarkus Armbruster                                   const char *template, ...)
1653306459aSMarkus Armbruster {
1663d16042cSMarkus Armbruster     va_list ap;
1673d16042cSMarkus Armbruster     QDict *req, *resp;
168ef9f5f0dSMarkus Armbruster     QDict *error;
1693306459aSMarkus Armbruster 
1703d16042cSMarkus Armbruster     va_start(ap, template);
1713d16042cSMarkus Armbruster     req = qdict_from_vjsonf_nofail(template, ap);
1723d16042cSMarkus Armbruster     va_end(ap);
1733d16042cSMarkus Armbruster 
17441725fa7SKevin Wolf     resp = qmp_dispatch(&qmp_commands, QOBJECT(req), allow_oob, NULL);
175ef9f5f0dSMarkus Armbruster     g_assert(resp);
176ef9f5f0dSMarkus Armbruster     error = qdict_get_qdict(resp, "error");
177ef9f5f0dSMarkus Armbruster     g_assert(error);
178ef9f5f0dSMarkus Armbruster     g_assert_cmpstr(qdict_get_try_str(error, "class"),
179ef9f5f0dSMarkus Armbruster                     ==, QapiErrorClass_str(cls));
180ef9f5f0dSMarkus Armbruster     g_assert(qdict_get_try_str(error, "desc"));
181ef9f5f0dSMarkus Armbruster     g_assert(qdict_size(error) == 2);
182ef9f5f0dSMarkus Armbruster     g_assert(qdict_size(resp) == 1);
1833306459aSMarkus Armbruster 
1843306459aSMarkus Armbruster     qobject_unref(resp);
1853d16042cSMarkus Armbruster     qobject_unref(req);
1863306459aSMarkus Armbruster }
1873306459aSMarkus Armbruster 
18869ed8366SMichael Roth /* test commands with no input and no return value */
18969ed8366SMichael Roth static void test_dispatch_cmd(void)
19069ed8366SMichael Roth {
191ef9f5f0dSMarkus Armbruster     QDict *ret;
19269ed8366SMichael Roth 
1933d16042cSMarkus Armbruster     ret = qobject_to(QDict,
1943d16042cSMarkus Armbruster                      do_qmp_dispatch(false,
19505ebf841SMarkus Armbruster                                      "{ 'execute': 'user-def-cmd' }"));
196ef9f5f0dSMarkus Armbruster     assert(ret && qdict_size(ret) == 0);
1973306459aSMarkus Armbruster     qobject_unref(ret);
19869ed8366SMichael Roth }
19969ed8366SMichael Roth 
200c55f070bSPeter Xu static void test_dispatch_cmd_oob(void)
201c55f070bSPeter Xu {
202ef9f5f0dSMarkus Armbruster     QDict *ret;
203c55f070bSPeter Xu 
2043d16042cSMarkus Armbruster     ret = qobject_to(QDict,
2053d16042cSMarkus Armbruster                      do_qmp_dispatch(true,
2063d16042cSMarkus Armbruster                                      "{ 'exec-oob': 'test-flags-command' }"));
207ef9f5f0dSMarkus Armbruster     assert(ret && qdict_size(ret) == 0);
2083306459aSMarkus Armbruster     qobject_unref(ret);
209c55f070bSPeter Xu }
210c55f070bSPeter Xu 
21169ed8366SMichael Roth /* test commands that return an error due to invalid parameters */
21251009170SPeter Maydell static void test_dispatch_cmd_failure(void)
21369ed8366SMichael Roth {
2143d16042cSMarkus Armbruster     /* missing arguments */
2153d16042cSMarkus Armbruster     do_qmp_dispatch_error(false, ERROR_CLASS_GENERIC_ERROR,
21605ebf841SMarkus Armbruster                           "{ 'execute': 'user-def-cmd2' }");
21769ed8366SMichael Roth 
2183d16042cSMarkus Armbruster     /* extra arguments */
2193d16042cSMarkus Armbruster     do_qmp_dispatch_error(false, ERROR_CLASS_GENERIC_ERROR,
22005ebf841SMarkus Armbruster                           "{ 'execute': 'user-def-cmd',"
2213d16042cSMarkus Armbruster                           " 'arguments': { 'a': 66 } }");
22269ed8366SMichael Roth }
22369ed8366SMichael Roth 
224ae6bf766SMarc-André Lureau static void test_dispatch_cmd_success_response(void)
225ae6bf766SMarc-André Lureau {
226ae6bf766SMarc-André Lureau     QDict *req = qdict_new();
227ae6bf766SMarc-André Lureau     QDict *resp;
228ae6bf766SMarc-André Lureau 
229ae6bf766SMarc-André Lureau     qdict_put_str(req, "execute", "cmd-success-response");
23041725fa7SKevin Wolf     resp = qmp_dispatch(&qmp_commands, QOBJECT(req), false, NULL);
231ae6bf766SMarc-André Lureau     g_assert_null(resp);
232ae6bf766SMarc-André Lureau     qobject_unref(req);
233ae6bf766SMarc-André Lureau }
234ae6bf766SMarc-André Lureau 
23569ed8366SMichael Roth /* test commands that involve both input parameters and return values */
23669ed8366SMichael Roth static void test_dispatch_cmd_io(void)
23769ed8366SMichael Roth {
238357765feSMarkus Armbruster     QDict *ret, *ret_dict, *ret_dict_dict, *ret_dict_dict_userdef;
239357765feSMarkus Armbruster     QDict *ret_dict_dict2, *ret_dict_dict2_userdef;
24001b2ffceSMarc-André Lureau     QNum *ret3;
24101b2ffceSMarc-André Lureau     int64_t val;
24269ed8366SMichael Roth 
2433d16042cSMarkus Armbruster     ret = qobject_to(QDict, do_qmp_dispatch(false,
24405ebf841SMarkus Armbruster         "{ 'execute': 'user-def-cmd2', 'arguments': {"
2453d16042cSMarkus Armbruster         " 'ud1a': { 'integer': 42, 'string': 'hello' },"
2463d16042cSMarkus Armbruster         " 'ud1b': { 'integer': 422, 'string': 'hello2' } } }"));
24769ed8366SMichael Roth 
248b6fcf32dSEric Blake     assert(!strcmp(qdict_get_str(ret, "string0"), "blah1"));
249b6fcf32dSEric Blake     ret_dict = qdict_get_qdict(ret, "dict1");
250b6fcf32dSEric Blake     assert(!strcmp(qdict_get_str(ret_dict, "string1"), "blah2"));
251b6fcf32dSEric Blake     ret_dict_dict = qdict_get_qdict(ret_dict, "dict2");
252357765feSMarkus Armbruster     ret_dict_dict_userdef = qdict_get_qdict(ret_dict_dict, "userdef");
253357765feSMarkus Armbruster     assert(qdict_get_int(ret_dict_dict_userdef, "integer") == 42);
254357765feSMarkus Armbruster     assert(!strcmp(qdict_get_str(ret_dict_dict_userdef, "string"), "hello"));
255357765feSMarkus Armbruster     assert(!strcmp(qdict_get_str(ret_dict_dict, "string"), "blah3"));
256b6fcf32dSEric Blake     ret_dict_dict2 = qdict_get_qdict(ret_dict, "dict3");
257357765feSMarkus Armbruster     ret_dict_dict2_userdef = qdict_get_qdict(ret_dict_dict2, "userdef");
258357765feSMarkus Armbruster     assert(qdict_get_int(ret_dict_dict2_userdef, "integer") == 422);
259357765feSMarkus Armbruster     assert(!strcmp(qdict_get_str(ret_dict_dict2_userdef, "string"), "hello2"));
260357765feSMarkus Armbruster     assert(!strcmp(qdict_get_str(ret_dict_dict2, "string"), "blah4"));
261cb3e7f08SMarc-André Lureau     qobject_unref(ret);
262c2216a8aSMarkus Armbruster 
2633d16042cSMarkus Armbruster     ret3 = qobject_to(QNum, do_qmp_dispatch(false,
2643d16042cSMarkus Armbruster         "{ 'execute': 'guest-get-time', 'arguments': { 'a': 66 } }"));
26501b2ffceSMarc-André Lureau     g_assert(qnum_get_try_int(ret3, &val));
26601b2ffceSMarc-André Lureau     g_assert_cmpint(val, ==, 66);
267cb3e7f08SMarc-André Lureau     qobject_unref(ret3);
26869ed8366SMichael Roth }
26969ed8366SMichael Roth 
270d2032598SMarkus Armbruster static void test_dispatch_cmd_deprecated(void)
271d2032598SMarkus Armbruster {
272d2032598SMarkus Armbruster     const char *cmd = "{ 'execute': 'test-command-features1' }";
273d2032598SMarkus Armbruster     QDict *ret;
274d2032598SMarkus Armbruster 
275d2032598SMarkus Armbruster     memset(&compat_policy, 0, sizeof(compat_policy));
276d2032598SMarkus Armbruster 
277d2032598SMarkus Armbruster     /* accept */
278d2032598SMarkus Armbruster     ret = qobject_to(QDict, do_qmp_dispatch(false, cmd));
279d2032598SMarkus Armbruster     assert(ret && qdict_size(ret) == 0);
280d2032598SMarkus Armbruster     qobject_unref(ret);
281d2032598SMarkus Armbruster 
282d2032598SMarkus Armbruster     compat_policy.has_deprecated_input = true;
283d2032598SMarkus Armbruster     compat_policy.deprecated_input = COMPAT_POLICY_INPUT_ACCEPT;
284d2032598SMarkus Armbruster     ret = qobject_to(QDict, do_qmp_dispatch(false, cmd));
285d2032598SMarkus Armbruster     assert(ret && qdict_size(ret) == 0);
286d2032598SMarkus Armbruster     qobject_unref(ret);
287d2032598SMarkus Armbruster 
288d2032598SMarkus Armbruster     compat_policy.deprecated_input = COMPAT_POLICY_INPUT_REJECT;
289d2032598SMarkus Armbruster     do_qmp_dispatch_error(false, ERROR_CLASS_COMMAND_NOT_FOUND, cmd);
290d2032598SMarkus Armbruster }
291d2032598SMarkus Armbruster 
292db291641SMarkus Armbruster static void test_dispatch_cmd_arg_deprecated(void)
293db291641SMarkus Armbruster {
294db291641SMarkus Armbruster     const char *cmd = "{ 'execute': 'test-features0',"
295db291641SMarkus Armbruster         " 'arguments': { 'fs1': { 'foo': 42 } } }";
296db291641SMarkus Armbruster     QDict *ret;
297db291641SMarkus Armbruster 
298db291641SMarkus Armbruster     memset(&compat_policy, 0, sizeof(compat_policy));
299db291641SMarkus Armbruster 
300db291641SMarkus Armbruster     /* accept */
301db291641SMarkus Armbruster     ret = qobject_to(QDict, do_qmp_dispatch(false, cmd));
302db291641SMarkus Armbruster     assert(ret && qdict_size(ret) == 1);
303db291641SMarkus Armbruster     qobject_unref(ret);
304db291641SMarkus Armbruster 
305db291641SMarkus Armbruster     compat_policy.has_deprecated_input = true;
306db291641SMarkus Armbruster     compat_policy.deprecated_input = COMPAT_POLICY_INPUT_ACCEPT;
307db291641SMarkus Armbruster     ret = qobject_to(QDict, do_qmp_dispatch(false, cmd));
308db291641SMarkus Armbruster     assert(ret && qdict_size(ret) == 1);
309db291641SMarkus Armbruster     qobject_unref(ret);
310db291641SMarkus Armbruster 
311db291641SMarkus Armbruster     compat_policy.deprecated_input = COMPAT_POLICY_INPUT_REJECT;
312db291641SMarkus Armbruster     do_qmp_dispatch_error(false, ERROR_CLASS_GENERIC_ERROR, cmd);
313db291641SMarkus Armbruster }
314db291641SMarkus Armbruster 
31591fa93e5SMarkus Armbruster static void test_dispatch_cmd_ret_deprecated(void)
31691fa93e5SMarkus Armbruster {
31791fa93e5SMarkus Armbruster     const char *cmd = "{ 'execute': 'test-features0' }";
31891fa93e5SMarkus Armbruster     QDict *ret;
31991fa93e5SMarkus Armbruster 
32091fa93e5SMarkus Armbruster     memset(&compat_policy, 0, sizeof(compat_policy));
32191fa93e5SMarkus Armbruster 
32291fa93e5SMarkus Armbruster     /* default accept */
32391fa93e5SMarkus Armbruster     ret = qobject_to(QDict, do_qmp_dispatch(false, cmd));
32491fa93e5SMarkus Armbruster     assert(ret && qdict_size(ret) == 1);
32591fa93e5SMarkus Armbruster     qobject_unref(ret);
32691fa93e5SMarkus Armbruster 
32791fa93e5SMarkus Armbruster     compat_policy.has_deprecated_output = true;
32891fa93e5SMarkus Armbruster     compat_policy.deprecated_output = COMPAT_POLICY_OUTPUT_ACCEPT;
32991fa93e5SMarkus Armbruster     ret = qobject_to(QDict, do_qmp_dispatch(false, cmd));
33091fa93e5SMarkus Armbruster     assert(ret && qdict_size(ret) == 1);
33191fa93e5SMarkus Armbruster     qobject_unref(ret);
33291fa93e5SMarkus Armbruster 
33391fa93e5SMarkus Armbruster     compat_policy.deprecated_output = COMPAT_POLICY_OUTPUT_HIDE;
33491fa93e5SMarkus Armbruster     ret = qobject_to(QDict, do_qmp_dispatch(false, cmd));
33591fa93e5SMarkus Armbruster     assert(ret && qdict_size(ret) == 0);
33691fa93e5SMarkus Armbruster     qobject_unref(ret);
33791fa93e5SMarkus Armbruster }
33891fa93e5SMarkus Armbruster 
3395cd5f0d0SMichael Roth /* test generated dealloc functions for generated types */
3405cd5f0d0SMichael Roth static void test_dealloc_types(void)
3415cd5f0d0SMichael Roth {
3425cd5f0d0SMichael Roth     UserDefOne *ud1test, *ud1a, *ud1b;
3435cd5f0d0SMichael Roth     UserDefOneList *ud1list;
3445cd5f0d0SMichael Roth 
345b21e2380SMarkus Armbruster     ud1test = g_new0(UserDefOne, 1);
346ddf21908SEric Blake     ud1test->integer = 42;
3475cd5f0d0SMichael Roth     ud1test->string = g_strdup("hi there 42");
3485cd5f0d0SMichael Roth 
3495cd5f0d0SMichael Roth     qapi_free_UserDefOne(ud1test);
3505cd5f0d0SMichael Roth 
351b21e2380SMarkus Armbruster     ud1a = g_new0(UserDefOne, 1);
352ddf21908SEric Blake     ud1a->integer = 43;
3535cd5f0d0SMichael Roth     ud1a->string = g_strdup("hi there 43");
3545cd5f0d0SMichael Roth 
355b21e2380SMarkus Armbruster     ud1b = g_new0(UserDefOne, 1);
356ddf21908SEric Blake     ud1b->integer = 44;
3575cd5f0d0SMichael Roth     ud1b->string = g_strdup("hi there 44");
3585cd5f0d0SMichael Roth 
359b21e2380SMarkus Armbruster     ud1list = g_new0(UserDefOneList, 1);
3605cd5f0d0SMichael Roth     ud1list->value = ud1a;
361b21e2380SMarkus Armbruster     ud1list->next = g_new0(UserDefOneList, 1);
3625cd5f0d0SMichael Roth     ud1list->next->value = ud1b;
3635cd5f0d0SMichael Roth 
3645cd5f0d0SMichael Roth     qapi_free_UserDefOneList(ud1list);
3655cd5f0d0SMichael Roth }
3665cd5f0d0SMichael Roth 
367d98150f0SLaszlo Ersek /* test generated deallocation on an object whose construction was prematurely
368d98150f0SLaszlo Ersek  * terminated due to an error */
369d98150f0SLaszlo Ersek static void test_dealloc_partial(void)
370d98150f0SLaszlo Ersek {
371d98150f0SLaszlo Ersek     static const char text[] = "don't leak me";
372d98150f0SLaszlo Ersek 
373d98150f0SLaszlo Ersek     UserDefTwo *ud2 = NULL;
374d98150f0SLaszlo Ersek     Error *err = NULL;
375d98150f0SLaszlo Ersek 
376d98150f0SLaszlo Ersek     /* create partial object */
377d98150f0SLaszlo Ersek     {
378d98150f0SLaszlo Ersek         QDict *ud2_dict;
379b70ce101SEric Blake         Visitor *v;
380d98150f0SLaszlo Ersek 
381d98150f0SLaszlo Ersek         ud2_dict = qdict_new();
38246f5ac20SEric Blake         qdict_put_str(ud2_dict, "string0", text);
383d98150f0SLaszlo Ersek 
384048abb7bSMarkus Armbruster         v = qobject_input_visitor_new(QOBJECT(ud2_dict));
385b70ce101SEric Blake         visit_type_UserDefTwo(v, NULL, &ud2, &err);
386b70ce101SEric Blake         visit_free(v);
387cb3e7f08SMarc-André Lureau         qobject_unref(ud2_dict);
388d98150f0SLaszlo Ersek     }
389d98150f0SLaszlo Ersek 
39068ab47e4SEric Blake     /* verify that visit_type_XXX() cleans up properly on error */
391a12a5a1aSEric Blake     error_free_or_abort(&err);
39268ab47e4SEric Blake     assert(!ud2);
39368ab47e4SEric Blake 
39468ab47e4SEric Blake     /* Manually create a partial object, leaving ud2->dict1 at NULL */
39568ab47e4SEric Blake     ud2 = g_new0(UserDefTwo, 1);
39668ab47e4SEric Blake     ud2->string0 = g_strdup(text);
397d98150f0SLaszlo Ersek 
398d98150f0SLaszlo Ersek     /* tear down partial object */
399d98150f0SLaszlo Ersek     qapi_free_UserDefTwo(ud2);
400d98150f0SLaszlo Ersek }
401d98150f0SLaszlo Ersek 
402d98150f0SLaszlo Ersek 
40369ed8366SMichael Roth int main(int argc, char **argv)
40469ed8366SMichael Roth {
40569ed8366SMichael Roth     g_test_init(&argc, &argv, NULL);
40669ed8366SMichael Roth 
407214e4a5bSMarc-André Lureau     g_test_add_func("/qmp/dispatch_cmd", test_dispatch_cmd);
408c55f070bSPeter Xu     g_test_add_func("/qmp/dispatch_cmd_oob", test_dispatch_cmd_oob);
409214e4a5bSMarc-André Lureau     g_test_add_func("/qmp/dispatch_cmd_failure", test_dispatch_cmd_failure);
410214e4a5bSMarc-André Lureau     g_test_add_func("/qmp/dispatch_cmd_io", test_dispatch_cmd_io);
411ae6bf766SMarc-André Lureau     g_test_add_func("/qmp/dispatch_cmd_success_response",
412ae6bf766SMarc-André Lureau                     test_dispatch_cmd_success_response);
413d2032598SMarkus Armbruster     g_test_add_func("/qmp/dispatch_cmd_deprecated",
414d2032598SMarkus Armbruster                     test_dispatch_cmd_deprecated);
415db291641SMarkus Armbruster     g_test_add_func("/qmp/dispatch_cmd_arg_deprecated",
416db291641SMarkus Armbruster                     test_dispatch_cmd_arg_deprecated);
41791fa93e5SMarkus Armbruster     g_test_add_func("/qmp/dispatch_cmd_ret_deprecated",
41891fa93e5SMarkus Armbruster                     test_dispatch_cmd_ret_deprecated);
419214e4a5bSMarc-André Lureau     g_test_add_func("/qmp/dealloc_types", test_dealloc_types);
420214e4a5bSMarc-André Lureau     g_test_add_func("/qmp/dealloc_partial", test_dealloc_partial);
42169ed8366SMichael Roth 
4221527badbSMarkus Armbruster     test_qmp_init_marshal(&qmp_commands);
42369ed8366SMichael Roth     g_test_run();
42469ed8366SMichael Roth 
42569ed8366SMichael Roth     return 0;
42669ed8366SMichael Roth }
427