1#!/usr/bin/env python3
2#
3# Copyright (c) 2020 Intel Corporation.
4#
5# SPDX-License-Identifier: Apache-2.0
6"""
7Script to parse CTF data and print to the screen in a custom and colorful
8format.
9
10Generate trace using samples/subsys/tracing for example:
11
12    west build -b qemu_x86 samples/subsys/tracing  -t run \
13      -- -DCONF_FILE=prj_uart_ctf.conf
14
15    mkdir ctf
16    cp build/channel0_0 ctf/
17    cp subsys/tracing/ctf/tsdl/metadata ctf/
18    ./scripts/tracing/parse_ctf.py -t ctf
19"""
20
21import sys
22import datetime
23import colorama
24from colorama import Fore
25import argparse
26try:
27    import bt2
28except ImportError:
29    sys.exit("Missing dependency: You need to install python bindings of babeltrace.")
30
31def parse_args():
32    parser = argparse.ArgumentParser(
33            description=__doc__,
34            formatter_class=argparse.RawDescriptionHelpFormatter, allow_abbrev=False)
35    parser.add_argument("-t", "--trace",
36            required=True,
37            help="tracing data (directory with metadata and trace file)")
38    args = parser.parse_args()
39    return args
40
41def main():
42    colorama.init()
43
44    args = parse_args()
45
46    msg_it = bt2.TraceCollectionMessageIterator(args.trace)
47    last_event_ns_from_origin = None
48    timeline = []
49
50    def get_thread(name):
51        for t in timeline:
52            if t.get('name', None) == name and t.get('in', 0 ) != 0 and not t.get('out', None):
53                return t
54        return {}
55
56    for msg in msg_it:
57
58        if not isinstance(msg, bt2._EventMessageConst):
59            continue
60
61        ns_from_origin = msg.default_clock_snapshot.ns_from_origin
62        event = msg.event
63        # Compute the time difference since the last event message.
64        diff_s = 0
65
66        if last_event_ns_from_origin is not None:
67            diff_s = (ns_from_origin - last_event_ns_from_origin) / 1e9
68
69        dt = datetime.datetime.fromtimestamp(ns_from_origin / 1e9)
70
71        if event.name in [
72                'thread_switched_out',
73                'thread_switched_in',
74                'thread_pending',
75                'thread_ready',
76                'thread_resume',
77                'thread_suspend',
78                'thread_create',
79                'thread_abort'
80                ]:
81
82            cpu = event.payload_field.get("cpu", None)
83            thread_id = event.payload_field.get("thread_id", None)
84            thread_name = event.payload_field.get("name", None)
85
86            th = {}
87            if event.name in ['thread_switched_out', 'thread_switched_in'] and cpu is not None:
88                cpu_string = f"(cpu: {cpu})"
89            else:
90                cpu_string = ""
91
92            if thread_name:
93                print(f"{dt} (+{diff_s:.6f} s): {event.name}: {thread_name} {cpu_string}")
94            elif thread_id:
95                print(f"{dt} (+{diff_s:.6f} s): {event.name}: {thread_id} {cpu_string}")
96            else:
97                print(f"{dt} (+{diff_s:.6f} s): {event.name}")
98
99            if event.name in ['thread_switched_out', 'thread_switched_in']:
100                if thread_name:
101                    th = get_thread(thread_name)
102                    if not th:
103                        th['name'] = thread_name
104                else:
105                    th = get_thread(thread_id)
106                    if not th:
107                        th['name'] = thread_id
108
109                if event.name in ['thread_switched_out']:
110                    th['out'] = ns_from_origin
111                    tin = th.get('in', None)
112                    tout = th.get('out', None)
113                    if tout is not None and tin is not None:
114                        diff = tout - tin
115                        th['runtime'] = diff
116                elif event.name in ['thread_switched_in']:
117                    th['in'] = ns_from_origin
118
119                    timeline.append(th)
120
121        elif event.name in ['thread_info']:
122            stack_size = event.payload_field['stack_size']
123            print(f"{dt} (+{diff_s:.6f} s): {event.name} (Stack size: {stack_size})")
124        elif event.name in ['start_call', 'end_call']:
125            if event.payload_field['id'] == 39:
126                c = Fore.GREEN
127            elif event.payload_field['id'] in [37, 38]:
128                c = Fore.CYAN
129            else:
130                c = Fore.YELLOW
131            print(c + f"{dt} (+{diff_s:.6f} s): {event.name} {event.payload_field['id']}" + Fore.RESET)
132        elif event.name in ['semaphore_init', 'semaphore_take', 'semaphore_give']:
133            c = Fore.CYAN
134            print(c + f"{dt} (+{diff_s:.6f} s): {event.name} ({event.payload_field['id']})" + Fore.RESET)
135        elif event.name in ['mutex_init', 'mutex_take', 'mutex_give']:
136            c = Fore.MAGENTA
137            print(c + f"{dt} (+{diff_s:.6f} s): {event.name} ({event.payload_field['id']})" + Fore.RESET)
138
139        elif event.name in ['named_event']:
140            name = event.payload_field['name']
141            arg0 = event.payload_field['arg0']
142            arg1 = event.payload_field['arg1']
143            print(f"{dt} (+{diff_s:.6f} s): {event.name} (name: {name}, arg0: {arg0} arg1: {arg1})")
144        else:
145            print(f"{dt} (+{diff_s:.6f} s): {event.name}")
146
147        last_event_ns_from_origin = ns_from_origin
148
149if __name__=="__main__":
150    main()
151