lib: fully detach CTF IR and CTF writer implementations
[babeltrace.git] / include / babeltrace / ctf-ir / event-internal.h
CommitLineData
adc315b8
JG
1#ifndef BABELTRACE_CTF_IR_EVENT_INTERNAL_H
2#define BABELTRACE_CTF_IR_EVENT_INTERNAL_H
273b65be
JG
3
4/*
272df73e 5 * Babeltrace - CTF IR: Event internal
273b65be 6 *
de9dd397 7 * Copyright 2013, 2014 Jérémie Galarneau <jeremie.galarneau@efficios.com>
273b65be
JG
8 *
9 * Author: Jérémie Galarneau <jeremie.galarneau@efficios.com>
10 *
11 * Permission is hereby granted, free of charge, to any person obtaining a copy
12 * of this software and associated documentation files (the "Software"), to deal
13 * in the Software without restriction, including without limitation the rights
14 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
15 * copies of the Software, and to permit persons to whom the Software is
16 * furnished to do so, subject to the following conditions:
17 *
18 * The above copyright notice and this permission notice shall be included in
19 * all copies or substantial portions of the Software.
20 *
21 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
22 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
23 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
24 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
25 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
26 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
27 * SOFTWARE.
28 */
29
3dca2276 30#include <babeltrace/assert-pre-internal.h>
273b65be 31#include <babeltrace/babeltrace-internal.h>
dac5c838 32#include <babeltrace/values.h>
123fbdec
JG
33#include <babeltrace/ctf-ir/stream-class.h>
34#include <babeltrace/ctf-ir/stream.h>
26e21a82 35#include <babeltrace/ctf-ir/stream-internal.h>
5c3b707d 36#include <babeltrace/ctf-ir/packet.h>
6c677fb5 37#include <babeltrace/ctf-ir/packet-internal.h>
3dca2276
PP
38#include <babeltrace/ctf-ir/fields.h>
39#include <babeltrace/ctf-ir/fields-internal.h>
40#include <babeltrace/ctf-ir/event-class-internal.h>
e22b45d0 41#include <babeltrace/ctf-ir/clock-value-set-internal.h>
312c056a 42#include <babeltrace/ctf-ir/field-wrapper-internal.h>
3dca2276 43#include <babeltrace/ctf-ir/validation-internal.h>
83509119 44#include <babeltrace/object-internal.h>
f6ccaed9 45#include <babeltrace/assert-internal.h>
273b65be
JG
46#include <glib.h>
47
50842bdc 48struct bt_stream_pos;
dc3fffef 49
3dca2276 50struct bt_event_common {
83509119 51 struct bt_object base;
3dca2276 52 struct bt_event_class_common *class;
312c056a 53 struct bt_field_wrapper *header_field;
3dca2276
PP
54 struct bt_field_common *stream_event_context_field;
55 struct bt_field_common *context_field;
56 struct bt_field_common *payload_field;
57 int frozen;
58};
59
60struct bt_event {
61 struct bt_event_common common;
e22b45d0 62 struct bt_clock_value_set cv_set;
3dca2276 63 struct bt_packet *packet;
273b65be
JG
64};
65
273b65be 66BT_HIDDEN
3dca2276 67int _bt_event_common_validate(struct bt_event_common *event);
273b65be 68
4ce9f9d0 69BT_HIDDEN
6c677fb5
PP
70void _bt_event_common_set_is_frozen(struct bt_event_common *event,
71 bool is_frozen);
4ce9f9d0 72
f6ccaed9 73BT_HIDDEN
6c677fb5 74void _bt_event_set_is_frozen(struct bt_event *event, bool is_frozen);
f6ccaed9
PP
75
76#ifdef BT_DEV_MODE
6c677fb5
PP
77# define bt_event_common_validate _bt_event_common_validate
78# define bt_event_common_set_is_frozen _bt_event_common_set_is_frozen
79# define bt_event_set_is_frozen _bt_event_set_is_frozen
f6ccaed9 80#else
6c677fb5
PP
81# define bt_event_common_validate(_event) 0
82# define bt_event_common_set_is_frozen(_event, _is_frozen)
83# define bt_event_set_is_frozen(_event, _is_frozen)
f6ccaed9
PP
84#endif
85
3dca2276 86#define BT_ASSERT_PRE_EVENT_COMMON_HOT(_event, _name) \
e22b45d0 87 BT_ASSERT_PRE_HOT((_event), (_name), ": %!+_e", (_event))
3dca2276 88
dc3fffef 89static inline
3dca2276
PP
90struct bt_event_class_common *bt_event_common_borrow_class(
91 struct bt_event_common *event)
dc3fffef 92{
f6ccaed9 93 BT_ASSERT(event);
3dca2276
PP
94 return event->class;
95}
96
2dd764c1
PP
97typedef void *(*create_field_func)(void *);
98typedef void (*release_field_func)(void *);
99typedef void *(*create_header_field_func)(void *, void *);
100typedef void (*release_header_field_func)(void *, void *);
101
3dca2276
PP
102BT_HIDDEN
103int bt_event_common_initialize(struct bt_event_common *event,
104 struct bt_event_class_common *event_class,
105 struct bt_clock_class *init_expected_clock_class,
3fea54f6 106 bool is_shared_with_parent, bt_object_release_func release_func,
3dca2276
PP
107 bt_validation_flag_copy_field_type_func field_type_copy_func,
108 bool must_be_in_trace,
109 int (*map_clock_classes_func)(struct bt_stream_class_common *stream_class,
110 struct bt_field_type_common *packet_context_field_type,
111 struct bt_field_type_common *event_header_field_type),
2dd764c1
PP
112 create_field_func create_field_func,
113 release_field_func release_field_func,
114 create_header_field_func create_header_field_func,
115 release_header_field_func release_header_field_func);
3dca2276
PP
116
117static inline
094ff7c0 118struct bt_field_common *bt_event_common_borrow_payload(
3dca2276
PP
119 struct bt_event_common *event)
120{
121 struct bt_field_common *payload = NULL;
122
123 BT_ASSERT_PRE_NON_NULL(event, "Event");
124
125 if (!event->payload_field) {
126 BT_LOGV("Event has no current payload field: addr=%p, "
127 "event-class-name=\"%s\", event-class-id=%" PRId64,
128 event, bt_event_class_common_get_name(event->class),
129 bt_event_class_common_get_id(event->class));
130 goto end;
131 }
132
133 payload = event->payload_field;
3dca2276
PP
134
135end:
136 return payload;
137}
138
3dca2276 139static inline
094ff7c0 140struct bt_field_common *bt_event_common_borrow_header(
3dca2276
PP
141 struct bt_event_common *event)
142{
143 struct bt_field_common *header = NULL;
144
145 BT_ASSERT_PRE_NON_NULL(event, "Event");
146
147 if (!event->header_field) {
148 BT_LOGV("Event has no current header field: addr=%p, "
149 "event-class-name=\"%s\", event-class-id=%" PRId64,
150 event, bt_event_class_common_get_name(event->class),
151 bt_event_class_common_get_id(event->class));
152 goto end;
153 }
154
312c056a 155 header = event->header_field->field;
3dca2276
PP
156
157end:
158 return header;
159}
160
3dca2276 161static inline
094ff7c0 162struct bt_field_common *bt_event_common_borrow_context(
3dca2276
PP
163 struct bt_event_common *event)
164{
165 struct bt_field_common *context = NULL;
166
167 BT_ASSERT_PRE_NON_NULL(event, "Event");
168
169 if (!event->context_field) {
170 BT_LOGV("Event has no current context field: addr=%p, "
171 "event-class-name=\"%s\", event-class-id=%" PRId64,
172 event, bt_event_class_common_get_name(event->class),
173 bt_event_class_common_get_id(event->class));
174 goto end;
175 }
176
177 context = event->context_field;
3dca2276
PP
178
179end:
180 return context;
181}
182
3dca2276 183static inline
094ff7c0 184struct bt_field_common *bt_event_common_borrow_stream_event_context(
3dca2276
PP
185 struct bt_event_common *event)
186{
187 struct bt_field_common *stream_event_context = NULL;
188
189 BT_ASSERT_PRE_NON_NULL(event, "Event");
190
191 if (!event->stream_event_context_field) {
192 BT_LOGV("Event has no current stream event context field: addr=%p, "
193 "event-class-name=\"%s\", event-class-id=%" PRId64,
194 event, bt_event_class_common_get_name(event->class),
195 bt_event_class_common_get_id(event->class));
196 goto end;
197 }
198
199 stream_event_context = event->stream_event_context_field;
200
201end:
094ff7c0 202 return stream_event_context;
3dca2276
PP
203}
204
205static inline
312c056a
PP
206void bt_event_common_finalize(struct bt_object *obj,
207 void (*field_release_func)(void *),
208 void (*header_field_release_func)(void *, struct bt_event_common *))
3dca2276
PP
209{
210 struct bt_event_common *event = (void *) obj;
211
212 BT_LOGD("Destroying event: addr=%p, "
213 "event-class-name=\"%s\", event-class-id=%" PRId64,
312c056a
PP
214 event,
215 event->class ? bt_event_class_common_get_name(event->class) : NULL,
216 event->class ? bt_event_class_common_get_id(event->class) : INT64_C(-1));
217
218 if (event->header_field) {
219 BT_LOGD_STR("Releasing event's header field.");
220 header_field_release_func(event->header_field, event);
221 }
222
223 if (event->stream_event_context_field) {
224 BT_LOGD_STR("Releasing event's stream event context field.");
225 field_release_func(event->stream_event_context_field);
226 }
227
228 if (event->context_field) {
229 BT_LOGD_STR("Releasing event's context field.");
230 field_release_func(event->context_field);
231 }
232
233 if (event->payload_field) {
234 BT_LOGD_STR("Releasing event's payload field.");
235 field_release_func(event->payload_field);
236 }
3dca2276 237
312c056a
PP
238 /*
239 * Leave this after calling header_field_release_func() because
240 * this function receives the event object and could need its
241 * class to perform some cleanup.
242 */
3dca2276
PP
243 if (!event->base.parent) {
244 /*
245 * Event was keeping a reference to its class since it shared no
246 * common ancestor with it to guarantee they would both have the
247 * same lifetime.
248 */
249 bt_put(event->class);
250 }
dc3fffef
PP
251}
252
6c677fb5
PP
253BT_UNUSED
254static inline
255void _bt_event_reset_dev_mode(struct bt_event *event)
256{
6c677fb5 257 BT_ASSERT(event);
312c056a 258
6c677fb5
PP
259 if (event->common.header_field) {
260 bt_field_common_set_is_frozen_recursive(
261 event->common.header_field->field, false);
262 bt_field_common_reset_recursive(
263 event->common.header_field->field);
264 }
265
266 if (event->common.stream_event_context_field) {
267 bt_field_common_set_is_frozen_recursive(
268 event->common.stream_event_context_field, false);
269 bt_field_common_reset_recursive(
270 event->common.stream_event_context_field);
271 }
272
273 if (event->common.context_field) {
274 bt_field_common_set_is_frozen_recursive(
275 event->common.context_field, false);
276 bt_field_common_reset_recursive(event->common.context_field);
277 }
278
279 if (event->common.payload_field) {
280 bt_field_common_set_is_frozen_recursive(
281 event->common.payload_field, false);
282 bt_field_common_reset_recursive(event->common.payload_field);
283 }
6c677fb5
PP
284}
285
286#ifdef BT_DEV_MODE
287# define bt_event_reset_dev_mode _bt_event_reset_dev_mode
288#else
289# define bt_event_reset_dev_mode(_x)
290#endif
312c056a
PP
291
292BT_HIDDEN
293void bt_event_destroy(struct bt_event *event);
294
6c677fb5
PP
295static inline
296void bt_event_reset(struct bt_event *event)
297{
298 BT_ASSERT(event);
299 bt_event_set_is_frozen(event, false);
e22b45d0 300 bt_clock_value_set_reset(&event->cv_set);
6c677fb5
PP
301 bt_object_put_no_null_check(&event->packet->base);
302 event->packet = NULL;
303}
304
305static inline
306void bt_event_recycle(struct bt_event *event)
307{
308 struct bt_event_class *event_class;
309
310 BT_ASSERT(event);
311 BT_LIB_LOGD("Recycling event: %!+e", event);
312
313 /*
314 * Those are the important ordered steps:
315 *
316 * 1. Reset the event object (put any permanent reference it
317 * has, unfreeze it and its fields in developer mode, etc.),
318 * but do NOT put its class's reference. This event class
319 * contains the pool to which we're about to recycle this
320 * event object, so we must guarantee its existence thanks
321 * to this existing reference.
322 *
323 * 2. Move the event class reference to our `event_class`
324 * variable so that we can set the event's class member
325 * to NULL before recycling it. We CANNOT do this after
326 * we put the event class reference because this bt_put()
327 * could destroy the event class, also destroying its
328 * event pool, thus also destroying our event object (this
329 * would result in an invalid write access).
330 *
331 * 3. Recycle the event object.
332 *
333 * 4. Put our event class reference.
334 */
335 bt_event_reset(event);
336 event_class = BT_FROM_COMMON(event->common.class);
337 BT_ASSERT(event_class);
338 event->common.class = NULL;
339 bt_object_pool_recycle_object(&event_class->event_pool, event);
340 bt_object_put_no_null_check(&event_class->common.base);
341}
342
343static inline
344void bt_event_set_packet(struct bt_event *event, struct bt_packet *packet)
345{
346 BT_ASSERT_PRE_NON_NULL(event, "Event");
347 BT_ASSERT_PRE_NON_NULL(packet, "Packet");
348 BT_ASSERT_PRE_EVENT_COMMON_HOT(BT_TO_COMMON(event), "Event");
26e21a82
PP
349 BT_ASSERT_PRE(bt_event_class_borrow_stream_class(
350 BT_FROM_COMMON(event->common.class)) ==
351 BT_FROM_COMMON(packet->stream->common.stream_class),
352 "Packet's stream class and event's stream class differ: "
353 "%![event-]+e, %![packet-]+a",
354 event, packet);
355
6c677fb5
PP
356 BT_ASSERT(!event->packet);
357 event->packet = packet;
358 bt_object_get_no_null_check_no_parent_check(&event->packet->base);
359 BT_LOGV("Set event's packet: event-addr=%p, "
360 "event-class-name=\"%s\", event-class-id=%" PRId64 ", "
361 "packet-addr=%p",
362 event, bt_event_class_common_get_name(event->common.class),
363 bt_event_class_common_get_id(event->common.class), packet);
364}
365
366static inline
367struct bt_event *bt_event_create(struct bt_event_class *event_class,
368 struct bt_packet *packet)
369{
370 struct bt_event *event = NULL;
371
372 BT_ASSERT(event_class);
373 event = bt_object_pool_create_object(&event_class->event_pool);
374 if (unlikely(!event)) {
375 BT_LIB_LOGE("Cannot allocate one event from event class's event pool: "
376 "%![event-class-]+E", event_class);
377 goto end;
378 }
379
380 if (unlikely(!event->common.class)) {
381 event->common.class = BT_TO_COMMON(event_class);
382 bt_object_get_no_null_check(&event_class->common.base);
383 }
384
385 BT_ASSERT(packet);
386 bt_event_set_packet(event, packet);
387 goto end;
388
389end:
390 return event;
391}
392
312c056a 393BT_HIDDEN
6c677fb5 394struct bt_event *bt_event_new(struct bt_event_class *event_class);
312c056a 395
adc315b8 396#endif /* BABELTRACE_CTF_IR_EVENT_INTERNAL_H */
This page took 0.060871 seconds and 4 git commands to generate.