| /* | 
 |  * QLit literal qobject | 
 |  * | 
 |  * Copyright IBM, Corp. 2009 | 
 |  * Copyright (c) 2013, 2015, 2017 Red Hat Inc. | 
 |  * | 
 |  * Authors: | 
 |  *  Anthony Liguori   <aliguori@us.ibm.com> | 
 |  *  Markus Armbruster <armbru@redhat.com> | 
 |  *  Marc-André Lureau <marcandre.lureau@redhat.com> | 
 |  * | 
 |  * This work is licensed under the terms of the GNU LGPL, version 2.1 or later. | 
 |  * See the COPYING.LIB file in the top-level directory. | 
 |  */ | 
 |  | 
 | #include "qemu/osdep.h" | 
 |  | 
 | #include "qapi/qmp/qlit.h" | 
 | #include "qapi/qmp/qbool.h" | 
 | #include "qapi/qmp/qlist.h" | 
 | #include "qapi/qmp/qnum.h" | 
 | #include "qapi/qmp/qdict.h" | 
 | #include "qapi/qmp/qstring.h" | 
 | #include "qapi/qmp/qnull.h" | 
 |  | 
 | static bool qlit_equal_qdict(const QLitObject *lhs, const QDict *qdict) | 
 | { | 
 |     int i; | 
 |  | 
 |     for (i = 0; lhs->value.qdict[i].key; i++) { | 
 |         QObject *obj = qdict_get(qdict, lhs->value.qdict[i].key); | 
 |  | 
 |         if (!qlit_equal_qobject(&lhs->value.qdict[i].value, obj)) { | 
 |             return false; | 
 |         } | 
 |     } | 
 |  | 
 |     /* Note: the literal qdict must not contain duplicates, this is | 
 |      * considered a programming error and it isn't checked here. */ | 
 |     if (qdict_size(qdict) != i) { | 
 |         return false; | 
 |     } | 
 |  | 
 |     return true; | 
 | } | 
 |  | 
 | static bool qlit_equal_qlist(const QLitObject *lhs, const QList *qlist) | 
 | { | 
 |     QListEntry *e; | 
 |     int i = 0; | 
 |  | 
 |     QLIST_FOREACH_ENTRY(qlist, e) { | 
 |         QObject *obj = qlist_entry_obj(e); | 
 |  | 
 |         if (!qlit_equal_qobject(&lhs->value.qlist[i], obj)) { | 
 |             return false; | 
 |         } | 
 |         i++; | 
 |     } | 
 |  | 
 |     return !e && lhs->value.qlist[i].type == QTYPE_NONE; | 
 | } | 
 |  | 
 | bool qlit_equal_qobject(const QLitObject *lhs, const QObject *rhs) | 
 | { | 
 |     if (!rhs || lhs->type != qobject_type(rhs)) { | 
 |         return false; | 
 |     } | 
 |  | 
 |     switch (lhs->type) { | 
 |     case QTYPE_QBOOL: | 
 |         return lhs->value.qbool == qbool_get_bool(qobject_to(QBool, rhs)); | 
 |     case QTYPE_QNUM: | 
 |         return lhs->value.qnum ==  qnum_get_int(qobject_to(QNum, rhs)); | 
 |     case QTYPE_QSTRING: | 
 |         return (strcmp(lhs->value.qstr, | 
 |                        qstring_get_str(qobject_to(QString, rhs))) == 0); | 
 |     case QTYPE_QDICT: | 
 |         return qlit_equal_qdict(lhs, qobject_to(QDict, rhs)); | 
 |     case QTYPE_QLIST: | 
 |         return qlit_equal_qlist(lhs, qobject_to(QList, rhs)); | 
 |     case QTYPE_QNULL: | 
 |         return true; | 
 |     default: | 
 |         break; | 
 |     } | 
 |  | 
 |     return false; | 
 | } | 
 |  | 
 | QObject *qobject_from_qlit(const QLitObject *qlit) | 
 | { | 
 |     switch (qlit->type) { | 
 |     case QTYPE_QNULL: | 
 |         return QOBJECT(qnull()); | 
 |     case QTYPE_QNUM: | 
 |         return QOBJECT(qnum_from_int(qlit->value.qnum)); | 
 |     case QTYPE_QSTRING: | 
 |         return QOBJECT(qstring_from_str(qlit->value.qstr)); | 
 |     case QTYPE_QDICT: { | 
 |         QDict *qdict = qdict_new(); | 
 |         QLitDictEntry *e; | 
 |  | 
 |         for (e = qlit->value.qdict; e->key; e++) { | 
 |             qdict_put_obj(qdict, e->key, qobject_from_qlit(&e->value)); | 
 |         } | 
 |         return QOBJECT(qdict); | 
 |     } | 
 |     case QTYPE_QLIST: { | 
 |         QList *qlist = qlist_new(); | 
 |         QLitObject *e; | 
 |  | 
 |         for (e = qlit->value.qlist; e->type != QTYPE_NONE; e++) { | 
 |             qlist_append_obj(qlist, qobject_from_qlit(e)); | 
 |         } | 
 |         return QOBJECT(qlist); | 
 |     } | 
 |     case QTYPE_QBOOL: | 
 |         return QOBJECT(qbool_from_bool(qlit->value.qbool)); | 
 |     default: | 
 |         assert(0); | 
 |     } | 
 |  | 
 |     return NULL; | 
 | } |