Refactor the component class and component API
[babeltrace.git] / converter / babeltrace.c
CommitLineData
34ac0e6c
MD
1/*
2 * babeltrace.c
3 *
4 * Babeltrace Trace Converter
5 *
64fa3fec
MD
6 * Copyright 2010-2011 EfficiOS Inc. and Linux Foundation
7 *
8 * Author: Mathieu Desnoyers <mathieu.desnoyers@efficios.com>
34ac0e6c
MD
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.
c462e188
MD
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.
34ac0e6c 27 */
4c8bfb7e 28
95d36295 29#include <babeltrace/babeltrace.h>
7c7c0433 30#include <babeltrace/plugin/plugin.h>
33b34c43
PP
31#include <babeltrace/component/component.h>
32#include <babeltrace/component/source.h>
33#include <babeltrace/component/sink.h>
34#include <babeltrace/component/filter.h>
35#include <babeltrace/component/component-class.h>
36#include <babeltrace/component/notification/iterator.h>
2e339de1
JG
37#include <babeltrace/ref.h>
38#include <babeltrace/values.h>
34ac0e6c 39#include <stdlib.h>
a44bc4c9 40#include <babeltrace/ctf-ir/metadata.h> /* for clocks */
7f26a816
PP
41#include <popt.h>
42#include <string.h>
43#include <stdio.h>
33b34c43 44#include <glib.h>
c42c79ea 45#include "babeltrace-cfg.h"
c1870f57 46#include "default-cfg.h"
34ac0e6c 47
33b34c43
PP
48GPtrArray *loaded_plugins;
49
50static
51void init_loaded_plugins_array(void)
52{
53 loaded_plugins = g_ptr_array_new_full(8, bt_put);
54}
55
56static
57void fini_loaded_plugins_array(void)
58{
59 g_ptr_array_free(loaded_plugins, TRUE);
60}
61
62static
63struct bt_plugin *find_plugin(const char *name)
64{
65 int i;
66 struct bt_plugin *plugin = NULL;
67
68 for (i = 0; i < loaded_plugins->len; i++) {
69 plugin = g_ptr_array_index(loaded_plugins, i);
70
71 if (strcmp(name, bt_plugin_get_name(plugin)) == 0) {
72 break;
73 }
74
75 plugin = NULL;
76 }
77
78 return bt_get(plugin);
79}
80
81static
82struct bt_component_class *find_component_class(const char *plugin_name,
83 const char *comp_class_name,
d3e4dcd8 84 enum bt_component_class_type comp_class_type)
33b34c43
PP
85{
86 struct bt_component_class *comp_class = NULL;
87 struct bt_plugin *plugin = find_plugin(plugin_name);
88
89 if (!plugin) {
90 goto end;
91 }
92
93 comp_class = bt_plugin_get_component_class_by_name_and_type(plugin,
94 comp_class_name, comp_class_type);
95 BT_PUT(plugin);
96end:
97 return comp_class;
98}
6c2f3ee5 99
7c7c0433 100static
d3e4dcd8 101const char *component_type_str(enum bt_component_class_type type)
7c7c0433
JG
102{
103 switch (type) {
d3e4dcd8 104 case BT_COMPONENT_CLASS_TYPE_SOURCE:
7c7c0433 105 return "source";
d3e4dcd8 106 case BT_COMPONENT_CLASS_TYPE_SINK:
7c7c0433 107 return "sink";
d3e4dcd8 108 case BT_COMPONENT_CLASS_TYPE_FILTER:
7c7c0433 109 return "filter";
d3e4dcd8 110 case BT_COMPONENT_CLASS_TYPE_UNKNOWN:
7c7c0433
JG
111 default:
112 return "unknown";
113 }
114}
115
116static
33b34c43 117void print_component_classes_found(void)
7c7c0433 118{
33b34c43 119 int plugins_count, component_classes_count = 0, i;
7c7c0433
JG
120
121 if (!babeltrace_verbose) {
122 return;
123 }
124
33b34c43
PP
125 plugins_count = loaded_plugins->len;
126 if (plugins_count == 0) {
127 fprintf(stderr, "No plugins found. Please make sure your plug-in search path is set correctly.\n");
7c7c0433
JG
128 return;
129 }
130
33b34c43
PP
131 for (i = 0; i < plugins_count; i++) {
132 struct bt_plugin *plugin = g_ptr_array_index(loaded_plugins, i);
133
134 component_classes_count += bt_plugin_get_component_class_count(plugin);
135 }
0cc9e945 136
33b34c43
PP
137 printf_verbose("Found %d component classes in %d plugins.\n",
138 component_classes_count, plugins_count);
139
140 for (i = 0; i < plugins_count; i++) {
141 int j;
142 struct bt_plugin *plugin = g_ptr_array_index(loaded_plugins, i);
143
144 component_classes_count =
145 bt_plugin_get_component_class_count(plugin);
146
147 for (j = 0; j < component_classes_count; j++) {
148 struct bt_component_class *comp_class =
149 bt_plugin_get_component_class(plugin, j);
150 const char *plugin_name = bt_plugin_get_name(plugin);
151 const char *comp_class_name =
152 bt_component_class_get_name(comp_class);
153 const char *path = bt_plugin_get_path(plugin);
154 const char *author = bt_plugin_get_author(plugin);
155 const char *license = bt_plugin_get_license(plugin);
156 const char *plugin_description =
157 bt_plugin_get_description(plugin);
158 const char *comp_class_description =
159 bt_component_class_get_description(comp_class);
d3e4dcd8 160 enum bt_component_class_type type =
33b34c43
PP
161 bt_component_class_get_type(comp_class);
162
163 printf_verbose("[%s - %s (%s)]\n", plugin_name,
164 comp_class_name, component_type_str(type));
165 printf_verbose("\tpath: %s\n", path ? path : "None");
166 printf_verbose("\tauthor: %s\n",
167 author ? author : "Unknown");
168 printf_verbose("\tlicense: %s\n",
169 license ? license : "Unknown");
170 printf_verbose("\tplugin description: %s\n",
171 plugin_description ? plugin_description : "None");
172 printf_verbose("\tcomponent description: %s\n",
173 comp_class_description ? comp_class_description : "None");
174 bt_put(comp_class);
175 }
7c7c0433
JG
176 }
177}
178
c42c79ea
PP
179static
180void print_indent(size_t indent)
181{
182 size_t i;
183
184 for (i = 0; i < indent; i++) {
8d24f1c0 185 printf_verbose(" ");
c42c79ea
PP
186 }
187}
188
189static
190void print_value(struct bt_value *, size_t, bool);
191
192static
193bool print_map_value(const char *key, struct bt_value *object, void *data)
194{
195 size_t indent = (size_t) data;
196
197 print_indent(indent);
8d24f1c0 198 printf_verbose("\"%s\": ", key);
c42c79ea
PP
199 print_value(object, indent, false);
200
201 return true;
202}
203
204static
205void print_value(struct bt_value *value, size_t indent, bool do_indent)
206{
207 bool bool_val;
208 int64_t int_val;
209 double dbl_val;
210 const char *str_val;
211 int size;
212 int i;
213
214 if (!value) {
215 return;
216 }
217
218 if (do_indent) {
219 print_indent(indent);
220 }
221
222 switch (bt_value_get_type(value)) {
223 case BT_VALUE_TYPE_NULL:
8d24f1c0 224 printf_verbose("null\n");
c42c79ea
PP
225 break;
226 case BT_VALUE_TYPE_BOOL:
227 bt_value_bool_get(value, &bool_val);
8d24f1c0 228 printf_verbose("%s\n", bool_val ? "true" : "false");
c42c79ea
PP
229 break;
230 case BT_VALUE_TYPE_INTEGER:
231 bt_value_integer_get(value, &int_val);
8d24f1c0 232 printf_verbose("%" PRId64 "\n", int_val);
c42c79ea
PP
233 break;
234 case BT_VALUE_TYPE_FLOAT:
235 bt_value_float_get(value, &dbl_val);
8d24f1c0 236 printf_verbose("%lf\n", dbl_val);
c42c79ea
PP
237 break;
238 case BT_VALUE_TYPE_STRING:
239 bt_value_string_get(value, &str_val);
8d24f1c0 240 printf_verbose("\"%s\"\n", str_val);
c42c79ea
PP
241 break;
242 case BT_VALUE_TYPE_ARRAY:
243 size = bt_value_array_size(value);
8d24f1c0 244 printf_verbose("[\n");
c42c79ea
PP
245
246 for (i = 0; i < size; i++) {
247 struct bt_value *element =
248 bt_value_array_get(value, i);
249
250 print_value(element, indent + 2, true);
251 BT_PUT(element);
252 }
253
254 print_indent(indent);
8d24f1c0 255 printf_verbose("]\n");
c42c79ea
PP
256 break;
257 case BT_VALUE_TYPE_MAP:
258 if (bt_value_map_is_empty(value)) {
8d24f1c0 259 printf_verbose("{}\n");
c42c79ea
PP
260 return;
261 }
262
8d24f1c0 263 printf_verbose("{\n");
c42c79ea
PP
264 bt_value_map_foreach(value, print_map_value,
265 (void *) (indent + 2));
266 print_indent(indent);
8d24f1c0 267 printf_verbose("}\n");
c42c79ea
PP
268 break;
269 default:
270 assert(false);
271 }
272}
273
274static
275void print_bt_config_component(struct bt_config_component *bt_config_component)
276{
8d24f1c0 277 printf_verbose(" %s.%s\n", bt_config_component->plugin_name->str,
c42c79ea 278 bt_config_component->component_name->str);
8d24f1c0 279 printf_verbose(" params:\n");
c42c79ea
PP
280 print_value(bt_config_component->params, 6, true);
281}
282
283static
284void print_bt_config_components(GPtrArray *array)
285{
286 size_t i;
287
288 for (i = 0; i < array->len; i++) {
289 struct bt_config_component *cfg_component =
e5bc7f81 290 bt_config_get_component(array, i);
c42c79ea
PP
291 print_bt_config_component(cfg_component);
292 BT_PUT(cfg_component);
293 }
294}
295
296static
297void print_cfg(struct bt_config *cfg)
298{
8d24f1c0
JG
299 printf_verbose("debug: %d\n", cfg->debug);
300 printf_verbose("verbose: %d\n", cfg->verbose);
301 printf_verbose("do list: %d\n", cfg->do_list);
302 printf_verbose("force correlate: %d\n", cfg->force_correlate);
303 printf_verbose("plugin paths:\n");
c42c79ea 304 print_value(cfg->plugin_paths, 2, true);
8d24f1c0 305 printf_verbose("sources:\n");
c42c79ea 306 print_bt_config_components(cfg->sources);
8d24f1c0 307 printf_verbose("sinks:\n");
c42c79ea
PP
308 print_bt_config_components(cfg->sinks);
309}
310
6c2f3ee5
JG
311static
312struct bt_component *create_trimmer(struct bt_config_component *source_cfg)
313{
314 struct bt_component *trimmer = NULL;
315 struct bt_component_class *trimmer_class = NULL;
316 struct bt_value *trimmer_params = NULL;
528debdf 317 struct bt_value *value;
6c2f3ee5 318
6c2f3ee5
JG
319 trimmer_params = bt_value_map_create();
320 if (!trimmer_params) {
321 goto end;
322 }
323
528debdf
MD
324 value = bt_value_map_get(source_cfg->params, "begin");
325 if (value) {
6c2f3ee5 326 enum bt_value_status ret;
6c2f3ee5 327
528debdf 328 ret = bt_value_map_insert(trimmer_params, "begin",
6c2f3ee5
JG
329 value);
330 BT_PUT(value);
331 if (ret) {
332 goto end;
333 }
334 }
528debdf
MD
335 value = bt_value_map_get(source_cfg->params, "end");
336 if (value) {
6c2f3ee5 337 enum bt_value_status ret;
6c2f3ee5 338
528debdf
MD
339 ret = bt_value_map_insert(trimmer_params, "end",
340 value);
341 BT_PUT(value);
342 if (ret) {
6c2f3ee5
JG
343 goto end;
344 }
528debdf
MD
345 }
346 value = bt_value_map_get(source_cfg->params, "clock-gmt");
347 if (value) {
348 enum bt_value_status ret;
6c2f3ee5 349
528debdf 350 ret = bt_value_map_insert(trimmer_params, "clock-gmt",
6c2f3ee5
JG
351 value);
352 BT_PUT(value);
353 if (ret) {
354 goto end;
355 }
356 }
357
33b34c43 358 trimmer_class = find_component_class("utils", "trimmer",
d3e4dcd8 359 BT_COMPONENT_CLASS_TYPE_FILTER);
6c2f3ee5
JG
360 if (!trimmer_class) {
361 fprintf(stderr, "Could not find trimmer component class. Aborting...\n");
362 goto end;
363 }
364 trimmer = bt_component_create(trimmer_class, "source_trimmer",
365 trimmer_params);
366 if (!trimmer) {
367 goto end;
368 }
369end:
370 bt_put(trimmer_params);
371 bt_put(trimmer_class);
372 return trimmer;
373}
374
375static
376int connect_source_sink(struct bt_component *source,
377 struct bt_config_component *source_cfg,
378 struct bt_component *sink)
379{
380 int ret = 0;
381 enum bt_component_status sink_status;
382 struct bt_component *trimmer = NULL;
383 struct bt_notification_iterator *source_it = NULL;
384 struct bt_notification_iterator *to_sink_it = NULL;
385
386 source_it = bt_component_source_create_iterator(source);
387 if (!source_it) {
388 fprintf(stderr, "Failed to instantiate source iterator. Aborting...\n");
389 ret = -1;
390 goto end;
391 }
392
528debdf
MD
393 if (bt_value_map_has_key(source_cfg->params, "begin")
394 || bt_value_map_has_key(source_cfg->params, "end")) {
6c2f3ee5
JG
395 /* A trimmer must be inserted in the graph. */
396 enum bt_component_status trimmer_status;
397
398 trimmer = create_trimmer(source_cfg);
399 if (!trimmer) {
400 fprintf(stderr, "Failed to create trimmer component. Aborting...\n");
401 ret = -1;
402 goto end;
403 }
404
405 trimmer_status = bt_component_filter_add_iterator(trimmer,
406 source_it);
407 BT_PUT(source_it);
408 if (trimmer_status != BT_COMPONENT_STATUS_OK) {
409 fprintf(stderr, "Failed to connect source to trimmer. Aborting...\n");
410 ret = -1;
411 goto end;
412 }
413
414 to_sink_it = bt_component_filter_create_iterator(trimmer);
415 if (!to_sink_it) {
416 fprintf(stderr, "Failed to instantiate trimmer iterator. Aborting...\n");
417 ret = -1;
418 goto end;
419 }
420 } else {
421 BT_MOVE(to_sink_it, source_it);
422 }
423
424 sink_status = bt_component_sink_add_iterator(sink, to_sink_it);
425 if (sink_status != BT_COMPONENT_STATUS_OK) {
426 fprintf(stderr, "Failed to connect to sink component. Aborting...\n");
427 ret = -1;
428 goto end;
429 }
430end:
431 bt_put(trimmer);
432 bt_put(source_it);
433 bt_put(to_sink_it);
434 return ret;
435}
436
33b34c43
PP
437static
438void add_to_loaded_plugins(struct bt_plugin **plugins)
98ecef32 439{
33b34c43
PP
440 while (*plugins) {
441 struct bt_plugin *plugin = *plugins;
442 /* Check if it's already loaded (from another path). */
443 struct bt_plugin *loaded_plugin =
444 find_plugin(bt_plugin_get_name(plugin));
445
446 if (loaded_plugin) {
447 printf_verbose("Not loading plugin `%s`: already loaded from `%s`\n",
448 bt_plugin_get_path(plugin),
449 bt_plugin_get_path(loaded_plugin));
450 BT_PUT(loaded_plugin);
451 BT_PUT(plugin);
452 } else {
453 /* Transfer ownership to global array. */
454 g_ptr_array_add(loaded_plugins, plugin);
455 }
456 *(plugins++) = NULL;
457 }
458}
459
460static
461int load_dynamic_plugins(struct bt_config *cfg)
462{
463 int nr_paths, i, ret = 0;
98ecef32
MD
464
465 nr_paths = bt_value_array_size(cfg->plugin_paths);
466 if (nr_paths < 0) {
33b34c43
PP
467 ret = -1;
468 goto end;
98ecef32 469 }
33b34c43 470
98ecef32
MD
471 for (i = 0; i < nr_paths; i++) {
472 struct bt_value *plugin_path_value = NULL;
473 const char *plugin_path;
33b34c43 474 struct bt_plugin **plugins;
98ecef32
MD
475
476 plugin_path_value = bt_value_array_get(cfg->plugin_paths, i);
477 if (bt_value_string_get(plugin_path_value,
478 &plugin_path)) {
479 BT_PUT(plugin_path_value);
480 continue;
481 }
33b34c43
PP
482
483 plugins = bt_plugin_create_all_from_dir(plugin_path, true);
484 if (!plugins) {
98ecef32
MD
485 printf_debug("Unable to dynamically load plugins from path %s.\n",
486 plugin_path);
33b34c43
PP
487 BT_PUT(plugin_path_value);
488 continue;
98ecef32 489 }
33b34c43
PP
490
491 add_to_loaded_plugins(plugins);
492 free(plugins);
493
98ecef32
MD
494 BT_PUT(plugin_path_value);
495 }
33b34c43
PP
496end:
497 return ret;
498}
499
500static
501int load_static_plugins(void)
502{
503 int ret = 0;
504 struct bt_plugin **plugins;
505
506 plugins = bt_plugin_create_all_from_static();
507 if (!plugins) {
508 printf_debug("Unable to load static plugins.\n");
509 ret = -1;
510 goto end;
511 }
512
513 add_to_loaded_plugins(plugins);
514 free(plugins);
515end:
516 return ret;
98ecef32
MD
517}
518
528debdf 519int main(int argc, const char **argv)
34ac0e6c 520{
7f26a816 521 int ret;
c42c79ea
PP
522 struct bt_component_class *source_class = NULL;
523 struct bt_component_class *sink_class = NULL;
7c7c0433
JG
524 struct bt_component *source = NULL, *sink = NULL;
525 struct bt_value *source_params = NULL, *sink_params = NULL;
c42c79ea 526 struct bt_config *cfg;
fec2a9f2 527 enum bt_component_status sink_status;
e5bc7f81 528 struct bt_config_component *source_cfg = NULL, *sink_cfg = NULL;
c42c79ea 529
33b34c43
PP
530 init_loaded_plugins_array();
531
c1870f57
JG
532 cfg = bt_config_create();
533 if (!cfg) {
534 fprintf(stderr, "Failed to create Babeltrace configuration\n");
535 ret = 1;
536 goto end;
537 }
538
539 ret = set_default_config(cfg);
540 if (ret) {
541 goto end;
542 }
543
33b34c43 544 ret = bt_config_init_from_args(cfg, argc, argv);
c1870f57 545 if (ret == 0) {
8d24f1c0
JG
546 babeltrace_verbose = cfg->verbose;
547 babeltrace_debug = cfg->debug;
c42c79ea
PP
548 print_cfg(cfg);
549 } else {
56a1cced
JG
550 goto end;
551 }
552
c42c79ea
PP
553 /* TODO handle more than 1 source and 1 sink. */
554 if (cfg->sources->len != 1 || cfg->sinks->len != 1) {
56a1cced
JG
555 ret = -1;
556 goto end;
557 }
558
34ac0e6c
MD
559 printf_verbose("Verbose mode active.\n");
560 printf_debug("Debug mode active.\n");
33bceaf8 561
33b34c43
PP
562 if (load_dynamic_plugins(cfg)) {
563 fprintf(stderr, "Failed to load dynamic plugins.\n");
98ecef32
MD
564 ret = -1;
565 goto end;
33bceaf8
JG
566 }
567
33b34c43 568 if (load_static_plugins()) {
cba174d5
JG
569 fprintf(stderr, "Failed to load static plugins.\n");
570 goto end;
571 }
572
33b34c43 573 print_component_classes_found();
e5bc7f81
JG
574 source_cfg = bt_config_get_component(cfg->sources, 0);
575 source_params = bt_get(source_cfg->params);
33b34c43
PP
576 source_class = find_component_class(source_cfg->plugin_name->str,
577 source_cfg->component_name->str,
d3e4dcd8 578 BT_COMPONENT_CLASS_TYPE_SOURCE);
56a1cced 579 if (!source_class) {
e5bc7f81
JG
580 fprintf(stderr, "Could not find %s.%s source component class. Aborting...\n",
581 source_cfg->plugin_name->str,
582 source_cfg->component_name->str);
56a1cced
JG
583 ret = -1;
584 goto end;
585 }
7c7c0433 586
e5bc7f81
JG
587 sink_cfg = bt_config_get_component(cfg->sinks, 0);
588 sink_params = bt_get(sink_cfg->params);
33b34c43
PP
589 sink_class = find_component_class(sink_cfg->plugin_name->str,
590 sink_cfg->component_name->str,
d3e4dcd8 591 BT_COMPONENT_CLASS_TYPE_SINK);
7c7c0433 592 if (!sink_class) {
e5bc7f81
JG
593 fprintf(stderr, "Could not find %s.%s output component class. Aborting...\n",
594 sink_cfg->plugin_name->str,
595 sink_cfg->component_name->str);
7c7c0433
JG
596 ret = -1;
597 goto end;
598 }
599
e5bc7f81 600 source = bt_component_create(source_class, "source", source_params);
56a1cced 601 if (!source) {
e5bc7f81 602 fprintf(stderr, "Failed to instantiate selected source component. Aborting...\n");
c42c79ea
PP
603 ret = -1;
604 goto end;
605 }
56a1cced 606
e5bc7f81 607 sink = bt_component_create(sink_class, "sink", sink_params);
7c7c0433 608 if (!sink) {
e5bc7f81 609 fprintf(stderr, "Failed to instantiate selected output component. Aborting...\n");
7c7c0433 610 ret = -1;
2e339de1
JG
611 goto end;
612 }
613
6c2f3ee5
JG
614 ret = connect_source_sink(source, source_cfg, sink);
615 if (ret) {
fec2a9f2
JG
616 goto end;
617 }
78586d8a 618
fec2a9f2
JG
619 while (true) {
620 sink_status = bt_component_sink_consume(sink);
fec2a9f2
JG
621 switch (sink_status) {
622 case BT_COMPONENT_STATUS_AGAIN:
623 /* Wait for an arbitraty 500 ms. */
624 usleep(500000);
78586d8a 625 break;
fec2a9f2
JG
626 case BT_COMPONENT_STATUS_OK:
627 break;
628 case BT_COMPONENT_STATUS_END:
629 goto end;
630 default:
631 fprintf(stderr, "Sink component returned an error, aborting...\n");
632 ret = -1;
633 goto end;
78586d8a 634 }
fec2a9f2 635 }
11e1d048 636end:
7c7c0433
JG
637 BT_PUT(sink_class);
638 BT_PUT(source_class);
639 BT_PUT(source);
640 BT_PUT(sink);
641 BT_PUT(source_params);
642 BT_PUT(sink_params);
c42c79ea 643 BT_PUT(cfg);
e5bc7f81
JG
644 BT_PUT(sink_cfg);
645 BT_PUT(source_cfg);
33b34c43 646 fini_loaded_plugins_array();
7f26a816 647 return ret ? 1 : 0;
4c8bfb7e 648}
This page took 0.070728 seconds and 4 git commands to generate.