blob: 701b517ec1ab87473622fdf5e1471f9791d75281 [file] [log] [blame]
Stefan Hajnoczi94a420b2010-05-22 17:52:39 +01001#!/bin/sh
2#
3# Code generator for trace events
4#
5# Copyright IBM, Corp. 2010
6#
7# This work is licensed under the terms of the GNU GPL, version 2. See
8# the COPYING file in the top-level directory.
9
10# Disable pathname expansion, makes processing text with '*' characters simpler
11set -f
12
13usage()
14{
15 cat >&2 <<EOF
Fabien Chouteau320fba22011-01-27 10:24:41 +010016usage: $0 [--nop | --simple | --stderr | --ust | --dtrace] [-h | -c]
Stefan Hajnoczi94a420b2010-05-22 17:52:39 +010017Generate tracing code for a file on stdin.
18
19Backends:
Stefan Hajnoczi26f72272010-05-22 19:24:51 +010020 --nop Tracing disabled
21 --simple Simple built-in backend
Fabien Chouteau320fba22011-01-27 10:24:41 +010022 --stderr Stderr built-in backend
Stefan Hajnoczi7e24e922010-05-22 21:11:33 +010023 --ust LTTng User Space Tracing backend
Daniel P. Berrangeb3d08c02010-11-12 13:20:24 +000024 --dtrace DTrace/SystemTAP backend
Stefan Hajnoczi94a420b2010-05-22 17:52:39 +010025
26Output formats:
Daniel P. Berrangec276b172010-11-12 13:20:25 +000027 -h Generate .h file
28 -c Generate .c file
29 -d Generate .d file (DTrace only)
30 --stap Generate .stp file (DTrace with SystemTAP only)
31
32Options:
Jes Sorensene323c932011-02-17 13:26:05 +010033 --binary [path] Full path to QEMU binary
34 --target-arch [arch] QEMU emulator target arch
35 --target-type [type] QEMU emulator target type ('system' or 'user')
36 --probe-prefix [prefix] Prefix for dtrace probe names
37 (default: qemu-\$targettype-\$targetarch)
Daniel P. Berrangec276b172010-11-12 13:20:25 +000038
Stefan Hajnoczi94a420b2010-05-22 17:52:39 +010039EOF
40 exit 1
41}
42
Stefan Hajnoczi913540a2011-09-13 13:34:35 +010043# Print a line without interpreting backslash escapes
44#
45# The built-in echo command may interpret backslash escapes without an option
46# to disable this behavior.
47puts()
48{
49 printf "%s\n" "$1"
50}
51
Stefan Hajnoczi94a420b2010-05-22 17:52:39 +010052# Get the name of a trace event
53get_name()
54{
Lluís49926042011-08-31 20:31:10 +020055 local name
56 name=${1%%\(*}
57 echo "${name##* }"
58}
59
60# Get the given property of a trace event
61# 1: trace-events line
62# 2: property name
63# -> return 0 if property is present, or 1 otherwise
64has_property()
65{
66 local props prop
67 props=${1%%\(*}
68 props=${props% *}
69 for prop in $props; do
70 if [ "$prop" = "$2" ]; then
71 return 0
72 fi
73 done
74 return 1
Stefan Hajnoczi94a420b2010-05-22 17:52:39 +010075}
76
77# Get the argument list of a trace event, including types and names
78get_args()
79{
80 local args
81 args=${1#*\(}
Paolo Bonzini1a96dd42011-04-15 15:23:59 +020082 args=${args%%\)*}
Stefan Hajnoczi94a420b2010-05-22 17:52:39 +010083 echo "$args"
84}
85
86# Get the argument name list of a trace event
87get_argnames()
88{
Daniel P. Berrangeb3d08c02010-11-12 13:20:24 +000089 local nfields field name sep
Stefan Hajnoczi94a420b2010-05-22 17:52:39 +010090 nfields=0
Daniel P. Berrangeb3d08c02010-11-12 13:20:24 +000091 sep="$2"
Stefan Hajnoczi94a420b2010-05-22 17:52:39 +010092 for field in $(get_args "$1"); do
93 nfields=$((nfields + 1))
94
95 # Drop pointer star
96 field=${field#\*}
97
98 # Only argument names have commas at the end
99 name=${field%,}
100 test "$field" = "$name" && continue
101
Daniel P. Berrangeb3d08c02010-11-12 13:20:24 +0000102 printf "%s%s " $name $sep
Stefan Hajnoczi94a420b2010-05-22 17:52:39 +0100103 done
104
105 # Last argument name
106 if [ "$nfields" -gt 1 ]
107 then
108 printf "%s" "$name"
109 fi
110}
111
Stefan Hajnoczi26f72272010-05-22 19:24:51 +0100112# Get the number of arguments to a trace event
113get_argc()
114{
115 local name argc
116 argc=0
Daniel P. Berrangeb3d08c02010-11-12 13:20:24 +0000117 for name in $(get_argnames "$1", ","); do
Stefan Hajnoczi26f72272010-05-22 19:24:51 +0100118 argc=$((argc + 1))
119 done
120 echo $argc
121}
122
Stefan Hajnoczi913540a2011-09-13 13:34:35 +0100123# Get the format string including double quotes for a trace event
Stefan Hajnoczi94a420b2010-05-22 17:52:39 +0100124get_fmt()
125{
Stefan Hajnoczi913540a2011-09-13 13:34:35 +0100126 puts "${1#*)}"
Stefan Hajnoczi94a420b2010-05-22 17:52:39 +0100127}
128
129linetoh_begin_nop()
130{
131 return
132}
133
134linetoh_nop()
135{
136 local name args
137 name=$(get_name "$1")
138 args=$(get_args "$1")
139
140 # Define an empty function for the trace event
141 cat <<EOF
142static inline void trace_$name($args)
143{
144}
145EOF
146}
147
148linetoh_end_nop()
149{
150 return
151}
152
153linetoc_begin_nop()
154{
155 return
156}
157
158linetoc_nop()
159{
160 # No need for function definitions in nop backend
161 return
162}
163
164linetoc_end_nop()
165{
166 return
167}
168
Stefan Hajnoczi26f72272010-05-22 19:24:51 +0100169linetoh_begin_simple()
170{
171 cat <<EOF
Lluísedb47ec2011-08-31 20:30:57 +0200172#include "trace/simple.h"
Stefan Hajnoczi26f72272010-05-22 19:24:51 +0100173EOF
174
175 simple_event_num=0
176}
177
178cast_args_to_uint64_t()
179{
180 local arg
Daniel P. Berrangeb3d08c02010-11-12 13:20:24 +0000181 for arg in $(get_argnames "$1", ","); do
Stefan Hajnoczi26f72272010-05-22 19:24:51 +0100182 printf "%s" "(uint64_t)(uintptr_t)$arg"
183 done
184}
185
186linetoh_simple()
187{
Lluís49926042011-08-31 20:31:10 +0200188 local name args argc trace_args
Stefan Hajnoczi26f72272010-05-22 19:24:51 +0100189 name=$(get_name "$1")
190 args=$(get_args "$1")
191 argc=$(get_argc "$1")
192
193 trace_args="$simple_event_num"
194 if [ "$argc" -gt 0 ]
195 then
196 trace_args="$trace_args, $(cast_args_to_uint64_t "$1")"
197 fi
198
199 cat <<EOF
200static inline void trace_$name($args)
201{
202 trace$argc($trace_args);
203}
204EOF
205
206 simple_event_num=$((simple_event_num + 1))
207}
208
209linetoh_end_simple()
210{
Prerna Saxena22890ab2010-06-24 17:04:53 +0530211 cat <<EOF
212#define NR_TRACE_EVENTS $simple_event_num
213extern TraceEvent trace_list[NR_TRACE_EVENTS];
214EOF
Stefan Hajnoczi26f72272010-05-22 19:24:51 +0100215}
216
217linetoc_begin_simple()
218{
Prerna Saxena22890ab2010-06-24 17:04:53 +0530219 cat <<EOF
220#include "trace.h"
221
222TraceEvent trace_list[] = {
223EOF
224 simple_event_num=0
225
Stefan Hajnoczi26f72272010-05-22 19:24:51 +0100226}
227
228linetoc_simple()
229{
Lluís03727e62011-08-31 20:31:45 +0200230 local name
Prerna Saxena22890ab2010-06-24 17:04:53 +0530231 name=$(get_name "$1")
232 cat <<EOF
Lluís03727e62011-08-31 20:31:45 +0200233{.tp_name = "$name", .state=0},
Prerna Saxena22890ab2010-06-24 17:04:53 +0530234EOF
235 simple_event_num=$((simple_event_num + 1))
Stefan Hajnoczi26f72272010-05-22 19:24:51 +0100236}
237
238linetoc_end_simple()
239{
Prerna Saxena22890ab2010-06-24 17:04:53 +0530240 cat <<EOF
241};
242EOF
Stefan Hajnoczi26f72272010-05-22 19:24:51 +0100243}
244
Fabien Chouteau320fba22011-01-27 10:24:41 +0100245#STDERR
246linetoh_begin_stderr()
247{
248 cat <<EOF
249#include <stdio.h>
Lluís9a82b6a2011-08-31 20:31:51 +0200250#include "trace/stderr.h"
251
252extern TraceEvent trace_list[];
Fabien Chouteau320fba22011-01-27 10:24:41 +0100253EOF
Lluís9a82b6a2011-08-31 20:31:51 +0200254
255 stderr_event_num=0
Fabien Chouteau320fba22011-01-27 10:24:41 +0100256}
257
258linetoh_stderr()
259{
260 local name args argnames argc fmt
261 name=$(get_name "$1")
262 args=$(get_args "$1")
263 argnames=$(get_argnames "$1" ",")
264 argc=$(get_argc "$1")
265 fmt=$(get_fmt "$1")
266
267 if [ "$argc" -gt 0 ]; then
268 argnames=", $argnames"
269 fi
270
271 cat <<EOF
272static inline void trace_$name($args)
273{
Lluís9a82b6a2011-08-31 20:31:51 +0200274 if (trace_list[$stderr_event_num].state != 0) {
Stefan Hajnoczi913540a2011-09-13 13:34:35 +0100275 fprintf(stderr, "$name " $fmt "\n" $argnames);
Lluís9a82b6a2011-08-31 20:31:51 +0200276 }
Fabien Chouteau320fba22011-01-27 10:24:41 +0100277}
278EOF
Lluís9a82b6a2011-08-31 20:31:51 +0200279 stderr_event_num=$((stderr_event_num + 1))
280
Fabien Chouteau320fba22011-01-27 10:24:41 +0100281}
282
283linetoh_end_stderr()
284{
Lluís9a82b6a2011-08-31 20:31:51 +0200285 cat <<EOF
286#define NR_TRACE_EVENTS $stderr_event_num
287EOF
Fabien Chouteau320fba22011-01-27 10:24:41 +0100288}
289
290linetoc_begin_stderr()
291{
Lluís9a82b6a2011-08-31 20:31:51 +0200292 cat <<EOF
293#include "trace.h"
294
295TraceEvent trace_list[] = {
296EOF
297 stderr_event_num=0
Fabien Chouteau320fba22011-01-27 10:24:41 +0100298}
299
300linetoc_stderr()
301{
Lluís9a82b6a2011-08-31 20:31:51 +0200302 local name
303 name=$(get_name "$1")
304 cat <<EOF
305{.tp_name = "$name", .state=0},
306EOF
307 stderr_event_num=$(($stderr_event_num + 1))
Fabien Chouteau320fba22011-01-27 10:24:41 +0100308}
309
310linetoc_end_stderr()
311{
Lluís9a82b6a2011-08-31 20:31:51 +0200312 cat <<EOF
313};
314EOF
Fabien Chouteau320fba22011-01-27 10:24:41 +0100315}
316#END OF STDERR
317
Stefan Hajnoczi7e24e922010-05-22 21:11:33 +0100318# Clean up after UST headers which pollute the namespace
319ust_clean_namespace() {
320 cat <<EOF
321#undef mutex_lock
322#undef mutex_unlock
323#undef inline
324#undef wmb
325EOF
326}
327
328linetoh_begin_ust()
329{
330 echo "#include <ust/tracepoint.h>"
331 ust_clean_namespace
332}
333
334linetoh_ust()
335{
336 local name args argnames
337 name=$(get_name "$1")
338 args=$(get_args "$1")
Daniel P. Berrangeb3d08c02010-11-12 13:20:24 +0000339 argnames=$(get_argnames "$1", ",")
Stefan Hajnoczi7e24e922010-05-22 21:11:33 +0100340
341 cat <<EOF
Stefan Hajnocziea9c1692010-10-05 14:28:52 +0100342DECLARE_TRACE(ust_$name, TP_PROTO($args), TP_ARGS($argnames));
Stefan Hajnoczi7e24e922010-05-22 21:11:33 +0100343#define trace_$name trace_ust_$name
344EOF
345}
346
347linetoh_end_ust()
348{
349 return
350}
351
352linetoc_begin_ust()
353{
354 cat <<EOF
355#include <ust/marker.h>
356$(ust_clean_namespace)
357#include "trace.h"
358EOF
359}
360
361linetoc_ust()
362{
363 local name args argnames fmt
364 name=$(get_name "$1")
365 args=$(get_args "$1")
Daniel P. Berrangeb3d08c02010-11-12 13:20:24 +0000366 argnames=$(get_argnames "$1", ",")
Lluísfa2d4802011-04-06 20:34:03 +0200367 [ -z "$argnames" ] || argnames=", $argnames"
Stefan Hajnoczi7e24e922010-05-22 21:11:33 +0100368 fmt=$(get_fmt "$1")
369
370 cat <<EOF
371DEFINE_TRACE(ust_$name);
372
373static void ust_${name}_probe($args)
374{
Stefan Hajnoczi913540a2011-09-13 13:34:35 +0100375 trace_mark(ust, $name, $fmt$argnames);
Stefan Hajnoczi7e24e922010-05-22 21:11:33 +0100376}
377EOF
378
379 # Collect names for later
380 names="$names $name"
381}
382
383linetoc_end_ust()
384{
385 cat <<EOF
386static void __attribute__((constructor)) trace_init(void)
387{
388EOF
389
390 for name in $names; do
391 cat <<EOF
392 register_trace_ust_$name(ust_${name}_probe);
393EOF
394 done
395
396 echo "}"
397}
398
Daniel P. Berrangeb3d08c02010-11-12 13:20:24 +0000399linetoh_begin_dtrace()
400{
401 cat <<EOF
402#include "trace-dtrace.h"
403EOF
404}
405
406linetoh_dtrace()
407{
Lluís49926042011-08-31 20:31:10 +0200408 local name args argnames nameupper
Daniel P. Berrangeb3d08c02010-11-12 13:20:24 +0000409 name=$(get_name "$1")
410 args=$(get_args "$1")
411 argnames=$(get_argnames "$1", ",")
Daniel P. Berrangeb3d08c02010-11-12 13:20:24 +0000412
413 nameupper=`echo $name | tr '[:lower:]' '[:upper:]'`
414
415 # Define an empty function for the trace event
416 cat <<EOF
417static inline void trace_$name($args) {
418 if (QEMU_${nameupper}_ENABLED()) {
419 QEMU_${nameupper}($argnames);
420 }
421}
422EOF
423}
424
425linetoh_end_dtrace()
426{
427 return
428}
429
430linetoc_begin_dtrace()
431{
432 return
433}
434
435linetoc_dtrace()
436{
437 # No need for function definitions in dtrace backend
438 return
439}
440
441linetoc_end_dtrace()
442{
443 return
444}
445
446linetod_begin_dtrace()
447{
448 cat <<EOF
449provider qemu {
450EOF
451}
452
453linetod_dtrace()
454{
Lluís49926042011-08-31 20:31:10 +0200455 local name args
Daniel P. Berrangeb3d08c02010-11-12 13:20:24 +0000456 name=$(get_name "$1")
457 args=$(get_args "$1")
Daniel P. Berrangeb3d08c02010-11-12 13:20:24 +0000458
459 # DTrace provider syntax expects foo() for empty
460 # params, not foo(void)
461 if [ "$args" = "void" ]; then
462 args=""
463 fi
464
465 # Define prototype for probe arguments
466 cat <<EOF
467 probe $name($args);
468EOF
469}
470
471linetod_end_dtrace()
472{
473 cat <<EOF
474};
475EOF
476}
477
Daniel P. Berrangec276b172010-11-12 13:20:25 +0000478linetostap_begin_dtrace()
479{
480 return
481}
482
483linetostap_dtrace()
484{
Lluís49926042011-08-31 20:31:10 +0200485 local i arg name args arglist
Daniel P. Berrangec276b172010-11-12 13:20:25 +0000486 name=$(get_name "$1")
487 args=$(get_args "$1")
488 arglist=$(get_argnames "$1", "")
Daniel P. Berrangec276b172010-11-12 13:20:25 +0000489
490 # Define prototype for probe arguments
491 cat <<EOF
Jes Sorensene323c932011-02-17 13:26:05 +0100492probe $probeprefix.$name = process("$binary").mark("$name")
Daniel P. Berrangec276b172010-11-12 13:20:25 +0000493{
494EOF
495
496 i=1
497 for arg in $arglist
498 do
499 # 'limit' is a reserved keyword
500 if [ "$arg" = "limit" ]; then
501 arg="_limit"
502 fi
503 cat <<EOF
504 $arg = \$arg$i;
505EOF
Lluísfa2d4802011-04-06 20:34:03 +0200506 i="$((i+1))"
Daniel P. Berrangec276b172010-11-12 13:20:25 +0000507 done
508
509 cat <<EOF
510}
511EOF
512}
513
514linetostap_end_dtrace()
515{
516 return
517}
518
Stefan Hajnoczi94a420b2010-05-22 17:52:39 +0100519# Process stdin by calling begin, line, and end functions for the backend
520convert()
521{
Lluís Vilanovab7d66a72011-12-06 17:38:15 +0100522 local begin process_line end str name NAME enabled
Stefan Hajnoczi94a420b2010-05-22 17:52:39 +0100523 begin="lineto$1_begin_$backend"
524 process_line="lineto$1_$backend"
525 end="lineto$1_end_$backend"
526
527 "$begin"
528
529 while read -r str; do
530 # Skip comments and empty lines
Stefan Hajnoczi5eb55272010-10-05 14:28:51 +0100531 test -z "${str%%#*}" && continue
Stefan Hajnoczi94a420b2010-05-22 17:52:39 +0100532
533 echo
Lluísdd215f62011-08-31 20:31:38 +0200534 # Process the line. The nop backend handles disabled lines.
535 if has_property "$str" "disable"; then
536 "lineto$1_nop" "$str"
Lluís Vilanovab7d66a72011-12-06 17:38:15 +0100537 enabled=0
Stefan Hajnoczi1e2cf2b2010-05-24 11:32:09 +0100538 else
539 "$process_line" "$str"
Lluís Vilanovab7d66a72011-12-06 17:38:15 +0100540 enabled=1
541 fi
542 if [ "$1" = "h" ]; then
543 name=$(get_name "$str")
544 NAME=$(echo $name | tr '[:lower:]' '[:upper:]')
545 echo "#define TRACE_${NAME}_ENABLED ${enabled}"
Stefan Hajnoczi1e2cf2b2010-05-24 11:32:09 +0100546 fi
Stefan Hajnoczi94a420b2010-05-22 17:52:39 +0100547 done
548
549 echo
550 "$end"
551}
552
553tracetoh()
554{
555 cat <<EOF
556#ifndef TRACE_H
557#define TRACE_H
558
559/* This file is autogenerated by tracetool, do not edit. */
560
561#include "qemu-common.h"
562EOF
563 convert h
564 echo "#endif /* TRACE_H */"
565}
566
567tracetoc()
568{
569 echo "/* This file is autogenerated by tracetool, do not edit. */"
570 convert c
571}
572
Daniel P. Berrangeb3d08c02010-11-12 13:20:24 +0000573tracetod()
574{
575 if [ $backend != "dtrace" ]; then
576 echo "DTrace probe generator not applicable to $backend backend"
577 exit 1
578 fi
579 echo "/* This file is autogenerated by tracetool, do not edit. */"
580 convert d
581}
582
Daniel P. Berrangec276b172010-11-12 13:20:25 +0000583tracetostap()
584{
585 if [ $backend != "dtrace" ]; then
586 echo "SystemTAP tapset generator not applicable to $backend backend"
587 exit 1
588 fi
589 if [ -z "$binary" ]; then
590 echo "--binary is required for SystemTAP tapset generator"
591 exit 1
592 fi
Jes Sorensene323c932011-02-17 13:26:05 +0100593 if [ -z "$probeprefix" -a -z "$targettype" ]; then
Daniel P. Berrangec276b172010-11-12 13:20:25 +0000594 echo "--target-type is required for SystemTAP tapset generator"
595 exit 1
596 fi
Jes Sorensene323c932011-02-17 13:26:05 +0100597 if [ -z "$probeprefix" -a -z "$targetarch" ]; then
Daniel P. Berrangec276b172010-11-12 13:20:25 +0000598 echo "--target-arch is required for SystemTAP tapset generator"
599 exit 1
600 fi
Jes Sorensene323c932011-02-17 13:26:05 +0100601 if [ -z "$probeprefix" ]; then
Lluísfa2d4802011-04-06 20:34:03 +0200602 probeprefix="qemu.$targettype.$targetarch";
Jes Sorensene323c932011-02-17 13:26:05 +0100603 fi
Daniel P. Berrangec276b172010-11-12 13:20:25 +0000604 echo "/* This file is autogenerated by tracetool, do not edit. */"
605 convert stap
606}
Stefan Hajnoczi94a420b2010-05-22 17:52:39 +0100607
Daniel P. Berrangec276b172010-11-12 13:20:25 +0000608
609backend=
610output=
611binary=
612targettype=
613targetarch=
Jes Sorensene323c932011-02-17 13:26:05 +0100614probeprefix=
Daniel P. Berrangec276b172010-11-12 13:20:25 +0000615
616
617until [ -z "$1" ]
618do
619 case "$1" in
Fabien Chouteau320fba22011-01-27 10:24:41 +0100620 "--nop" | "--simple" | "--stderr" | "--ust" | "--dtrace") backend="${1#--}" ;;
Daniel P. Berrangec276b172010-11-12 13:20:25 +0000621
622 "--binary") shift ; binary="$1" ;;
623 "--target-arch") shift ; targetarch="$1" ;;
624 "--target-type") shift ; targettype="$1" ;;
Jes Sorensene323c932011-02-17 13:26:05 +0100625 "--probe-prefix") shift ; probeprefix="$1" ;;
Daniel P. Berrangec276b172010-11-12 13:20:25 +0000626
627 "-h" | "-c" | "-d") output="${1#-}" ;;
628 "--stap") output="${1#--}" ;;
629
630 "--check-backend") exit 0 ;; # used by ./configure to test for backend
631
Fabien Chouteau320fba22011-01-27 10:24:41 +0100632 "--list-backends") # used by ./configure to list available backends
633 echo "nop simple stderr ust dtrace"
634 exit 0
635 ;;
636
Daniel P. Berrangec276b172010-11-12 13:20:25 +0000637 *)
638 usage;;
639 esac
640 shift
641done
642
643if [ "$backend" = "" -o "$output" = "" ]; then
644 usage
645fi
646
647gen="traceto$output"
648"$gen"
Stefan Hajnoczi94a420b2010-05-22 17:52:39 +0100649
650exit 0