qapi: Improve reporting of redefinition
[qemu/armbru.git] / tests / test-qobject-output-visitor.c
blob3e993e5ba82abaffb7ca4c2280caf585ca988242
1 /*
2 * QObject Output Visitor unit-tests.
4 * Copyright (C) 2011-2016 Red Hat Inc.
6 * Authors:
7 * Luiz Capitulino <lcapitulino@redhat.com>
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.
13 #include "qemu/osdep.h"
15 #include "qemu-common.h"
16 #include "qapi/error.h"
17 #include "qapi/qobject-output-visitor.h"
18 #include "test-qapi-visit.h"
19 #include "qapi/qmp/qbool.h"
20 #include "qapi/qmp/qdict.h"
21 #include "qapi/qmp/qlist.h"
22 #include "qapi/qmp/qnull.h"
23 #include "qapi/qmp/qnum.h"
24 #include "qapi/qmp/qstring.h"
26 typedef struct TestOutputVisitorData {
27 Visitor *ov;
28 QObject *obj;
29 } TestOutputVisitorData;
31 static void visitor_output_setup(TestOutputVisitorData *data,
32 const void *unused)
34 data->ov = qobject_output_visitor_new(&data->obj);
35 g_assert(data->ov);
38 static void visitor_output_teardown(TestOutputVisitorData *data,
39 const void *unused)
41 visit_free(data->ov);
42 data->ov = NULL;
43 qobject_unref(data->obj);
44 data->obj = NULL;
47 static QObject *visitor_get(TestOutputVisitorData *data)
49 visit_complete(data->ov, &data->obj);
50 g_assert(data->obj);
51 return data->obj;
54 static void visitor_reset(TestOutputVisitorData *data)
56 visitor_output_teardown(data, NULL);
57 visitor_output_setup(data, NULL);
60 static void test_visitor_out_int(TestOutputVisitorData *data,
61 const void *unused)
63 int64_t value = -42;
64 int64_t val;
65 QNum *qnum;
67 visit_type_int(data->ov, NULL, &value, &error_abort);
69 qnum = qobject_to(QNum, visitor_get(data));
70 g_assert(qnum);
71 g_assert(qnum_get_try_int(qnum, &val));
72 g_assert_cmpint(val, ==, value);
75 static void test_visitor_out_bool(TestOutputVisitorData *data,
76 const void *unused)
78 bool value = true;
79 QBool *qbool;
81 visit_type_bool(data->ov, NULL, &value, &error_abort);
83 qbool = qobject_to(QBool, visitor_get(data));
84 g_assert(qbool);
85 g_assert(qbool_get_bool(qbool) == value);
88 static void test_visitor_out_number(TestOutputVisitorData *data,
89 const void *unused)
91 double value = 3.14;
92 QNum *qnum;
94 visit_type_number(data->ov, NULL, &value, &error_abort);
96 qnum = qobject_to(QNum, visitor_get(data));
97 g_assert(qnum);
98 g_assert(qnum_get_double(qnum) == value);
101 static void test_visitor_out_string(TestOutputVisitorData *data,
102 const void *unused)
104 char *string = (char *) "Q E M U";
105 QString *qstr;
107 visit_type_str(data->ov, NULL, &string, &error_abort);
109 qstr = qobject_to(QString, visitor_get(data));
110 g_assert(qstr);
111 g_assert_cmpstr(qstring_get_str(qstr), ==, string);
114 static void test_visitor_out_no_string(TestOutputVisitorData *data,
115 const void *unused)
117 char *string = NULL;
118 QString *qstr;
120 /* A null string should return "" */
121 visit_type_str(data->ov, NULL, &string, &error_abort);
123 qstr = qobject_to(QString, visitor_get(data));
124 g_assert(qstr);
125 g_assert_cmpstr(qstring_get_str(qstr), ==, "");
128 static void test_visitor_out_enum(TestOutputVisitorData *data,
129 const void *unused)
131 EnumOne i;
132 QString *qstr;
134 for (i = 0; i < ENUM_ONE__MAX; i++) {
135 visit_type_EnumOne(data->ov, "unused", &i, &error_abort);
137 qstr = qobject_to(QString, visitor_get(data));
138 g_assert(qstr);
139 g_assert_cmpstr(qstring_get_str(qstr), ==, EnumOne_str(i));
140 visitor_reset(data);
144 static void test_visitor_out_enum_errors(TestOutputVisitorData *data,
145 const void *unused)
147 EnumOne i, bad_values[] = { ENUM_ONE__MAX, -1 };
148 Error *err;
150 for (i = 0; i < ARRAY_SIZE(bad_values) ; i++) {
151 err = NULL;
152 visit_type_EnumOne(data->ov, "unused", &bad_values[i], &err);
153 error_free_or_abort(&err);
154 visitor_reset(data);
159 static void test_visitor_out_struct(TestOutputVisitorData *data,
160 const void *unused)
162 TestStruct test_struct = { .integer = 42,
163 .boolean = false,
164 .string = (char *) "foo"};
165 TestStruct *p = &test_struct;
166 QDict *qdict;
168 visit_type_TestStruct(data->ov, NULL, &p, &error_abort);
170 qdict = qobject_to(QDict, visitor_get(data));
171 g_assert(qdict);
172 g_assert_cmpint(qdict_size(qdict), ==, 3);
173 g_assert_cmpint(qdict_get_int(qdict, "integer"), ==, 42);
174 g_assert_cmpint(qdict_get_bool(qdict, "boolean"), ==, false);
175 g_assert_cmpstr(qdict_get_str(qdict, "string"), ==, "foo");
178 static void test_visitor_out_struct_nested(TestOutputVisitorData *data,
179 const void *unused)
181 int64_t value = 42;
182 UserDefTwo *ud2;
183 QDict *qdict, *dict1, *dict2, *dict3, *userdef;
184 const char *string = "user def string";
185 const char *strings[] = { "forty two", "forty three", "forty four",
186 "forty five" };
188 ud2 = g_malloc0(sizeof(*ud2));
189 ud2->string0 = g_strdup(strings[0]);
191 ud2->dict1 = g_malloc0(sizeof(*ud2->dict1));
192 ud2->dict1->string1 = g_strdup(strings[1]);
194 ud2->dict1->dict2 = g_malloc0(sizeof(*ud2->dict1->dict2));
195 ud2->dict1->dict2->userdef = g_new0(UserDefOne, 1);
196 ud2->dict1->dict2->userdef->string = g_strdup(string);
197 ud2->dict1->dict2->userdef->integer = value;
198 ud2->dict1->dict2->string = g_strdup(strings[2]);
200 ud2->dict1->dict3 = g_malloc0(sizeof(*ud2->dict1->dict3));
201 ud2->dict1->has_dict3 = true;
202 ud2->dict1->dict3->userdef = g_new0(UserDefOne, 1);
203 ud2->dict1->dict3->userdef->string = g_strdup(string);
204 ud2->dict1->dict3->userdef->integer = value;
205 ud2->dict1->dict3->string = g_strdup(strings[3]);
207 visit_type_UserDefTwo(data->ov, "unused", &ud2, &error_abort);
209 qdict = qobject_to(QDict, visitor_get(data));
210 g_assert(qdict);
211 g_assert_cmpint(qdict_size(qdict), ==, 2);
212 g_assert_cmpstr(qdict_get_str(qdict, "string0"), ==, strings[0]);
214 dict1 = qdict_get_qdict(qdict, "dict1");
215 g_assert_cmpint(qdict_size(dict1), ==, 3);
216 g_assert_cmpstr(qdict_get_str(dict1, "string1"), ==, strings[1]);
218 dict2 = qdict_get_qdict(dict1, "dict2");
219 g_assert_cmpint(qdict_size(dict2), ==, 2);
220 g_assert_cmpstr(qdict_get_str(dict2, "string"), ==, strings[2]);
221 userdef = qdict_get_qdict(dict2, "userdef");
222 g_assert_cmpint(qdict_size(userdef), ==, 2);
223 g_assert_cmpint(qdict_get_int(userdef, "integer"), ==, value);
224 g_assert_cmpstr(qdict_get_str(userdef, "string"), ==, string);
226 dict3 = qdict_get_qdict(dict1, "dict3");
227 g_assert_cmpint(qdict_size(dict3), ==, 2);
228 g_assert_cmpstr(qdict_get_str(dict3, "string"), ==, strings[3]);
229 userdef = qdict_get_qdict(dict3, "userdef");
230 g_assert_cmpint(qdict_size(userdef), ==, 2);
231 g_assert_cmpint(qdict_get_int(userdef, "integer"), ==, value);
232 g_assert_cmpstr(qdict_get_str(userdef, "string"), ==, string);
234 qapi_free_UserDefTwo(ud2);
237 static void test_visitor_out_struct_errors(TestOutputVisitorData *data,
238 const void *unused)
240 EnumOne bad_values[] = { ENUM_ONE__MAX, -1 };
241 UserDefOne u = {0};
242 UserDefOne *pu = &u;
243 Error *err;
244 int i;
246 for (i = 0; i < ARRAY_SIZE(bad_values) ; i++) {
247 err = NULL;
248 u.has_enum1 = true;
249 u.enum1 = bad_values[i];
250 visit_type_UserDefOne(data->ov, "unused", &pu, &err);
251 error_free_or_abort(&err);
252 visitor_reset(data);
257 static void test_visitor_out_list(TestOutputVisitorData *data,
258 const void *unused)
260 const char *value_str = "list value";
261 TestStructList *p, *head = NULL;
262 const int max_items = 10;
263 bool value_bool = true;
264 int value_int = 10;
265 QListEntry *entry;
266 QList *qlist;
267 int i;
269 /* Build the list in reverse order... */
270 for (i = 0; i < max_items; i++) {
271 p = g_malloc0(sizeof(*p));
272 p->value = g_malloc0(sizeof(*p->value));
273 p->value->integer = value_int + (max_items - i - 1);
274 p->value->boolean = value_bool;
275 p->value->string = g_strdup(value_str);
277 p->next = head;
278 head = p;
281 visit_type_TestStructList(data->ov, NULL, &head, &error_abort);
283 qlist = qobject_to(QList, visitor_get(data));
284 g_assert(qlist);
285 g_assert(!qlist_empty(qlist));
287 /* ...and ensure that the visitor sees it in order */
288 i = 0;
289 QLIST_FOREACH_ENTRY(qlist, entry) {
290 QDict *qdict;
292 qdict = qobject_to(QDict, entry->value);
293 g_assert(qdict);
294 g_assert_cmpint(qdict_size(qdict), ==, 3);
295 g_assert_cmpint(qdict_get_int(qdict, "integer"), ==, value_int + i);
296 g_assert_cmpint(qdict_get_bool(qdict, "boolean"), ==, value_bool);
297 g_assert_cmpstr(qdict_get_str(qdict, "string"), ==, value_str);
298 i++;
300 g_assert_cmpint(i, ==, max_items);
302 qapi_free_TestStructList(head);
305 static void test_visitor_out_list_qapi_free(TestOutputVisitorData *data,
306 const void *unused)
308 UserDefTwoList *p, *head = NULL;
309 const char string[] = "foo bar";
310 int i, max_count = 1024;
312 for (i = 0; i < max_count; i++) {
313 p = g_malloc0(sizeof(*p));
314 p->value = g_malloc0(sizeof(*p->value));
316 p->value->string0 = g_strdup(string);
317 p->value->dict1 = g_new0(UserDefTwoDict, 1);
318 p->value->dict1->string1 = g_strdup(string);
319 p->value->dict1->dict2 = g_new0(UserDefTwoDictDict, 1);
320 p->value->dict1->dict2->userdef = g_new0(UserDefOne, 1);
321 p->value->dict1->dict2->userdef->string = g_strdup(string);
322 p->value->dict1->dict2->userdef->integer = 42;
323 p->value->dict1->dict2->string = g_strdup(string);
324 p->value->dict1->has_dict3 = false;
326 p->next = head;
327 head = p;
330 qapi_free_UserDefTwoList(head);
333 static void test_visitor_out_any(TestOutputVisitorData *data,
334 const void *unused)
336 QObject *qobj;
337 QNum *qnum;
338 QBool *qbool;
339 QString *qstring;
340 QDict *qdict;
341 int64_t val;
343 qobj = QOBJECT(qnum_from_int(-42));
344 visit_type_any(data->ov, NULL, &qobj, &error_abort);
345 qnum = qobject_to(QNum, visitor_get(data));
346 g_assert(qnum);
347 g_assert(qnum_get_try_int(qnum, &val));
348 g_assert_cmpint(val, ==, -42);
349 qobject_unref(qobj);
351 visitor_reset(data);
352 qdict = qdict_new();
353 qdict_put_int(qdict, "integer", -42);
354 qdict_put_bool(qdict, "boolean", true);
355 qdict_put_str(qdict, "string", "foo");
356 qobj = QOBJECT(qdict);
357 visit_type_any(data->ov, NULL, &qobj, &error_abort);
358 qobject_unref(qobj);
359 qdict = qobject_to(QDict, visitor_get(data));
360 g_assert(qdict);
361 qnum = qobject_to(QNum, qdict_get(qdict, "integer"));
362 g_assert(qnum);
363 g_assert(qnum_get_try_int(qnum, &val));
364 g_assert_cmpint(val, ==, -42);
365 qbool = qobject_to(QBool, qdict_get(qdict, "boolean"));
366 g_assert(qbool);
367 g_assert(qbool_get_bool(qbool) == true);
368 qstring = qobject_to(QString, qdict_get(qdict, "string"));
369 g_assert(qstring);
370 g_assert_cmpstr(qstring_get_str(qstring), ==, "foo");
373 static void test_visitor_out_union_flat(TestOutputVisitorData *data,
374 const void *unused)
376 QDict *qdict;
378 UserDefFlatUnion *tmp = g_malloc0(sizeof(UserDefFlatUnion));
379 tmp->enum1 = ENUM_ONE_VALUE1;
380 tmp->string = g_strdup("str");
381 tmp->integer = 41;
382 tmp->u.value1.boolean = true;
384 visit_type_UserDefFlatUnion(data->ov, NULL, &tmp, &error_abort);
385 qdict = qobject_to(QDict, visitor_get(data));
386 g_assert(qdict);
387 g_assert_cmpstr(qdict_get_str(qdict, "enum1"), ==, "value1");
388 g_assert_cmpstr(qdict_get_str(qdict, "string"), ==, "str");
389 g_assert_cmpint(qdict_get_int(qdict, "integer"), ==, 41);
390 g_assert_cmpint(qdict_get_bool(qdict, "boolean"), ==, true);
392 qapi_free_UserDefFlatUnion(tmp);
395 static void test_visitor_out_alternate(TestOutputVisitorData *data,
396 const void *unused)
398 UserDefAlternate *tmp;
399 QNum *qnum;
400 QString *qstr;
401 QDict *qdict;
402 int64_t val;
404 tmp = g_new0(UserDefAlternate, 1);
405 tmp->type = QTYPE_QNUM;
406 tmp->u.i = 42;
408 visit_type_UserDefAlternate(data->ov, NULL, &tmp, &error_abort);
409 qnum = qobject_to(QNum, visitor_get(data));
410 g_assert(qnum);
411 g_assert(qnum_get_try_int(qnum, &val));
412 g_assert_cmpint(val, ==, 42);
414 qapi_free_UserDefAlternate(tmp);
416 visitor_reset(data);
417 tmp = g_new0(UserDefAlternate, 1);
418 tmp->type = QTYPE_QSTRING;
419 tmp->u.e = ENUM_ONE_VALUE1;
421 visit_type_UserDefAlternate(data->ov, NULL, &tmp, &error_abort);
422 qstr = qobject_to(QString, visitor_get(data));
423 g_assert(qstr);
424 g_assert_cmpstr(qstring_get_str(qstr), ==, "value1");
426 qapi_free_UserDefAlternate(tmp);
428 visitor_reset(data);
429 tmp = g_new0(UserDefAlternate, 1);
430 tmp->type = QTYPE_QNULL;
431 tmp->u.n = qnull();
433 visit_type_UserDefAlternate(data->ov, NULL, &tmp, &error_abort);
434 g_assert_cmpint(qobject_type(visitor_get(data)), ==, QTYPE_QNULL);
436 qapi_free_UserDefAlternate(tmp);
438 visitor_reset(data);
439 tmp = g_new0(UserDefAlternate, 1);
440 tmp->type = QTYPE_QDICT;
441 tmp->u.udfu.integer = 1;
442 tmp->u.udfu.string = g_strdup("str");
443 tmp->u.udfu.enum1 = ENUM_ONE_VALUE1;
444 tmp->u.udfu.u.value1.boolean = true;
446 visit_type_UserDefAlternate(data->ov, NULL, &tmp, &error_abort);
447 qdict = qobject_to(QDict, visitor_get(data));
448 g_assert(qdict);
449 g_assert_cmpint(qdict_size(qdict), ==, 4);
450 g_assert_cmpint(qdict_get_int(qdict, "integer"), ==, 1);
451 g_assert_cmpstr(qdict_get_str(qdict, "string"), ==, "str");
452 g_assert_cmpstr(qdict_get_str(qdict, "enum1"), ==, "value1");
453 g_assert_cmpint(qdict_get_bool(qdict, "boolean"), ==, true);
455 qapi_free_UserDefAlternate(tmp);
458 static void test_visitor_out_null(TestOutputVisitorData *data,
459 const void *unused)
461 QNull *null = NULL;
462 QDict *qdict;
463 QObject *nil;
465 visit_start_struct(data->ov, NULL, NULL, 0, &error_abort);
466 visit_type_null(data->ov, "a", &null, &error_abort);
467 visit_check_struct(data->ov, &error_abort);
468 visit_end_struct(data->ov, NULL);
469 qdict = qobject_to(QDict, visitor_get(data));
470 g_assert(qdict);
471 g_assert_cmpint(qdict_size(qdict), ==, 1);
472 nil = qdict_get(qdict, "a");
473 g_assert(nil);
474 g_assert(qobject_type(nil) == QTYPE_QNULL);
477 static void init_list_union(UserDefListUnion *cvalue)
479 int i;
480 switch (cvalue->type) {
481 case USER_DEF_LIST_UNION_KIND_INTEGER: {
482 intList **list = &cvalue->u.integer.data;
483 for (i = 0; i < 32; i++) {
484 *list = g_new0(intList, 1);
485 (*list)->value = i;
486 (*list)->next = NULL;
487 list = &(*list)->next;
489 break;
491 case USER_DEF_LIST_UNION_KIND_S8: {
492 int8List **list = &cvalue->u.s8.data;
493 for (i = 0; i < 32; i++) {
494 *list = g_new0(int8List, 1);
495 (*list)->value = i;
496 (*list)->next = NULL;
497 list = &(*list)->next;
499 break;
501 case USER_DEF_LIST_UNION_KIND_S16: {
502 int16List **list = &cvalue->u.s16.data;
503 for (i = 0; i < 32; i++) {
504 *list = g_new0(int16List, 1);
505 (*list)->value = i;
506 (*list)->next = NULL;
507 list = &(*list)->next;
509 break;
511 case USER_DEF_LIST_UNION_KIND_S32: {
512 int32List **list = &cvalue->u.s32.data;
513 for (i = 0; i < 32; i++) {
514 *list = g_new0(int32List, 1);
515 (*list)->value = i;
516 (*list)->next = NULL;
517 list = &(*list)->next;
519 break;
521 case USER_DEF_LIST_UNION_KIND_S64: {
522 int64List **list = &cvalue->u.s64.data;
523 for (i = 0; i < 32; i++) {
524 *list = g_new0(int64List, 1);
525 (*list)->value = i;
526 (*list)->next = NULL;
527 list = &(*list)->next;
529 break;
531 case USER_DEF_LIST_UNION_KIND_U8: {
532 uint8List **list = &cvalue->u.u8.data;
533 for (i = 0; i < 32; i++) {
534 *list = g_new0(uint8List, 1);
535 (*list)->value = i;
536 (*list)->next = NULL;
537 list = &(*list)->next;
539 break;
541 case USER_DEF_LIST_UNION_KIND_U16: {
542 uint16List **list = &cvalue->u.u16.data;
543 for (i = 0; i < 32; i++) {
544 *list = g_new0(uint16List, 1);
545 (*list)->value = i;
546 (*list)->next = NULL;
547 list = &(*list)->next;
549 break;
551 case USER_DEF_LIST_UNION_KIND_U32: {
552 uint32List **list = &cvalue->u.u32.data;
553 for (i = 0; i < 32; i++) {
554 *list = g_new0(uint32List, 1);
555 (*list)->value = i;
556 (*list)->next = NULL;
557 list = &(*list)->next;
559 break;
561 case USER_DEF_LIST_UNION_KIND_U64: {
562 uint64List **list = &cvalue->u.u64.data;
563 for (i = 0; i < 32; i++) {
564 *list = g_new0(uint64List, 1);
565 (*list)->value = i;
566 (*list)->next = NULL;
567 list = &(*list)->next;
569 break;
571 case USER_DEF_LIST_UNION_KIND_BOOLEAN: {
572 boolList **list = &cvalue->u.boolean.data;
573 for (i = 0; i < 32; i++) {
574 *list = g_new0(boolList, 1);
575 (*list)->value = QEMU_IS_ALIGNED(i, 3);
576 (*list)->next = NULL;
577 list = &(*list)->next;
579 break;
581 case USER_DEF_LIST_UNION_KIND_STRING: {
582 strList **list = &cvalue->u.string.data;
583 for (i = 0; i < 32; i++) {
584 *list = g_new0(strList, 1);
585 (*list)->value = g_strdup_printf("%d", i);
586 (*list)->next = NULL;
587 list = &(*list)->next;
589 break;
591 case USER_DEF_LIST_UNION_KIND_NUMBER: {
592 numberList **list = &cvalue->u.number.data;
593 for (i = 0; i < 32; i++) {
594 *list = g_new0(numberList, 1);
595 (*list)->value = (double)i / 3;
596 (*list)->next = NULL;
597 list = &(*list)->next;
599 break;
601 default:
602 g_assert_not_reached();
606 static void check_list_union(QObject *qobj,
607 UserDefListUnionKind kind)
609 QDict *qdict;
610 QList *qlist;
611 int i;
613 qdict = qobject_to(QDict, qobj);
614 g_assert(qdict);
615 g_assert(qdict_haskey(qdict, "data"));
616 qlist = qlist_copy(qobject_to(QList, qdict_get(qdict, "data")));
618 switch (kind) {
619 case USER_DEF_LIST_UNION_KIND_U8:
620 case USER_DEF_LIST_UNION_KIND_U16:
621 case USER_DEF_LIST_UNION_KIND_U32:
622 case USER_DEF_LIST_UNION_KIND_U64:
623 for (i = 0; i < 32; i++) {
624 QObject *tmp;
625 QNum *qvalue;
626 uint64_t val;
628 tmp = qlist_peek(qlist);
629 g_assert(tmp);
630 qvalue = qobject_to(QNum, tmp);
631 g_assert(qnum_get_try_uint(qvalue, &val));
632 g_assert_cmpint(val, ==, i);
633 qobject_unref(qlist_pop(qlist));
635 break;
637 case USER_DEF_LIST_UNION_KIND_S8:
638 case USER_DEF_LIST_UNION_KIND_S16:
639 case USER_DEF_LIST_UNION_KIND_S32:
640 case USER_DEF_LIST_UNION_KIND_S64:
642 * All integer elements in JSON arrays get stored into QNums
643 * when we convert to QObjects, so we can check them all in
644 * the same fashion, so simply fall through here.
646 case USER_DEF_LIST_UNION_KIND_INTEGER:
647 for (i = 0; i < 32; i++) {
648 QObject *tmp;
649 QNum *qvalue;
650 int64_t val;
652 tmp = qlist_peek(qlist);
653 g_assert(tmp);
654 qvalue = qobject_to(QNum, tmp);
655 g_assert(qnum_get_try_int(qvalue, &val));
656 g_assert_cmpint(val, ==, i);
657 qobject_unref(qlist_pop(qlist));
659 break;
660 case USER_DEF_LIST_UNION_KIND_BOOLEAN:
661 for (i = 0; i < 32; i++) {
662 QObject *tmp;
663 QBool *qvalue;
664 tmp = qlist_peek(qlist);
665 g_assert(tmp);
666 qvalue = qobject_to(QBool, tmp);
667 g_assert_cmpint(qbool_get_bool(qvalue), ==, i % 3 == 0);
668 qobject_unref(qlist_pop(qlist));
670 break;
671 case USER_DEF_LIST_UNION_KIND_STRING:
672 for (i = 0; i < 32; i++) {
673 QObject *tmp;
674 QString *qvalue;
675 gchar str[8];
676 tmp = qlist_peek(qlist);
677 g_assert(tmp);
678 qvalue = qobject_to(QString, tmp);
679 sprintf(str, "%d", i);
680 g_assert_cmpstr(qstring_get_str(qvalue), ==, str);
681 qobject_unref(qlist_pop(qlist));
683 break;
684 case USER_DEF_LIST_UNION_KIND_NUMBER:
685 for (i = 0; i < 32; i++) {
686 QObject *tmp;
687 QNum *qvalue;
688 GString *double_expected = g_string_new("");
689 GString *double_actual = g_string_new("");
691 tmp = qlist_peek(qlist);
692 g_assert(tmp);
693 qvalue = qobject_to(QNum, tmp);
694 g_string_printf(double_expected, "%.6f", (double)i / 3);
695 g_string_printf(double_actual, "%.6f", qnum_get_double(qvalue));
696 g_assert_cmpstr(double_actual->str, ==, double_expected->str);
698 qobject_unref(qlist_pop(qlist));
699 g_string_free(double_expected, true);
700 g_string_free(double_actual, true);
702 break;
703 default:
704 g_assert_not_reached();
706 qobject_unref(qlist);
709 static void test_list_union(TestOutputVisitorData *data,
710 const void *unused,
711 UserDefListUnionKind kind)
713 UserDefListUnion *cvalue = g_new0(UserDefListUnion, 1);
714 QObject *obj;
716 cvalue->type = kind;
717 init_list_union(cvalue);
719 visit_type_UserDefListUnion(data->ov, NULL, &cvalue, &error_abort);
721 obj = visitor_get(data);
722 check_list_union(obj, cvalue->type);
723 qapi_free_UserDefListUnion(cvalue);
726 static void test_visitor_out_list_union_int(TestOutputVisitorData *data,
727 const void *unused)
729 test_list_union(data, unused, USER_DEF_LIST_UNION_KIND_INTEGER);
732 static void test_visitor_out_list_union_int8(TestOutputVisitorData *data,
733 const void *unused)
735 test_list_union(data, unused, USER_DEF_LIST_UNION_KIND_S8);
738 static void test_visitor_out_list_union_int16(TestOutputVisitorData *data,
739 const void *unused)
741 test_list_union(data, unused, USER_DEF_LIST_UNION_KIND_S16);
744 static void test_visitor_out_list_union_int32(TestOutputVisitorData *data,
745 const void *unused)
747 test_list_union(data, unused, USER_DEF_LIST_UNION_KIND_S32);
750 static void test_visitor_out_list_union_int64(TestOutputVisitorData *data,
751 const void *unused)
753 test_list_union(data, unused, USER_DEF_LIST_UNION_KIND_S64);
756 static void test_visitor_out_list_union_uint8(TestOutputVisitorData *data,
757 const void *unused)
759 test_list_union(data, unused, USER_DEF_LIST_UNION_KIND_U8);
762 static void test_visitor_out_list_union_uint16(TestOutputVisitorData *data,
763 const void *unused)
765 test_list_union(data, unused, USER_DEF_LIST_UNION_KIND_U16);
768 static void test_visitor_out_list_union_uint32(TestOutputVisitorData *data,
769 const void *unused)
771 test_list_union(data, unused, USER_DEF_LIST_UNION_KIND_U32);
774 static void test_visitor_out_list_union_uint64(TestOutputVisitorData *data,
775 const void *unused)
777 test_list_union(data, unused, USER_DEF_LIST_UNION_KIND_U64);
780 static void test_visitor_out_list_union_bool(TestOutputVisitorData *data,
781 const void *unused)
783 test_list_union(data, unused, USER_DEF_LIST_UNION_KIND_BOOLEAN);
786 static void test_visitor_out_list_union_str(TestOutputVisitorData *data,
787 const void *unused)
789 test_list_union(data, unused, USER_DEF_LIST_UNION_KIND_STRING);
792 static void test_visitor_out_list_union_number(TestOutputVisitorData *data,
793 const void *unused)
795 test_list_union(data, unused, USER_DEF_LIST_UNION_KIND_NUMBER);
798 static void output_visitor_test_add(const char *testpath,
799 TestOutputVisitorData *data,
800 void (*test_func)(TestOutputVisitorData *data, const void *user_data))
802 g_test_add(testpath, TestOutputVisitorData, data, visitor_output_setup,
803 test_func, visitor_output_teardown);
806 int main(int argc, char **argv)
808 TestOutputVisitorData out_visitor_data;
810 g_test_init(&argc, &argv, NULL);
812 output_visitor_test_add("/visitor/output/int",
813 &out_visitor_data, test_visitor_out_int);
814 output_visitor_test_add("/visitor/output/bool",
815 &out_visitor_data, test_visitor_out_bool);
816 output_visitor_test_add("/visitor/output/number",
817 &out_visitor_data, test_visitor_out_number);
818 output_visitor_test_add("/visitor/output/string",
819 &out_visitor_data, test_visitor_out_string);
820 output_visitor_test_add("/visitor/output/no-string",
821 &out_visitor_data, test_visitor_out_no_string);
822 output_visitor_test_add("/visitor/output/enum",
823 &out_visitor_data, test_visitor_out_enum);
824 output_visitor_test_add("/visitor/output/enum-errors",
825 &out_visitor_data, test_visitor_out_enum_errors);
826 output_visitor_test_add("/visitor/output/struct",
827 &out_visitor_data, test_visitor_out_struct);
828 output_visitor_test_add("/visitor/output/struct-nested",
829 &out_visitor_data, test_visitor_out_struct_nested);
830 output_visitor_test_add("/visitor/output/struct-errors",
831 &out_visitor_data, test_visitor_out_struct_errors);
832 output_visitor_test_add("/visitor/output/list",
833 &out_visitor_data, test_visitor_out_list);
834 output_visitor_test_add("/visitor/output/any",
835 &out_visitor_data, test_visitor_out_any);
836 output_visitor_test_add("/visitor/output/list-qapi-free",
837 &out_visitor_data, test_visitor_out_list_qapi_free);
838 output_visitor_test_add("/visitor/output/union-flat",
839 &out_visitor_data, test_visitor_out_union_flat);
840 output_visitor_test_add("/visitor/output/alternate",
841 &out_visitor_data, test_visitor_out_alternate);
842 output_visitor_test_add("/visitor/output/null",
843 &out_visitor_data, test_visitor_out_null);
844 output_visitor_test_add("/visitor/output/list_union/int",
845 &out_visitor_data,
846 test_visitor_out_list_union_int);
847 output_visitor_test_add("/visitor/output/list_union/int8",
848 &out_visitor_data,
849 test_visitor_out_list_union_int8);
850 output_visitor_test_add("/visitor/output/list_union/int16",
851 &out_visitor_data,
852 test_visitor_out_list_union_int16);
853 output_visitor_test_add("/visitor/output/list_union/int32",
854 &out_visitor_data,
855 test_visitor_out_list_union_int32);
856 output_visitor_test_add("/visitor/output/list_union/int64",
857 &out_visitor_data,
858 test_visitor_out_list_union_int64);
859 output_visitor_test_add("/visitor/output/list_union/uint8",
860 &out_visitor_data,
861 test_visitor_out_list_union_uint8);
862 output_visitor_test_add("/visitor/output/list_union/uint16",
863 &out_visitor_data,
864 test_visitor_out_list_union_uint16);
865 output_visitor_test_add("/visitor/output/list_union/uint32",
866 &out_visitor_data,
867 test_visitor_out_list_union_uint32);
868 output_visitor_test_add("/visitor/output/list_union/uint64",
869 &out_visitor_data,
870 test_visitor_out_list_union_uint64);
871 output_visitor_test_add("/visitor/output/list_union/bool",
872 &out_visitor_data,
873 test_visitor_out_list_union_bool);
874 output_visitor_test_add("/visitor/output/list_union/string",
875 &out_visitor_data,
876 test_visitor_out_list_union_str);
877 output_visitor_test_add("/visitor/output/list_union/number",
878 &out_visitor_data,
879 test_visitor_out_list_union_number);
881 g_test_run();
883 return 0;