c3ccd95c57edf569498c3b725c4be2d6e0de6e49
[babeltrace.git] / lib / component / component.c
1 /*
2 * component.c
3 *
4 * Babeltrace Plugin Component
5 *
6 * Copyright 2015 Jérémie Galarneau <jeremie.galarneau@efficios.com>
7 *
8 * Author: Jérémie Galarneau <jeremie.galarneau@efficios.com>
9 *
10 * Permission is hereby granted, free of charge, to any person obtaining a copy
11 * of this software and associated documentation files (the "Software"), to deal
12 * in the Software without restriction, including without limitation the rights
13 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
14 * copies of the Software, and to permit persons to whom the Software is
15 * furnished to do so, subject to the following conditions:
16 *
17 * The above copyright notice and this permission notice shall be included in
18 * all copies or substantial portions of the Software.
19 *
20 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
21 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
22 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
23 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
24 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
25 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
26 * SOFTWARE.
27 */
28
29 #include <babeltrace/component/component.h>
30 #include <babeltrace/component/component-internal.h>
31 #include <babeltrace/component/component-class-internal.h>
32 #include <babeltrace/component/component-source-internal.h>
33 #include <babeltrace/component/component-filter-internal.h>
34 #include <babeltrace/component/component-sink-internal.h>
35 #include <babeltrace/component/notification/iterator-internal.h>
36 #include <babeltrace/babeltrace-internal.h>
37 #include <babeltrace/compiler.h>
38 #include <babeltrace/ref.h>
39
40 static
41 struct bt_component * (* const component_create_funcs[])(
42 struct bt_component_class *, struct bt_value *) = {
43 [BT_COMPONENT_CLASS_TYPE_SOURCE] = bt_component_source_create,
44 [BT_COMPONENT_CLASS_TYPE_SINK] = bt_component_sink_create,
45 [BT_COMPONENT_CLASS_TYPE_FILTER] = bt_component_filter_create,
46 };
47
48 static
49 enum bt_component_status (* const component_validation_funcs[])(
50 struct bt_component *) = {
51 [BT_COMPONENT_CLASS_TYPE_SOURCE] = bt_component_source_validate,
52 [BT_COMPONENT_CLASS_TYPE_SINK] = bt_component_sink_validate,
53 [BT_COMPONENT_CLASS_TYPE_FILTER] = bt_component_filter_validate,
54 };
55
56 static
57 void bt_component_destroy(struct bt_object *obj)
58 {
59 struct bt_component *component = NULL;
60 struct bt_component_class *component_class = NULL;
61
62 if (!obj) {
63 return;
64 }
65
66 component = container_of(obj, struct bt_component, base);
67 component_class = component->class;
68
69 /*
70 * User data is destroyed first, followed by the concrete component
71 * instance.
72 */
73 if (component->class->methods.destroy) {
74 component->class->methods.destroy(component);
75 }
76
77 if (component->destroy) {
78 component->destroy(component);
79 }
80
81 g_string_free(component->name, TRUE);
82 bt_put(component_class);
83 g_free(component);
84 }
85
86 BT_HIDDEN
87 enum bt_component_status bt_component_init(struct bt_component *component,
88 bt_component_class_destroy_method destroy)
89 {
90 enum bt_component_status ret = BT_COMPONENT_STATUS_OK;
91
92 if (!component) {
93 ret = BT_COMPONENT_STATUS_INVALID;
94 goto end;
95 }
96
97 component->destroy = destroy;
98 end:
99 return ret;
100 }
101
102 enum bt_component_class_type bt_component_get_class_type(
103 struct bt_component *component)
104 {
105 return component ? component->class->type : BT_COMPONENT_CLASS_TYPE_UNKNOWN;
106 }
107
108 BT_HIDDEN
109 struct bt_notification_iterator *bt_component_create_iterator(
110 struct bt_component *component)
111 {
112 enum bt_notification_iterator_status ret_iterator;
113 enum bt_component_class_type type;
114 struct bt_notification_iterator *iterator = NULL;
115 struct bt_component_class *class = component->class;
116
117 if (!component) {
118 goto error;
119 }
120
121 type = bt_component_get_class_type(component);
122 if (type != BT_COMPONENT_CLASS_TYPE_SOURCE &&
123 type != BT_COMPONENT_CLASS_TYPE_FILTER) {
124 /* Unsupported operation. */
125 goto error;
126 }
127
128 iterator = bt_notification_iterator_create(component);
129 if (!iterator) {
130 goto error;
131 }
132
133 switch (type) {
134 case BT_COMPONENT_CLASS_TYPE_SOURCE:
135 {
136 struct bt_component_class_source *source_class;
137 enum bt_component_status ret_component;
138
139 source_class = container_of(class, struct bt_component_class_source, parent);
140 assert(source_class->methods.init_iterator);
141 ret_component =
142 source_class->methods.init_iterator(component, iterator);
143 if (ret_component != BT_COMPONENT_STATUS_OK) {
144 goto error;
145 }
146 break;
147
148 break;
149 }
150 case BT_COMPONENT_CLASS_TYPE_FILTER:
151 {
152 struct bt_component_class_filter *filter_class;
153 enum bt_component_status ret_component;
154
155 filter_class = container_of(class, struct bt_component_class_filter, parent);
156 assert(filter_class->methods.init_iterator);
157 ret_component =
158 filter_class->methods.init_iterator(component, iterator);
159 if (ret_component != BT_COMPONENT_STATUS_OK) {
160 goto error;
161 }
162 break;
163 }
164 default:
165 /* Unreachable. */
166 assert(0);
167 }
168
169 ret_iterator = bt_notification_iterator_validate(iterator);
170 if (ret_iterator != BT_NOTIFICATION_ITERATOR_STATUS_OK) {
171 goto error;
172 }
173
174 return iterator;
175 error:
176 BT_PUT(iterator);
177 return iterator;
178 }
179
180 struct bt_component *bt_component_create_with_init_method_data(
181 struct bt_component_class *component_class, const char *name,
182 struct bt_value *params, void *init_method_data)
183 {
184 int ret;
185 struct bt_component *component = NULL;
186 enum bt_component_class_type type;
187
188 if (!component_class) {
189 goto end;
190 }
191
192 type = bt_component_class_get_type(component_class);
193 if (type <= BT_COMPONENT_CLASS_TYPE_UNKNOWN ||
194 type > BT_COMPONENT_CLASS_TYPE_FILTER) {
195 goto end;
196 }
197
198 component = component_create_funcs[type](component_class, params);
199 if (!component) {
200 goto end;
201 }
202
203 bt_object_init(component, bt_component_destroy);
204 component->name = g_string_new(name);
205 if (!component->name) {
206 BT_PUT(component);
207 goto end;
208 }
209
210 component->initializing = true;
211
212 if (component_class->methods.init) {
213 ret = component_class->methods.init(component, params,
214 init_method_data);
215 component->initializing = false;
216 if (ret != BT_COMPONENT_STATUS_OK) {
217 BT_PUT(component);
218 goto end;
219 }
220 }
221
222 component->initializing = false;
223 ret = component_validation_funcs[type](component);
224 if (ret != BT_COMPONENT_STATUS_OK) {
225 BT_PUT(component);
226 goto end;
227 }
228
229 bt_component_class_freeze(component->class);
230 end:
231 return component;
232 }
233
234 struct bt_component *bt_component_create(
235 struct bt_component_class *component_class, const char *name,
236 struct bt_value *params)
237 {
238 return bt_component_create_with_init_method_data(component_class, name,
239 params, NULL);
240 }
241
242 const char *bt_component_get_name(struct bt_component *component)
243 {
244 const char *ret = NULL;
245
246 if (!component) {
247 goto end;
248 }
249
250 ret = component->name->str;
251 end:
252 return ret;
253 }
254
255 enum bt_component_status bt_component_set_name(struct bt_component *component,
256 const char *name)
257 {
258 enum bt_component_status ret = BT_COMPONENT_STATUS_OK;
259
260 if (!component || !name || name[0] == '\0') {
261 ret = BT_COMPONENT_STATUS_INVALID;
262 goto end;
263 }
264
265 g_string_assign(component->name, name);
266 end:
267 return ret;
268 }
269
270 struct bt_component_class *bt_component_get_class(
271 struct bt_component *component)
272 {
273 return component ? bt_get(component->class) : NULL;
274 }
275
276 void *bt_component_get_private_data(struct bt_component *component)
277 {
278 return component ? component->user_data : NULL;
279 }
280
281 enum bt_component_status
282 bt_component_set_private_data(struct bt_component *component,
283 void *data)
284 {
285 enum bt_component_status ret = BT_COMPONENT_STATUS_OK;
286
287 if (!component || !component->initializing) {
288 ret = BT_COMPONENT_STATUS_INVALID;
289 goto end;
290 }
291
292 component->user_data = data;
293 end:
294 return ret;
295 }
This page took 0.040171 seconds and 3 git commands to generate.