flt.lttng-utils.debug-info: adapt debug-info component to API changes
[babeltrace.git] / lib / trace-ir / stream-class.c
CommitLineData
11b0cdc8 1/*
e2f7325d 2 * Copyright 2017-2018 Philippe Proulx <pproulx@efficios.com>
de9dd397 3 * Copyright 2013, 2014 Jérémie Galarneau <jeremie.galarneau@efficios.com>
11b0cdc8 4 *
11b0cdc8
JG
5 * Permission is hereby granted, free of charge, to any person obtaining a copy
6 * of this software and associated documentation files (the "Software"), to deal
7 * in the Software without restriction, including without limitation the rights
8 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 * copies of the Software, and to permit persons to whom the Software is
10 * furnished to do so, subject to the following conditions:
11 *
12 * The above copyright notice and this permission notice shall be included in
13 * all copies or substantial portions of the Software.
14 *
15 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21 * SOFTWARE.
22 */
23
d2f71f12
PP
24#define BT_LOG_TAG "STREAM-CLASS"
25#include <babeltrace/lib-logging-internal.h>
26
3dca2276 27#include <babeltrace/assert-pre-internal.h>
56e18c4c
PP
28#include <babeltrace/trace-ir/clock-class-internal.h>
29#include <babeltrace/trace-ir/event-class-internal.h>
c6bd8523
PP
30#include <babeltrace/trace-ir/field-class-internal.h>
31#include <babeltrace/trace-ir/field-internal.h>
56e18c4c 32#include <babeltrace/trace-ir/stream-class-internal.h>
40f4ba76 33#include <babeltrace/trace-ir/trace-const.h>
e5be10ef 34#include <babeltrace/trace-ir/trace-internal.h>
56e18c4c
PP
35#include <babeltrace/trace-ir/utils-internal.h>
36#include <babeltrace/trace-ir/field-wrapper-internal.h>
37#include <babeltrace/trace-ir/resolve-field-path-internal.h>
3d9990ac
PP
38#include <babeltrace/compiler-internal.h>
39#include <babeltrace/align-internal.h>
40#include <babeltrace/endian-internal.h>
f6ccaed9 41#include <babeltrace/assert-internal.h>
44c440bc 42#include <babeltrace/property-internal.h>
dc3fffef 43#include <inttypes.h>
544d0515 44#include <stdint.h>
e011d2c1 45#include <stdbool.h>
11b0cdc8 46
44c440bc
PP
47#define BT_ASSERT_PRE_STREAM_CLASS_HOT(_sc) \
48 BT_ASSERT_PRE_HOT((_sc), "Stream class", ": %!+S", (_sc))
142c5610 49
cb6f1f7d 50static
44c440bc 51void destroy_stream_class(struct bt_object *obj)
3ea33115 52{
cb6f1f7d
PP
53 struct bt_stream_class *stream_class = (void *) obj;
54
44c440bc
PP
55 BT_LIB_LOGD("Destroying stream class: %!+S", stream_class);
56 BT_LOGD_STR("Putting default clock class.");
238b7404 57 BT_OBJECT_PUT_REF_AND_RESET(stream_class->default_clock_class);
3ea33115 58
3dca2276
PP
59 if (stream_class->event_classes) {
60 BT_LOGD_STR("Destroying event classes.");
61 g_ptr_array_free(stream_class->event_classes, TRUE);
238b7404 62 stream_class->event_classes = NULL;
d2f71f12
PP
63 }
64
44c440bc
PP
65 if (stream_class->name.str) {
66 g_string_free(stream_class->name.str, TRUE);
238b7404
PP
67 stream_class->name.str = NULL;
68 stream_class->name.value = NULL;
3ea33115
JG
69 }
70
e6276565 71 BT_LOGD_STR("Putting packet context field class.");
238b7404 72 BT_OBJECT_PUT_REF_AND_RESET(stream_class->packet_context_fc);
e6276565 73 BT_LOGD_STR("Putting event common context field class.");
238b7404 74 BT_OBJECT_PUT_REF_AND_RESET(stream_class->event_common_context_fc);
312c056a 75 bt_object_pool_finalize(&stream_class->packet_context_field_pool);
3dca2276 76 g_free(stream_class);
3ea33115
JG
77}
78
312c056a
PP
79static
80void free_field_wrapper(struct bt_field_wrapper *field_wrapper,
81 struct bt_stream_class *stream_class)
82{
83 bt_field_wrapper_destroy((void *) field_wrapper);
84}
85
44c440bc
PP
86BT_ASSERT_PRE_FUNC
87static
862ca4ed 88bool stream_class_id_is_unique(const struct bt_trace_class *tc, uint64_t id)
44c440bc
PP
89{
90 uint64_t i;
91 bool is_unique = true;
92
862ca4ed 93 for (i = 0; i < tc->stream_classes->len; i++) {
40f4ba76 94 const struct bt_stream_class *sc =
862ca4ed 95 tc->stream_classes->pdata[i];
44c440bc
PP
96
97 if (sc->id == id) {
98 is_unique = false;
99 goto end;
100 }
101 }
102
103end:
104 return is_unique;
105}
106
107static
862ca4ed
PP
108struct bt_stream_class *create_stream_class_with_id(
109 struct bt_trace_class *tc, uint64_t id)
2f100782 110{
3dca2276
PP
111 struct bt_stream_class *stream_class = NULL;
112 int ret;
2f100782 113
862ca4ed
PP
114 BT_ASSERT(tc);
115 BT_ASSERT_PRE(stream_class_id_is_unique(tc, id),
116 "Duplicate stream class ID: %![tc-]+T, id=%" PRIu64, tc, id);
117 BT_LIB_LOGD("Creating stream class object: %![tc-]+T, id=%" PRIu64,
118 tc, id);
3dca2276 119 stream_class = g_new0(struct bt_stream_class, 1);
d2f71f12 120 if (!stream_class) {
3dca2276
PP
121 BT_LOGE_STR("Failed to allocate one stream class.");
122 goto error;
d2f71f12
PP
123 }
124
44c440bc
PP
125 bt_object_init_shared_with_parent(&stream_class->base,
126 destroy_stream_class);
127
128 stream_class->name.str = g_string_new(NULL);
129 if (!stream_class->name.str) {
130 BT_LOGE_STR("Failed to allocate a GString.");
131 ret = -1;
132 goto end;
133 }
134
135 stream_class->id = id;
136 stream_class->assigns_automatic_event_class_id = true;
137 stream_class->assigns_automatic_stream_id = true;
138 stream_class->event_classes = g_ptr_array_new_with_free_func(
139 (GDestroyNotify) bt_object_try_spec_release);
140 if (!stream_class->event_classes) {
141 BT_LOGE_STR("Failed to allocate a GPtrArray.");
3dca2276 142 goto error;
2f100782
JG
143 }
144
312c056a
PP
145 ret = bt_object_pool_initialize(&stream_class->packet_context_field_pool,
146 (bt_object_pool_new_object_func) bt_field_wrapper_new,
147 (bt_object_pool_destroy_object_func) free_field_wrapper,
148 stream_class);
149 if (ret) {
150 BT_LOGE("Failed to initialize packet context field pool: ret=%d",
151 ret);
152 goto error;
153 }
154
862ca4ed
PP
155 bt_object_set_parent(&stream_class->base, &tc->base);
156 g_ptr_array_add(tc->stream_classes, stream_class);
157 bt_trace_class_freeze(tc);
44c440bc 158 BT_LIB_LOGD("Created stream class object: %!+S", stream_class);
312c056a
PP
159 goto end;
160
161error:
65300d60 162 BT_OBJECT_PUT_REF_AND_RESET(stream_class);
312c056a
PP
163
164end:
44c440bc 165 return stream_class;
312c056a
PP
166}
167
862ca4ed 168struct bt_stream_class *bt_stream_class_create(struct bt_trace_class *tc)
312c056a 169{
862ca4ed
PP
170 BT_ASSERT_PRE_NON_NULL(tc, "Trace class");
171 BT_ASSERT_PRE(tc->assigns_automatic_stream_class_id,
172 "Trace class does not automatically assigns stream class IDs: "
173 "%![sc-]+T", tc);
174 return create_stream_class_with_id(tc,
175 (uint64_t) tc->stream_classes->len);
44c440bc 176}
312c056a 177
40f4ba76 178struct bt_stream_class *bt_stream_class_create_with_id(
862ca4ed 179 struct bt_trace_class *tc, uint64_t id)
44c440bc 180{
862ca4ed
PP
181 BT_ASSERT_PRE_NON_NULL(tc, "Trace class");
182 BT_ASSERT_PRE(!tc->assigns_automatic_stream_class_id,
183 "Trace class automatically assigns stream class IDs: "
184 "%![sc-]+T", tc);
185 return create_stream_class_with_id(tc, id);
312c056a
PP
186}
187
862ca4ed 188struct bt_trace_class *bt_stream_class_borrow_trace_class(
40f4ba76 189 struct bt_stream_class *stream_class)
11b0cdc8 190{
44c440bc 191 BT_ASSERT_PRE_NON_NULL(stream_class, "Stream class");
862ca4ed 192 return bt_stream_class_borrow_trace_class_inline(stream_class);
11b0cdc8
JG
193}
194
862ca4ed 195const struct bt_trace_class *bt_stream_class_borrow_trace_class_const(
40f4ba76 196 const struct bt_stream_class *stream_class)
e5be10ef 197{
862ca4ed 198 return bt_stream_class_borrow_trace_class((void *) stream_class);
e5be10ef
PP
199}
200
40f4ba76 201const char *bt_stream_class_get_name(const struct bt_stream_class *stream_class)
2f100782 202{
cb6f1f7d 203 BT_ASSERT_PRE_NON_NULL(stream_class, "Stream class");
44c440bc 204 return stream_class->name.value;
2f100782
JG
205}
206
a6ae8edc 207enum bt_stream_class_status bt_stream_class_set_name(
40f4ba76 208 struct bt_stream_class *stream_class,
3dca2276 209 const char *name)
5ca83563 210{
44c440bc
PP
211 BT_ASSERT_PRE_NON_NULL(stream_class, "Stream class");
212 BT_ASSERT_PRE_NON_NULL(name, "Name");
213 BT_ASSERT_PRE_STREAM_CLASS_HOT(stream_class);
214 g_string_assign(stream_class->name.str, name);
215 stream_class->name.value = stream_class->name.str->str;
216 BT_LIB_LOGV("Set stream class's name: %!+S", stream_class);
a6ae8edc 217 return BT_STREAM_CLASS_STATUS_OK;
5ca83563
JG
218}
219
40f4ba76 220uint64_t bt_stream_class_get_id(const struct bt_stream_class *stream_class)
2f100782 221{
cb6f1f7d 222 BT_ASSERT_PRE_NON_NULL(stream_class, "Stream class");
44c440bc 223 return stream_class->id;
2f100782
JG
224}
225
44c440bc 226uint64_t bt_stream_class_get_event_class_count(
40f4ba76 227 const struct bt_stream_class *stream_class)
29664b2a 228{
44c440bc
PP
229 BT_ASSERT_PRE_NON_NULL(stream_class, "Stream class");
230 return (uint64_t) stream_class->event_classes->len;
29664b2a
PP
231}
232
44c440bc
PP
233struct bt_event_class *bt_stream_class_borrow_event_class_by_index(
234 struct bt_stream_class *stream_class, uint64_t index)
0d23acbe 235{
44c440bc
PP
236 BT_ASSERT_PRE_NON_NULL(stream_class, "Stream class");
237 BT_ASSERT_PRE_VALID_INDEX(index, stream_class->event_classes->len);
238 return g_ptr_array_index(stream_class->event_classes, index);
0d23acbe
PP
239}
240
40f4ba76
PP
241const struct bt_event_class *
242bt_stream_class_borrow_event_class_by_index_const(
243 const struct bt_stream_class *stream_class, uint64_t index)
e5be10ef 244{
40f4ba76 245 return bt_stream_class_borrow_event_class_by_index(
e5be10ef
PP
246 (void *) stream_class, index);
247}
248
44c440bc 249struct bt_event_class *bt_stream_class_borrow_event_class_by_id(
e5be10ef 250 struct bt_stream_class *stream_class, uint64_t id)
11b0cdc8 251{
44c440bc
PP
252 struct bt_event_class *event_class = NULL;
253 uint64_t i;
0b9ce69f 254
862ca4ed 255 BT_ASSERT_PRE_NON_NULL(stream_class, "Stream class");
11b0cdc8 256
e5be10ef 257 for (i = 0; i < stream_class->event_classes->len; i++) {
44c440bc 258 struct bt_event_class *event_class_candidate =
e5be10ef 259 g_ptr_array_index(stream_class->event_classes, i);
e6a8e8e4 260
44c440bc
PP
261 if (event_class_candidate->id == id) {
262 event_class = event_class_candidate;
09840de5
PP
263 goto end;
264 }
69dc4535
JG
265 }
266
69dc4535 267end:
44c440bc 268 return event_class;
0863f950
PP
269}
270
40f4ba76
PP
271const struct bt_event_class *
272bt_stream_class_borrow_event_class_by_id_const(
273 const struct bt_stream_class *stream_class, uint64_t id)
e5be10ef 274{
40f4ba76 275 return bt_stream_class_borrow_event_class_by_id(
e5be10ef
PP
276 (void *) stream_class, id);
277}
278
40f4ba76
PP
279const struct bt_field_class *
280bt_stream_class_borrow_packet_context_field_class_const(
281 const struct bt_stream_class *stream_class)
12c8a1a3 282{
cb6f1f7d 283 BT_ASSERT_PRE_NON_NULL(stream_class, "Stream class");
5cd6d0e5 284 return stream_class->packet_context_fc;
12c8a1a3
JG
285}
286
a6ae8edc 287enum bt_stream_class_status bt_stream_class_set_packet_context_field_class(
40f4ba76
PP
288 struct bt_stream_class *stream_class,
289 struct bt_field_class *field_class)
12c8a1a3 290{
44c440bc
PP
291 int ret;
292 struct bt_resolve_field_path_context resolve_ctx = {
5cd6d0e5 293 .packet_context = field_class,
44c440bc
PP
294 .event_common_context = NULL,
295 .event_specific_context = NULL,
296 .event_payload = NULL,
297 };
cb6f1f7d 298
44c440bc 299 BT_ASSERT_PRE_NON_NULL(stream_class, "Stream class");
5cd6d0e5 300 BT_ASSERT_PRE_NON_NULL(field_class, "Field class");
44c440bc 301 BT_ASSERT_PRE_STREAM_CLASS_HOT(stream_class);
864cad70
PP
302 BT_ASSERT_PRE(bt_field_class_get_type(field_class) ==
303 BT_FIELD_CLASS_TYPE_STRUCTURE,
e6276565 304 "Packet context field class is not a structure field class: %!+F",
5cd6d0e5 305 field_class);
5cd6d0e5 306 ret = bt_resolve_field_paths(field_class, &resolve_ctx);
44c440bc 307 if (ret) {
a6ae8edc
PP
308 /*
309 * This is the only reason for which
310 * bt_resolve_field_paths() can fail: anything else
311 * would be because a precondition is not satisfied.
312 */
313 ret = BT_STREAM_CLASS_STATUS_NOMEM;
cb6f1f7d
PP
314 goto end;
315 }
316
862ca4ed 317 bt_field_class_make_part_of_trace_class(field_class);
65300d60 318 bt_object_put_ref(stream_class->packet_context_fc);
398454ed
PP
319 stream_class->packet_context_fc = field_class;
320 bt_object_get_no_null_check(stream_class->packet_context_fc);
5cd6d0e5 321 bt_field_class_freeze(field_class);
e6276565 322 BT_LIB_LOGV("Set stream class's packet context field class: %!+S",
44c440bc 323 stream_class);
cb6f1f7d
PP
324
325end:
326 return ret;
12c8a1a3
JG
327}
328
40f4ba76
PP
329const struct bt_field_class *
330bt_stream_class_borrow_event_common_context_field_class_const(
331 const struct bt_stream_class *stream_class)
af181248 332{
cb6f1f7d 333 BT_ASSERT_PRE_NON_NULL(stream_class, "Stream class");
5cd6d0e5 334 return stream_class->event_common_context_fc;
af181248
JG
335}
336
a6ae8edc
PP
337enum bt_stream_class_status
338bt_stream_class_set_event_common_context_field_class(
40f4ba76
PP
339 struct bt_stream_class *stream_class,
340 struct bt_field_class *field_class)
af181248 341{
44c440bc
PP
342 int ret;
343 struct bt_resolve_field_path_context resolve_ctx = {
44c440bc 344 .packet_context = NULL,
5cd6d0e5 345 .event_common_context = field_class,
44c440bc
PP
346 .event_specific_context = NULL,
347 .event_payload = NULL,
348 };
cb6f1f7d 349
44c440bc 350 BT_ASSERT_PRE_NON_NULL(stream_class, "Stream class");
5cd6d0e5 351 BT_ASSERT_PRE_NON_NULL(field_class, "Field class");
44c440bc 352 BT_ASSERT_PRE_STREAM_CLASS_HOT(stream_class);
864cad70
PP
353 BT_ASSERT_PRE(bt_field_class_get_type(field_class) ==
354 BT_FIELD_CLASS_TYPE_STRUCTURE,
e6276565 355 "Event common context field class is not a structure field class: %!+F",
5cd6d0e5 356 field_class);
5cd6d0e5 357 resolve_ctx.packet_context = stream_class->packet_context_fc;
5cd6d0e5 358 ret = bt_resolve_field_paths(field_class, &resolve_ctx);
44c440bc 359 if (ret) {
a6ae8edc
PP
360 /*
361 * This is the only reason for which
362 * bt_resolve_field_paths() can fail: anything else
363 * would be because a precondition is not satisfied.
364 */
365 ret = BT_STREAM_CLASS_STATUS_NOMEM;
cb6f1f7d
PP
366 goto end;
367 }
368
862ca4ed 369 bt_field_class_make_part_of_trace_class(field_class);
65300d60 370 bt_object_put_ref(stream_class->event_common_context_fc);
398454ed
PP
371 stream_class->event_common_context_fc = field_class;
372 bt_object_get_no_null_check(stream_class->event_common_context_fc);
5cd6d0e5 373 bt_field_class_freeze(field_class);
e6276565 374 BT_LIB_LOGV("Set stream class's event common context field class: %!+S",
44c440bc 375 stream_class);
cb6f1f7d 376
cb6f1f7d
PP
377end:
378 return ret;
11b0cdc8
JG
379}
380
44c440bc 381BT_HIDDEN
40f4ba76 382void _bt_stream_class_freeze(const struct bt_stream_class *stream_class)
8bf65fbd 383{
5cd6d0e5 384 /* The field classes and default clock class are already frozen */
44c440bc
PP
385 BT_ASSERT(stream_class);
386 BT_LIB_LOGD("Freezing stream class: %!+S", stream_class);
40f4ba76 387 ((struct bt_stream_class *) stream_class)->frozen = true;
8bf65fbd
JG
388}
389
a6ae8edc 390enum bt_stream_class_status bt_stream_class_set_default_clock_class(
40f4ba76 391 struct bt_stream_class *stream_class,
44c440bc 392 struct bt_clock_class *clock_class)
8bf65fbd 393{
44c440bc
PP
394 BT_ASSERT_PRE_NON_NULL(stream_class, "Stream class");
395 BT_ASSERT_PRE_NON_NULL(clock_class, "Clock class");
396 BT_ASSERT_PRE_STREAM_CLASS_HOT(stream_class);
65300d60 397 bt_object_put_ref(stream_class->default_clock_class);
398454ed
PP
398 stream_class->default_clock_class = clock_class;
399 bt_object_get_no_null_check(stream_class->default_clock_class);
44c440bc
PP
400 bt_clock_class_freeze(clock_class);
401 BT_LIB_LOGV("Set stream class's default clock class: %!+S",
402 stream_class);
a6ae8edc 403 return BT_STREAM_CLASS_STATUS_OK;
8bf65fbd
JG
404}
405
44c440bc
PP
406struct bt_clock_class *bt_stream_class_borrow_default_clock_class(
407 struct bt_stream_class *stream_class)
8bf65fbd 408{
44c440bc
PP
409 BT_ASSERT_PRE_NON_NULL(stream_class, "Stream class");
410 return stream_class->default_clock_class;
411}
8bf65fbd 412
40f4ba76
PP
413const struct bt_clock_class *bt_stream_class_borrow_default_clock_class_const(
414 const struct bt_stream_class *stream_class)
415{
416 BT_ASSERT_PRE_NON_NULL(stream_class, "Stream class");
417 return stream_class->default_clock_class;
418}
419
44c440bc 420bt_bool bt_stream_class_assigns_automatic_event_class_id(
40f4ba76 421 const struct bt_stream_class *stream_class)
44c440bc
PP
422{
423 BT_ASSERT_PRE_NON_NULL(stream_class, "Stream class");
424 return (bt_bool) stream_class->assigns_automatic_event_class_id;
8bf65fbd
JG
425}
426
40f4ba76
PP
427void bt_stream_class_set_assigns_automatic_event_class_id(
428 struct bt_stream_class *stream_class,
e5be10ef 429 bt_bool value)
8bf65fbd 430{
44c440bc
PP
431 BT_ASSERT_PRE_NON_NULL(stream_class, "Stream class");
432 BT_ASSERT_PRE_STREAM_CLASS_HOT(stream_class);
433 stream_class->assigns_automatic_event_class_id = (bool) value;
434 BT_LIB_LOGV("Set stream class's automatic event class ID "
435 "assignment property: %!+S", stream_class);
44c440bc 436}
8bf65fbd 437
44c440bc 438bt_bool bt_stream_class_assigns_automatic_stream_id(
40f4ba76 439 const struct bt_stream_class *stream_class)
44c440bc
PP
440{
441 BT_ASSERT_PRE_NON_NULL(stream_class, "Stream class");
442 return (bt_bool) stream_class->assigns_automatic_stream_id;
443}
8bf65fbd 444
40f4ba76
PP
445void bt_stream_class_set_assigns_automatic_stream_id(
446 struct bt_stream_class *stream_class,
e5be10ef 447 bt_bool value)
44c440bc
PP
448{
449 BT_ASSERT_PRE_NON_NULL(stream_class, "Stream class");
450 BT_ASSERT_PRE_STREAM_CLASS_HOT(stream_class);
451 stream_class->assigns_automatic_stream_id = (bool) value;
452 BT_LIB_LOGV("Set stream class's automatic stream ID "
453 "assignment property: %!+S", stream_class);
44c440bc 454}
3dca2276 455
44c440bc 456bt_bool bt_stream_class_default_clock_is_always_known(
40f4ba76 457 const struct bt_stream_class *stream_class)
44c440bc 458{
dc68f16d 459 /* BT_CLOCK_SNAPSHOT_STATE_UNKNOWN is not supported as of 2.0 */
44c440bc 460 return BT_TRUE;
2a3ced3c 461}
c5b9b441
PP
462
463void bt_stream_class_get_ref(const struct bt_stream_class *stream_class)
464{
465 bt_object_get_ref(stream_class);
466}
467
468void bt_stream_class_put_ref(const struct bt_stream_class *stream_class)
469{
470 bt_object_put_ref(stream_class);
471}
This page took 0.077724 seconds and 4 git commands to generate.