Site logo
Stories around the Genode Operating System RSS feed
Sebastian Sumpf avatar

Simple tracing within Qemu


I recently discovered Qemu's tracing feature. It is useful for examining the state of device models, like interrupt controllers or the CPU, or for observing the behavior of guest operating systems. As a simple example, lets enable tracing for ARM's interrupt controller called GIC. The device model resides in Qemu in the file hw/intc/arm_gic.c. In order to find out which trace events exist one can either grep for trace- or examine the file trace_events which resides in each directory that supports tracing. The file hw/intc/trace_events contains the following entries for ARM's GIC:

# hw/intc/arm_gic.c
gic_enable_irq(int irq) "irq %d enabled"
gic_disable_irq(int irq) "irq %d disabled"
gic_set_irq(int irq, int level, int cpumask, int target) "irq %d level %d cpumask 0x%x target 0x%x"
gic_update_bestirq(const char *s, int cpu, int irq, int prio, int priority_mask, int running_priority) "%s %d irq %d priority %d cpu priority mask %d cpu running priority %d"
gic_update_set_irq(int cpu, const char *name, int level) "cpu[%d]: %s = %d"
gic_acknowledge_irq(const char *s, int cpu, int irq) "%s %d acknowledged irq %d"
gic_cpu_write(const char *s, int cpu, int addr, uint32_t val) "%s %d iface write at 0x%08x 0x%08" PRIx32
gic_cpu_read(const char *s, int cpu, int addr, uint32_t val) "%s %d iface read at 0x%08x: 0x%08" PRIx32
gic_hyp_read(int addr, uint32_t val) "hyp read at 0x%08x: 0x%08" PRIx32
gic_hyp_write(int addr, uint32_t val) "hyp write at 0x%08x: 0x%08" PRIx32
gic_dist_read(int addr, unsigned int size, uint32_t val) "dist read at 0x%08x size %u: 0x%08" PRIx32
gic_dist_write(int addr, unsigned int size, uint32_t val) "dist write at 0x%08x size %u: 0x%08" PRIx32
gic_lr_entry(int cpu, int entry, uint32_t val) "cpu %d: new lr entry %d: 0x%08" PRIx32
gic_update_maintenance_irq(int cpu, int val) "cpu %d: maintenance = %d"

Additionally,

qemu-system-arm --trace help

can be used to print all available trace events.

I will choose to trace gic_cpu_read and gic_cpu_write by writing the event names to a file:

echo gic_cpu_read > /tmp/events
echo gic_cpu_write >> /tmp/events

To enable tracing on Genode, add the following to QEMU_OPT in your build.conf file

QEMU_OPT += --trace events=/tmp/events

and execute, for example, a Genode run script for an ARM board:

 cd <Genode directory>/build/arm_v7
 make KERNEL=hw BOARD=pbxa9 run/log

This will produce the following tracing output:

15452@1572600542.840985:gic_cpu_read cpu 0 iface read at 0x00000004: 0x00000000
15452@1572600542.841024:gic_cpu_write cpu 0 iface write at 0x00000004 0x000000ff
15452@1572600542.841034:gic_cpu_read cpu 0 iface read at 0x00000008: 0x00000000
15452@1572600542.841063:gic_cpu_write cpu 0 iface write at 0x00000008 0x00000007
15452@1572600542.841071:gic_cpu_write cpu 0 iface write at 0x00000000 0x00000001
15452@1572600542.865729:gic_cpu_read cpu 0 iface read at 0x0000000c: 0x0000001d
15452@1572600542.865816:gic_cpu_write cpu 0 iface write at 0x00000010 0x0000001d
15452@1572600542.896840:gic_cpu_read cpu 0 iface read at 0x0000000c: 0x0000001d
15452@1572600542.896878:gic_cpu_write cpu 0 iface write at 0x00000010 0x0000001d

Qemu also supports wildcards.

 echo gic_* > /tmp/events

will enable all GIC trace points and produce more output like:

16177@1572600951.108290:gic_set_irq irq 29 level 1 cpumask 0x1 target 0x1
16177@1572600951.108303:gic_update_bestirq cpu 0 irq 29 priority 0 cpu priority mask 255 cpu running priority 256
16177@1572600951.108310:gic_update_set_irq cpu[0]: irq = 1
16177@1572600951.108392:gic_acknowledge_irq cpu 0 acknowledged irq 29
16177@1572600951.108403:gic_cpu_read cpu 0 iface read at 0x0000000c: 0x0000001d
16177@1572600951.108416:gic_cpu_write cpu 0 iface write at 0x00000010 0x0000001d
16177@1572600951.108424:gic_update_bestirq cpu 0 irq 29 priority 0 cpu priority mask 255 cpu running priority 256
16177@1572600951.108431:gic_update_set_irq cpu[0]: irq = 1
16177@1572600951.108493:gic_set_irq irq 29 level 0 cpumask 0x1 target 0x1
16177@1572600951.118627:gic_set_irq irq 29 level 1 cpumask 0x1 target 0x1
16177@1572600951.118637:gic_update_bestirq cpu 0 irq 29 priority 0 cpu priority mask 255 cpu running priority 256
16177@1572600951.118644:gic_update_set_irq cpu[0]: irq = 1
16177@1572600951.118726:gic_acknowledge_irq cpu 0 acknowledged irq 29
16177@1572600951.118737:gic_cpu_read cpu 0 iface read at 0x0000000c: 0x0000001d
16177@1572600951.118748:gic_cpu_write cpu 0 iface write at 0x00000010 0x0000001d
16177@1572600951.118755:gic_update_bestirq cpu 0 irq 29 priority 0 cpu priority mask 255 cpu running priority 256
16177@1572600951.118761:gic_update_set_irq cpu[0]: irq = 1
16177@1572600951.118802:gic_set_irq irq 29 level 0 cpumask 0x1 target 0x1
16177@1572600951.134421:gic_set_irq irq 29 level 1 cpumask 0x1 target 0x1
16177@1572600951.134432:gic_update_bestirq cpu 0 irq 29 priority 0 cpu priority mask 255 cpu running priority 256
16177@1572600951.134439:gic_update_set_irq cpu[0]: irq = 1
16177@1572600951.134523:gic_acknowledge_irq cpu 0 acknowledged irq 29
16177@1572600951.134534:gic_cpu_read cpu 0 iface read at 0x0000000c: 0x0000001d
16177@1572600951.134545:gic_cpu_write cpu 0 iface write at 0x00000010 0x0000001d
16177@1572600951.134552:gic_update_bestirq cpu 0 irq 29 priority 0 cpu priority mask 255 cpu running priority 256
16177@1572600951.134559:gic_update_set_irq cpu[0]: irq = 1
16177@1572600951.134601:gic_set_irq irq 29 level 0 cpumask 0x1 target 0x1
16177@1572600951.188595:gic_set_irq irq 29 level 1 cpumask 0x1 target 0x1
16177@1572600951.188815:gic_update_bestirq cpu 0 irq 29 priority 0 cpu priority mask 255 cpu running priority 256

Some Qemu builds might not have tracing enabled per default. In this case Qemu needs to be rebuild with the following options:

 configure --enable-trace-backends=simple
 make