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 "qapi/dealloc-visitor.h"
15 #include "qemu/queue.h"
16 #include "qemu-common.h"
17 #include "qapi/qmp/types.h"
18 #include "qapi/visitor-impl.h"
20 typedef struct StackEntry
24 QTAILQ_ENTRY(StackEntry
) node
;
27 struct QapiDeallocVisitor
30 QTAILQ_HEAD(, StackEntry
) stack
;
34 static QapiDeallocVisitor
*to_qov(Visitor
*v
)
36 return container_of(v
, QapiDeallocVisitor
, visitor
);
39 static void qapi_dealloc_push(QapiDeallocVisitor
*qov
, void *value
)
41 StackEntry
*e
= g_malloc0(sizeof(*e
));
45 /* see if we're just pushing a list head tracker */
47 e
->is_list_head
= true;
49 QTAILQ_INSERT_HEAD(&qov
->stack
, e
, node
);
52 static void *qapi_dealloc_pop(QapiDeallocVisitor
*qov
)
54 StackEntry
*e
= QTAILQ_FIRST(&qov
->stack
);
56 QTAILQ_REMOVE(&qov
->stack
, e
, node
);
62 static void qapi_dealloc_start_struct(Visitor
*v
, void **obj
, const char *kind
,
63 const char *name
, size_t unused
,
66 QapiDeallocVisitor
*qov
= to_qov(v
);
67 qapi_dealloc_push(qov
, obj
);
70 static void qapi_dealloc_end_struct(Visitor
*v
, Error
**errp
)
72 QapiDeallocVisitor
*qov
= to_qov(v
);
73 void **obj
= qapi_dealloc_pop(qov
);
79 static void qapi_dealloc_start_implicit_struct(Visitor
*v
,
84 QapiDeallocVisitor
*qov
= to_qov(v
);
85 qapi_dealloc_push(qov
, obj
);
88 static void qapi_dealloc_end_implicit_struct(Visitor
*v
, Error
**errp
)
90 QapiDeallocVisitor
*qov
= to_qov(v
);
91 void **obj
= qapi_dealloc_pop(qov
);
97 static void qapi_dealloc_start_list(Visitor
*v
, const char *name
, Error
**errp
)
99 QapiDeallocVisitor
*qov
= to_qov(v
);
100 qapi_dealloc_push(qov
, NULL
);
103 static GenericList
*qapi_dealloc_next_list(Visitor
*v
, GenericList
**listp
,
106 GenericList
*list
= *listp
;
107 QapiDeallocVisitor
*qov
= to_qov(v
);
108 StackEntry
*e
= QTAILQ_FIRST(&qov
->stack
);
110 if (e
&& e
->is_list_head
) {
111 e
->is_list_head
= false;
124 static void qapi_dealloc_end_list(Visitor
*v
, Error
**errp
)
126 QapiDeallocVisitor
*qov
= to_qov(v
);
127 void *obj
= qapi_dealloc_pop(qov
);
128 assert(obj
== NULL
); /* should've been list head tracker with no payload */
131 static void qapi_dealloc_type_str(Visitor
*v
, char **obj
, const char *name
,
139 static void qapi_dealloc_type_int(Visitor
*v
, int64_t *obj
, const char *name
,
144 static void qapi_dealloc_type_bool(Visitor
*v
, bool *obj
, const char *name
,
149 static void qapi_dealloc_type_number(Visitor
*v
, double *obj
, const char *name
,
154 static void qapi_dealloc_type_size(Visitor
*v
, uint64_t *obj
, const char *name
,
159 static void qapi_dealloc_type_enum(Visitor
*v
, int *obj
, const char *strings
[],
160 const char *kind
, const char *name
,
165 /* If there's no data present, the dealloc visitor has nothing to free.
166 * Thus, indicate to visitor code that the subsequent union fields can
167 * be skipped. This is not an error condition, since the cleanup of the
168 * rest of an object can continue unhindered, so leave errp unset in
171 * NOTE: In cases where we're attempting to deallocate an object that
172 * may have missing fields, the field indicating the union type may
173 * be missing. In such a case, it's possible we don't have enough
174 * information to differentiate data_present == false from a case where
175 * data *is* present but happens to be a scalar with a value of 0.
176 * This is okay, since in the case of the dealloc visitor there's no
177 * work that needs to done in either situation.
179 * The current inability in QAPI code to more thoroughly verify a union
180 * type in such cases will likely need to be addressed if we wish to
181 * implement this interface for other types of visitors in the future,
184 static bool qapi_dealloc_start_union(Visitor
*v
, bool data_present
,
190 Visitor
*qapi_dealloc_get_visitor(QapiDeallocVisitor
*v
)
195 void qapi_dealloc_visitor_cleanup(QapiDeallocVisitor
*v
)
200 QapiDeallocVisitor
*qapi_dealloc_visitor_new(void)
202 QapiDeallocVisitor
*v
;
204 v
= g_malloc0(sizeof(*v
));
206 v
->visitor
.start_struct
= qapi_dealloc_start_struct
;
207 v
->visitor
.end_struct
= qapi_dealloc_end_struct
;
208 v
->visitor
.start_implicit_struct
= qapi_dealloc_start_implicit_struct
;
209 v
->visitor
.end_implicit_struct
= qapi_dealloc_end_implicit_struct
;
210 v
->visitor
.start_list
= qapi_dealloc_start_list
;
211 v
->visitor
.next_list
= qapi_dealloc_next_list
;
212 v
->visitor
.end_list
= qapi_dealloc_end_list
;
213 v
->visitor
.type_enum
= qapi_dealloc_type_enum
;
214 v
->visitor
.type_int
= qapi_dealloc_type_int
;
215 v
->visitor
.type_bool
= qapi_dealloc_type_bool
;
216 v
->visitor
.type_str
= qapi_dealloc_type_str
;
217 v
->visitor
.type_number
= qapi_dealloc_type_number
;
218 v
->visitor
.type_size
= qapi_dealloc_type_size
;
219 v
->visitor
.start_union
= qapi_dealloc_start_union
;
221 QTAILQ_INIT(&v
->stack
);