]>
Commit | Line | Data |
---|---|---|
9bbc853b | 1 | #include "qemu/osdep.h" |
da34e65c | 2 | #include "qapi/error.h" |
269e09f3 IM |
3 | #include "qom/object_interfaces.h" |
4 | #include "qemu/module.h" | |
90998d58 DB |
5 | #include "qapi-visit.h" |
6 | #include "qapi/qmp-output-visitor.h" | |
7 | #include "qapi/opts-visitor.h" | |
269e09f3 IM |
8 | |
9 | void user_creatable_complete(Object *obj, Error **errp) | |
10 | { | |
11 | ||
12 | UserCreatableClass *ucc; | |
13 | UserCreatable *uc = | |
14 | (UserCreatable *)object_dynamic_cast(obj, TYPE_USER_CREATABLE); | |
15 | ||
16 | if (!uc) { | |
17 | return; | |
18 | } | |
19 | ||
20 | ucc = USER_CREATABLE_GET_CLASS(uc); | |
21 | if (ucc->complete) { | |
22 | ucc->complete(uc, errp); | |
23 | } | |
24 | } | |
25 | ||
d6edb155 LM |
26 | bool user_creatable_can_be_deleted(UserCreatable *uc, Error **errp) |
27 | { | |
28 | ||
29 | UserCreatableClass *ucc = USER_CREATABLE_GET_CLASS(uc); | |
30 | ||
31 | if (ucc->can_be_deleted) { | |
32 | return ucc->can_be_deleted(uc, errp); | |
33 | } else { | |
34 | return true; | |
35 | } | |
36 | } | |
37 | ||
90998d58 DB |
38 | |
39 | Object *user_creatable_add(const QDict *qdict, | |
40 | Visitor *v, Error **errp) | |
41 | { | |
42 | char *type = NULL; | |
43 | char *id = NULL; | |
44 | Object *obj = NULL; | |
15c2f669 | 45 | Error *local_err = NULL; |
90998d58 DB |
46 | QDict *pdict; |
47 | ||
48 | pdict = qdict_clone_shallow(qdict); | |
49 | ||
50 | visit_start_struct(v, NULL, NULL, 0, &local_err); | |
51 | if (local_err) { | |
52 | goto out; | |
53 | } | |
54 | ||
55 | qdict_del(pdict, "qom-type"); | |
56 | visit_type_str(v, "qom-type", &type, &local_err); | |
57 | if (local_err) { | |
58 | goto out_visit; | |
59 | } | |
60 | ||
61 | qdict_del(pdict, "id"); | |
62 | visit_type_str(v, "id", &id, &local_err); | |
63 | if (local_err) { | |
64 | goto out_visit; | |
65 | } | |
15c2f669 | 66 | visit_check_struct(v, &local_err); |
90998d58 DB |
67 | if (local_err) { |
68 | goto out_visit; | |
69 | } | |
70 | ||
15c2f669 EB |
71 | obj = user_creatable_add_type(type, id, pdict, v, &local_err); |
72 | ||
73 | out_visit: | |
74 | visit_end_struct(v); | |
90998d58 DB |
75 | |
76 | out: | |
77 | QDECREF(pdict); | |
78 | g_free(id); | |
79 | g_free(type); | |
80 | if (local_err) { | |
81 | error_propagate(errp, local_err); | |
82 | object_unref(obj); | |
83 | return NULL; | |
84 | } | |
85 | return obj; | |
86 | } | |
87 | ||
88 | ||
89 | Object *user_creatable_add_type(const char *type, const char *id, | |
90 | const QDict *qdict, | |
91 | Visitor *v, Error **errp) | |
92 | { | |
93 | Object *obj; | |
94 | ObjectClass *klass; | |
95 | const QDictEntry *e; | |
96 | Error *local_err = NULL; | |
97 | ||
98 | klass = object_class_by_name(type); | |
99 | if (!klass) { | |
100 | error_setg(errp, "invalid object type: %s", type); | |
101 | return NULL; | |
102 | } | |
103 | ||
104 | if (!object_class_dynamic_cast(klass, TYPE_USER_CREATABLE)) { | |
105 | error_setg(errp, "object type '%s' isn't supported by object-add", | |
106 | type); | |
107 | return NULL; | |
108 | } | |
109 | ||
110 | if (object_class_is_abstract(klass)) { | |
111 | error_setg(errp, "object type '%s' is abstract", type); | |
112 | return NULL; | |
113 | } | |
114 | ||
ad739706 | 115 | assert(qdict); |
90998d58 | 116 | obj = object_new(type); |
ad739706 EB |
117 | visit_start_struct(v, NULL, NULL, 0, &local_err); |
118 | if (local_err) { | |
119 | goto out; | |
120 | } | |
121 | for (e = qdict_first(qdict); e; e = qdict_next(qdict, e)) { | |
122 | object_property_set(obj, v, e->key, &local_err); | |
123 | if (local_err) { | |
124 | break; | |
90998d58 DB |
125 | } |
126 | } | |
15c2f669 EB |
127 | if (!local_err) { |
128 | visit_check_struct(v, &local_err); | |
129 | } | |
130 | visit_end_struct(v); | |
ad739706 EB |
131 | if (local_err) { |
132 | goto out; | |
133 | } | |
90998d58 DB |
134 | |
135 | object_property_add_child(object_get_objects_root(), | |
136 | id, obj, &local_err); | |
137 | if (local_err) { | |
138 | goto out; | |
139 | } | |
140 | ||
141 | user_creatable_complete(obj, &local_err); | |
142 | if (local_err) { | |
143 | object_property_del(object_get_objects_root(), | |
144 | id, &error_abort); | |
145 | goto out; | |
146 | } | |
147 | out: | |
148 | if (local_err) { | |
149 | error_propagate(errp, local_err); | |
150 | object_unref(obj); | |
151 | return NULL; | |
152 | } | |
153 | return obj; | |
154 | } | |
155 | ||
156 | ||
157 | Object *user_creatable_add_opts(QemuOpts *opts, Error **errp) | |
158 | { | |
159 | OptsVisitor *ov; | |
160 | QDict *pdict; | |
161 | Object *obj = NULL; | |
162 | ||
163 | ov = opts_visitor_new(opts); | |
164 | pdict = qemu_opts_to_qdict(opts, NULL); | |
165 | ||
166 | obj = user_creatable_add(pdict, opts_get_visitor(ov), errp); | |
167 | opts_visitor_cleanup(ov); | |
168 | QDECREF(pdict); | |
169 | return obj; | |
170 | } | |
171 | ||
172 | ||
173 | int user_creatable_add_opts_foreach(void *opaque, QemuOpts *opts, Error **errp) | |
174 | { | |
175 | bool (*type_predicate)(const char *) = opaque; | |
176 | Object *obj = NULL; | |
51b9b478 | 177 | Error *err = NULL; |
90998d58 DB |
178 | const char *type; |
179 | ||
180 | type = qemu_opt_get(opts, "qom-type"); | |
181 | if (type && type_predicate && | |
182 | !type_predicate(type)) { | |
183 | return 0; | |
184 | } | |
185 | ||
51b9b478 | 186 | obj = user_creatable_add_opts(opts, &err); |
90998d58 | 187 | if (!obj) { |
51b9b478 | 188 | error_report_err(err); |
90998d58 DB |
189 | return -1; |
190 | } | |
191 | object_unref(obj); | |
192 | return 0; | |
193 | } | |
194 | ||
195 | ||
196 | void user_creatable_del(const char *id, Error **errp) | |
197 | { | |
198 | Object *container; | |
199 | Object *obj; | |
200 | ||
201 | container = object_get_objects_root(); | |
202 | obj = object_resolve_path_component(container, id); | |
203 | if (!obj) { | |
204 | error_setg(errp, "object '%s' not found", id); | |
205 | return; | |
206 | } | |
207 | ||
208 | if (!user_creatable_can_be_deleted(USER_CREATABLE(obj), errp)) { | |
209 | error_setg(errp, "object '%s' is in use, can not be deleted", id); | |
210 | return; | |
211 | } | |
212 | object_unparent(obj); | |
213 | } | |
214 | ||
269e09f3 IM |
215 | static void register_types(void) |
216 | { | |
217 | static const TypeInfo uc_interface_info = { | |
218 | .name = TYPE_USER_CREATABLE, | |
219 | .parent = TYPE_INTERFACE, | |
220 | .class_size = sizeof(UserCreatableClass), | |
221 | }; | |
222 | ||
223 | type_register_static(&uc_interface_info); | |
224 | } | |
225 | ||
226 | type_init(register_types) |