Babeltrace 2 C API 2.1.0
Open-source trace manipulation framework
Loading...
Searching...
No Matches
Simple source component class

This example shows a basic source component class packaged as a shared object plugin.

The name of the plugin is dust and the name of the source component class is input. Therefore the component class is identified in the babeltrace2 command-line tool as source.dust.input.

A source.dust.input component reads a text file having this fictitious format:

1578694237 154215 send-msg Jowl pig filet mignon, turducken capicola.
1578694237 200774 recv-msg Pork belly pig burgdoggen venison bacon.
1578694241 001831 send-msg Bacon ipsum dolor amet strip steak.
1578694241 944187 send-msg Spare ribs filet mignon boudin bresaola.
1578694245 115406 recv-msg Rump cow t-bone hamburger short tenderloin.

That is:

  • Each line represents an event record.
  • For a given line:
    • The first token is the Unix timestamp (seconds since the Unix epoch) of the record.
    • The second token is a number of microseconds to add to the Unix timestamp.
    • The third token is the name of the event record: only send-msg and recv-msg are possible.
    • The remaining characters form the messsage of the event record (payload).

A source.dust.input component accepts a single initialization parameter, path, which is the path of the file to open and read.

A source.dust.input component creates a single output port named out.

For each line of the input file, the message iterator of a source.dust.input component emits an event message.

To simplify this example, a source.dust.input component isn't resilient and needs a valid input and valid initialization parameters. The code also doesn't check the return status codes of API functions for simplicity, but you must check them in production code.

The source component class implementation and the shared object plugin macros are in the same file, dust.c:

