test-qobject-output-visitor.c 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657
  1. /*
  2. * QObject Output Visitor unit-tests.
  3. *
  4. * Copyright (C) 2011-2016 Red Hat Inc.
  5. *
  6. * Authors:
  7. * Luiz Capitulino <lcapitulino@redhat.com>
  8. *
  9. * This work is licensed under the terms of the GNU GPL, version 2 or later.
  10. * See the COPYING file in the top-level directory.
  11. */
  12. #include "qemu/osdep.h"
  13. #include "qapi/error.h"
  14. #include "qapi/qobject-output-visitor.h"
  15. #include "test-qapi-visit.h"
  16. #include "qapi/qmp/qbool.h"
  17. #include "qapi/qmp/qdict.h"
  18. #include "qapi/qmp/qlist.h"
  19. #include "qapi/qmp/qnull.h"
  20. #include "qapi/qmp/qnum.h"
  21. #include "qapi/qmp/qstring.h"
  22. typedef struct TestOutputVisitorData {
  23. Visitor *ov;
  24. QObject *obj;
  25. } TestOutputVisitorData;
  26. static void visitor_output_setup(TestOutputVisitorData *data,
  27. const void *unused)
  28. {
  29. data->ov = qobject_output_visitor_new(&data->obj);
  30. g_assert(data->ov);
  31. }
  32. static void visitor_output_teardown(TestOutputVisitorData *data,
  33. const void *unused)
  34. {
  35. visit_free(data->ov);
  36. data->ov = NULL;
  37. qobject_unref(data->obj);
  38. data->obj = NULL;
  39. }
  40. static QObject *visitor_get(TestOutputVisitorData *data)
  41. {
  42. visit_complete(data->ov, &data->obj);
  43. g_assert(data->obj);
  44. return data->obj;
  45. }
  46. static void visitor_reset(TestOutputVisitorData *data)
  47. {
  48. visitor_output_teardown(data, NULL);
  49. visitor_output_setup(data, NULL);
  50. }
  51. static void test_visitor_out_int(TestOutputVisitorData *data,
  52. const void *unused)
  53. {
  54. int64_t value = -42;
  55. int64_t val;
  56. QNum *qnum;
  57. visit_type_int(data->ov, NULL, &value, &error_abort);
  58. qnum = qobject_to(QNum, visitor_get(data));
  59. g_assert(qnum);
  60. g_assert(qnum_get_try_int(qnum, &val));
  61. g_assert_cmpint(val, ==, value);
  62. }
  63. static void test_visitor_out_bool(TestOutputVisitorData *data,
  64. const void *unused)
  65. {
  66. bool value = true;
  67. QBool *qbool;
  68. visit_type_bool(data->ov, NULL, &value, &error_abort);
  69. qbool = qobject_to(QBool, visitor_get(data));
  70. g_assert(qbool);
  71. g_assert(qbool_get_bool(qbool) == value);
  72. }
  73. static void test_visitor_out_number(TestOutputVisitorData *data,
  74. const void *unused)
  75. {
  76. double value = 3.14;
  77. QNum *qnum;
  78. visit_type_number(data->ov, NULL, &value, &error_abort);
  79. qnum = qobject_to(QNum, visitor_get(data));
  80. g_assert(qnum);
  81. g_assert(qnum_get_double(qnum) == value);
  82. }
  83. static void test_visitor_out_string(TestOutputVisitorData *data,
  84. const void *unused)
  85. {
  86. char *string = (char *) "Q E M U";
  87. QString *qstr;
  88. visit_type_str(data->ov, NULL, &string, &error_abort);
  89. qstr = qobject_to(QString, visitor_get(data));
  90. g_assert(qstr);
  91. g_assert_cmpstr(qstring_get_str(qstr), ==, string);
  92. }
  93. static void test_visitor_out_no_string(TestOutputVisitorData *data,
  94. const void *unused)
  95. {
  96. char *string = NULL;
  97. QString *qstr;
  98. /* A null string should return "" */
  99. visit_type_str(data->ov, NULL, &string, &error_abort);
  100. qstr = qobject_to(QString, visitor_get(data));
  101. g_assert(qstr);
  102. g_assert_cmpstr(qstring_get_str(qstr), ==, "");
  103. }
  104. static void test_visitor_out_enum(TestOutputVisitorData *data,
  105. const void *unused)
  106. {
  107. EnumOne i;
  108. QString *qstr;
  109. for (i = 0; i < ENUM_ONE__MAX; i++) {
  110. visit_type_EnumOne(data->ov, "unused", &i, &error_abort);
  111. qstr = qobject_to(QString, visitor_get(data));
  112. g_assert(qstr);
  113. g_assert_cmpstr(qstring_get_str(qstr), ==, EnumOne_str(i));
  114. visitor_reset(data);
  115. }
  116. }
  117. static void test_visitor_out_struct(TestOutputVisitorData *data,
  118. const void *unused)
  119. {
  120. TestStruct test_struct = { .integer = 42,
  121. .boolean = false,
  122. .string = (char *) "foo"};
  123. TestStruct *p = &test_struct;
  124. QDict *qdict;
  125. visit_type_TestStruct(data->ov, NULL, &p, &error_abort);
  126. qdict = qobject_to(QDict, visitor_get(data));
  127. g_assert(qdict);
  128. g_assert_cmpint(qdict_size(qdict), ==, 3);
  129. g_assert_cmpint(qdict_get_int(qdict, "integer"), ==, 42);
  130. g_assert_cmpint(qdict_get_bool(qdict, "boolean"), ==, false);
  131. g_assert_cmpstr(qdict_get_str(qdict, "string"), ==, "foo");
  132. }
  133. static void test_visitor_out_struct_nested(TestOutputVisitorData *data,
  134. const void *unused)
  135. {
  136. int64_t value = 42;
  137. UserDefTwo *ud2;
  138. QDict *qdict, *dict1, *dict2, *dict3, *userdef;
  139. const char *string = "user def string";
  140. const char *strings[] = { "forty two", "forty three", "forty four",
  141. "forty five" };
  142. ud2 = g_malloc0(sizeof(*ud2));
  143. ud2->string0 = g_strdup(strings[0]);
  144. ud2->dict1 = g_malloc0(sizeof(*ud2->dict1));
  145. ud2->dict1->string1 = g_strdup(strings[1]);
  146. ud2->dict1->dict2 = g_malloc0(sizeof(*ud2->dict1->dict2));
  147. ud2->dict1->dict2->userdef = g_new0(UserDefOne, 1);
  148. ud2->dict1->dict2->userdef->string = g_strdup(string);
  149. ud2->dict1->dict2->userdef->integer = value;
  150. ud2->dict1->dict2->string = g_strdup(strings[2]);
  151. ud2->dict1->dict3 = g_malloc0(sizeof(*ud2->dict1->dict3));
  152. ud2->dict1->dict3->userdef = g_new0(UserDefOne, 1);
  153. ud2->dict1->dict3->userdef->string = g_strdup(string);
  154. ud2->dict1->dict3->userdef->integer = value;
  155. ud2->dict1->dict3->string = g_strdup(strings[3]);
  156. visit_type_UserDefTwo(data->ov, "unused", &ud2, &error_abort);
  157. qdict = qobject_to(QDict, visitor_get(data));
  158. g_assert(qdict);
  159. g_assert_cmpint(qdict_size(qdict), ==, 2);
  160. g_assert_cmpstr(qdict_get_str(qdict, "string0"), ==, strings[0]);
  161. dict1 = qdict_get_qdict(qdict, "dict1");
  162. g_assert_cmpint(qdict_size(dict1), ==, 3);
  163. g_assert_cmpstr(qdict_get_str(dict1, "string1"), ==, strings[1]);
  164. dict2 = qdict_get_qdict(dict1, "dict2");
  165. g_assert_cmpint(qdict_size(dict2), ==, 2);
  166. g_assert_cmpstr(qdict_get_str(dict2, "string"), ==, strings[2]);
  167. userdef = qdict_get_qdict(dict2, "userdef");
  168. g_assert_cmpint(qdict_size(userdef), ==, 2);
  169. g_assert_cmpint(qdict_get_int(userdef, "integer"), ==, value);
  170. g_assert_cmpstr(qdict_get_str(userdef, "string"), ==, string);
  171. dict3 = qdict_get_qdict(dict1, "dict3");
  172. g_assert_cmpint(qdict_size(dict3), ==, 2);
  173. g_assert_cmpstr(qdict_get_str(dict3, "string"), ==, strings[3]);
  174. userdef = qdict_get_qdict(dict3, "userdef");
  175. g_assert_cmpint(qdict_size(userdef), ==, 2);
  176. g_assert_cmpint(qdict_get_int(userdef, "integer"), ==, value);
  177. g_assert_cmpstr(qdict_get_str(userdef, "string"), ==, string);
  178. qapi_free_UserDefTwo(ud2);
  179. }
  180. static void test_visitor_out_list(TestOutputVisitorData *data,
  181. const void *unused)
  182. {
  183. const char *value_str = "list value";
  184. TestStruct *value;
  185. TestStructList *head = NULL;
  186. const int max_items = 10;
  187. bool value_bool = true;
  188. int value_int = 10;
  189. QListEntry *entry;
  190. QList *qlist;
  191. int i;
  192. /* Build the list in reverse order... */
  193. for (i = 0; i < max_items; i++) {
  194. value = g_malloc0(sizeof(*value));
  195. value->integer = value_int + (max_items - i - 1);
  196. value->boolean = value_bool;
  197. value->string = g_strdup(value_str);
  198. QAPI_LIST_PREPEND(head, value);
  199. }
  200. visit_type_TestStructList(data->ov, NULL, &head, &error_abort);
  201. qlist = qobject_to(QList, visitor_get(data));
  202. g_assert(qlist);
  203. g_assert(!qlist_empty(qlist));
  204. /* ...and ensure that the visitor sees it in order */
  205. i = 0;
  206. QLIST_FOREACH_ENTRY(qlist, entry) {
  207. QDict *qdict;
  208. qdict = qobject_to(QDict, entry->value);
  209. g_assert(qdict);
  210. g_assert_cmpint(qdict_size(qdict), ==, 3);
  211. g_assert_cmpint(qdict_get_int(qdict, "integer"), ==, value_int + i);
  212. g_assert_cmpint(qdict_get_bool(qdict, "boolean"), ==, value_bool);
  213. g_assert_cmpstr(qdict_get_str(qdict, "string"), ==, value_str);
  214. i++;
  215. }
  216. g_assert_cmpint(i, ==, max_items);
  217. qapi_free_TestStructList(head);
  218. }
  219. static void test_visitor_out_list_qapi_free(TestOutputVisitorData *data,
  220. const void *unused)
  221. {
  222. UserDefTwo *value;
  223. UserDefTwoList *head = NULL;
  224. const char string[] = "foo bar";
  225. int i, max_count = 1024;
  226. for (i = 0; i < max_count; i++) {
  227. value = g_malloc0(sizeof(*value));
  228. value->string0 = g_strdup(string);
  229. value->dict1 = g_new0(UserDefTwoDict, 1);
  230. value->dict1->string1 = g_strdup(string);
  231. value->dict1->dict2 = g_new0(UserDefTwoDictDict, 1);
  232. value->dict1->dict2->userdef = g_new0(UserDefOne, 1);
  233. value->dict1->dict2->userdef->string = g_strdup(string);
  234. value->dict1->dict2->userdef->integer = 42;
  235. value->dict1->dict2->string = g_strdup(string);
  236. QAPI_LIST_PREPEND(head, value);
  237. }
  238. qapi_free_UserDefTwoList(head);
  239. }
  240. static void test_visitor_out_any(TestOutputVisitorData *data,
  241. const void *unused)
  242. {
  243. QObject *qobj;
  244. QNum *qnum;
  245. QBool *qbool;
  246. QString *qstring;
  247. QDict *qdict;
  248. int64_t val;
  249. qobj = QOBJECT(qnum_from_int(-42));
  250. visit_type_any(data->ov, NULL, &qobj, &error_abort);
  251. qnum = qobject_to(QNum, visitor_get(data));
  252. g_assert(qnum);
  253. g_assert(qnum_get_try_int(qnum, &val));
  254. g_assert_cmpint(val, ==, -42);
  255. qobject_unref(qobj);
  256. visitor_reset(data);
  257. qdict = qdict_new();
  258. qdict_put_int(qdict, "integer", -42);
  259. qdict_put_bool(qdict, "boolean", true);
  260. qdict_put_str(qdict, "string", "foo");
  261. qobj = QOBJECT(qdict);
  262. visit_type_any(data->ov, NULL, &qobj, &error_abort);
  263. qobject_unref(qobj);
  264. qdict = qobject_to(QDict, visitor_get(data));
  265. g_assert(qdict);
  266. qnum = qobject_to(QNum, qdict_get(qdict, "integer"));
  267. g_assert(qnum);
  268. g_assert(qnum_get_try_int(qnum, &val));
  269. g_assert_cmpint(val, ==, -42);
  270. qbool = qobject_to(QBool, qdict_get(qdict, "boolean"));
  271. g_assert(qbool);
  272. g_assert(qbool_get_bool(qbool) == true);
  273. qstring = qobject_to(QString, qdict_get(qdict, "string"));
  274. g_assert(qstring);
  275. g_assert_cmpstr(qstring_get_str(qstring), ==, "foo");
  276. }
  277. static void test_visitor_out_union_flat(TestOutputVisitorData *data,
  278. const void *unused)
  279. {
  280. QDict *qdict;
  281. UserDefFlatUnion *tmp = g_new0(UserDefFlatUnion, 1);
  282. tmp->enum1 = ENUM_ONE_VALUE1;
  283. tmp->string = g_strdup("str");
  284. tmp->integer = 41;
  285. tmp->u.value1.boolean = true;
  286. visit_type_UserDefFlatUnion(data->ov, NULL, &tmp, &error_abort);
  287. qdict = qobject_to(QDict, visitor_get(data));
  288. g_assert(qdict);
  289. g_assert_cmpstr(qdict_get_str(qdict, "enum1"), ==, "value1");
  290. g_assert_cmpstr(qdict_get_str(qdict, "string"), ==, "str");
  291. g_assert_cmpint(qdict_get_int(qdict, "integer"), ==, 41);
  292. g_assert_cmpint(qdict_get_bool(qdict, "boolean"), ==, true);
  293. qapi_free_UserDefFlatUnion(tmp);
  294. }
  295. static void test_visitor_out_union_in_union(TestOutputVisitorData *data,
  296. const void *unused)
  297. {
  298. QDict *qdict;
  299. TestUnionInUnion *tmp = g_new0(TestUnionInUnion, 1);
  300. tmp->type = TEST_UNION_ENUM_VALUE_A;
  301. tmp->u.value_a.type_a = TEST_UNION_ENUMA_VALUE_A1;
  302. tmp->u.value_a.u.value_a1.integer = 42;
  303. tmp->u.value_a.u.value_a1.name = g_strdup("fish");
  304. visit_type_TestUnionInUnion(data->ov, NULL, &tmp, &error_abort);
  305. qdict = qobject_to(QDict, visitor_get(data));
  306. g_assert(qdict);
  307. g_assert_cmpstr(qdict_get_str(qdict, "type"), ==, "value-a");
  308. g_assert_cmpstr(qdict_get_str(qdict, "type-a"), ==, "value-a1");
  309. g_assert_cmpint(qdict_get_int(qdict, "integer"), ==, 42);
  310. g_assert_cmpstr(qdict_get_str(qdict, "name"), ==, "fish");
  311. qapi_free_TestUnionInUnion(tmp);
  312. visitor_reset(data);
  313. tmp = g_new0(TestUnionInUnion, 1);
  314. tmp->type = TEST_UNION_ENUM_VALUE_A;
  315. tmp->u.value_a.type_a = TEST_UNION_ENUMA_VALUE_A2;
  316. tmp->u.value_a.u.value_a2.integer = 1729;
  317. tmp->u.value_a.u.value_a2.size = 87539319;
  318. visit_type_TestUnionInUnion(data->ov, NULL, &tmp, &error_abort);
  319. qdict = qobject_to(QDict, visitor_get(data));
  320. g_assert(qdict);
  321. g_assert_cmpstr(qdict_get_str(qdict, "type"), ==, "value-a");
  322. g_assert_cmpstr(qdict_get_str(qdict, "type-a"), ==, "value-a2");
  323. g_assert_cmpint(qdict_get_int(qdict, "integer"), ==, 1729);
  324. g_assert_cmpint(qdict_get_int(qdict, "size"), ==, 87539319);
  325. qapi_free_TestUnionInUnion(tmp);
  326. visitor_reset(data);
  327. tmp = g_new0(TestUnionInUnion, 1);
  328. tmp->type = TEST_UNION_ENUM_VALUE_B;
  329. tmp->u.value_b.integer = 1729;
  330. tmp->u.value_b.onoff = true;
  331. visit_type_TestUnionInUnion(data->ov, NULL, &tmp, &error_abort);
  332. qdict = qobject_to(QDict, visitor_get(data));
  333. g_assert(qdict);
  334. g_assert_cmpstr(qdict_get_str(qdict, "type"), ==, "value-b");
  335. g_assert_cmpint(qdict_get_int(qdict, "integer"), ==, 1729);
  336. g_assert_cmpint(qdict_get_bool(qdict, "onoff"), ==, true);
  337. qapi_free_TestUnionInUnion(tmp);
  338. }
  339. static void test_visitor_out_alternate(TestOutputVisitorData *data,
  340. const void *unused)
  341. {
  342. UserDefAlternate *tmp;
  343. QNum *qnum;
  344. QString *qstr;
  345. QDict *qdict;
  346. int64_t val;
  347. tmp = g_new0(UserDefAlternate, 1);
  348. tmp->type = QTYPE_QNUM;
  349. tmp->u.i = 42;
  350. visit_type_UserDefAlternate(data->ov, NULL, &tmp, &error_abort);
  351. qnum = qobject_to(QNum, visitor_get(data));
  352. g_assert(qnum);
  353. g_assert(qnum_get_try_int(qnum, &val));
  354. g_assert_cmpint(val, ==, 42);
  355. qapi_free_UserDefAlternate(tmp);
  356. visitor_reset(data);
  357. tmp = g_new0(UserDefAlternate, 1);
  358. tmp->type = QTYPE_QSTRING;
  359. tmp->u.e = ENUM_ONE_VALUE1;
  360. visit_type_UserDefAlternate(data->ov, NULL, &tmp, &error_abort);
  361. qstr = qobject_to(QString, visitor_get(data));
  362. g_assert(qstr);
  363. g_assert_cmpstr(qstring_get_str(qstr), ==, "value1");
  364. qapi_free_UserDefAlternate(tmp);
  365. visitor_reset(data);
  366. tmp = g_new0(UserDefAlternate, 1);
  367. tmp->type = QTYPE_QNULL;
  368. tmp->u.n = qnull();
  369. visit_type_UserDefAlternate(data->ov, NULL, &tmp, &error_abort);
  370. g_assert_cmpint(qobject_type(visitor_get(data)), ==, QTYPE_QNULL);
  371. qapi_free_UserDefAlternate(tmp);
  372. visitor_reset(data);
  373. tmp = g_new0(UserDefAlternate, 1);
  374. tmp->type = QTYPE_QDICT;
  375. tmp->u.udfu.integer = 1;
  376. tmp->u.udfu.string = g_strdup("str");
  377. tmp->u.udfu.enum1 = ENUM_ONE_VALUE1;
  378. tmp->u.udfu.u.value1.boolean = true;
  379. visit_type_UserDefAlternate(data->ov, NULL, &tmp, &error_abort);
  380. qdict = qobject_to(QDict, visitor_get(data));
  381. g_assert(qdict);
  382. g_assert_cmpint(qdict_size(qdict), ==, 4);
  383. g_assert_cmpint(qdict_get_int(qdict, "integer"), ==, 1);
  384. g_assert_cmpstr(qdict_get_str(qdict, "string"), ==, "str");
  385. g_assert_cmpstr(qdict_get_str(qdict, "enum1"), ==, "value1");
  386. g_assert_cmpint(qdict_get_bool(qdict, "boolean"), ==, true);
  387. qapi_free_UserDefAlternate(tmp);
  388. }
  389. static void test_visitor_out_null(TestOutputVisitorData *data,
  390. const void *unused)
  391. {
  392. QNull *null = NULL;
  393. QDict *qdict;
  394. QObject *nil;
  395. visit_start_struct(data->ov, NULL, NULL, 0, &error_abort);
  396. visit_type_null(data->ov, "a", &null, &error_abort);
  397. visit_check_struct(data->ov, &error_abort);
  398. visit_end_struct(data->ov, NULL);
  399. qdict = qobject_to(QDict, visitor_get(data));
  400. g_assert(qdict);
  401. g_assert_cmpint(qdict_size(qdict), ==, 1);
  402. nil = qdict_get(qdict, "a");
  403. g_assert(nil);
  404. g_assert(qobject_type(nil) == QTYPE_QNULL);
  405. }
  406. static void test_visitor_out_list_struct(TestOutputVisitorData *data,
  407. const void *unused)
  408. {
  409. const char *int_member[] = {
  410. "integer", "s8", "s16", "s32", "s64", "u8", "u16", "u32", "u64" };
  411. g_autoptr(ArrayStruct) arrs = g_new0(ArrayStruct, 1);
  412. int i, j;
  413. QDict *qdict;
  414. QList *qlist;
  415. QListEntry *e;
  416. for (i = 31; i >= 0; i--) {
  417. QAPI_LIST_PREPEND(arrs->integer, i);
  418. }
  419. for (i = 31; i >= 0; i--) {
  420. QAPI_LIST_PREPEND(arrs->s8, i);
  421. }
  422. for (i = 31; i >= 0; i--) {
  423. QAPI_LIST_PREPEND(arrs->s16, i);
  424. }
  425. for (i = 31; i >= 0; i--) {
  426. QAPI_LIST_PREPEND(arrs->s32, i);
  427. }
  428. for (i = 31; i >= 0; i--) {
  429. QAPI_LIST_PREPEND(arrs->s64, i);
  430. }
  431. for (i = 31; i >= 0; i--) {
  432. QAPI_LIST_PREPEND(arrs->u8, i);
  433. }
  434. for (i = 31; i >= 0; i--) {
  435. QAPI_LIST_PREPEND(arrs->u16, i);
  436. }
  437. for (i = 31; i >= 0; i--) {
  438. QAPI_LIST_PREPEND(arrs->u32, i);
  439. }
  440. for (i = 31; i >= 0; i--) {
  441. QAPI_LIST_PREPEND(arrs->u64, i);
  442. }
  443. for (i = 31; i >= 0; i--) {
  444. QAPI_LIST_PREPEND(arrs->number, (double)i / 3);
  445. }
  446. for (i = 31; i >= 0; i--) {
  447. QAPI_LIST_PREPEND(arrs->boolean, QEMU_IS_ALIGNED(i, 3));
  448. }
  449. for (i = 31; i >= 0; i--) {
  450. QAPI_LIST_PREPEND(arrs->string, g_strdup_printf("%d", i));
  451. }
  452. visit_type_ArrayStruct(data->ov, NULL, &arrs, &error_abort);
  453. qdict = qobject_to(QDict, visitor_get(data));
  454. g_assert(qdict);
  455. for (i = 0; i < G_N_ELEMENTS(int_member); i++) {
  456. qlist = qdict_get_qlist(qdict, int_member[i]);
  457. g_assert(qlist);
  458. j = 0;
  459. QLIST_FOREACH_ENTRY(qlist, e) {
  460. QNum *qvalue = qobject_to(QNum, qlist_entry_obj(e));
  461. g_assert(qvalue);
  462. g_assert_cmpint(qnum_get_int(qvalue), ==, j);
  463. j++;
  464. }
  465. }
  466. qlist = qdict_get_qlist(qdict, "number");
  467. g_assert(qlist);
  468. i = 0;
  469. QLIST_FOREACH_ENTRY(qlist, e) {
  470. QNum *qvalue = qobject_to(QNum, qlist_entry_obj(e));
  471. char expected[32], actual[32];
  472. g_assert(qvalue);
  473. sprintf(expected, "%.6f", (double)i / 3);
  474. sprintf(actual, "%.6f", qnum_get_double(qvalue));
  475. g_assert_cmpstr(actual, ==, expected);
  476. i++;
  477. }
  478. qlist = qdict_get_qlist(qdict, "boolean");
  479. g_assert(qlist);
  480. i = 0;
  481. QLIST_FOREACH_ENTRY(qlist, e) {
  482. QBool *qvalue = qobject_to(QBool, qlist_entry_obj(e));
  483. g_assert(qvalue);
  484. g_assert_cmpint(qbool_get_bool(qvalue), ==, i % 3 == 0);
  485. i++;
  486. }
  487. qlist = qdict_get_qlist(qdict, "string");
  488. g_assert(qlist);
  489. i = 0;
  490. QLIST_FOREACH_ENTRY(qlist, e) {
  491. QString *qvalue = qobject_to(QString, qlist_entry_obj(e));
  492. char expected[32];
  493. g_assert(qvalue);
  494. sprintf(expected, "%d", i);
  495. g_assert_cmpstr(qstring_get_str(qvalue), ==, expected);
  496. i++;
  497. }
  498. }
  499. static void output_visitor_test_add(const char *testpath,
  500. TestOutputVisitorData *data,
  501. void (*test_func)(TestOutputVisitorData *data, const void *user_data))
  502. {
  503. g_test_add(testpath, TestOutputVisitorData, data, visitor_output_setup,
  504. test_func, visitor_output_teardown);
  505. }
  506. int main(int argc, char **argv)
  507. {
  508. TestOutputVisitorData out_visitor_data;
  509. g_test_init(&argc, &argv, NULL);
  510. output_visitor_test_add("/visitor/output/int",
  511. &out_visitor_data, test_visitor_out_int);
  512. output_visitor_test_add("/visitor/output/bool",
  513. &out_visitor_data, test_visitor_out_bool);
  514. output_visitor_test_add("/visitor/output/number",
  515. &out_visitor_data, test_visitor_out_number);
  516. output_visitor_test_add("/visitor/output/string",
  517. &out_visitor_data, test_visitor_out_string);
  518. output_visitor_test_add("/visitor/output/no-string",
  519. &out_visitor_data, test_visitor_out_no_string);
  520. output_visitor_test_add("/visitor/output/enum",
  521. &out_visitor_data, test_visitor_out_enum);
  522. output_visitor_test_add("/visitor/output/struct",
  523. &out_visitor_data, test_visitor_out_struct);
  524. output_visitor_test_add("/visitor/output/struct-nested",
  525. &out_visitor_data, test_visitor_out_struct_nested);
  526. output_visitor_test_add("/visitor/output/list",
  527. &out_visitor_data, test_visitor_out_list);
  528. output_visitor_test_add("/visitor/output/any",
  529. &out_visitor_data, test_visitor_out_any);
  530. output_visitor_test_add("/visitor/output/list-qapi-free",
  531. &out_visitor_data, test_visitor_out_list_qapi_free);
  532. output_visitor_test_add("/visitor/output/union-flat",
  533. &out_visitor_data, test_visitor_out_union_flat);
  534. output_visitor_test_add("/visitor/output/union-in-union",
  535. &out_visitor_data, test_visitor_out_union_in_union);
  536. output_visitor_test_add("/visitor/output/alternate",
  537. &out_visitor_data, test_visitor_out_alternate);
  538. output_visitor_test_add("/visitor/output/null",
  539. &out_visitor_data, test_visitor_out_null);
  540. output_visitor_test_add("/visitor/output/list_struct",
  541. &out_visitor_data, test_visitor_out_list_struct);
  542. g_test_run();
  543. return 0;
  544. }