2
0

test-qobject-output-visitor.c 26 KB

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