cpp-common/bt2c/fmt.hpp: use `wise_enum::string_type` in `EnableIfIsWiseEnum` definition
[babeltrace.git] / src / ctf-writer / writer.c
1 /*
2 * SPDX-License-Identifier: MIT
3 *
4 * Copyright 2013, 2014 Jérémie Galarneau <jeremie.galarneau@efficios.com>
5 *
6 * Babeltrace CTF Writer
7 */
8
9 #define BT_LOG_TAG "CTF-WRITER"
10 #include "logging.h"
11
12 #include <errno.h>
13 #include <fcntl.h>
14 #include <inttypes.h>
15 #include <stdio.h>
16 #include <stdlib.h>
17 #include <sys/stat.h>
18 #include <unistd.h>
19
20 #include <babeltrace2-ctf-writer/object.h>
21
22 #include "common/assert.h"
23 #include "compat/compiler.h"
24 #include "compat/endian.h"
25 #include "common/uuid.h"
26
27 #include "clock.h"
28 #include "fields.h"
29 #include "field-types.h"
30 #include "functor.h"
31 #include "stream-class.h"
32 #include "stream.h"
33 #include "trace.h"
34 #include "writer.h"
35
36 static
37 void bt_ctf_writer_destroy(struct bt_ctf_object *obj);
38
39 static
40 int init_trace_packet_header(struct bt_ctf_trace *trace)
41 {
42 int ret = 0;
43 struct bt_ctf_field_type *_uint32_t =
44 get_field_type(FIELD_TYPE_ALIAS_UINT32_T);
45 struct bt_ctf_field_type *_uint8_t =
46 get_field_type(FIELD_TYPE_ALIAS_UINT8_T);
47 struct bt_ctf_field_type *trace_packet_header_type =
48 bt_ctf_field_type_structure_create();
49 struct bt_ctf_field_type *uuid_array_type =
50 bt_ctf_field_type_array_create(_uint8_t, 16);
51
52 if (!trace_packet_header_type || !uuid_array_type) {
53 ret = -1;
54 goto end;
55 }
56
57 ret = bt_ctf_field_type_structure_add_field(trace_packet_header_type,
58 _uint32_t, "magic");
59 if (ret) {
60 goto end;
61 }
62
63 ret = bt_ctf_field_type_structure_add_field(trace_packet_header_type,
64 uuid_array_type, "uuid");
65 if (ret) {
66 goto end;
67 }
68
69 ret = bt_ctf_field_type_structure_add_field(trace_packet_header_type,
70 _uint32_t, "stream_id");
71 if (ret) {
72 goto end;
73 }
74
75 ret = bt_ctf_trace_set_packet_header_field_type(trace,
76 trace_packet_header_type);
77 if (ret) {
78 goto end;
79 }
80 end:
81 bt_ctf_object_put_ref(uuid_array_type);
82 bt_ctf_object_put_ref(_uint32_t);
83 bt_ctf_object_put_ref(_uint8_t);
84 bt_ctf_object_put_ref(trace_packet_header_type);
85 return ret;
86 }
87
88 BT_EXPORT
89 struct bt_ctf_writer *bt_ctf_writer_create(const char *path)
90 {
91 int ret;
92 struct bt_ctf_writer *writer = NULL;
93 bt_uuid_t uuid;
94 char *metadata_path = NULL;
95
96 if (!path) {
97 goto error;
98 }
99
100 writer = g_new0(struct bt_ctf_writer, 1);
101 if (!writer) {
102 goto error;
103 }
104
105 metadata_path = g_build_filename(path, "metadata", NULL);
106
107 bt_ctf_object_init_shared(&writer->base, bt_ctf_writer_destroy);
108 writer->path = g_string_new(path);
109 if (!writer->path) {
110 goto error_destroy;
111 }
112
113 writer->trace = bt_ctf_trace_create();
114 if (!writer->trace) {
115 goto error_destroy;
116 }
117
118 ret = init_trace_packet_header(writer->trace);
119 if (ret) {
120 goto error_destroy;
121 }
122
123 /* Generate a UUID for this writer's trace */
124 bt_uuid_generate(uuid);
125
126 ret = bt_ctf_trace_set_uuid(writer->trace, uuid);
127 if (ret) {
128 goto error_destroy;
129 }
130
131 bt_ctf_object_set_parent(&writer->trace->common.base, &writer->base);
132 bt_ctf_object_put_ref(writer->trace);
133
134 /* Default to little-endian */
135 ret = bt_ctf_writer_set_byte_order(writer, BT_CTF_BYTE_ORDER_NATIVE);
136 BT_ASSERT_DBG(ret == 0);
137
138 /* Create trace directory if necessary and open a metadata file */
139 if (g_mkdir_with_parents(path, S_IRWXU | S_IRWXG)) {
140 perror("g_mkdir_with_parents");
141 goto error_destroy;
142 }
143
144 writer->metadata_fd = open(metadata_path,
145 O_WRONLY | O_CREAT | O_TRUNC,
146 S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP);
147 if (writer->metadata_fd < 0) {
148 perror("open");
149 goto error_destroy;
150 }
151
152 g_free(metadata_path);
153 return writer;
154
155 error_destroy:
156 BT_CTF_OBJECT_PUT_REF_AND_RESET(writer);
157 error:
158 g_free(metadata_path);
159 return writer;
160 }
161
162 BT_EXPORT
163 void bt_ctf_writer_destroy(struct bt_ctf_object *obj)
164 {
165 struct bt_ctf_writer *writer;
166
167 writer = container_of(obj, struct bt_ctf_writer, base);
168 bt_ctf_writer_flush_metadata(writer);
169 if (writer->path) {
170 g_string_free(writer->path, TRUE);
171 }
172
173 if (writer->metadata_fd > 0) {
174 if (close(writer->metadata_fd)) {
175 perror("close");
176 }
177 }
178
179 bt_ctf_object_try_spec_release(&writer->trace->common.base);
180 g_free(writer);
181 }
182
183 BT_EXPORT
184 struct bt_ctf_trace *bt_ctf_writer_get_trace(struct bt_ctf_writer *writer)
185 {
186 struct bt_ctf_trace *trace = NULL;
187
188 if (!writer) {
189 goto end;
190 }
191
192 trace = writer->trace;
193 bt_ctf_object_get_ref(trace);
194 end:
195 return trace;
196 }
197
198 BT_EXPORT
199 struct bt_ctf_stream *bt_ctf_writer_create_stream(struct bt_ctf_writer *writer,
200 struct bt_ctf_stream_class *stream_class)
201 {
202 struct bt_ctf_stream *stream = NULL;
203 int stream_class_count;
204 bt_ctf_bool stream_class_found = BT_CTF_FALSE;
205 int i;
206
207 if (!writer || !stream_class) {
208 goto error;
209 }
210
211 /* Make sure the stream class is part of the writer's trace */
212 stream_class_count = bt_ctf_trace_get_stream_class_count(writer->trace);
213 if (stream_class_count < 0) {
214 goto error;
215 }
216
217 for (i = 0; i < stream_class_count; i++) {
218 struct bt_ctf_stream_class *existing_stream_class =
219 bt_ctf_trace_get_stream_class_by_index(
220 writer->trace, i);
221
222 if (existing_stream_class == stream_class) {
223 stream_class_found = BT_CTF_TRUE;
224 }
225
226 BT_CTF_OBJECT_PUT_REF_AND_RESET(existing_stream_class);
227
228 if (stream_class_found) {
229 break;
230 }
231 }
232
233 if (!stream_class_found) {
234 int ret = bt_ctf_trace_add_stream_class(writer->trace,
235 stream_class);
236
237 if (ret) {
238 goto error;
239 }
240 }
241
242 stream = bt_ctf_stream_create_with_id(stream_class, NULL, -1ULL);
243 if (!stream) {
244 goto error;
245 }
246
247 return stream;
248
249 error:
250 BT_CTF_OBJECT_PUT_REF_AND_RESET(stream);
251 return stream;
252 }
253
254 BT_EXPORT
255 int bt_ctf_writer_add_environment_field(struct bt_ctf_writer *writer,
256 const char *name,
257 const char *value)
258 {
259 int ret = -1;
260
261 if (!writer || !name || !value) {
262 goto end;
263 }
264
265 ret = bt_ctf_trace_set_environment_field_string(writer->trace,
266 name, value);
267 end:
268 return ret;
269 }
270
271 BT_EXPORT
272 int bt_ctf_writer_add_environment_field_int64(struct bt_ctf_writer *writer,
273 const char *name, int64_t value)
274 {
275 int ret = -1;
276
277 if (!writer || !name) {
278 goto end;
279 }
280
281 ret = bt_ctf_trace_set_environment_field_integer(writer->trace, name,
282 value);
283 end:
284 return ret;
285 }
286
287 BT_EXPORT
288 int bt_ctf_writer_add_clock(struct bt_ctf_writer *writer,
289 struct bt_ctf_clock *clock)
290 {
291 int ret = -1;
292
293 if (!writer || !clock) {
294 goto end;
295 }
296
297 ret = bt_ctf_trace_add_clock_class(writer->trace, clock->clock_class);
298 end:
299 return ret;
300 }
301
302 BT_EXPORT
303 char *bt_ctf_writer_get_metadata_string(struct bt_ctf_writer *writer)
304 {
305 char *metadata_string = NULL;
306
307 if (!writer) {
308 goto end;
309 }
310
311 metadata_string = bt_ctf_trace_get_metadata_string(
312 writer->trace);
313 end:
314 return metadata_string;
315 }
316
317 BT_EXPORT
318 void bt_ctf_writer_flush_metadata(struct bt_ctf_writer *writer)
319 {
320 int ret;
321 char *metadata_string = NULL;
322
323 if (!writer) {
324 goto end;
325 }
326
327 metadata_string = bt_ctf_trace_get_metadata_string(
328 writer->trace);
329 if (!metadata_string) {
330 goto end;
331 }
332
333 if (lseek(writer->metadata_fd, 0, SEEK_SET) == (off_t)-1) {
334 perror("lseek");
335 goto end;
336 }
337
338 if (ftruncate(writer->metadata_fd, 0)) {
339 perror("ftruncate");
340 goto end;
341 }
342
343 ret = write(writer->metadata_fd, metadata_string,
344 strlen(metadata_string));
345 if (ret < 0) {
346 perror("write");
347 goto end;
348 }
349 end:
350 g_free(metadata_string);
351 }
352
353 BT_EXPORT
354 int bt_ctf_writer_set_byte_order(struct bt_ctf_writer *writer,
355 enum bt_ctf_byte_order byte_order)
356 {
357 int ret = 0;
358
359 if (!writer || writer->frozen) {
360 ret = -1;
361 goto end;
362 }
363
364 if (byte_order == BT_CTF_BYTE_ORDER_NATIVE) {
365 if (BYTE_ORDER == LITTLE_ENDIAN) {
366 byte_order = BT_CTF_BYTE_ORDER_LITTLE_ENDIAN;
367 } else {
368 byte_order = BT_CTF_BYTE_ORDER_BIG_ENDIAN;
369 }
370 }
371
372 ret = bt_ctf_trace_set_native_byte_order(writer->trace,
373 byte_order);
374 end:
375 return ret;
376 }
377
378 void bt_ctf_writer_freeze(struct bt_ctf_writer *writer)
379 {
380 writer->frozen = 1;
381 }
382
383 static
384 const unsigned int field_type_aliases_alignments[] = {
385 [FIELD_TYPE_ALIAS_UINT5_T] = 1,
386 [FIELD_TYPE_ALIAS_UINT8_T ... FIELD_TYPE_ALIAS_UINT16_T] = 8,
387 [FIELD_TYPE_ALIAS_UINT27_T] = 1,
388 [FIELD_TYPE_ALIAS_UINT32_T ... FIELD_TYPE_ALIAS_UINT64_T] = 8,
389 };
390
391 static
392 const unsigned int field_type_aliases_sizes[] = {
393 [FIELD_TYPE_ALIAS_UINT5_T] = 5,
394 [FIELD_TYPE_ALIAS_UINT8_T] = 8,
395 [FIELD_TYPE_ALIAS_UINT16_T] = 16,
396 [FIELD_TYPE_ALIAS_UINT27_T] = 27,
397 [FIELD_TYPE_ALIAS_UINT32_T] = 32,
398 [FIELD_TYPE_ALIAS_UINT64_T] = 64,
399 };
400
401 struct bt_ctf_field_type *get_field_type(enum field_type_alias alias)
402 {
403 int ret;
404 unsigned int alignment, size;
405 struct bt_ctf_field_type *field_type = NULL;
406
407 if (alias >= NR_FIELD_TYPE_ALIAS) {
408 goto end;
409 }
410
411 alignment = field_type_aliases_alignments[alias];
412 size = field_type_aliases_sizes[alias];
413 field_type = bt_ctf_field_type_integer_create(size);
414 ret = bt_ctf_field_type_set_alignment(field_type, alignment);
415 if (ret) {
416 BT_CTF_OBJECT_PUT_REF_AND_RESET(field_type);
417 }
418 end:
419 return field_type;
420 }
421
422 const char *bt_ctf_get_byte_order_string(enum bt_ctf_byte_order byte_order)
423 {
424 const char *string;
425
426 switch (byte_order) {
427 case BT_CTF_BYTE_ORDER_LITTLE_ENDIAN:
428 string = "le";
429 break;
430 case BT_CTF_BYTE_ORDER_BIG_ENDIAN:
431 string = "be";
432 break;
433 case BT_CTF_BYTE_ORDER_NATIVE:
434 string = "native";
435 break;
436 default:
437 bt_common_abort();
438 }
439
440 return string;
441 }
This page took 0.038059 seconds and 4 git commands to generate.