#include <stdlib.h>
#include <stdio.h>
#include <stdint.h>
#include <inttypes.h>
#include <string.h>
#include <babeltrace2/babeltrace.h>
/* Source component's private data */
struct dust_in {
/* Input file path parameter value (owned by this) */
const bt_value *path_value;
/* Stream (owned by this) */
bt_stream *stream;
/* Event classes for each type of event (owned by this) */
bt_event_class *send_msg_event_class;
bt_event_class *recv_msg_event_class;
};
/*
* Creates an event class within `stream_class` named `name`.
*/
static
bt_event_class *create_event_class(bt_stream_class *stream_class,
const char *name)
{
/* Borrow trace class from stream class */
bt_trace_class *trace_class =
/* Create a default event class */
bt_event_class *event_class = bt_event_class_create(stream_class);
/* Name the event class */
bt_event_class_set_name(event_class, name);
/*
* Create an empty structure field class to be used as the
* event class's payload field class.
*/
bt_field_class *payload_field_class =
/*
* Create a string field class to be used as the payload field
* class's `msg` member.
*/
bt_field_class *msg_field_class =
/*
* Append the string field class to the structure field class as the
* `msg` member.
*/
bt_field_class_structure_append_member(payload_field_class, "msg",
msg_field_class);
/* Set the event class's payload field class */
bt_event_class_set_payload_field_class(event_class, payload_field_class);
/* Put the references we don't need anymore */
bt_field_class_put_ref(payload_field_class);
bt_field_class_put_ref(msg_field_class);
return event_class;
}
/*
* Creates the source component's metadata and stream objects.
*/
static
void create_metadata_and_stream(bt_self_component *self_component,
struct dust_in *dust_in)
{
/* Create a default trace class */
bt_trace_class *trace_class = bt_trace_class_create(self_component);
/* Create a stream trace class within `trace_class` */
bt_stream_class *stream_class = bt_stream_class_create(trace_class);
/* Create a default clock class (1 GHz frequency) */
bt_clock_class *clock_class = bt_clock_class_create(self_component);
/*
* Set `clock_class` as the default clock class of `stream_class`.
*
* This means all the streams created from `stream_class` have a
* conceptual default clock which is an instance of `clock_class`.
* Any event message created for such a stream has a snapshot of the
* stream's default clock.
*/
bt_stream_class_set_default_clock_class(stream_class, clock_class);
/* Create the two event classes we need */
dust_in->send_msg_event_class = create_event_class(stream_class,
"send-msg");
dust_in->recv_msg_event_class = create_event_class(stream_class,
"recv-msg");
/* Create a default trace from (instance of `trace_class`) */
bt_trace *trace = bt_trace_create(trace_class);
/*
* Create the source component's stream (instance of `stream_class`
* within `trace`).
*/
dust_in->stream = bt_stream_create(stream_class, trace);
/* Put the references we don't need anymore */
bt_clock_class_put_ref(clock_class);
bt_stream_class_put_ref(stream_class);
bt_trace_class_put_ref(trace_class);
}
/*
* Initializes the source component.
*/
static
bt_self_component_source *self_component_source,
const bt_value *params, void *initialize_method_data)
{
/* Allocate a private data structure */
struct dust_in *dust_in = malloc(sizeof(*dust_in));
/*
* Keep a reference of the `path` string value parameter so that the
* initialization method of a message iterator can read its string
* value to open the file.
*/
dust_in->path_value =
bt_value_get_ref(dust_in->path_value);
/* Upcast `self_component_source` to the `bt_self_component` type */
bt_self_component *self_component =
/* Create the source component's metadata and stream objects */
create_metadata_and_stream(self_component, dust_in);
/* Set the component's user data to our private data structure */
bt_self_component_set_data(self_component, dust_in);
/*
* Add an output port named `out` to the source component.
*
* This is needed so that this source component can be connected to
* a filter or a sink component. Once a downstream component is
* connected, it can create our message iterator.
*/
"out", NULL, NULL);
}
/*
* Finalizes the source component.
*/
static
void dust_in_finalize(bt_self_component_source *self_component_source)
{
/* Retrieve our private data from the component's user data */
struct dust_in *dust_in = bt_self_component_get_data(
/* Put all references */
bt_value_put_ref(dust_in->path_value);
bt_event_class_put_ref(dust_in->send_msg_event_class);
bt_event_class_put_ref(dust_in->recv_msg_event_class);
bt_stream_put_ref(dust_in->stream);
/* Free the allocated structure */
free(dust_in);
}
/* State of a message iterator */
enum dust_in_message_iterator_state {
/* Emit a stream beginning message */
DUST_IN_MESSAGE_ITERATOR_STATE_STREAM_BEGINNING,
/* Emit an event message */
DUST_IN_MESSAGE_ITERATOR_STATE_EVENT,
/* Message iterator is ended */
DUST_IN_MESSAGE_ITERATOR_STATE_ENDED,
};
/* Message iterator's private data */
struct dust_in_message_iterator {
/* (Weak) link to the component's private data */
struct dust_in *dust_in;
/* Current message iterator's state */
enum dust_in_message_iterator_state state;
/* Input file */
FILE *file;
/* Buffers to read data from the input file */
char name_buffer[32];
char msg_buffer[1024];
};
/*
* Initializes the message iterator.
*/
static
dust_in_message_iterator_initialize(
bt_self_message_iterator *self_message_iterator,
{
/* Allocate a private data structure */
struct dust_in_message_iterator *dust_in_iter =
malloc(sizeof(*dust_in_iter));
/* Retrieve the component's private data from its user data */
struct dust_in *dust_in = bt_self_component_get_data(
/* Keep a link to the component's private data */
dust_in_iter->dust_in = dust_in;
/* Set the message iterator's initial state */
dust_in_iter->state = DUST_IN_MESSAGE_ITERATOR_STATE_STREAM_BEGINNING;
/* Get the raw value of the input file path string value */
const char *path = bt_value_string_get(dust_in->path_value);
/* Open the input file in text mode */
dust_in_iter->file = fopen(path, "r");
/* Set the message iterator's user data to our private data structure */
bt_self_message_iterator_set_data(self_message_iterator, dust_in_iter);
}
/*
* Finalizes the message iterator.
*/
static
void dust_in_message_iterator_finalize(
bt_self_message_iterator *self_message_iterator)
{
/* Retrieve our private data from the message iterator's user data */
struct dust_in_message_iterator *dust_in_iter =
bt_self_message_iterator_get_data(self_message_iterator);
/* Close the input file */
fclose(dust_in_iter->file);
/* Free the allocated structure */
free(dust_in_iter);
}
/*
* Creates a message from the message iterator's input file's current
* line.
*
* If there's a line to process, then this function creates an event
* message. Otherwise it creates a stream end message and sets the
* message iterator's state accordingly.
*/
static
bt_message *create_message_from_line(
struct dust_in_message_iterator *dust_in_iter,
bt_self_message_iterator *self_message_iterator)
{
uint64_t timestamp;
uint64_t extra_us;
bt_message *message;
/* Try to read a line from the input file into individual tokens */
int count = fscanf(dust_in_iter->file, "%" PRIu64 " %" PRIu64 " %s %[^\n]",
&timestamp, &extra_us, &dust_in_iter->name_buffer[0],
&dust_in_iter->msg_buffer[0]);
/* Reached the end of the file? */
if (count == EOF || feof(dust_in_iter->file)) {
/*
* Reached the end of the file: create a stream end message and
* set the message iterator's state to
* `DUST_IN_MESSAGE_ITERATOR_STATE_ENDED` so that the next call
* to dust_in_message_iterator_next() returns
* `BT_MESSAGE_ITERATOR_CLASS_NEXT_METHOD_STATUS_END` (end of
* iteration).
*/
message = bt_message_stream_end_create(self_message_iterator,
dust_in_iter->dust_in->stream);
dust_in_iter->state = DUST_IN_MESSAGE_ITERATOR_STATE_ENDED;
goto end;
}
/* Choose the correct event class, depending on the event name token */
bt_event_class *event_class;
if (strcmp(dust_in_iter->name_buffer, "send-msg") == 0) {
event_class = dust_in_iter->dust_in->send_msg_event_class;
} else {
event_class = dust_in_iter->dust_in->recv_msg_event_class;
}
/*
* At this point `timestamp` contains seconds since the Unix epoch.
* Multiply it by 1,000,000,000 to get nanoseconds since the Unix
* epoch because the stream's clock's frequency is 1 GHz.
*/
timestamp *= UINT64_C(1000000000);
/* Add the extra microseconds (as nanoseconds) to `timestamp` */
timestamp += extra_us * UINT64_C(1000);
/* Create the event message */
self_message_iterator, event_class, dust_in_iter->dust_in->stream,
timestamp);
/*
* At this point `message` is an event message which contains
* an empty event object.
*
* We need to fill its fields.
*
* The only field to fill is the payload field's `msg` field
* which is the event record's message.
*
* All the references below are borrowed references, therefore we
* don't need to put them.
*/
bt_field *payload_field = bt_event_borrow_payload_field(event);
payload_field, 0);
bt_field_string_set_value(msg_field, dust_in_iter->msg_buffer);
end:
return message;
}
/*
* Returns the next message to the message iterator's user.
*
* This method can fill the `messages` array with up to `capacity`
* messages.
*
* To keep this example simple, we put a single message into `messages`
* and set `*count` to 1 (if the message iterator isn't ended).
*/
static
bt_message_iterator_class_next_method_status dust_in_message_iterator_next(
bt_self_message_iterator *self_message_iterator,
bt_message_array_const messages, uint64_t capacity,
uint64_t *count)
{
/* Retrieve our private data from the message iterator's user data */
struct dust_in_message_iterator *dust_in_iter =
bt_self_message_iterator_get_data(self_message_iterator);
/*
* This is the message to return (by moving it to the `messages`
* array).
*
* We initialize it to `NULL`. If it's not `NULL` after the
* processing below, then we move it to the message array.
*/
bt_message *message = NULL;
/* Initialize the return status to a success */
switch (dust_in_iter->state) {
case DUST_IN_MESSAGE_ITERATOR_STATE_STREAM_BEGINNING:
/* Create a stream beginning message */
message = bt_message_stream_beginning_create(self_message_iterator,
dust_in_iter->dust_in->stream);
/* Next state: an event message */
dust_in_iter->state = DUST_IN_MESSAGE_ITERATOR_STATE_EVENT;
break;
case DUST_IN_MESSAGE_ITERATOR_STATE_EVENT:
/*
* Create an event or a stream end message from the message
* iterator's input file's current line.
*
* This function also updates the message iterator's state if
* needed.
*/
message = create_message_from_line(dust_in_iter,
self_message_iterator);
break;
case DUST_IN_MESSAGE_ITERATOR_STATE_ENDED:
/* Message iterator is ended: return the corresponding status */
status =
goto end;
}
if (message) {
/*
* We created a message above: move it to the beginning of the
* `messages` array, setting `*count` to 1 to indicate that the
* array contains a single message.
*/
messages[0] = message;
*count = 1;
}
end:
return status;
}
/* Mandatory */
/* Define the `dust` plugin */
BT_PLUGIN(dust);
/* Define the `input` source component class */
BT_PLUGIN_SOURCE_COMPONENT_CLASS(input, dust_in_message_iterator_next);
/* Set some of the `input` source component class's optional methods */
dust_in_message_iterator_initialize);
dust_in_message_iterator_finalize);
bt_message const ** bt_message_array_const
Array of constant messages.
Definition types.h:182
bt_component_class_initialize_method_status
Status codes for bt_component_class_source_initialize_method, bt_component_class_filter_initialize_me...
Definition component-class-dev.h:891
@ BT_COMPONENT_CLASS_INITIALIZE_METHOD_STATUS_OK
Success.
Definition component-class-dev.h:896
bt_message_iterator_class_initialize_method_status
Status codes for bt_message_iterator_class_initialize_method.
Definition message-iterator-class.h:661
bt_message_iterator_class_next_method_status
Status codes for bt_message_iterator_class_next_method.
Definition message-iterator-class.h:718
@ BT_MESSAGE_ITERATOR_CLASS_INITIALIZE_METHOD_STATUS_OK
Success.
Definition message-iterator-class.h:666
@ BT_MESSAGE_ITERATOR_CLASS_NEXT_METHOD_STATUS_OK
Success.
Definition message-iterator-class.h:723
@ BT_MESSAGE_ITERATOR_CLASS_NEXT_METHOD_STATUS_END
End of iteration.
Definition message-iterator-class.h:729
struct bt_message bt_message
Message.
Definition types.h:58
bt_message * bt_message_stream_beginning_create(bt_self_message_iterator *self_message_iterator, const bt_stream *stream)
Creates a stream beginning message for the stream stream from the message iterator self_message_itera...
bt_message * bt_message_event_create_with_default_clock_snapshot(bt_self_message_iterator *self_message_iterator, const bt_event_class *event_class, const bt_stream *stream, uint64_t clock_snapshot_value)
Creates an event message, having an instance of the event class event_class and a default clock snaps...
bt_event * bt_message_event_borrow_event(bt_message *message)
Borrows the event of the event message message.
bt_message * bt_message_stream_end_create(bt_self_message_iterator *self_message_iterator, const bt_stream *stream)
Creates a stream end message for the stream stream from the message iterator self_message_iterator.
#define BT_PLUGIN_SOURCE_COMPONENT_CLASS(_name, _message_iterator_class_next_method)
Alias of BT_PLUGIN_SOURCE_COMPONENT_CLASS_WITH_ID() with the _plugin_id parameter set to auto,...
Definition plugin-dev.h:773
#define BT_PLUGIN_MODULE()
Defines a plugin module.
Definition plugin-dev.h:335
#define BT_PLUGIN_SOURCE_COMPONENT_CLASS_FINALIZE_METHOD(_name, _method)
Alias of BT_PLUGIN_SOURCE_COMPONENT_CLASS_FINALIZE_METHOD_WITH_ID() with the _plugin_id parameter set...
Definition plugin-dev.h:1242
#define BT_PLUGIN_SOURCE_COMPONENT_CLASS_MESSAGE_ITERATOR_CLASS_INITIALIZE_METHOD(_name, _method)
Alias of BT_PLUGIN_SOURCE_COMPONENT_CLASS_MESSAGE_ITERATOR_CLASS_INITIALIZE_METHOD_WITH_ID() with the...
Definition plugin-dev.h:1425
#define BT_PLUGIN_SOURCE_COMPONENT_CLASS_MESSAGE_ITERATOR_CLASS_FINALIZE_METHOD(_name, _method)
Alias of BT_PLUGIN_SOURCE_COMPONENT_CLASS_MESSAGE_ITERATOR_CLASS_FINALIZE_METHOD_WITH_ID() with the _...
Definition plugin-dev.h:1379
#define BT_PLUGIN(_name)
Alias of BT_PLUGIN_WITH_ID() with the _id parameter set to auto.
Definition plugin-dev.h:423
#define BT_PLUGIN_SOURCE_COMPONENT_CLASS_INITIALIZE_METHOD(_name, _method)
Alias of BT_PLUGIN_SOURCE_COMPONENT_CLASS_INITIALIZE_METHOD_WITH_ID() with the _plugin_id parameter s...
Definition plugin-dev.h:1333
struct bt_self_component_port_output bt_self_component_port_output
Self component output port.
Definition types.h:81
struct bt_self_component bt_self_component
Self component.
Definition types.h:71
bt_self_component_add_port_status bt_self_component_source_add_output_port(bt_self_component_source *self_component, const char *name, void *user_data, bt_self_component_port_output **self_component_port)
Adds an output port named name and having the user data user_data to the source component self_compon...
void * bt_self_component_get_data(const bt_self_component *self_component)
Returns the user data of the component self_component.
struct bt_self_component_source bt_self_component_source
Self source component.
Definition types.h:84
static bt_self_component * bt_self_component_source_as_self_component(bt_self_component_source *self_component)
Upcasts the self source component self_component to the common bt_self_component type.
Definition self-component.h:811
void bt_self_component_set_data(bt_self_component *self_component, void *user_data)
Sets the user data of the component self_component to data.
struct bt_self_component_source_configuration bt_self_component_source_configuration
Self source component configuration.
Definition types.h:85
struct bt_self_message_iterator_configuration bt_self_message_iterator_configuration
Self message iterator configuration.
Definition types.h:87
bt_self_component * bt_self_message_iterator_borrow_component(bt_self_message_iterator *self_message_iterator)
Borrows the component which provides the message iterator self_message_iterator.
void * bt_self_message_iterator_get_data(const bt_self_message_iterator *self_message_iterator)
Returns the user data of the message iterator self_message_iterator.
struct bt_self_message_iterator bt_self_message_iterator
Self message iterator.
Definition types.h:86
void bt_self_message_iterator_set_data(bt_self_message_iterator *self_message_iterator, void *user_data)
Sets the user data of the message iterator self_message_iterator to data.
struct bt_clock_class bt_clock_class
Clock class.
Definition types.h:22
void bt_clock_class_put_ref(const bt_clock_class *clock_class)
Decrements the reference count of the clock class clock_class.
bt_clock_class * bt_clock_class_create(bt_self_component *self_component)
Creates a default clock class from the self component self_component.
void bt_event_class_put_ref(const bt_event_class *event_class)
Decrements the reference count of the event class event_class.
bt_event_class * bt_event_class_create(bt_stream_class *stream_class)
Creates a default event class and adds it to the stream class stream_class.
bt_event_class_set_field_class_status bt_event_class_set_payload_field_class(bt_event_class *event_class, bt_field_class *field_class)
Sets the payload field class of the event class event_class to field_class.
bt_event_class_set_name_status bt_event_class_set_name(bt_event_class *event_class, const char *name)
Sets the name of the event class event_class to a copy of name.
struct bt_event_class bt_event_class
Event class.
Definition types.h:37
bt_field * bt_event_borrow_payload_field(bt_event *event)
Borrows the payload field of the event event.
struct bt_event bt_event
Event.
Definition types.h:36
void bt_field_class_put_ref(const bt_field_class *field_class)
Decrements the reference count of the field class field_class.
struct bt_field_class bt_field_class
Field class.
Definition types.h:39
bt_field_class * bt_field_class_structure_create(bt_trace_class *trace_class)
Creates a structure field class from the trace class trace_class.
bt_field_class * bt_field_class_string_create(bt_trace_class *trace_class)
Creates a string field class from the trace class trace_class.
bt_field_class_structure_append_member_status bt_field_class_structure_append_member(bt_field_class *field_class, const char *name, bt_field_class *member_field_class)
Appends a member to the structure field class field_class having the name name and the field class me...
bt_field_string_set_value_status bt_field_string_set_value(bt_field *field, const char *value)
Sets the value of the string field field to a copy of value.
bt_field * bt_field_structure_borrow_member_field_by_index(bt_field *field, uint64_t index)
Borrows the field of the member at index index from the structure field field.
struct bt_field bt_field
Field.
Definition types.h:38
bt_stream_class_set_default_clock_class_status bt_stream_class_set_default_clock_class(bt_stream_class *stream_class, bt_clock_class *clock_class)
Sets the default clock class of the stream class stream_class to clock_class.
struct bt_stream_class bt_stream_class
Stream class.
Definition types.h:90
bt_stream_class * bt_stream_class_create(bt_trace_class *trace_class)
Creates a default stream class and adds it to the trace class trace_class.
bt_trace_class * bt_stream_class_borrow_trace_class(bt_stream_class *stream_class)
Borrows the trace class which contains the stream class stream_class.
void bt_stream_class_put_ref(const bt_stream_class *stream_class)
Decrements the reference count of the stream class stream_class.
void bt_stream_put_ref(const bt_stream *stream)
Decrements the reference count of the stream stream.
bt_stream * bt_stream_create(bt_stream_class *stream_class, bt_trace *trace)
Creates a stream from the stream class stream_class and adds it to the trace trace.
struct bt_stream bt_stream
Stream.
Definition types.h:89
void bt_trace_class_put_ref(const bt_trace_class *trace_class)
Decrements the reference count of the trace class trace_class.
bt_trace_class * bt_trace_class_create(bt_self_component *self_component)
Creates a default trace class from the self component self_component.
struct bt_trace_class bt_trace_class
Trace class.
Definition types.h:92
bt_trace * bt_trace_create(bt_trace_class *trace_class)
Creates a default trace from the trace class trace_class.
void bt_trace_put_ref(const bt_trace *trace)
Decrements the reference count of the trace trace.
struct bt_trace bt_trace
Trace.
Definition types.h:91
struct bt_value bt_value
Value.
Definition types.h:93
void bt_value_get_ref(const bt_value *value)
Increments the reference count of the value value.
void bt_value_put_ref(const bt_value *value)
Decrements the reference count of the value value.
const char * bt_value_string_get(const bt_value *value)
Returns the raw value of the string value value.
const bt_value * bt_value_map_borrow_entry_value_const(const bt_value *value, const char *key)
Borrows the value of the entry with the key key in the map value value (const version).

As per the Compile and link a Babeltrace 2 shared object plugin guide, you can build the shared object plugin as such:

$ cc dust.c -fPIC -c $(pkg-config --cflags babeltrace2)
$ ld dust.o -o dust.so -shared $(pkg-config --libs babeltrace2)

With the babeltrace2 tool, assuming you have a valid input file named dust, you can view the event messages that a source.dust.input message iterator emits:

$ babeltrace2 --plugin-path=. --component=source.dust.input --params='path="dust"'

The output is similar to:

[17:10:37.154215000] (+?.?????????) send-msg: { msg = "Jowl pig filet mignon, turducken capicola." }
[17:10:37.200774000] (+0.046559000) recv-msg: { msg = "Pork belly pig burgdoggen venison bacon." }
[17:10:41.001831000] (+3.801057000) send-msg: { msg = "Bacon ipsum dolor amet strip steak." }
[17:10:41.944187000] (+0.942356000) send-msg: { msg = "Spare ribs filet mignon boudin bresaola." }
[17:10:45.115406000] (+3.171219000) recv-msg: { msg = "Rump cow t-bone hamburger short tenderloin." }

You can also view more details with

$ babeltrace2 --plugin-path=. --component=source.dust.input --params='path="dust"' \
--component=sink.text.details