Reduce the number of Makefiles in 'src/plugins/utils'
[babeltrace.git] / src / ctf-writer / writer.c
... / ...
CommitLineData
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
36static
37void bt_ctf_writer_destroy(struct bt_ctf_object *obj);
38
39static
40int 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 }
80end:
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
88BT_EXPORT
89struct 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
155error_destroy:
156 BT_CTF_OBJECT_PUT_REF_AND_RESET(writer);
157error:
158 g_free(metadata_path);
159 return writer;
160}
161
162BT_EXPORT
163void 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
183BT_EXPORT
184struct 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);
194end:
195 return trace;
196}
197
198BT_EXPORT
199struct 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
249error:
250 BT_CTF_OBJECT_PUT_REF_AND_RESET(stream);
251 return stream;
252}
253
254BT_EXPORT
255int 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);
267end:
268 return ret;
269}
270
271BT_EXPORT
272int 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);
283end:
284 return ret;
285}
286
287BT_EXPORT
288int 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);
298end:
299 return ret;
300}
301
302BT_EXPORT
303char *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);
313end:
314 return metadata_string;
315}
316
317BT_EXPORT
318void 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 }
349end:
350 g_free(metadata_string);
351}
352
353BT_EXPORT
354int 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);
374end:
375 return ret;
376}
377
378void bt_ctf_writer_freeze(struct bt_ctf_writer *writer)
379{
380 writer->frozen = 1;
381}
382
383static
384const 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
391static
392const 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
401struct 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 }
418end:
419 return field_type;
420}
421
422const 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.023928 seconds and 4 git commands to generate.