2
0

test-qobject-output-visitor.c 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884
  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 "qemu-common.h"
  14. #include "qapi/error.h"
  15. #include "qapi/qobject-output-visitor.h"
  16. #include "test-qapi-visit.h"
  17. #include "qapi/qmp/qbool.h"
  18. #include "qapi/qmp/qdict.h"
  19. #include "qapi/qmp/qlist.h"
  20. #include "qapi/qmp/qnull.h"
  21. #include "qapi/qmp/qnum.h"
  22. #include "qapi/qmp/qstring.h"
  23. typedef struct TestOutputVisitorData {
  24. Visitor *ov;
  25. QObject *obj;
  26. } TestOutputVisitorData;
  27. static void visitor_output_setup(TestOutputVisitorData *data,
  28. const void *unused)
  29. {
  30. data->ov = qobject_output_visitor_new(&data->obj);
  31. g_assert(data->ov);
  32. }
  33. static void visitor_output_teardown(TestOutputVisitorData *data,
  34. const void *unused)
  35. {
  36. visit_free(data->ov);
  37. data->ov = NULL;
  38. qobject_unref(data->obj);
  39. data->obj = NULL;
  40. }
  41. static QObject *visitor_get(TestOutputVisitorData *data)
  42. {
  43. visit_complete(data->ov, &data->obj);
  44. g_assert(data->obj);
  45. return data->obj;
  46. }
  47. static void visitor_reset(TestOutputVisitorData *data)
  48. {
  49. visitor_output_teardown(data, NULL);
  50. visitor_output_setup(data, NULL);
  51. }
  52. static void test_visitor_out_int(TestOutputVisitorData *data,
  53. const void *unused)
  54. {
  55. int64_t value = -42;
  56. int64_t val;
  57. QNum *qnum;
  58. visit_type_int(data->ov, NULL, &value, &error_abort);
  59. qnum = qobject_to(QNum, visitor_get(data));
  60. g_assert(qnum);
  61. g_assert(qnum_get_try_int(qnum, &val));
  62. g_assert_cmpint(val, ==, value);
  63. }
  64. static void test_visitor_out_bool(TestOutputVisitorData *data,
  65. const void *unused)
  66. {
  67. bool value = true;
  68. QBool *qbool;
  69. visit_type_bool(data->ov, NULL, &value, &error_abort);
  70. qbool = qobject_to(QBool, visitor_get(data));
  71. g_assert(qbool);
  72. g_assert(qbool_get_bool(qbool) == value);
  73. }
  74. static void test_visitor_out_number(TestOutputVisitorData *data,
  75. const void *unused)
  76. {
  77. double value = 3.14;
  78. QNum *qnum;
  79. visit_type_number(data->ov, NULL, &value, &error_abort);
  80. qnum = qobject_to(QNum, visitor_get(data));
  81. g_assert(qnum);
  82. g_assert(qnum_get_double(qnum) == value);
  83. }
  84. static void test_visitor_out_string(TestOutputVisitorData *data,
  85. const void *unused)
  86. {
  87. char *string = (char *) "Q E M U";
  88. QString *qstr;
  89. visit_type_str(data->ov, NULL, &string, &error_abort);
  90. qstr = qobject_to(QString, visitor_get(data));
  91. g_assert(qstr);
  92. g_assert_cmpstr(qstring_get_str(qstr), ==, string);
  93. }
  94. static void test_visitor_out_no_string(TestOutputVisitorData *data,
  95. const void *unused)
  96. {
  97. char *string = NULL;
  98. QString *qstr;
  99. /* A null string should return "" */
  100. visit_type_str(data->ov, NULL, &string, &error_abort);
  101. qstr = qobject_to(QString, visitor_get(data));
  102. g_assert(qstr);
  103. g_assert_cmpstr(qstring_get_str(qstr), ==, "");
  104. }
  105. static void test_visitor_out_enum(TestOutputVisitorData *data,
  106. const void *unused)
  107. {
  108. EnumOne i;
  109. QString *qstr;
  110. for (i = 0; i < ENUM_ONE__MAX; i++) {
  111. visit_type_EnumOne(data->ov, "unused", &i, &error_abort);
  112. qstr = qobject_to(QString, visitor_get(data));
  113. g_assert(qstr);
  114. g_assert_cmpstr(qstring_get_str(qstr), ==, EnumOne_str(i));
  115. visitor_reset(data);
  116. }
  117. }
  118. static void test_visitor_out_enum_errors(TestOutputVisitorData *data,
  119. const void *unused)
  120. {
  121. EnumOne i, bad_values[] = { ENUM_ONE__MAX, -1 };
  122. Error *err;
  123. for (i = 0; i < ARRAY_SIZE(bad_values) ; i++) {
  124. err = NULL;
  125. visit_type_EnumOne(data->ov, "unused", &bad_values[i], &err);
  126. error_free_or_abort(&err);
  127. visitor_reset(data);
  128. }
  129. }
  130. static void test_visitor_out_struct(TestOutputVisitorData *data,
  131. const void *unused)
  132. {
  133. TestStruct test_struct = { .integer = 42,
  134. .boolean = false,
  135. .string = (char *) "foo"};
  136. TestStruct *p = &test_struct;
  137. QDict *qdict;
  138. visit_type_TestStruct(data->ov, NULL, &p, &error_abort);
  139. qdict = qobject_to(QDict, visitor_get(data));
  140. g_assert(qdict);
  141. g_assert_cmpint(qdict_size(qdict), ==, 3);
  142. g_assert_cmpint(qdict_get_int(qdict, "integer"), ==, 42);
  143. g_assert_cmpint(qdict_get_bool(qdict, "boolean"), ==, false);
  144. g_assert_cmpstr(qdict_get_str(qdict, "string"), ==, "foo");
  145. }
  146. static void test_visitor_out_struct_nested(TestOutputVisitorData *data,
  147. const void *unused)
  148. {
  149. int64_t value = 42;
  150. UserDefTwo *ud2;
  151. QDict *qdict, *dict1, *dict2, *dict3, *userdef;
  152. const char *string = "user def string";
  153. const char *strings[] = { "forty two", "forty three", "forty four",
  154. "forty five" };
  155. ud2 = g_malloc0(sizeof(*ud2));
  156. ud2->string0 = g_strdup(strings[0]);
  157. ud2->dict1 = g_malloc0(sizeof(*ud2->dict1));
  158. ud2->dict1->string1 = g_strdup(strings[1]);
  159. ud2->dict1->dict2 = g_malloc0(sizeof(*ud2->dict1->dict2));
  160. ud2->dict1->dict2->userdef = g_new0(UserDefOne, 1);
  161. ud2->dict1->dict2->userdef->string = g_strdup(string);
  162. ud2->dict1->dict2->userdef->integer = value;
  163. ud2->dict1->dict2->string = g_strdup(strings[2]);
  164. ud2->dict1->dict3 = g_malloc0(sizeof(*ud2->dict1->dict3));
  165. ud2->dict1->has_dict3 = true;
  166. ud2->dict1->dict3->userdef = g_new0(UserDefOne, 1);
  167. ud2->dict1->dict3->userdef->string = g_strdup(string);
  168. ud2->dict1->dict3->userdef->integer = value;
  169. ud2->dict1->dict3->string = g_strdup(strings[3]);
  170. visit_type_UserDefTwo(data->ov, "unused", &ud2, &error_abort);
  171. qdict = qobject_to(QDict, visitor_get(data));
  172. g_assert(qdict);
  173. g_assert_cmpint(qdict_size(qdict), ==, 2);
  174. g_assert_cmpstr(qdict_get_str(qdict, "string0"), ==, strings[0]);
  175. dict1 = qdict_get_qdict(qdict, "dict1");
  176. g_assert_cmpint(qdict_size(dict1), ==, 3);
  177. g_assert_cmpstr(qdict_get_str(dict1, "string1"), ==, strings[1]);
  178. dict2 = qdict_get_qdict(dict1, "dict2");
  179. g_assert_cmpint(qdict_size(dict2), ==, 2);
  180. g_assert_cmpstr(qdict_get_str(dict2, "string"), ==, strings[2]);
  181. userdef = qdict_get_qdict(dict2, "userdef");
  182. g_assert_cmpint(qdict_size(userdef), ==, 2);
  183. g_assert_cmpint(qdict_get_int(userdef, "integer"), ==, value);
  184. g_assert_cmpstr(qdict_get_str(userdef, "string"), ==, string);
  185. dict3 = qdict_get_qdict(dict1, "dict3");
  186. g_assert_cmpint(qdict_size(dict3), ==, 2);
  187. g_assert_cmpstr(qdict_get_str(dict3, "string"), ==, strings[3]);
  188. userdef = qdict_get_qdict(dict3, "userdef");
  189. g_assert_cmpint(qdict_size(userdef), ==, 2);
  190. g_assert_cmpint(qdict_get_int(userdef, "integer"), ==, value);
  191. g_assert_cmpstr(qdict_get_str(userdef, "string"), ==, string);
  192. qapi_free_UserDefTwo(ud2);
  193. }
  194. static void test_visitor_out_struct_errors(TestOutputVisitorData *data,
  195. const void *unused)
  196. {
  197. EnumOne bad_values[] = { ENUM_ONE__MAX, -1 };
  198. UserDefOne u = {0};
  199. UserDefOne *pu = &u;
  200. Error *err;
  201. int i;
  202. for (i = 0; i < ARRAY_SIZE(bad_values) ; i++) {
  203. err = NULL;
  204. u.has_enum1 = true;
  205. u.enum1 = bad_values[i];
  206. visit_type_UserDefOne(data->ov, "unused", &pu, &err);
  207. error_free_or_abort(&err);
  208. visitor_reset(data);
  209. }
  210. }
  211. static void test_visitor_out_list(TestOutputVisitorData *data,
  212. const void *unused)
  213. {
  214. const char *value_str = "list value";
  215. TestStructList *p, *head = NULL;
  216. const int max_items = 10;
  217. bool value_bool = true;
  218. int value_int = 10;
  219. QListEntry *entry;
  220. QList *qlist;
  221. int i;
  222. /* Build the list in reverse order... */
  223. for (i = 0; i < max_items; i++) {
  224. p = g_malloc0(sizeof(*p));
  225. p->value = g_malloc0(sizeof(*p->value));
  226. p->value->integer = value_int + (max_items - i - 1);
  227. p->value->boolean = value_bool;
  228. p->value->string = g_strdup(value_str);
  229. p->next = head;
  230. head = p;
  231. }
  232. visit_type_TestStructList(data->ov, NULL, &head, &error_abort);
  233. qlist = qobject_to(QList, visitor_get(data));
  234. g_assert(qlist);
  235. g_assert(!qlist_empty(qlist));
  236. /* ...and ensure that the visitor sees it in order */
  237. i = 0;
  238. QLIST_FOREACH_ENTRY(qlist, entry) {
  239. QDict *qdict;
  240. qdict = qobject_to(QDict, entry->value);
  241. g_assert(qdict);
  242. g_assert_cmpint(qdict_size(qdict), ==, 3);
  243. g_assert_cmpint(qdict_get_int(qdict, "integer"), ==, value_int + i);
  244. g_assert_cmpint(qdict_get_bool(qdict, "boolean"), ==, value_bool);
  245. g_assert_cmpstr(qdict_get_str(qdict, "string"), ==, value_str);
  246. i++;
  247. }
  248. g_assert_cmpint(i, ==, max_items);
  249. qapi_free_TestStructList(head);
  250. }
  251. static void test_visitor_out_list_qapi_free(TestOutputVisitorData *data,
  252. const void *unused)
  253. {
  254. UserDefTwoList *p, *head = NULL;
  255. const char string[] = "foo bar";
  256. int i, max_count = 1024;
  257. for (i = 0; i < max_count; i++) {
  258. p = g_malloc0(sizeof(*p));
  259. p->value = g_malloc0(sizeof(*p->value));
  260. p->value->string0 = g_strdup(string);
  261. p->value->dict1 = g_new0(UserDefTwoDict, 1);
  262. p->value->dict1->string1 = g_strdup(string);
  263. p->value->dict1->dict2 = g_new0(UserDefTwoDictDict, 1);
  264. p->value->dict1->dict2->userdef = g_new0(UserDefOne, 1);
  265. p->value->dict1->dict2->userdef->string = g_strdup(string);
  266. p->value->dict1->dict2->userdef->integer = 42;
  267. p->value->dict1->dict2->string = g_strdup(string);
  268. p->value->dict1->has_dict3 = false;
  269. p->next = head;
  270. head = p;
  271. }
  272. qapi_free_UserDefTwoList(head);
  273. }
  274. static void test_visitor_out_any(TestOutputVisitorData *data,
  275. const void *unused)
  276. {
  277. QObject *qobj;
  278. QNum *qnum;
  279. QBool *qbool;
  280. QString *qstring;
  281. QDict *qdict;
  282. int64_t val;
  283. qobj = QOBJECT(qnum_from_int(-42));
  284. visit_type_any(data->ov, NULL, &qobj, &error_abort);
  285. qnum = qobject_to(QNum, visitor_get(data));
  286. g_assert(qnum);
  287. g_assert(qnum_get_try_int(qnum, &val));
  288. g_assert_cmpint(val, ==, -42);
  289. qobject_unref(qobj);
  290. visitor_reset(data);
  291. qdict = qdict_new();
  292. qdict_put_int(qdict, "integer", -42);
  293. qdict_put_bool(qdict, "boolean", true);
  294. qdict_put_str(qdict, "string", "foo");
  295. qobj = QOBJECT(qdict);
  296. visit_type_any(data->ov, NULL, &qobj, &error_abort);
  297. qobject_unref(qobj);
  298. qdict = qobject_to(QDict, visitor_get(data));
  299. g_assert(qdict);
  300. qnum = qobject_to(QNum, qdict_get(qdict, "integer"));
  301. g_assert(qnum);
  302. g_assert(qnum_get_try_int(qnum, &val));
  303. g_assert_cmpint(val, ==, -42);
  304. qbool = qobject_to(QBool, qdict_get(qdict, "boolean"));
  305. g_assert(qbool);
  306. g_assert(qbool_get_bool(qbool) == true);
  307. qstring = qobject_to(QString, qdict_get(qdict, "string"));
  308. g_assert(qstring);
  309. g_assert_cmpstr(qstring_get_str(qstring), ==, "foo");
  310. }
  311. static void test_visitor_out_union_flat(TestOutputVisitorData *data,
  312. const void *unused)
  313. {
  314. QDict *qdict;
  315. UserDefFlatUnion *tmp = g_malloc0(sizeof(UserDefFlatUnion));
  316. tmp->enum1 = ENUM_ONE_VALUE1;
  317. tmp->string = g_strdup("str");
  318. tmp->integer = 41;
  319. tmp->u.value1.boolean = true;
  320. visit_type_UserDefFlatUnion(data->ov, NULL, &tmp, &error_abort);
  321. qdict = qobject_to(QDict, visitor_get(data));
  322. g_assert(qdict);
  323. g_assert_cmpstr(qdict_get_str(qdict, "enum1"), ==, "value1");
  324. g_assert_cmpstr(qdict_get_str(qdict, "string"), ==, "str");
  325. g_assert_cmpint(qdict_get_int(qdict, "integer"), ==, 41);
  326. g_assert_cmpint(qdict_get_bool(qdict, "boolean"), ==, true);
  327. qapi_free_UserDefFlatUnion(tmp);
  328. }
  329. static void test_visitor_out_alternate(TestOutputVisitorData *data,
  330. const void *unused)
  331. {
  332. UserDefAlternate *tmp;
  333. QNum *qnum;
  334. QString *qstr;
  335. QDict *qdict;
  336. int64_t val;
  337. tmp = g_new0(UserDefAlternate, 1);
  338. tmp->type = QTYPE_QNUM;
  339. tmp->u.i = 42;
  340. visit_type_UserDefAlternate(data->ov, NULL, &tmp, &error_abort);
  341. qnum = qobject_to(QNum, visitor_get(data));
  342. g_assert(qnum);
  343. g_assert(qnum_get_try_int(qnum, &val));
  344. g_assert_cmpint(val, ==, 42);
  345. qapi_free_UserDefAlternate(tmp);
  346. visitor_reset(data);
  347. tmp = g_new0(UserDefAlternate, 1);
  348. tmp->type = QTYPE_QSTRING;
  349. tmp->u.e = ENUM_ONE_VALUE1;
  350. visit_type_UserDefAlternate(data->ov, NULL, &tmp, &error_abort);
  351. qstr = qobject_to(QString, visitor_get(data));
  352. g_assert(qstr);
  353. g_assert_cmpstr(qstring_get_str(qstr), ==, "value1");
  354. qapi_free_UserDefAlternate(tmp);
  355. visitor_reset(data);
  356. tmp = g_new0(UserDefAlternate, 1);
  357. tmp->type = QTYPE_QNULL;
  358. tmp->u.n = qnull();
  359. visit_type_UserDefAlternate(data->ov, NULL, &tmp, &error_abort);
  360. g_assert_cmpint(qobject_type(visitor_get(data)), ==, QTYPE_QNULL);
  361. qapi_free_UserDefAlternate(tmp);
  362. visitor_reset(data);
  363. tmp = g_new0(UserDefAlternate, 1);
  364. tmp->type = QTYPE_QDICT;
  365. tmp->u.udfu.integer = 1;
  366. tmp->u.udfu.string = g_strdup("str");
  367. tmp->u.udfu.enum1 = ENUM_ONE_VALUE1;
  368. tmp->u.udfu.u.value1.boolean = true;
  369. visit_type_UserDefAlternate(data->ov, NULL, &tmp, &error_abort);
  370. qdict = qobject_to(QDict, visitor_get(data));
  371. g_assert(qdict);
  372. g_assert_cmpint(qdict_size(qdict), ==, 4);
  373. g_assert_cmpint(qdict_get_int(qdict, "integer"), ==, 1);
  374. g_assert_cmpstr(qdict_get_str(qdict, "string"), ==, "str");
  375. g_assert_cmpstr(qdict_get_str(qdict, "enum1"), ==, "value1");
  376. g_assert_cmpint(qdict_get_bool(qdict, "boolean"), ==, true);
  377. qapi_free_UserDefAlternate(tmp);
  378. }
  379. static void test_visitor_out_null(TestOutputVisitorData *data,
  380. const void *unused)
  381. {
  382. QNull *null = NULL;
  383. QDict *qdict;
  384. QObject *nil;
  385. visit_start_struct(data->ov, NULL, NULL, 0, &error_abort);
  386. visit_type_null(data->ov, "a", &null, &error_abort);
  387. visit_check_struct(data->ov, &error_abort);
  388. visit_end_struct(data->ov, NULL);
  389. qdict = qobject_to(QDict, visitor_get(data));
  390. g_assert(qdict);
  391. g_assert_cmpint(qdict_size(qdict), ==, 1);
  392. nil = qdict_get(qdict, "a");
  393. g_assert(nil);
  394. g_assert(qobject_type(nil) == QTYPE_QNULL);
  395. }
  396. static void init_list_union(UserDefListUnion *cvalue)
  397. {
  398. int i;
  399. switch (cvalue->type) {
  400. case USER_DEF_LIST_UNION_KIND_INTEGER: {
  401. intList **list = &cvalue->u.integer.data;
  402. for (i = 0; i < 32; i++) {
  403. *list = g_new0(intList, 1);
  404. (*list)->value = i;
  405. (*list)->next = NULL;
  406. list = &(*list)->next;
  407. }
  408. break;
  409. }
  410. case USER_DEF_LIST_UNION_KIND_S8: {
  411. int8List **list = &cvalue->u.s8.data;
  412. for (i = 0; i < 32; i++) {
  413. *list = g_new0(int8List, 1);
  414. (*list)->value = i;
  415. (*list)->next = NULL;
  416. list = &(*list)->next;
  417. }
  418. break;
  419. }
  420. case USER_DEF_LIST_UNION_KIND_S16: {
  421. int16List **list = &cvalue->u.s16.data;
  422. for (i = 0; i < 32; i++) {
  423. *list = g_new0(int16List, 1);
  424. (*list)->value = i;
  425. (*list)->next = NULL;
  426. list = &(*list)->next;
  427. }
  428. break;
  429. }
  430. case USER_DEF_LIST_UNION_KIND_S32: {
  431. int32List **list = &cvalue->u.s32.data;
  432. for (i = 0; i < 32; i++) {
  433. *list = g_new0(int32List, 1);
  434. (*list)->value = i;
  435. (*list)->next = NULL;
  436. list = &(*list)->next;
  437. }
  438. break;
  439. }
  440. case USER_DEF_LIST_UNION_KIND_S64: {
  441. int64List **list = &cvalue->u.s64.data;
  442. for (i = 0; i < 32; i++) {
  443. *list = g_new0(int64List, 1);
  444. (*list)->value = i;
  445. (*list)->next = NULL;
  446. list = &(*list)->next;
  447. }
  448. break;
  449. }
  450. case USER_DEF_LIST_UNION_KIND_U8: {
  451. uint8List **list = &cvalue->u.u8.data;
  452. for (i = 0; i < 32; i++) {
  453. *list = g_new0(uint8List, 1);
  454. (*list)->value = i;
  455. (*list)->next = NULL;
  456. list = &(*list)->next;
  457. }
  458. break;
  459. }
  460. case USER_DEF_LIST_UNION_KIND_U16: {
  461. uint16List **list = &cvalue->u.u16.data;
  462. for (i = 0; i < 32; i++) {
  463. *list = g_new0(uint16List, 1);
  464. (*list)->value = i;
  465. (*list)->next = NULL;
  466. list = &(*list)->next;
  467. }
  468. break;
  469. }
  470. case USER_DEF_LIST_UNION_KIND_U32: {
  471. uint32List **list = &cvalue->u.u32.data;
  472. for (i = 0; i < 32; i++) {
  473. *list = g_new0(uint32List, 1);
  474. (*list)->value = i;
  475. (*list)->next = NULL;
  476. list = &(*list)->next;
  477. }
  478. break;
  479. }
  480. case USER_DEF_LIST_UNION_KIND_U64: {
  481. uint64List **list = &cvalue->u.u64.data;
  482. for (i = 0; i < 32; i++) {
  483. *list = g_new0(uint64List, 1);
  484. (*list)->value = i;
  485. (*list)->next = NULL;
  486. list = &(*list)->next;
  487. }
  488. break;
  489. }
  490. case USER_DEF_LIST_UNION_KIND_BOOLEAN: {
  491. boolList **list = &cvalue->u.boolean.data;
  492. for (i = 0; i < 32; i++) {
  493. *list = g_new0(boolList, 1);
  494. (*list)->value = QEMU_IS_ALIGNED(i, 3);
  495. (*list)->next = NULL;
  496. list = &(*list)->next;
  497. }
  498. break;
  499. }
  500. case USER_DEF_LIST_UNION_KIND_STRING: {
  501. strList **list = &cvalue->u.string.data;
  502. for (i = 0; i < 32; i++) {
  503. *list = g_new0(strList, 1);
  504. (*list)->value = g_strdup_printf("%d", i);
  505. (*list)->next = NULL;
  506. list = &(*list)->next;
  507. }
  508. break;
  509. }
  510. case USER_DEF_LIST_UNION_KIND_NUMBER: {
  511. numberList **list = &cvalue->u.number.data;
  512. for (i = 0; i < 32; i++) {
  513. *list = g_new0(numberList, 1);
  514. (*list)->value = (double)i / 3;
  515. (*list)->next = NULL;
  516. list = &(*list)->next;
  517. }
  518. break;
  519. }
  520. default:
  521. g_assert_not_reached();
  522. }
  523. }
  524. static void check_list_union(QObject *qobj,
  525. UserDefListUnionKind kind)
  526. {
  527. QDict *qdict;
  528. QList *qlist;
  529. int i;
  530. qdict = qobject_to(QDict, qobj);
  531. g_assert(qdict);
  532. g_assert(qdict_haskey(qdict, "data"));
  533. qlist = qlist_copy(qobject_to(QList, qdict_get(qdict, "data")));
  534. switch (kind) {
  535. case USER_DEF_LIST_UNION_KIND_U8:
  536. case USER_DEF_LIST_UNION_KIND_U16:
  537. case USER_DEF_LIST_UNION_KIND_U32:
  538. case USER_DEF_LIST_UNION_KIND_U64:
  539. for (i = 0; i < 32; i++) {
  540. QObject *tmp;
  541. QNum *qvalue;
  542. uint64_t val;
  543. tmp = qlist_peek(qlist);
  544. g_assert(tmp);
  545. qvalue = qobject_to(QNum, tmp);
  546. g_assert(qnum_get_try_uint(qvalue, &val));
  547. g_assert_cmpint(val, ==, i);
  548. qobject_unref(qlist_pop(qlist));
  549. }
  550. break;
  551. case USER_DEF_LIST_UNION_KIND_S8:
  552. case USER_DEF_LIST_UNION_KIND_S16:
  553. case USER_DEF_LIST_UNION_KIND_S32:
  554. case USER_DEF_LIST_UNION_KIND_S64:
  555. /*
  556. * All integer elements in JSON arrays get stored into QNums
  557. * when we convert to QObjects, so we can check them all in
  558. * the same fashion, so simply fall through here.
  559. */
  560. case USER_DEF_LIST_UNION_KIND_INTEGER:
  561. for (i = 0; i < 32; i++) {
  562. QObject *tmp;
  563. QNum *qvalue;
  564. int64_t val;
  565. tmp = qlist_peek(qlist);
  566. g_assert(tmp);
  567. qvalue = qobject_to(QNum, tmp);
  568. g_assert(qnum_get_try_int(qvalue, &val));
  569. g_assert_cmpint(val, ==, i);
  570. qobject_unref(qlist_pop(qlist));
  571. }
  572. break;
  573. case USER_DEF_LIST_UNION_KIND_BOOLEAN:
  574. for (i = 0; i < 32; i++) {
  575. QObject *tmp;
  576. QBool *qvalue;
  577. tmp = qlist_peek(qlist);
  578. g_assert(tmp);
  579. qvalue = qobject_to(QBool, tmp);
  580. g_assert_cmpint(qbool_get_bool(qvalue), ==, i % 3 == 0);
  581. qobject_unref(qlist_pop(qlist));
  582. }
  583. break;
  584. case USER_DEF_LIST_UNION_KIND_STRING:
  585. for (i = 0; i < 32; i++) {
  586. QObject *tmp;
  587. QString *qvalue;
  588. gchar str[8];
  589. tmp = qlist_peek(qlist);
  590. g_assert(tmp);
  591. qvalue = qobject_to(QString, tmp);
  592. sprintf(str, "%d", i);
  593. g_assert_cmpstr(qstring_get_str(qvalue), ==, str);
  594. qobject_unref(qlist_pop(qlist));
  595. }
  596. break;
  597. case USER_DEF_LIST_UNION_KIND_NUMBER:
  598. for (i = 0; i < 32; i++) {
  599. QObject *tmp;
  600. QNum *qvalue;
  601. GString *double_expected = g_string_new("");
  602. GString *double_actual = g_string_new("");
  603. tmp = qlist_peek(qlist);
  604. g_assert(tmp);
  605. qvalue = qobject_to(QNum, tmp);
  606. g_string_printf(double_expected, "%.6f", (double)i / 3);
  607. g_string_printf(double_actual, "%.6f", qnum_get_double(qvalue));
  608. g_assert_cmpstr(double_actual->str, ==, double_expected->str);
  609. qobject_unref(qlist_pop(qlist));
  610. g_string_free(double_expected, true);
  611. g_string_free(double_actual, true);
  612. }
  613. break;
  614. default:
  615. g_assert_not_reached();
  616. }
  617. qobject_unref(qlist);
  618. }
  619. static void test_list_union(TestOutputVisitorData *data,
  620. const void *unused,
  621. UserDefListUnionKind kind)
  622. {
  623. UserDefListUnion *cvalue = g_new0(UserDefListUnion, 1);
  624. QObject *obj;
  625. cvalue->type = kind;
  626. init_list_union(cvalue);
  627. visit_type_UserDefListUnion(data->ov, NULL, &cvalue, &error_abort);
  628. obj = visitor_get(data);
  629. check_list_union(obj, cvalue->type);
  630. qapi_free_UserDefListUnion(cvalue);
  631. }
  632. static void test_visitor_out_list_union_int(TestOutputVisitorData *data,
  633. const void *unused)
  634. {
  635. test_list_union(data, unused, USER_DEF_LIST_UNION_KIND_INTEGER);
  636. }
  637. static void test_visitor_out_list_union_int8(TestOutputVisitorData *data,
  638. const void *unused)
  639. {
  640. test_list_union(data, unused, USER_DEF_LIST_UNION_KIND_S8);
  641. }
  642. static void test_visitor_out_list_union_int16(TestOutputVisitorData *data,
  643. const void *unused)
  644. {
  645. test_list_union(data, unused, USER_DEF_LIST_UNION_KIND_S16);
  646. }
  647. static void test_visitor_out_list_union_int32(TestOutputVisitorData *data,
  648. const void *unused)
  649. {
  650. test_list_union(data, unused, USER_DEF_LIST_UNION_KIND_S32);
  651. }
  652. static void test_visitor_out_list_union_int64(TestOutputVisitorData *data,
  653. const void *unused)
  654. {
  655. test_list_union(data, unused, USER_DEF_LIST_UNION_KIND_S64);
  656. }
  657. static void test_visitor_out_list_union_uint8(TestOutputVisitorData *data,
  658. const void *unused)
  659. {
  660. test_list_union(data, unused, USER_DEF_LIST_UNION_KIND_U8);
  661. }
  662. static void test_visitor_out_list_union_uint16(TestOutputVisitorData *data,
  663. const void *unused)
  664. {
  665. test_list_union(data, unused, USER_DEF_LIST_UNION_KIND_U16);
  666. }
  667. static void test_visitor_out_list_union_uint32(TestOutputVisitorData *data,
  668. const void *unused)
  669. {
  670. test_list_union(data, unused, USER_DEF_LIST_UNION_KIND_U32);
  671. }
  672. static void test_visitor_out_list_union_uint64(TestOutputVisitorData *data,
  673. const void *unused)
  674. {
  675. test_list_union(data, unused, USER_DEF_LIST_UNION_KIND_U64);
  676. }
  677. static void test_visitor_out_list_union_bool(TestOutputVisitorData *data,
  678. const void *unused)
  679. {
  680. test_list_union(data, unused, USER_DEF_LIST_UNION_KIND_BOOLEAN);
  681. }
  682. static void test_visitor_out_list_union_str(TestOutputVisitorData *data,
  683. const void *unused)
  684. {
  685. test_list_union(data, unused, USER_DEF_LIST_UNION_KIND_STRING);
  686. }
  687. static void test_visitor_out_list_union_number(TestOutputVisitorData *data,
  688. const void *unused)
  689. {
  690. test_list_union(data, unused, USER_DEF_LIST_UNION_KIND_NUMBER);
  691. }
  692. static void output_visitor_test_add(const char *testpath,
  693. TestOutputVisitorData *data,
  694. void (*test_func)(TestOutputVisitorData *data, const void *user_data))
  695. {
  696. g_test_add(testpath, TestOutputVisitorData, data, visitor_output_setup,
  697. test_func, visitor_output_teardown);
  698. }
  699. int main(int argc, char **argv)
  700. {
  701. TestOutputVisitorData out_visitor_data;
  702. g_test_init(&argc, &argv, NULL);
  703. output_visitor_test_add("/visitor/output/int",
  704. &out_visitor_data, test_visitor_out_int);
  705. output_visitor_test_add("/visitor/output/bool",
  706. &out_visitor_data, test_visitor_out_bool);
  707. output_visitor_test_add("/visitor/output/number",
  708. &out_visitor_data, test_visitor_out_number);
  709. output_visitor_test_add("/visitor/output/string",
  710. &out_visitor_data, test_visitor_out_string);
  711. output_visitor_test_add("/visitor/output/no-string",
  712. &out_visitor_data, test_visitor_out_no_string);
  713. output_visitor_test_add("/visitor/output/enum",
  714. &out_visitor_data, test_visitor_out_enum);
  715. output_visitor_test_add("/visitor/output/enum-errors",
  716. &out_visitor_data, test_visitor_out_enum_errors);
  717. output_visitor_test_add("/visitor/output/struct",
  718. &out_visitor_data, test_visitor_out_struct);
  719. output_visitor_test_add("/visitor/output/struct-nested",
  720. &out_visitor_data, test_visitor_out_struct_nested);
  721. output_visitor_test_add("/visitor/output/struct-errors",
  722. &out_visitor_data, test_visitor_out_struct_errors);
  723. output_visitor_test_add("/visitor/output/list",
  724. &out_visitor_data, test_visitor_out_list);
  725. output_visitor_test_add("/visitor/output/any",
  726. &out_visitor_data, test_visitor_out_any);
  727. output_visitor_test_add("/visitor/output/list-qapi-free",
  728. &out_visitor_data, test_visitor_out_list_qapi_free);
  729. output_visitor_test_add("/visitor/output/union-flat",
  730. &out_visitor_data, test_visitor_out_union_flat);
  731. output_visitor_test_add("/visitor/output/alternate",
  732. &out_visitor_data, test_visitor_out_alternate);
  733. output_visitor_test_add("/visitor/output/null",
  734. &out_visitor_data, test_visitor_out_null);
  735. output_visitor_test_add("/visitor/output/list_union/int",
  736. &out_visitor_data,
  737. test_visitor_out_list_union_int);
  738. output_visitor_test_add("/visitor/output/list_union/int8",
  739. &out_visitor_data,
  740. test_visitor_out_list_union_int8);
  741. output_visitor_test_add("/visitor/output/list_union/int16",
  742. &out_visitor_data,
  743. test_visitor_out_list_union_int16);
  744. output_visitor_test_add("/visitor/output/list_union/int32",
  745. &out_visitor_data,
  746. test_visitor_out_list_union_int32);
  747. output_visitor_test_add("/visitor/output/list_union/int64",
  748. &out_visitor_data,
  749. test_visitor_out_list_union_int64);
  750. output_visitor_test_add("/visitor/output/list_union/uint8",
  751. &out_visitor_data,
  752. test_visitor_out_list_union_uint8);
  753. output_visitor_test_add("/visitor/output/list_union/uint16",
  754. &out_visitor_data,
  755. test_visitor_out_list_union_uint16);
  756. output_visitor_test_add("/visitor/output/list_union/uint32",
  757. &out_visitor_data,
  758. test_visitor_out_list_union_uint32);
  759. output_visitor_test_add("/visitor/output/list_union/uint64",
  760. &out_visitor_data,
  761. test_visitor_out_list_union_uint64);
  762. output_visitor_test_add("/visitor/output/list_union/bool",
  763. &out_visitor_data,
  764. test_visitor_out_list_union_bool);
  765. output_visitor_test_add("/visitor/output/list_union/string",
  766. &out_visitor_data,
  767. test_visitor_out_list_union_str);
  768. output_visitor_test_add("/visitor/output/list_union/number",
  769. &out_visitor_data,
  770. test_visitor_out_list_union_number);
  771. g_test_run();
  772. return 0;
  773. }