4 * Copyright IBM, Corp. 2011
7 * Michael Roth <mdroth@linux.vnet.ibm.com>
9 * This work is licensed under the terms of the GNU LGPL, version 2.1 or later.
10 * See the COPYING.LIB file in the top-level directory.
14 #include "qemu/osdep.h"
15 #include "qapi/dealloc-visitor.h"
16 #include "qemu/queue.h"
17 #include "qemu-common.h"
18 #include "qapi/qmp/types.h"
19 #include "qapi/visitor-impl.h"
21 typedef struct StackEntry
25 QTAILQ_ENTRY(StackEntry
) node
;
28 struct QapiDeallocVisitor
31 QTAILQ_HEAD(, StackEntry
) stack
;
35 static QapiDeallocVisitor
*to_qov(Visitor
*v
)
37 return container_of(v
, QapiDeallocVisitor
, visitor
);
40 static void qapi_dealloc_push(QapiDeallocVisitor
*qov
, void *value
)
42 StackEntry
*e
= g_malloc0(sizeof(*e
));
46 /* see if we're just pushing a list head tracker */
48 e
->is_list_head
= true;
50 QTAILQ_INSERT_HEAD(&qov
->stack
, e
, node
);
53 static void *qapi_dealloc_pop(QapiDeallocVisitor
*qov
)
55 StackEntry
*e
= QTAILQ_FIRST(&qov
->stack
);
57 QTAILQ_REMOVE(&qov
->stack
, e
, node
);
63 static void qapi_dealloc_start_struct(Visitor
*v
, void **obj
, const char *kind
,
64 const char *name
, size_t unused
,
67 QapiDeallocVisitor
*qov
= to_qov(v
);
68 qapi_dealloc_push(qov
, obj
);
71 static void qapi_dealloc_end_struct(Visitor
*v
, Error
**errp
)
73 QapiDeallocVisitor
*qov
= to_qov(v
);
74 void **obj
= qapi_dealloc_pop(qov
);
80 static void qapi_dealloc_start_implicit_struct(Visitor
*v
,
85 QapiDeallocVisitor
*qov
= to_qov(v
);
86 qapi_dealloc_push(qov
, obj
);
89 static void qapi_dealloc_end_implicit_struct(Visitor
*v
, Error
**errp
)
91 QapiDeallocVisitor
*qov
= to_qov(v
);
92 void **obj
= qapi_dealloc_pop(qov
);
98 static void qapi_dealloc_start_list(Visitor
*v
, const char *name
, Error
**errp
)
100 QapiDeallocVisitor
*qov
= to_qov(v
);
101 qapi_dealloc_push(qov
, NULL
);
104 static GenericList
*qapi_dealloc_next_list(Visitor
*v
, GenericList
**listp
,
107 GenericList
*list
= *listp
;
108 QapiDeallocVisitor
*qov
= to_qov(v
);
109 StackEntry
*e
= QTAILQ_FIRST(&qov
->stack
);
111 if (e
&& e
->is_list_head
) {
112 e
->is_list_head
= false;
125 static void qapi_dealloc_end_list(Visitor
*v
, Error
**errp
)
127 QapiDeallocVisitor
*qov
= to_qov(v
);
128 void *obj
= qapi_dealloc_pop(qov
);
129 assert(obj
== NULL
); /* should've been list head tracker with no payload */
132 static void qapi_dealloc_type_str(Visitor
*v
, char **obj
, const char *name
,
140 static void qapi_dealloc_type_int(Visitor
*v
, int64_t *obj
, const char *name
,
145 static void qapi_dealloc_type_bool(Visitor
*v
, bool *obj
, const char *name
,
150 static void qapi_dealloc_type_number(Visitor
*v
, double *obj
, const char *name
,
155 static void qapi_dealloc_type_anything(Visitor
*v
, QObject
**obj
,
156 const char *name
, Error
**errp
)
159 qobject_decref(*obj
);
163 static void qapi_dealloc_type_size(Visitor
*v
, uint64_t *obj
, const char *name
,
168 static void qapi_dealloc_type_enum(Visitor
*v
, int *obj
,
169 const char * const strings
[],
170 const char *kind
, const char *name
,
175 /* If there's no data present, the dealloc visitor has nothing to free.
176 * Thus, indicate to visitor code that the subsequent union fields can
177 * be skipped. This is not an error condition, since the cleanup of the
178 * rest of an object can continue unhindered, so leave errp unset in
181 * NOTE: In cases where we're attempting to deallocate an object that
182 * may have missing fields, the field indicating the union type may
183 * be missing. In such a case, it's possible we don't have enough
184 * information to differentiate data_present == false from a case where
185 * data *is* present but happens to be a scalar with a value of 0.
186 * This is okay, since in the case of the dealloc visitor there's no
187 * work that needs to done in either situation.
189 * The current inability in QAPI code to more thoroughly verify a union
190 * type in such cases will likely need to be addressed if we wish to
191 * implement this interface for other types of visitors in the future,
194 static bool qapi_dealloc_start_union(Visitor
*v
, bool data_present
,
200 Visitor
*qapi_dealloc_get_visitor(QapiDeallocVisitor
*v
)
205 void qapi_dealloc_visitor_cleanup(QapiDeallocVisitor
*v
)
210 QapiDeallocVisitor
*qapi_dealloc_visitor_new(void)
212 QapiDeallocVisitor
*v
;
214 v
= g_malloc0(sizeof(*v
));
216 v
->visitor
.start_struct
= qapi_dealloc_start_struct
;
217 v
->visitor
.end_struct
= qapi_dealloc_end_struct
;
218 v
->visitor
.start_implicit_struct
= qapi_dealloc_start_implicit_struct
;
219 v
->visitor
.end_implicit_struct
= qapi_dealloc_end_implicit_struct
;
220 v
->visitor
.start_list
= qapi_dealloc_start_list
;
221 v
->visitor
.next_list
= qapi_dealloc_next_list
;
222 v
->visitor
.end_list
= qapi_dealloc_end_list
;
223 v
->visitor
.type_enum
= qapi_dealloc_type_enum
;
224 v
->visitor
.type_int
= qapi_dealloc_type_int
;
225 v
->visitor
.type_bool
= qapi_dealloc_type_bool
;
226 v
->visitor
.type_str
= qapi_dealloc_type_str
;
227 v
->visitor
.type_number
= qapi_dealloc_type_number
;
228 v
->visitor
.type_any
= qapi_dealloc_type_anything
;
229 v
->visitor
.type_size
= qapi_dealloc_type_size
;
230 v
->visitor
.start_union
= qapi_dealloc_start_union
;
232 QTAILQ_INIT(&v
->stack
);