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