| /* |
| * 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; |
| } |