blob: 84246f7038759ed501eeacf8be3e36ca68d9aad0 [file] [log] [blame]
Pekka Paalanenc6c67c12008-05-12 21:20:59 +02001 In-kernel memory-mapped I/O tracing
2
3
4Home page and links to optional user space tools:
5
6 http://nouveau.freedesktop.org/wiki/MmioTrace
7
8MMIO tracing was originally developed by Intel around 2003 for their Fault
9Injection Test Harness. In Dec 2006 - Jan 2007, using the code from Intel,
10Jeff Muizelaar created a tool for tracing MMIO accesses with the Nouveau
11project in mind. Since then many people have contributed.
12
13Mmiotrace was built for reverse engineering any memory-mapped IO device with
14the Nouveau project as the first real user. Only x86 and x86_64 architectures
15are supported.
16
17Out-of-tree mmiotrace was originally modified for mainline inclusion and
18ftrace framework by Pekka Paalanen <pq@iki.fi>.
19
20
21Preparation
22-----------
23
24Mmiotrace feature is compiled in by the CONFIG_MMIOTRACE option. Tracing is
25disabled by default, so it is safe to have this set to yes. SMP systems are
26supported, but tracing is unreliable and may miss events if more than one CPU
27is on-line, therefore mmiotrace takes all but one CPU off-line during run-time
28activation [not implemented].
29
30
31Usage Quick Reference
32---------------------
33
34$ mount -t debugfs debugfs /debug
35$ echo mmiotrace > /debug/tracing/current_tracer
36$ cat /debug/tracing/trace_pipe > mydump.txt &
37Start X or whatever.
38$ echo "X is up" > /debug/tracing/marker
39$ echo none > /debug/tracing/current_tracer
40Check kernel log.
41
42
43Usage
44-----
45
46Make sure debugfs is mounted to /debug. If not, (requires root privileges)
47$ mount -t debugfs debugfs /debug
48
49Check that the driver you are about to trace is not loaded.
50
51Activate mmiotrace (requires root privileges):
52$ echo mmiotrace > /debug/tracing/current_tracer
53
54Start storing the trace:
55$ cat /debug/tracing/trace_pipe > mydump.txt &
56The 'cat' process should stay running (sleeping) in the background.
57
58Load the driver you want to trace and use it. Mmiotrace will only catch MMIO
59accesses to areas that are ioremapped while mmiotrace is active.
60
61[Unimplemented feature:]
62During tracing you can place comments (markers) into the trace by
63$ echo "X is up" > /debug/tracing/marker
64This makes it easier to see which part of the (huge) trace corresponds to
65which action. It is recommended to place descriptive markers about what you
66do.
67
68Shut down mmiotrace (requires root privileges):
69$ echo none > /debug/tracing/current_tracer
70The 'cat' process exits. If it does not, kill it by 'fg' and pressing ctrl+c.
71
72[This feature is not implemented yet!]
73Check your kernel log. If there are messages about mmiotrace losing events,
74this is due to buffer overrun, and the trace is incomplete. You should enlarge
75the buffers and try again. [How?]
76
77If you are doing a trace for a driver project, e.g. Nouveau, you should also
78do the following before sending your results:
79$ lspci -vvv > lspci.txt
80$ dmesg > dmesg.txt
81$ tar zcf pciid-nick-mmiotrace.tar.gz mydump.txt lspci.txt dmesg.txt
82and then send the .tar.gz file. The trace compresses considerably. Replace
83"pciid" and "nick" with the PCI ID or model name of your piece of hardware
84under investigation and your nick name.
85
86
87How Mmiotrace Works
88-------------------
89
90Access to hardware IO-memory is gained by mapping addresses from PCI bus by
91calling one of the ioremap_*() functions. Mmiotrace is hooked into the
92__ioremap() function and gets called whenever a mapping is created. Mapping is
93an event that is recorded into the trace log. Note, that ISA range mappings
94are not caught, since the mapping always exists and is returned directly.
95
96MMIO accesses are recorded via page faults. Just before __ioremap() returns,
97the mapped pages are marked as not present. Any access to the pages causes a
98fault. The page fault handler calls mmiotrace to handle the fault. Mmiotrace
99marks the page present, sets TF flag to achieve single stepping and exits the
100fault handler. The instruction that faulted is executed and debug trap is
101entered. Here mmiotrace again marks the page as not present. The instruction
102is decoded to get the type of operation (read/write), data width and the value
103read or written. These are stored to the trace log.
104
105Setting the page present in the page fault handler has a race condition on SMP
106machines. During the single stepping other CPUs may run freely on that page
107and events can be missed without a notice. Re-enabling other CPUs during
108tracing is discouraged.
109
110
111Trace Log Format
112----------------
113
114The raw log is text and easily filtered with e.g. grep and awk. One record is
115one line in the log. A record starts with a keyword, followed by keyword
116dependant arguments. Arguments are separated by a space, or continue until the
117end of line. The format for version 20070824 is as follows:
118
119Explanation Keyword Space separated arguments
120---------------------------------------------------------------------------
121
122read event R width, timestamp, map id, physical, value, PC, PID
123write event W width, timestamp, map id, physical, value, PC, PID
124ioremap event MAP timestamp, map id, physical, virtual, length, PC, PID
125iounmap event UNMAP timestamp, map id, PC, PID
126marker MARK timestamp, text
127version VERSION the string "20070824"
128info for reader LSPCI one line from lspci -v
129PCI address map PCIDEV space separated /proc/bus/pci/devices data
130unk. opcode UNKNOWN timestamp, map id, physical, data, PC, PID
131
132Timestamp is in seconds with decimals. Physical is a PCI bus address, virtual
133is a kernel virtual address. Width is the data width in bytes and value is the
134data value. Map id is an arbitrary id number identifying the mapping that was
135used in an operation. PC is the program counter and PID is process id. PC is
136zero if it is not recorded. PID is always zero as tracing MMIO accesses
137originating in user space memory is not yet supported.
138
139For instance, the following awk filter will pass all 32-bit writes that target
140physical addresses in the range [0xfb73ce40, 0xfb800000[
141
142$ awk '/W 4 / { adr=strtonum($5); if (adr >= 0xfb73ce40 &&
143adr < 0xfb800000) print; }'
144
145
146Tools for Developers
147--------------------
148
149The user space tools include utilities for:
150- replacing numeric addresses and values with hardware register names
151- replaying MMIO logs, i.e., re-executing the recorded writes
152
153