Commit | Line | Data |
---|---|---|
f984b51e PP |
1 | /* |
2 | * Memory mapped I/O tracing | |
3 | * | |
4 | * Copyright (C) 2008 Pekka Paalanen <pq@iki.fi> | |
5 | */ | |
6 | ||
7 | #define DEBUG 1 | |
8 | ||
9 | #include <linux/kernel.h> | |
10 | #include <linux/mmiotrace.h> | |
11 | ||
12 | #include "trace.h" | |
13 | ||
14 | extern void | |
15 | __trace_special(void *__tr, void *__data, | |
16 | unsigned long arg1, unsigned long arg2, unsigned long arg3); | |
17 | ||
18 | static struct trace_array *mmio_trace_array; | |
19 | ||
20 | ||
21 | static void mmio_trace_init(struct trace_array *tr) | |
22 | { | |
23 | pr_debug("in %s\n", __func__); | |
24 | mmio_trace_array = tr; | |
25 | if (tr->ctrl) | |
26 | enable_mmiotrace(); | |
27 | } | |
28 | ||
29 | static void mmio_trace_reset(struct trace_array *tr) | |
30 | { | |
31 | pr_debug("in %s\n", __func__); | |
32 | if (tr->ctrl) | |
33 | disable_mmiotrace(); | |
34 | } | |
35 | ||
36 | static void mmio_trace_ctrl_update(struct trace_array *tr) | |
37 | { | |
38 | pr_debug("in %s\n", __func__); | |
39 | if (tr->ctrl) | |
40 | enable_mmiotrace(); | |
41 | else | |
42 | disable_mmiotrace(); | |
43 | } | |
44 | ||
45 | static struct tracer mmio_tracer __read_mostly = | |
46 | { | |
47 | .name = "mmiotrace", | |
48 | .init = mmio_trace_init, | |
49 | .reset = mmio_trace_reset, | |
50 | .ctrl_update = mmio_trace_ctrl_update, | |
51 | }; | |
52 | ||
53 | __init static int init_mmio_trace(void) | |
54 | { | |
55 | int ret = init_mmiotrace(); | |
56 | if (ret) | |
57 | return ret; | |
58 | return register_tracer(&mmio_tracer); | |
59 | } | |
60 | device_initcall(init_mmio_trace); | |
61 | ||
62 | void mmio_trace_record(u32 type, unsigned long addr, unsigned long arg) | |
63 | { | |
64 | struct trace_array *tr = mmio_trace_array; | |
65 | struct trace_array_cpu *data = tr->data[smp_processor_id()]; | |
66 | ||
67 | if (!current || current->pid == 0) { | |
68 | /* | |
69 | * XXX: This is a problem. We need to able to record, no | |
70 | * matter what. tracing_generic_entry_update() would crash. | |
71 | */ | |
72 | static unsigned limit; | |
73 | if (limit++ < 12) | |
74 | pr_err("Error in %s: no current.\n", __func__); | |
75 | return; | |
76 | } | |
77 | if (!tr || !data) { | |
78 | static unsigned limit; | |
79 | if (limit++ < 12) | |
80 | pr_err("%s: no tr or data\n", __func__); | |
81 | return; | |
82 | } | |
83 | __trace_special(tr, data, type, addr, arg); | |
84 | } |