xref: /openbmc/linux/tools/perf/util/data-convert-bt.c (revision 2f0f2441b4a10948e2ec042b48fef13680387f7c)
1 // SPDX-License-Identifier: GPL-2.0-only
2 /*
3  * CTF writing support via babeltrace.
4  *
5  * Copyright (C) 2014, Jiri Olsa <jolsa@redhat.com>
6  * Copyright (C) 2014, Sebastian Andrzej Siewior <bigeasy@linutronix.de>
7  */
8 
9 #include <errno.h>
10 #include <inttypes.h>
11 #include <linux/compiler.h>
12 #include <linux/kernel.h>
13 #include <babeltrace/ctf-writer/writer.h>
14 #include <babeltrace/ctf-writer/clock.h>
15 #include <babeltrace/ctf-writer/stream.h>
16 #include <babeltrace/ctf-writer/event.h>
17 #include <babeltrace/ctf-writer/event-types.h>
18 #include <babeltrace/ctf-writer/event-fields.h>
19 #include <babeltrace/ctf-ir/utils.h>
20 #include <babeltrace/ctf/events.h>
21 #include <traceevent/event-parse.h>
22 #include "asm/bug.h"
23 #include "data-convert-bt.h"
24 #include "session.h"
25 #include "util.h"
26 #include "debug.h"
27 #include "tool.h"
28 #include "evlist.h"
29 #include "evsel.h"
30 #include "machine.h"
31 #include "config.h"
32 #include "sane_ctype.h"
33 
34 #define pr_N(n, fmt, ...) \
35 	eprintf(n, debug_data_convert, fmt, ##__VA_ARGS__)
36 
37 #define pr(fmt, ...)  pr_N(1, pr_fmt(fmt), ##__VA_ARGS__)
38 #define pr2(fmt, ...) pr_N(2, pr_fmt(fmt), ##__VA_ARGS__)
39 
40 #define pr_time2(t, fmt, ...) pr_time_N(2, debug_data_convert, t, pr_fmt(fmt), ##__VA_ARGS__)
41 
42 struct evsel_priv {
43 	struct bt_ctf_event_class *event_class;
44 };
45 
46 #define MAX_CPUS	4096
47 
48 struct ctf_stream {
49 	struct bt_ctf_stream *stream;
50 	int cpu;
51 	u32 count;
52 };
53 
54 struct ctf_writer {
55 	/* writer primitives */
56 	struct bt_ctf_writer		 *writer;
57 	struct ctf_stream		**stream;
58 	int				  stream_cnt;
59 	struct bt_ctf_stream_class	 *stream_class;
60 	struct bt_ctf_clock		 *clock;
61 
62 	/* data types */
63 	union {
64 		struct {
65 			struct bt_ctf_field_type	*s64;
66 			struct bt_ctf_field_type	*u64;
67 			struct bt_ctf_field_type	*s32;
68 			struct bt_ctf_field_type	*u32;
69 			struct bt_ctf_field_type	*string;
70 			struct bt_ctf_field_type	*u32_hex;
71 			struct bt_ctf_field_type	*u64_hex;
72 		};
73 		struct bt_ctf_field_type *array[6];
74 	} data;
75 	struct bt_ctf_event_class	*comm_class;
76 	struct bt_ctf_event_class	*exit_class;
77 	struct bt_ctf_event_class	*fork_class;
78 	struct bt_ctf_event_class	*mmap_class;
79 	struct bt_ctf_event_class	*mmap2_class;
80 };
81 
82 struct convert {
83 	struct perf_tool	tool;
84 	struct ctf_writer	writer;
85 
86 	u64			events_size;
87 	u64			events_count;
88 	u64			non_sample_count;
89 
90 	/* Ordered events configured queue size. */
91 	u64			queue_size;
92 };
93 
94 static int value_set(struct bt_ctf_field_type *type,
95 		     struct bt_ctf_event *event,
96 		     const char *name, u64 val)
97 {
98 	struct bt_ctf_field *field;
99 	bool sign = bt_ctf_field_type_integer_get_signed(type);
100 	int ret;
101 
102 	field = bt_ctf_field_create(type);
103 	if (!field) {
104 		pr_err("failed to create a field %s\n", name);
105 		return -1;
106 	}
107 
108 	if (sign) {
109 		ret = bt_ctf_field_signed_integer_set_value(field, val);
110 		if (ret) {
111 			pr_err("failed to set field value %s\n", name);
112 			goto err;
113 		}
114 	} else {
115 		ret = bt_ctf_field_unsigned_integer_set_value(field, val);
116 		if (ret) {
117 			pr_err("failed to set field value %s\n", name);
118 			goto err;
119 		}
120 	}
121 
122 	ret = bt_ctf_event_set_payload(event, name, field);
123 	if (ret) {
124 		pr_err("failed to set payload %s\n", name);
125 		goto err;
126 	}
127 
128 	pr2("  SET [%s = %" PRIu64 "]\n", name, val);
129 
130 err:
131 	bt_ctf_field_put(field);
132 	return ret;
133 }
134 
135 #define __FUNC_VALUE_SET(_name, _val_type)				\
136 static __maybe_unused int value_set_##_name(struct ctf_writer *cw,	\
137 			     struct bt_ctf_event *event,		\
138 			     const char *name,				\
139 			     _val_type val)				\
140 {									\
141 	struct bt_ctf_field_type *type = cw->data._name;		\
142 	return value_set(type, event, name, (u64) val);			\
143 }
144 
145 #define FUNC_VALUE_SET(_name) __FUNC_VALUE_SET(_name, _name)
146 
147 FUNC_VALUE_SET(s32)
148 FUNC_VALUE_SET(u32)
149 FUNC_VALUE_SET(s64)
150 FUNC_VALUE_SET(u64)
151 __FUNC_VALUE_SET(u64_hex, u64)
152 
153 static int string_set_value(struct bt_ctf_field *field, const char *string);
154 static __maybe_unused int
155 value_set_string(struct ctf_writer *cw, struct bt_ctf_event *event,
156 		 const char *name, const char *string)
157 {
158 	struct bt_ctf_field_type *type = cw->data.string;
159 	struct bt_ctf_field *field;
160 	int ret = 0;
161 
162 	field = bt_ctf_field_create(type);
163 	if (!field) {
164 		pr_err("failed to create a field %s\n", name);
165 		return -1;
166 	}
167 
168 	ret = string_set_value(field, string);
169 	if (ret) {
170 		pr_err("failed to set value %s\n", name);
171 		goto err_put_field;
172 	}
173 
174 	ret = bt_ctf_event_set_payload(event, name, field);
175 	if (ret)
176 		pr_err("failed to set payload %s\n", name);
177 
178 err_put_field:
179 	bt_ctf_field_put(field);
180 	return ret;
181 }
182 
183 static struct bt_ctf_field_type*
184 get_tracepoint_field_type(struct ctf_writer *cw, struct tep_format_field *field)
185 {
186 	unsigned long flags = field->flags;
187 
188 	if (flags & TEP_FIELD_IS_STRING)
189 		return cw->data.string;
190 
191 	if (!(flags & TEP_FIELD_IS_SIGNED)) {
192 		/* unsigned long are mostly pointers */
193 		if (flags & TEP_FIELD_IS_LONG || flags & TEP_FIELD_IS_POINTER)
194 			return cw->data.u64_hex;
195 	}
196 
197 	if (flags & TEP_FIELD_IS_SIGNED) {
198 		if (field->size == 8)
199 			return cw->data.s64;
200 		else
201 			return cw->data.s32;
202 	}
203 
204 	if (field->size == 8)
205 		return cw->data.u64;
206 	else
207 		return cw->data.u32;
208 }
209 
210 static unsigned long long adjust_signedness(unsigned long long value_int, int size)
211 {
212 	unsigned long long value_mask;
213 
214 	/*
215 	 * value_mask = (1 << (size * 8 - 1)) - 1.
216 	 * Directly set value_mask for code readers.
217 	 */
218 	switch (size) {
219 	case 1:
220 		value_mask = 0x7fULL;
221 		break;
222 	case 2:
223 		value_mask = 0x7fffULL;
224 		break;
225 	case 4:
226 		value_mask = 0x7fffffffULL;
227 		break;
228 	case 8:
229 		/*
230 		 * For 64 bit value, return it self. There is no need
231 		 * to fill high bit.
232 		 */
233 		/* Fall through */
234 	default:
235 		/* BUG! */
236 		return value_int;
237 	}
238 
239 	/* If it is a positive value, don't adjust. */
240 	if ((value_int & (~0ULL - value_mask)) == 0)
241 		return value_int;
242 
243 	/* Fill upper part of value_int with 1 to make it a negative long long. */
244 	return (value_int & value_mask) | ~value_mask;
245 }
246 
247 static int string_set_value(struct bt_ctf_field *field, const char *string)
248 {
249 	char *buffer = NULL;
250 	size_t len = strlen(string), i, p;
251 	int err;
252 
253 	for (i = p = 0; i < len; i++, p++) {
254 		if (isprint(string[i])) {
255 			if (!buffer)
256 				continue;
257 			buffer[p] = string[i];
258 		} else {
259 			char numstr[5];
260 
261 			snprintf(numstr, sizeof(numstr), "\\x%02x",
262 				 (unsigned int)(string[i]) & 0xff);
263 
264 			if (!buffer) {
265 				buffer = zalloc(i + (len - i) * 4 + 2);
266 				if (!buffer) {
267 					pr_err("failed to set unprintable string '%s'\n", string);
268 					return bt_ctf_field_string_set_value(field, "UNPRINTABLE-STRING");
269 				}
270 				if (i > 0)
271 					strncpy(buffer, string, i);
272 			}
273 			memcpy(buffer + p, numstr, 4);
274 			p += 3;
275 		}
276 	}
277 
278 	if (!buffer)
279 		return bt_ctf_field_string_set_value(field, string);
280 	err = bt_ctf_field_string_set_value(field, buffer);
281 	free(buffer);
282 	return err;
283 }
284 
285 static int add_tracepoint_field_value(struct ctf_writer *cw,
286 				      struct bt_ctf_event_class *event_class,
287 				      struct bt_ctf_event *event,
288 				      struct perf_sample *sample,
289 				      struct tep_format_field *fmtf)
290 {
291 	struct bt_ctf_field_type *type;
292 	struct bt_ctf_field *array_field;
293 	struct bt_ctf_field *field;
294 	const char *name = fmtf->name;
295 	void *data = sample->raw_data;
296 	unsigned long flags = fmtf->flags;
297 	unsigned int n_items;
298 	unsigned int i;
299 	unsigned int offset;
300 	unsigned int len;
301 	int ret;
302 
303 	name = fmtf->alias;
304 	offset = fmtf->offset;
305 	len = fmtf->size;
306 	if (flags & TEP_FIELD_IS_STRING)
307 		flags &= ~TEP_FIELD_IS_ARRAY;
308 
309 	if (flags & TEP_FIELD_IS_DYNAMIC) {
310 		unsigned long long tmp_val;
311 
312 		tmp_val = tep_read_number(fmtf->event->tep,
313 					  data + offset, len);
314 		offset = tmp_val;
315 		len = offset >> 16;
316 		offset &= 0xffff;
317 	}
318 
319 	if (flags & TEP_FIELD_IS_ARRAY) {
320 
321 		type = bt_ctf_event_class_get_field_by_name(
322 				event_class, name);
323 		array_field = bt_ctf_field_create(type);
324 		bt_ctf_field_type_put(type);
325 		if (!array_field) {
326 			pr_err("Failed to create array type %s\n", name);
327 			return -1;
328 		}
329 
330 		len = fmtf->size / fmtf->arraylen;
331 		n_items = fmtf->arraylen;
332 	} else {
333 		n_items = 1;
334 		array_field = NULL;
335 	}
336 
337 	type = get_tracepoint_field_type(cw, fmtf);
338 
339 	for (i = 0; i < n_items; i++) {
340 		if (flags & TEP_FIELD_IS_ARRAY)
341 			field = bt_ctf_field_array_get_field(array_field, i);
342 		else
343 			field = bt_ctf_field_create(type);
344 
345 		if (!field) {
346 			pr_err("failed to create a field %s\n", name);
347 			return -1;
348 		}
349 
350 		if (flags & TEP_FIELD_IS_STRING)
351 			ret = string_set_value(field, data + offset + i * len);
352 		else {
353 			unsigned long long value_int;
354 
355 			value_int = tep_read_number(
356 					fmtf->event->tep,
357 					data + offset + i * len, len);
358 
359 			if (!(flags & TEP_FIELD_IS_SIGNED))
360 				ret = bt_ctf_field_unsigned_integer_set_value(
361 						field, value_int);
362 			else
363 				ret = bt_ctf_field_signed_integer_set_value(
364 						field, adjust_signedness(value_int, len));
365 		}
366 
367 		if (ret) {
368 			pr_err("failed to set file value %s\n", name);
369 			goto err_put_field;
370 		}
371 		if (!(flags & TEP_FIELD_IS_ARRAY)) {
372 			ret = bt_ctf_event_set_payload(event, name, field);
373 			if (ret) {
374 				pr_err("failed to set payload %s\n", name);
375 				goto err_put_field;
376 			}
377 		}
378 		bt_ctf_field_put(field);
379 	}
380 	if (flags & TEP_FIELD_IS_ARRAY) {
381 		ret = bt_ctf_event_set_payload(event, name, array_field);
382 		if (ret) {
383 			pr_err("Failed add payload array %s\n", name);
384 			return -1;
385 		}
386 		bt_ctf_field_put(array_field);
387 	}
388 	return 0;
389 
390 err_put_field:
391 	bt_ctf_field_put(field);
392 	return -1;
393 }
394 
395 static int add_tracepoint_fields_values(struct ctf_writer *cw,
396 					struct bt_ctf_event_class *event_class,
397 					struct bt_ctf_event *event,
398 					struct tep_format_field *fields,
399 					struct perf_sample *sample)
400 {
401 	struct tep_format_field *field;
402 	int ret;
403 
404 	for (field = fields; field; field = field->next) {
405 		ret = add_tracepoint_field_value(cw, event_class, event, sample,
406 				field);
407 		if (ret)
408 			return -1;
409 	}
410 	return 0;
411 }
412 
413 static int add_tracepoint_values(struct ctf_writer *cw,
414 				 struct bt_ctf_event_class *event_class,
415 				 struct bt_ctf_event *event,
416 				 struct perf_evsel *evsel,
417 				 struct perf_sample *sample)
418 {
419 	struct tep_format_field *common_fields = evsel->tp_format->format.common_fields;
420 	struct tep_format_field *fields        = evsel->tp_format->format.fields;
421 	int ret;
422 
423 	ret = add_tracepoint_fields_values(cw, event_class, event,
424 					   common_fields, sample);
425 	if (!ret)
426 		ret = add_tracepoint_fields_values(cw, event_class, event,
427 						   fields, sample);
428 
429 	return ret;
430 }
431 
432 static int
433 add_bpf_output_values(struct bt_ctf_event_class *event_class,
434 		      struct bt_ctf_event *event,
435 		      struct perf_sample *sample)
436 {
437 	struct bt_ctf_field_type *len_type, *seq_type;
438 	struct bt_ctf_field *len_field, *seq_field;
439 	unsigned int raw_size = sample->raw_size;
440 	unsigned int nr_elements = raw_size / sizeof(u32);
441 	unsigned int i;
442 	int ret;
443 
444 	if (nr_elements * sizeof(u32) != raw_size)
445 		pr_warning("Incorrect raw_size (%u) in bpf output event, skip %zu bytes\n",
446 			   raw_size, nr_elements * sizeof(u32) - raw_size);
447 
448 	len_type = bt_ctf_event_class_get_field_by_name(event_class, "raw_len");
449 	len_field = bt_ctf_field_create(len_type);
450 	if (!len_field) {
451 		pr_err("failed to create 'raw_len' for bpf output event\n");
452 		ret = -1;
453 		goto put_len_type;
454 	}
455 
456 	ret = bt_ctf_field_unsigned_integer_set_value(len_field, nr_elements);
457 	if (ret) {
458 		pr_err("failed to set field value for raw_len\n");
459 		goto put_len_field;
460 	}
461 	ret = bt_ctf_event_set_payload(event, "raw_len", len_field);
462 	if (ret) {
463 		pr_err("failed to set payload to raw_len\n");
464 		goto put_len_field;
465 	}
466 
467 	seq_type = bt_ctf_event_class_get_field_by_name(event_class, "raw_data");
468 	seq_field = bt_ctf_field_create(seq_type);
469 	if (!seq_field) {
470 		pr_err("failed to create 'raw_data' for bpf output event\n");
471 		ret = -1;
472 		goto put_seq_type;
473 	}
474 
475 	ret = bt_ctf_field_sequence_set_length(seq_field, len_field);
476 	if (ret) {
477 		pr_err("failed to set length of 'raw_data'\n");
478 		goto put_seq_field;
479 	}
480 
481 	for (i = 0; i < nr_elements; i++) {
482 		struct bt_ctf_field *elem_field =
483 			bt_ctf_field_sequence_get_field(seq_field, i);
484 
485 		ret = bt_ctf_field_unsigned_integer_set_value(elem_field,
486 				((u32 *)(sample->raw_data))[i]);
487 
488 		bt_ctf_field_put(elem_field);
489 		if (ret) {
490 			pr_err("failed to set raw_data[%d]\n", i);
491 			goto put_seq_field;
492 		}
493 	}
494 
495 	ret = bt_ctf_event_set_payload(event, "raw_data", seq_field);
496 	if (ret)
497 		pr_err("failed to set payload for raw_data\n");
498 
499 put_seq_field:
500 	bt_ctf_field_put(seq_field);
501 put_seq_type:
502 	bt_ctf_field_type_put(seq_type);
503 put_len_field:
504 	bt_ctf_field_put(len_field);
505 put_len_type:
506 	bt_ctf_field_type_put(len_type);
507 	return ret;
508 }
509 
510 static int
511 add_callchain_output_values(struct bt_ctf_event_class *event_class,
512 		      struct bt_ctf_event *event,
513 		      struct ip_callchain *callchain)
514 {
515 	struct bt_ctf_field_type *len_type, *seq_type;
516 	struct bt_ctf_field *len_field, *seq_field;
517 	unsigned int nr_elements = callchain->nr;
518 	unsigned int i;
519 	int ret;
520 
521 	len_type = bt_ctf_event_class_get_field_by_name(
522 			event_class, "perf_callchain_size");
523 	len_field = bt_ctf_field_create(len_type);
524 	if (!len_field) {
525 		pr_err("failed to create 'perf_callchain_size' for callchain output event\n");
526 		ret = -1;
527 		goto put_len_type;
528 	}
529 
530 	ret = bt_ctf_field_unsigned_integer_set_value(len_field, nr_elements);
531 	if (ret) {
532 		pr_err("failed to set field value for perf_callchain_size\n");
533 		goto put_len_field;
534 	}
535 	ret = bt_ctf_event_set_payload(event, "perf_callchain_size", len_field);
536 	if (ret) {
537 		pr_err("failed to set payload to perf_callchain_size\n");
538 		goto put_len_field;
539 	}
540 
541 	seq_type = bt_ctf_event_class_get_field_by_name(
542 			event_class, "perf_callchain");
543 	seq_field = bt_ctf_field_create(seq_type);
544 	if (!seq_field) {
545 		pr_err("failed to create 'perf_callchain' for callchain output event\n");
546 		ret = -1;
547 		goto put_seq_type;
548 	}
549 
550 	ret = bt_ctf_field_sequence_set_length(seq_field, len_field);
551 	if (ret) {
552 		pr_err("failed to set length of 'perf_callchain'\n");
553 		goto put_seq_field;
554 	}
555 
556 	for (i = 0; i < nr_elements; i++) {
557 		struct bt_ctf_field *elem_field =
558 			bt_ctf_field_sequence_get_field(seq_field, i);
559 
560 		ret = bt_ctf_field_unsigned_integer_set_value(elem_field,
561 				((u64 *)(callchain->ips))[i]);
562 
563 		bt_ctf_field_put(elem_field);
564 		if (ret) {
565 			pr_err("failed to set callchain[%d]\n", i);
566 			goto put_seq_field;
567 		}
568 	}
569 
570 	ret = bt_ctf_event_set_payload(event, "perf_callchain", seq_field);
571 	if (ret)
572 		pr_err("failed to set payload for raw_data\n");
573 
574 put_seq_field:
575 	bt_ctf_field_put(seq_field);
576 put_seq_type:
577 	bt_ctf_field_type_put(seq_type);
578 put_len_field:
579 	bt_ctf_field_put(len_field);
580 put_len_type:
581 	bt_ctf_field_type_put(len_type);
582 	return ret;
583 }
584 
585 static int add_generic_values(struct ctf_writer *cw,
586 			      struct bt_ctf_event *event,
587 			      struct perf_evsel *evsel,
588 			      struct perf_sample *sample)
589 {
590 	u64 type = evsel->attr.sample_type;
591 	int ret;
592 
593 	/*
594 	 * missing:
595 	 *   PERF_SAMPLE_TIME         - not needed as we have it in
596 	 *                              ctf event header
597 	 *   PERF_SAMPLE_READ         - TODO
598 	 *   PERF_SAMPLE_RAW          - tracepoint fields are handled separately
599 	 *   PERF_SAMPLE_BRANCH_STACK - TODO
600 	 *   PERF_SAMPLE_REGS_USER    - TODO
601 	 *   PERF_SAMPLE_STACK_USER   - TODO
602 	 */
603 
604 	if (type & PERF_SAMPLE_IP) {
605 		ret = value_set_u64_hex(cw, event, "perf_ip", sample->ip);
606 		if (ret)
607 			return -1;
608 	}
609 
610 	if (type & PERF_SAMPLE_TID) {
611 		ret = value_set_s32(cw, event, "perf_tid", sample->tid);
612 		if (ret)
613 			return -1;
614 
615 		ret = value_set_s32(cw, event, "perf_pid", sample->pid);
616 		if (ret)
617 			return -1;
618 	}
619 
620 	if ((type & PERF_SAMPLE_ID) ||
621 	    (type & PERF_SAMPLE_IDENTIFIER)) {
622 		ret = value_set_u64(cw, event, "perf_id", sample->id);
623 		if (ret)
624 			return -1;
625 	}
626 
627 	if (type & PERF_SAMPLE_STREAM_ID) {
628 		ret = value_set_u64(cw, event, "perf_stream_id", sample->stream_id);
629 		if (ret)
630 			return -1;
631 	}
632 
633 	if (type & PERF_SAMPLE_PERIOD) {
634 		ret = value_set_u64(cw, event, "perf_period", sample->period);
635 		if (ret)
636 			return -1;
637 	}
638 
639 	if (type & PERF_SAMPLE_WEIGHT) {
640 		ret = value_set_u64(cw, event, "perf_weight", sample->weight);
641 		if (ret)
642 			return -1;
643 	}
644 
645 	if (type & PERF_SAMPLE_DATA_SRC) {
646 		ret = value_set_u64(cw, event, "perf_data_src",
647 				sample->data_src);
648 		if (ret)
649 			return -1;
650 	}
651 
652 	if (type & PERF_SAMPLE_TRANSACTION) {
653 		ret = value_set_u64(cw, event, "perf_transaction",
654 				sample->transaction);
655 		if (ret)
656 			return -1;
657 	}
658 
659 	return 0;
660 }
661 
662 static int ctf_stream__flush(struct ctf_stream *cs)
663 {
664 	int err = 0;
665 
666 	if (cs) {
667 		err = bt_ctf_stream_flush(cs->stream);
668 		if (err)
669 			pr_err("CTF stream %d flush failed\n", cs->cpu);
670 
671 		pr("Flush stream for cpu %d (%u samples)\n",
672 		   cs->cpu, cs->count);
673 
674 		cs->count = 0;
675 	}
676 
677 	return err;
678 }
679 
680 static struct ctf_stream *ctf_stream__create(struct ctf_writer *cw, int cpu)
681 {
682 	struct ctf_stream *cs;
683 	struct bt_ctf_field *pkt_ctx   = NULL;
684 	struct bt_ctf_field *cpu_field = NULL;
685 	struct bt_ctf_stream *stream   = NULL;
686 	int ret;
687 
688 	cs = zalloc(sizeof(*cs));
689 	if (!cs) {
690 		pr_err("Failed to allocate ctf stream\n");
691 		return NULL;
692 	}
693 
694 	stream = bt_ctf_writer_create_stream(cw->writer, cw->stream_class);
695 	if (!stream) {
696 		pr_err("Failed to create CTF stream\n");
697 		goto out;
698 	}
699 
700 	pkt_ctx = bt_ctf_stream_get_packet_context(stream);
701 	if (!pkt_ctx) {
702 		pr_err("Failed to obtain packet context\n");
703 		goto out;
704 	}
705 
706 	cpu_field = bt_ctf_field_structure_get_field(pkt_ctx, "cpu_id");
707 	bt_ctf_field_put(pkt_ctx);
708 	if (!cpu_field) {
709 		pr_err("Failed to obtain cpu field\n");
710 		goto out;
711 	}
712 
713 	ret = bt_ctf_field_unsigned_integer_set_value(cpu_field, (u32) cpu);
714 	if (ret) {
715 		pr_err("Failed to update CPU number\n");
716 		goto out;
717 	}
718 
719 	bt_ctf_field_put(cpu_field);
720 
721 	cs->cpu    = cpu;
722 	cs->stream = stream;
723 	return cs;
724 
725 out:
726 	if (cpu_field)
727 		bt_ctf_field_put(cpu_field);
728 	if (stream)
729 		bt_ctf_stream_put(stream);
730 
731 	free(cs);
732 	return NULL;
733 }
734 
735 static void ctf_stream__delete(struct ctf_stream *cs)
736 {
737 	if (cs) {
738 		bt_ctf_stream_put(cs->stream);
739 		free(cs);
740 	}
741 }
742 
743 static struct ctf_stream *ctf_stream(struct ctf_writer *cw, int cpu)
744 {
745 	struct ctf_stream *cs = cw->stream[cpu];
746 
747 	if (!cs) {
748 		cs = ctf_stream__create(cw, cpu);
749 		cw->stream[cpu] = cs;
750 	}
751 
752 	return cs;
753 }
754 
755 static int get_sample_cpu(struct ctf_writer *cw, struct perf_sample *sample,
756 			  struct perf_evsel *evsel)
757 {
758 	int cpu = 0;
759 
760 	if (evsel->attr.sample_type & PERF_SAMPLE_CPU)
761 		cpu = sample->cpu;
762 
763 	if (cpu > cw->stream_cnt) {
764 		pr_err("Event was recorded for CPU %d, limit is at %d.\n",
765 			cpu, cw->stream_cnt);
766 		cpu = 0;
767 	}
768 
769 	return cpu;
770 }
771 
772 #define STREAM_FLUSH_COUNT 100000
773 
774 /*
775  * Currently we have no other way to determine the
776  * time for the stream flush other than keep track
777  * of the number of events and check it against
778  * threshold.
779  */
780 static bool is_flush_needed(struct ctf_stream *cs)
781 {
782 	return cs->count >= STREAM_FLUSH_COUNT;
783 }
784 
785 static int process_sample_event(struct perf_tool *tool,
786 				union perf_event *_event,
787 				struct perf_sample *sample,
788 				struct perf_evsel *evsel,
789 				struct machine *machine __maybe_unused)
790 {
791 	struct convert *c = container_of(tool, struct convert, tool);
792 	struct evsel_priv *priv = evsel->priv;
793 	struct ctf_writer *cw = &c->writer;
794 	struct ctf_stream *cs;
795 	struct bt_ctf_event_class *event_class;
796 	struct bt_ctf_event *event;
797 	int ret;
798 	unsigned long type = evsel->attr.sample_type;
799 
800 	if (WARN_ONCE(!priv, "Failed to setup all events.\n"))
801 		return 0;
802 
803 	event_class = priv->event_class;
804 
805 	/* update stats */
806 	c->events_count++;
807 	c->events_size += _event->header.size;
808 
809 	pr_time2(sample->time, "sample %" PRIu64 "\n", c->events_count);
810 
811 	event = bt_ctf_event_create(event_class);
812 	if (!event) {
813 		pr_err("Failed to create an CTF event\n");
814 		return -1;
815 	}
816 
817 	bt_ctf_clock_set_time(cw->clock, sample->time);
818 
819 	ret = add_generic_values(cw, event, evsel, sample);
820 	if (ret)
821 		return -1;
822 
823 	if (evsel->attr.type == PERF_TYPE_TRACEPOINT) {
824 		ret = add_tracepoint_values(cw, event_class, event,
825 					    evsel, sample);
826 		if (ret)
827 			return -1;
828 	}
829 
830 	if (type & PERF_SAMPLE_CALLCHAIN) {
831 		ret = add_callchain_output_values(event_class,
832 				event, sample->callchain);
833 		if (ret)
834 			return -1;
835 	}
836 
837 	if (perf_evsel__is_bpf_output(evsel)) {
838 		ret = add_bpf_output_values(event_class, event, sample);
839 		if (ret)
840 			return -1;
841 	}
842 
843 	cs = ctf_stream(cw, get_sample_cpu(cw, sample, evsel));
844 	if (cs) {
845 		if (is_flush_needed(cs))
846 			ctf_stream__flush(cs);
847 
848 		cs->count++;
849 		bt_ctf_stream_append_event(cs->stream, event);
850 	}
851 
852 	bt_ctf_event_put(event);
853 	return cs ? 0 : -1;
854 }
855 
856 #define __NON_SAMPLE_SET_FIELD(_name, _type, _field) 	\
857 do {							\
858 	ret = value_set_##_type(cw, event, #_field, _event->_name._field);\
859 	if (ret)					\
860 		return -1;				\
861 } while(0)
862 
863 #define __FUNC_PROCESS_NON_SAMPLE(_name, body) 	\
864 static int process_##_name##_event(struct perf_tool *tool,	\
865 				   union perf_event *_event,	\
866 				   struct perf_sample *sample,	\
867 				   struct machine *machine)	\
868 {								\
869 	struct convert *c = container_of(tool, struct convert, tool);\
870 	struct ctf_writer *cw = &c->writer;			\
871 	struct bt_ctf_event_class *event_class = cw->_name##_class;\
872 	struct bt_ctf_event *event;				\
873 	struct ctf_stream *cs;					\
874 	int ret;						\
875 								\
876 	c->non_sample_count++;					\
877 	c->events_size += _event->header.size;			\
878 	event = bt_ctf_event_create(event_class);		\
879 	if (!event) {						\
880 		pr_err("Failed to create an CTF event\n");	\
881 		return -1;					\
882 	}							\
883 								\
884 	bt_ctf_clock_set_time(cw->clock, sample->time);		\
885 	body							\
886 	cs = ctf_stream(cw, 0);					\
887 	if (cs) {						\
888 		if (is_flush_needed(cs))			\
889 			ctf_stream__flush(cs);			\
890 								\
891 		cs->count++;					\
892 		bt_ctf_stream_append_event(cs->stream, event);	\
893 	}							\
894 	bt_ctf_event_put(event);				\
895 								\
896 	return perf_event__process_##_name(tool, _event, sample, machine);\
897 }
898 
899 __FUNC_PROCESS_NON_SAMPLE(comm,
900 	__NON_SAMPLE_SET_FIELD(comm, u32, pid);
901 	__NON_SAMPLE_SET_FIELD(comm, u32, tid);
902 	__NON_SAMPLE_SET_FIELD(comm, string, comm);
903 )
904 __FUNC_PROCESS_NON_SAMPLE(fork,
905 	__NON_SAMPLE_SET_FIELD(fork, u32, pid);
906 	__NON_SAMPLE_SET_FIELD(fork, u32, ppid);
907 	__NON_SAMPLE_SET_FIELD(fork, u32, tid);
908 	__NON_SAMPLE_SET_FIELD(fork, u32, ptid);
909 	__NON_SAMPLE_SET_FIELD(fork, u64, time);
910 )
911 
912 __FUNC_PROCESS_NON_SAMPLE(exit,
913 	__NON_SAMPLE_SET_FIELD(fork, u32, pid);
914 	__NON_SAMPLE_SET_FIELD(fork, u32, ppid);
915 	__NON_SAMPLE_SET_FIELD(fork, u32, tid);
916 	__NON_SAMPLE_SET_FIELD(fork, u32, ptid);
917 	__NON_SAMPLE_SET_FIELD(fork, u64, time);
918 )
919 __FUNC_PROCESS_NON_SAMPLE(mmap,
920 	__NON_SAMPLE_SET_FIELD(mmap, u32, pid);
921 	__NON_SAMPLE_SET_FIELD(mmap, u32, tid);
922 	__NON_SAMPLE_SET_FIELD(mmap, u64_hex, start);
923 	__NON_SAMPLE_SET_FIELD(mmap, string, filename);
924 )
925 __FUNC_PROCESS_NON_SAMPLE(mmap2,
926 	__NON_SAMPLE_SET_FIELD(mmap2, u32, pid);
927 	__NON_SAMPLE_SET_FIELD(mmap2, u32, tid);
928 	__NON_SAMPLE_SET_FIELD(mmap2, u64_hex, start);
929 	__NON_SAMPLE_SET_FIELD(mmap2, string, filename);
930 )
931 #undef __NON_SAMPLE_SET_FIELD
932 #undef __FUNC_PROCESS_NON_SAMPLE
933 
934 /* If dup < 0, add a prefix. Else, add _dupl_X suffix. */
935 static char *change_name(char *name, char *orig_name, int dup)
936 {
937 	char *new_name = NULL;
938 	size_t len;
939 
940 	if (!name)
941 		name = orig_name;
942 
943 	if (dup >= 10)
944 		goto out;
945 	/*
946 	 * Add '_' prefix to potential keywork.  According to
947 	 * Mathieu Desnoyers (https://lkml.org/lkml/2015/1/23/652),
948 	 * futher CTF spec updating may require us to use '$'.
949 	 */
950 	if (dup < 0)
951 		len = strlen(name) + sizeof("_");
952 	else
953 		len = strlen(orig_name) + sizeof("_dupl_X");
954 
955 	new_name = malloc(len);
956 	if (!new_name)
957 		goto out;
958 
959 	if (dup < 0)
960 		snprintf(new_name, len, "_%s", name);
961 	else
962 		snprintf(new_name, len, "%s_dupl_%d", orig_name, dup);
963 
964 out:
965 	if (name != orig_name)
966 		free(name);
967 	return new_name;
968 }
969 
970 static int event_class_add_field(struct bt_ctf_event_class *event_class,
971 		struct bt_ctf_field_type *type,
972 		struct tep_format_field *field)
973 {
974 	struct bt_ctf_field_type *t = NULL;
975 	char *name;
976 	int dup = 1;
977 	int ret;
978 
979 	/* alias was already assigned */
980 	if (field->alias != field->name)
981 		return bt_ctf_event_class_add_field(event_class, type,
982 				(char *)field->alias);
983 
984 	name = field->name;
985 
986 	/* If 'name' is a keywork, add prefix. */
987 	if (bt_ctf_validate_identifier(name))
988 		name = change_name(name, field->name, -1);
989 
990 	if (!name) {
991 		pr_err("Failed to fix invalid identifier.");
992 		return -1;
993 	}
994 	while ((t = bt_ctf_event_class_get_field_by_name(event_class, name))) {
995 		bt_ctf_field_type_put(t);
996 		name = change_name(name, field->name, dup++);
997 		if (!name) {
998 			pr_err("Failed to create dup name for '%s'\n", field->name);
999 			return -1;
1000 		}
1001 	}
1002 
1003 	ret = bt_ctf_event_class_add_field(event_class, type, name);
1004 	if (!ret)
1005 		field->alias = name;
1006 
1007 	return ret;
1008 }
1009 
1010 static int add_tracepoint_fields_types(struct ctf_writer *cw,
1011 				       struct tep_format_field *fields,
1012 				       struct bt_ctf_event_class *event_class)
1013 {
1014 	struct tep_format_field *field;
1015 	int ret;
1016 
1017 	for (field = fields; field; field = field->next) {
1018 		struct bt_ctf_field_type *type;
1019 		unsigned long flags = field->flags;
1020 
1021 		pr2("  field '%s'\n", field->name);
1022 
1023 		type = get_tracepoint_field_type(cw, field);
1024 		if (!type)
1025 			return -1;
1026 
1027 		/*
1028 		 * A string is an array of chars. For this we use the string
1029 		 * type and don't care that it is an array. What we don't
1030 		 * support is an array of strings.
1031 		 */
1032 		if (flags & TEP_FIELD_IS_STRING)
1033 			flags &= ~TEP_FIELD_IS_ARRAY;
1034 
1035 		if (flags & TEP_FIELD_IS_ARRAY)
1036 			type = bt_ctf_field_type_array_create(type, field->arraylen);
1037 
1038 		ret = event_class_add_field(event_class, type, field);
1039 
1040 		if (flags & TEP_FIELD_IS_ARRAY)
1041 			bt_ctf_field_type_put(type);
1042 
1043 		if (ret) {
1044 			pr_err("Failed to add field '%s': %d\n",
1045 					field->name, ret);
1046 			return -1;
1047 		}
1048 	}
1049 
1050 	return 0;
1051 }
1052 
1053 static int add_tracepoint_types(struct ctf_writer *cw,
1054 				struct perf_evsel *evsel,
1055 				struct bt_ctf_event_class *class)
1056 {
1057 	struct tep_format_field *common_fields = evsel->tp_format->format.common_fields;
1058 	struct tep_format_field *fields        = evsel->tp_format->format.fields;
1059 	int ret;
1060 
1061 	ret = add_tracepoint_fields_types(cw, common_fields, class);
1062 	if (!ret)
1063 		ret = add_tracepoint_fields_types(cw, fields, class);
1064 
1065 	return ret;
1066 }
1067 
1068 static int add_bpf_output_types(struct ctf_writer *cw,
1069 				struct bt_ctf_event_class *class)
1070 {
1071 	struct bt_ctf_field_type *len_type = cw->data.u32;
1072 	struct bt_ctf_field_type *seq_base_type = cw->data.u32_hex;
1073 	struct bt_ctf_field_type *seq_type;
1074 	int ret;
1075 
1076 	ret = bt_ctf_event_class_add_field(class, len_type, "raw_len");
1077 	if (ret)
1078 		return ret;
1079 
1080 	seq_type = bt_ctf_field_type_sequence_create(seq_base_type, "raw_len");
1081 	if (!seq_type)
1082 		return -1;
1083 
1084 	return bt_ctf_event_class_add_field(class, seq_type, "raw_data");
1085 }
1086 
1087 static int add_generic_types(struct ctf_writer *cw, struct perf_evsel *evsel,
1088 			     struct bt_ctf_event_class *event_class)
1089 {
1090 	u64 type = evsel->attr.sample_type;
1091 
1092 	/*
1093 	 * missing:
1094 	 *   PERF_SAMPLE_TIME         - not needed as we have it in
1095 	 *                              ctf event header
1096 	 *   PERF_SAMPLE_READ         - TODO
1097 	 *   PERF_SAMPLE_CALLCHAIN    - TODO
1098 	 *   PERF_SAMPLE_RAW          - tracepoint fields and BPF output
1099 	 *                              are handled separately
1100 	 *   PERF_SAMPLE_BRANCH_STACK - TODO
1101 	 *   PERF_SAMPLE_REGS_USER    - TODO
1102 	 *   PERF_SAMPLE_STACK_USER   - TODO
1103 	 */
1104 
1105 #define ADD_FIELD(cl, t, n)						\
1106 	do {								\
1107 		pr2("  field '%s'\n", n);				\
1108 		if (bt_ctf_event_class_add_field(cl, t, n)) {		\
1109 			pr_err("Failed to add field '%s';\n", n);	\
1110 			return -1;					\
1111 		}							\
1112 	} while (0)
1113 
1114 	if (type & PERF_SAMPLE_IP)
1115 		ADD_FIELD(event_class, cw->data.u64_hex, "perf_ip");
1116 
1117 	if (type & PERF_SAMPLE_TID) {
1118 		ADD_FIELD(event_class, cw->data.s32, "perf_tid");
1119 		ADD_FIELD(event_class, cw->data.s32, "perf_pid");
1120 	}
1121 
1122 	if ((type & PERF_SAMPLE_ID) ||
1123 	    (type & PERF_SAMPLE_IDENTIFIER))
1124 		ADD_FIELD(event_class, cw->data.u64, "perf_id");
1125 
1126 	if (type & PERF_SAMPLE_STREAM_ID)
1127 		ADD_FIELD(event_class, cw->data.u64, "perf_stream_id");
1128 
1129 	if (type & PERF_SAMPLE_PERIOD)
1130 		ADD_FIELD(event_class, cw->data.u64, "perf_period");
1131 
1132 	if (type & PERF_SAMPLE_WEIGHT)
1133 		ADD_FIELD(event_class, cw->data.u64, "perf_weight");
1134 
1135 	if (type & PERF_SAMPLE_DATA_SRC)
1136 		ADD_FIELD(event_class, cw->data.u64, "perf_data_src");
1137 
1138 	if (type & PERF_SAMPLE_TRANSACTION)
1139 		ADD_FIELD(event_class, cw->data.u64, "perf_transaction");
1140 
1141 	if (type & PERF_SAMPLE_CALLCHAIN) {
1142 		ADD_FIELD(event_class, cw->data.u32, "perf_callchain_size");
1143 		ADD_FIELD(event_class,
1144 			bt_ctf_field_type_sequence_create(
1145 				cw->data.u64_hex, "perf_callchain_size"),
1146 			"perf_callchain");
1147 	}
1148 
1149 #undef ADD_FIELD
1150 	return 0;
1151 }
1152 
1153 static int add_event(struct ctf_writer *cw, struct perf_evsel *evsel)
1154 {
1155 	struct bt_ctf_event_class *event_class;
1156 	struct evsel_priv *priv;
1157 	const char *name = perf_evsel__name(evsel);
1158 	int ret;
1159 
1160 	pr("Adding event '%s' (type %d)\n", name, evsel->attr.type);
1161 
1162 	event_class = bt_ctf_event_class_create(name);
1163 	if (!event_class)
1164 		return -1;
1165 
1166 	ret = add_generic_types(cw, evsel, event_class);
1167 	if (ret)
1168 		goto err;
1169 
1170 	if (evsel->attr.type == PERF_TYPE_TRACEPOINT) {
1171 		ret = add_tracepoint_types(cw, evsel, event_class);
1172 		if (ret)
1173 			goto err;
1174 	}
1175 
1176 	if (perf_evsel__is_bpf_output(evsel)) {
1177 		ret = add_bpf_output_types(cw, event_class);
1178 		if (ret)
1179 			goto err;
1180 	}
1181 
1182 	ret = bt_ctf_stream_class_add_event_class(cw->stream_class, event_class);
1183 	if (ret) {
1184 		pr("Failed to add event class into stream.\n");
1185 		goto err;
1186 	}
1187 
1188 	priv = malloc(sizeof(*priv));
1189 	if (!priv)
1190 		goto err;
1191 
1192 	priv->event_class = event_class;
1193 	evsel->priv       = priv;
1194 	return 0;
1195 
1196 err:
1197 	bt_ctf_event_class_put(event_class);
1198 	pr_err("Failed to add event '%s'.\n", name);
1199 	return -1;
1200 }
1201 
1202 static int setup_events(struct ctf_writer *cw, struct perf_session *session)
1203 {
1204 	struct perf_evlist *evlist = session->evlist;
1205 	struct perf_evsel *evsel;
1206 	int ret;
1207 
1208 	evlist__for_each_entry(evlist, evsel) {
1209 		ret = add_event(cw, evsel);
1210 		if (ret)
1211 			return ret;
1212 	}
1213 	return 0;
1214 }
1215 
1216 #define __NON_SAMPLE_ADD_FIELD(t, n)						\
1217 	do {							\
1218 		pr2("  field '%s'\n", #n);			\
1219 		if (bt_ctf_event_class_add_field(event_class, cw->data.t, #n)) {\
1220 			pr_err("Failed to add field '%s';\n", #n);\
1221 			return -1;				\
1222 		}						\
1223 	} while(0)
1224 
1225 #define __FUNC_ADD_NON_SAMPLE_EVENT_CLASS(_name, body) 		\
1226 static int add_##_name##_event(struct ctf_writer *cw)		\
1227 {								\
1228 	struct bt_ctf_event_class *event_class;			\
1229 	int ret;						\
1230 								\
1231 	pr("Adding "#_name" event\n");				\
1232 	event_class = bt_ctf_event_class_create("perf_" #_name);\
1233 	if (!event_class)					\
1234 		return -1;					\
1235 	body							\
1236 								\
1237 	ret = bt_ctf_stream_class_add_event_class(cw->stream_class, event_class);\
1238 	if (ret) {						\
1239 		pr("Failed to add event class '"#_name"' into stream.\n");\
1240 		return ret;					\
1241 	}							\
1242 								\
1243 	cw->_name##_class = event_class;			\
1244 	bt_ctf_event_class_put(event_class);			\
1245 	return 0;						\
1246 }
1247 
1248 __FUNC_ADD_NON_SAMPLE_EVENT_CLASS(comm,
1249 	__NON_SAMPLE_ADD_FIELD(u32, pid);
1250 	__NON_SAMPLE_ADD_FIELD(u32, tid);
1251 	__NON_SAMPLE_ADD_FIELD(string, comm);
1252 )
1253 
1254 __FUNC_ADD_NON_SAMPLE_EVENT_CLASS(fork,
1255 	__NON_SAMPLE_ADD_FIELD(u32, pid);
1256 	__NON_SAMPLE_ADD_FIELD(u32, ppid);
1257 	__NON_SAMPLE_ADD_FIELD(u32, tid);
1258 	__NON_SAMPLE_ADD_FIELD(u32, ptid);
1259 	__NON_SAMPLE_ADD_FIELD(u64, time);
1260 )
1261 
1262 __FUNC_ADD_NON_SAMPLE_EVENT_CLASS(exit,
1263 	__NON_SAMPLE_ADD_FIELD(u32, pid);
1264 	__NON_SAMPLE_ADD_FIELD(u32, ppid);
1265 	__NON_SAMPLE_ADD_FIELD(u32, tid);
1266 	__NON_SAMPLE_ADD_FIELD(u32, ptid);
1267 	__NON_SAMPLE_ADD_FIELD(u64, time);
1268 )
1269 
1270 __FUNC_ADD_NON_SAMPLE_EVENT_CLASS(mmap,
1271 	__NON_SAMPLE_ADD_FIELD(u32, pid);
1272 	__NON_SAMPLE_ADD_FIELD(u32, tid);
1273 	__NON_SAMPLE_ADD_FIELD(u64_hex, start);
1274 	__NON_SAMPLE_ADD_FIELD(string, filename);
1275 )
1276 
1277 __FUNC_ADD_NON_SAMPLE_EVENT_CLASS(mmap2,
1278 	__NON_SAMPLE_ADD_FIELD(u32, pid);
1279 	__NON_SAMPLE_ADD_FIELD(u32, tid);
1280 	__NON_SAMPLE_ADD_FIELD(u64_hex, start);
1281 	__NON_SAMPLE_ADD_FIELD(string, filename);
1282 )
1283 #undef __NON_SAMPLE_ADD_FIELD
1284 #undef __FUNC_ADD_NON_SAMPLE_EVENT_CLASS
1285 
1286 static int setup_non_sample_events(struct ctf_writer *cw,
1287 				   struct perf_session *session __maybe_unused)
1288 {
1289 	int ret;
1290 
1291 	ret = add_comm_event(cw);
1292 	if (ret)
1293 		return ret;
1294 	ret = add_exit_event(cw);
1295 	if (ret)
1296 		return ret;
1297 	ret = add_fork_event(cw);
1298 	if (ret)
1299 		return ret;
1300 	ret = add_mmap_event(cw);
1301 	if (ret)
1302 		return ret;
1303 	ret = add_mmap2_event(cw);
1304 	if (ret)
1305 		return ret;
1306 	return 0;
1307 }
1308 
1309 static void cleanup_events(struct perf_session *session)
1310 {
1311 	struct perf_evlist *evlist = session->evlist;
1312 	struct perf_evsel *evsel;
1313 
1314 	evlist__for_each_entry(evlist, evsel) {
1315 		struct evsel_priv *priv;
1316 
1317 		priv = evsel->priv;
1318 		bt_ctf_event_class_put(priv->event_class);
1319 		zfree(&evsel->priv);
1320 	}
1321 
1322 	perf_evlist__delete(evlist);
1323 	session->evlist = NULL;
1324 }
1325 
1326 static int setup_streams(struct ctf_writer *cw, struct perf_session *session)
1327 {
1328 	struct ctf_stream **stream;
1329 	struct perf_header *ph = &session->header;
1330 	int ncpus;
1331 
1332 	/*
1333 	 * Try to get the number of cpus used in the data file,
1334 	 * if not present fallback to the MAX_CPUS.
1335 	 */
1336 	ncpus = ph->env.nr_cpus_avail ?: MAX_CPUS;
1337 
1338 	stream = zalloc(sizeof(*stream) * ncpus);
1339 	if (!stream) {
1340 		pr_err("Failed to allocate streams.\n");
1341 		return -ENOMEM;
1342 	}
1343 
1344 	cw->stream     = stream;
1345 	cw->stream_cnt = ncpus;
1346 	return 0;
1347 }
1348 
1349 static void free_streams(struct ctf_writer *cw)
1350 {
1351 	int cpu;
1352 
1353 	for (cpu = 0; cpu < cw->stream_cnt; cpu++)
1354 		ctf_stream__delete(cw->stream[cpu]);
1355 
1356 	free(cw->stream);
1357 }
1358 
1359 static int ctf_writer__setup_env(struct ctf_writer *cw,
1360 				 struct perf_session *session)
1361 {
1362 	struct perf_header *header = &session->header;
1363 	struct bt_ctf_writer *writer = cw->writer;
1364 
1365 #define ADD(__n, __v)							\
1366 do {									\
1367 	if (bt_ctf_writer_add_environment_field(writer, __n, __v))	\
1368 		return -1;						\
1369 } while (0)
1370 
1371 	ADD("host",    header->env.hostname);
1372 	ADD("sysname", "Linux");
1373 	ADD("release", header->env.os_release);
1374 	ADD("version", header->env.version);
1375 	ADD("machine", header->env.arch);
1376 	ADD("domain", "kernel");
1377 	ADD("tracer_name", "perf");
1378 
1379 #undef ADD
1380 	return 0;
1381 }
1382 
1383 static int ctf_writer__setup_clock(struct ctf_writer *cw)
1384 {
1385 	struct bt_ctf_clock *clock = cw->clock;
1386 
1387 	bt_ctf_clock_set_description(clock, "perf clock");
1388 
1389 #define SET(__n, __v)				\
1390 do {						\
1391 	if (bt_ctf_clock_set_##__n(clock, __v))	\
1392 		return -1;			\
1393 } while (0)
1394 
1395 	SET(frequency,   1000000000);
1396 	SET(offset_s,    0);
1397 	SET(offset,      0);
1398 	SET(precision,   10);
1399 	SET(is_absolute, 0);
1400 
1401 #undef SET
1402 	return 0;
1403 }
1404 
1405 static struct bt_ctf_field_type *create_int_type(int size, bool sign, bool hex)
1406 {
1407 	struct bt_ctf_field_type *type;
1408 
1409 	type = bt_ctf_field_type_integer_create(size);
1410 	if (!type)
1411 		return NULL;
1412 
1413 	if (sign &&
1414 	    bt_ctf_field_type_integer_set_signed(type, 1))
1415 		goto err;
1416 
1417 	if (hex &&
1418 	    bt_ctf_field_type_integer_set_base(type, BT_CTF_INTEGER_BASE_HEXADECIMAL))
1419 		goto err;
1420 
1421 #if __BYTE_ORDER == __BIG_ENDIAN
1422 	bt_ctf_field_type_set_byte_order(type, BT_CTF_BYTE_ORDER_BIG_ENDIAN);
1423 #else
1424 	bt_ctf_field_type_set_byte_order(type, BT_CTF_BYTE_ORDER_LITTLE_ENDIAN);
1425 #endif
1426 
1427 	pr2("Created type: INTEGER %d-bit %ssigned %s\n",
1428 	    size, sign ? "un" : "", hex ? "hex" : "");
1429 	return type;
1430 
1431 err:
1432 	bt_ctf_field_type_put(type);
1433 	return NULL;
1434 }
1435 
1436 static void ctf_writer__cleanup_data(struct ctf_writer *cw)
1437 {
1438 	unsigned int i;
1439 
1440 	for (i = 0; i < ARRAY_SIZE(cw->data.array); i++)
1441 		bt_ctf_field_type_put(cw->data.array[i]);
1442 }
1443 
1444 static int ctf_writer__init_data(struct ctf_writer *cw)
1445 {
1446 #define CREATE_INT_TYPE(type, size, sign, hex)		\
1447 do {							\
1448 	(type) = create_int_type(size, sign, hex);	\
1449 	if (!(type))					\
1450 		goto err;				\
1451 } while (0)
1452 
1453 	CREATE_INT_TYPE(cw->data.s64, 64, true,  false);
1454 	CREATE_INT_TYPE(cw->data.u64, 64, false, false);
1455 	CREATE_INT_TYPE(cw->data.s32, 32, true,  false);
1456 	CREATE_INT_TYPE(cw->data.u32, 32, false, false);
1457 	CREATE_INT_TYPE(cw->data.u32_hex, 32, false, true);
1458 	CREATE_INT_TYPE(cw->data.u64_hex, 64, false, true);
1459 
1460 	cw->data.string  = bt_ctf_field_type_string_create();
1461 	if (cw->data.string)
1462 		return 0;
1463 
1464 err:
1465 	ctf_writer__cleanup_data(cw);
1466 	pr_err("Failed to create data types.\n");
1467 	return -1;
1468 }
1469 
1470 static void ctf_writer__cleanup(struct ctf_writer *cw)
1471 {
1472 	ctf_writer__cleanup_data(cw);
1473 
1474 	bt_ctf_clock_put(cw->clock);
1475 	free_streams(cw);
1476 	bt_ctf_stream_class_put(cw->stream_class);
1477 	bt_ctf_writer_put(cw->writer);
1478 
1479 	/* and NULL all the pointers */
1480 	memset(cw, 0, sizeof(*cw));
1481 }
1482 
1483 static int ctf_writer__init(struct ctf_writer *cw, const char *path)
1484 {
1485 	struct bt_ctf_writer		*writer;
1486 	struct bt_ctf_stream_class	*stream_class;
1487 	struct bt_ctf_clock		*clock;
1488 	struct bt_ctf_field_type	*pkt_ctx_type;
1489 	int				ret;
1490 
1491 	/* CTF writer */
1492 	writer = bt_ctf_writer_create(path);
1493 	if (!writer)
1494 		goto err;
1495 
1496 	cw->writer = writer;
1497 
1498 	/* CTF clock */
1499 	clock = bt_ctf_clock_create("perf_clock");
1500 	if (!clock) {
1501 		pr("Failed to create CTF clock.\n");
1502 		goto err_cleanup;
1503 	}
1504 
1505 	cw->clock = clock;
1506 
1507 	if (ctf_writer__setup_clock(cw)) {
1508 		pr("Failed to setup CTF clock.\n");
1509 		goto err_cleanup;
1510 	}
1511 
1512 	/* CTF stream class */
1513 	stream_class = bt_ctf_stream_class_create("perf_stream");
1514 	if (!stream_class) {
1515 		pr("Failed to create CTF stream class.\n");
1516 		goto err_cleanup;
1517 	}
1518 
1519 	cw->stream_class = stream_class;
1520 
1521 	/* CTF clock stream setup */
1522 	if (bt_ctf_stream_class_set_clock(stream_class, clock)) {
1523 		pr("Failed to assign CTF clock to stream class.\n");
1524 		goto err_cleanup;
1525 	}
1526 
1527 	if (ctf_writer__init_data(cw))
1528 		goto err_cleanup;
1529 
1530 	/* Add cpu_id for packet context */
1531 	pkt_ctx_type = bt_ctf_stream_class_get_packet_context_type(stream_class);
1532 	if (!pkt_ctx_type)
1533 		goto err_cleanup;
1534 
1535 	ret = bt_ctf_field_type_structure_add_field(pkt_ctx_type, cw->data.u32, "cpu_id");
1536 	bt_ctf_field_type_put(pkt_ctx_type);
1537 	if (ret)
1538 		goto err_cleanup;
1539 
1540 	/* CTF clock writer setup */
1541 	if (bt_ctf_writer_add_clock(writer, clock)) {
1542 		pr("Failed to assign CTF clock to writer.\n");
1543 		goto err_cleanup;
1544 	}
1545 
1546 	return 0;
1547 
1548 err_cleanup:
1549 	ctf_writer__cleanup(cw);
1550 err:
1551 	pr_err("Failed to setup CTF writer.\n");
1552 	return -1;
1553 }
1554 
1555 static int ctf_writer__flush_streams(struct ctf_writer *cw)
1556 {
1557 	int cpu, ret = 0;
1558 
1559 	for (cpu = 0; cpu < cw->stream_cnt && !ret; cpu++)
1560 		ret = ctf_stream__flush(cw->stream[cpu]);
1561 
1562 	return ret;
1563 }
1564 
1565 static int convert__config(const char *var, const char *value, void *cb)
1566 {
1567 	struct convert *c = cb;
1568 
1569 	if (!strcmp(var, "convert.queue-size"))
1570 		return perf_config_u64(&c->queue_size, var, value);
1571 
1572 	return 0;
1573 }
1574 
1575 int bt_convert__perf2ctf(const char *input, const char *path,
1576 			 struct perf_data_convert_opts *opts)
1577 {
1578 	struct perf_session *session;
1579 	struct perf_data data = {
1580 		.path	   = input,
1581 		.mode      = PERF_DATA_MODE_READ,
1582 		.force     = opts->force,
1583 	};
1584 	struct convert c = {
1585 		.tool = {
1586 			.sample          = process_sample_event,
1587 			.mmap            = perf_event__process_mmap,
1588 			.mmap2           = perf_event__process_mmap2,
1589 			.comm            = perf_event__process_comm,
1590 			.exit            = perf_event__process_exit,
1591 			.fork            = perf_event__process_fork,
1592 			.lost            = perf_event__process_lost,
1593 			.tracing_data    = perf_event__process_tracing_data,
1594 			.build_id        = perf_event__process_build_id,
1595 			.namespaces      = perf_event__process_namespaces,
1596 			.ordered_events  = true,
1597 			.ordering_requires_timestamps = true,
1598 		},
1599 	};
1600 	struct ctf_writer *cw = &c.writer;
1601 	int err;
1602 
1603 	if (opts->all) {
1604 		c.tool.comm = process_comm_event;
1605 		c.tool.exit = process_exit_event;
1606 		c.tool.fork = process_fork_event;
1607 		c.tool.mmap = process_mmap_event;
1608 		c.tool.mmap2 = process_mmap2_event;
1609 	}
1610 
1611 	err = perf_config(convert__config, &c);
1612 	if (err)
1613 		return err;
1614 
1615 	/* CTF writer */
1616 	if (ctf_writer__init(cw, path))
1617 		return -1;
1618 
1619 	err = -1;
1620 	/* perf.data session */
1621 	session = perf_session__new(&data, 0, &c.tool);
1622 	if (!session)
1623 		goto free_writer;
1624 
1625 	if (c.queue_size) {
1626 		ordered_events__set_alloc_size(&session->ordered_events,
1627 					       c.queue_size);
1628 	}
1629 
1630 	/* CTF writer env/clock setup  */
1631 	if (ctf_writer__setup_env(cw, session))
1632 		goto free_session;
1633 
1634 	/* CTF events setup */
1635 	if (setup_events(cw, session))
1636 		goto free_session;
1637 
1638 	if (opts->all && setup_non_sample_events(cw, session))
1639 		goto free_session;
1640 
1641 	if (setup_streams(cw, session))
1642 		goto free_session;
1643 
1644 	err = perf_session__process_events(session);
1645 	if (!err)
1646 		err = ctf_writer__flush_streams(cw);
1647 	else
1648 		pr_err("Error during conversion.\n");
1649 
1650 	fprintf(stderr,
1651 		"[ perf data convert: Converted '%s' into CTF data '%s' ]\n",
1652 		data.path, path);
1653 
1654 	fprintf(stderr,
1655 		"[ perf data convert: Converted and wrote %.3f MB (%" PRIu64 " samples",
1656 		(double) c.events_size / 1024.0 / 1024.0,
1657 		c.events_count);
1658 
1659 	if (!c.non_sample_count)
1660 		fprintf(stderr, ") ]\n");
1661 	else
1662 		fprintf(stderr, ", %" PRIu64 " non-samples) ]\n", c.non_sample_count);
1663 
1664 	cleanup_events(session);
1665 	perf_session__delete(session);
1666 	ctf_writer__cleanup(cw);
1667 
1668 	return err;
1669 
1670 free_session:
1671 	perf_session__delete(session);
1672 free_writer:
1673 	ctf_writer__cleanup(cw);
1674 	pr_err("Error during conversion setup.\n");
1675 	return err;
1676 }
1677