blob: 70dac199b093d6727db43ac652aa92c26e41eb20 [file] [log] [blame]
Markus Metzgereee3af42008-01-30 13:31:09 +01001/*
2 * Debug Store (DS) support
3 *
4 * This provides a low-level interface to the hardware's Debug Store
Markus Metzger93fa7632008-04-08 11:01:58 +02005 * feature that is used for branch trace store (BTS) and
Markus Metzgereee3af42008-01-30 13:31:09 +01006 * precise-event based sampling (PEBS).
7 *
Markus Metzger93fa7632008-04-08 11:01:58 +02008 * It manages:
Markus Metzgerc2724772008-12-11 13:49:59 +01009 * - DS and BTS hardware configuration
Markus Metzger6abb11a2008-11-25 09:05:27 +010010 * - buffer overflow handling (to be done)
Markus Metzger93fa7632008-04-08 11:01:58 +020011 * - buffer access
12 *
Markus Metzgerc2724772008-12-11 13:49:59 +010013 * It does not do:
14 * - security checking (is the caller allowed to trace the task)
15 * - buffer allocation (memory accounting)
Markus Metzgereee3af42008-01-30 13:31:09 +010016 *
17 *
Markus Metzgerde79f542009-04-03 16:43:40 +020018 * Copyright (C) 2007-2009 Intel Corporation.
19 * Markus Metzger <markus.t.metzger@intel.com>, 2007-2009
Markus Metzgereee3af42008-01-30 13:31:09 +010020 */
21
H. Peter Anvin1965aae2008-10-22 22:26:29 -070022#ifndef _ASM_X86_DS_H
23#define _ASM_X86_DS_H
Markus Metzgereee3af42008-01-30 13:31:09 +010024
Markus Metzger93fa7632008-04-08 11:01:58 +020025
Markus Metzgereee3af42008-01-30 13:31:09 +010026#include <linux/types.h>
27#include <linux/init.h>
Markus Metzgerca0002a2008-11-25 09:01:25 +010028#include <linux/err.h>
Markus Metzgereee3af42008-01-30 13:31:09 +010029
Markus Metzgereee3af42008-01-30 13:31:09 +010030
Markus Metzgere5e8ca62008-11-25 08:47:19 +010031#ifdef CONFIG_X86_DS
32
Markus Metzger93fa7632008-04-08 11:01:58 +020033struct task_struct;
Markus Metzgerc2724772008-12-11 13:49:59 +010034struct ds_context;
Markus Metzgerca0002a2008-11-25 09:01:25 +010035struct ds_tracer;
36struct bts_tracer;
37struct pebs_tracer;
38
39typedef void (*bts_ovfl_callback_t)(struct bts_tracer *);
40typedef void (*pebs_ovfl_callback_t)(struct pebs_tracer *);
Markus Metzgereee3af42008-01-30 13:31:09 +010041
Markus Metzgerc2724772008-12-11 13:49:59 +010042
43/*
44 * A list of features plus corresponding macros to talk about them in
45 * the ds_request function's flags parameter.
46 *
47 * We use the enum to index an array of corresponding control bits;
48 * we use the macro to index a flags bit-vector.
49 */
50enum ds_feature {
51 dsf_bts = 0,
52 dsf_bts_kernel,
53#define BTS_KERNEL (1 << dsf_bts_kernel)
54 /* trace kernel-mode branches */
55
56 dsf_bts_user,
57#define BTS_USER (1 << dsf_bts_user)
58 /* trace user-mode branches */
59
60 dsf_bts_overflow,
61 dsf_bts_max,
62 dsf_pebs = dsf_bts_max,
63
64 dsf_pebs_max,
65 dsf_ctl_max = dsf_pebs_max,
66 dsf_bts_timestamps = dsf_ctl_max,
67#define BTS_TIMESTAMPS (1 << dsf_bts_timestamps)
68 /* add timestamps into BTS trace */
69
70#define BTS_USER_FLAGS (BTS_KERNEL | BTS_USER | BTS_TIMESTAMPS)
71};
72
73
Markus Metzger93fa7632008-04-08 11:01:58 +020074/*
75 * Request BTS or PEBS
Markus Metzgereee3af42008-01-30 13:31:09 +010076 *
Markus Metzger93fa7632008-04-08 11:01:58 +020077 * Due to alignement constraints, the actual buffer may be slightly
78 * smaller than the requested or provided buffer.
79 *
Markus Metzgerca0002a2008-11-25 09:01:25 +010080 * Returns a pointer to a tracer structure on success, or
81 * ERR_PTR(errcode) on failure.
82 *
83 * The interrupt threshold is independent from the overflow callback
84 * to allow users to use their own overflow interrupt handling mechanism.
Markus Metzger93fa7632008-04-08 11:01:58 +020085 *
Markus Metzgerde79f542009-04-03 16:43:40 +020086 * The function might sleep.
87 *
88 * task: the task to request recording for
89 * cpu: the cpu to request recording for
Markus Metzger93fa7632008-04-08 11:01:58 +020090 * base: the base pointer for the (non-pageable) buffer;
Markus Metzger6abb11a2008-11-25 09:05:27 +010091 * size: the size of the provided buffer in bytes
Markus Metzger93fa7632008-04-08 11:01:58 +020092 * ovfl: pointer to a function to be called on buffer overflow;
93 * NULL if cyclic buffer requested
Markus Metzgerca0002a2008-11-25 09:01:25 +010094 * th: the interrupt threshold in records from the end of the buffer;
95 * -1 if no interrupt threshold is requested.
Markus Metzgerc2724772008-12-11 13:49:59 +010096 * flags: a bit-mask of the above flags
Markus Metzgereee3af42008-01-30 13:31:09 +010097 */
Markus Metzgerde79f542009-04-03 16:43:40 +020098extern struct bts_tracer *ds_request_bts_task(struct task_struct *task,
99 void *base, size_t size,
100 bts_ovfl_callback_t ovfl,
101 size_t th, unsigned int flags);
102extern struct bts_tracer *ds_request_bts_cpu(int cpu, void *base, size_t size,
103 bts_ovfl_callback_t ovfl,
104 size_t th, unsigned int flags);
105extern struct pebs_tracer *ds_request_pebs_task(struct task_struct *task,
106 void *base, size_t size,
107 pebs_ovfl_callback_t ovfl,
108 size_t th, unsigned int flags);
109extern struct pebs_tracer *ds_request_pebs_cpu(int cpu,
110 void *base, size_t size,
111 pebs_ovfl_callback_t ovfl,
112 size_t th, unsigned int flags);
Markus Metzger93fa7632008-04-08 11:01:58 +0200113
114/*
115 * Release BTS or PEBS resources
Markus Metzgerc2724772008-12-11 13:49:59 +0100116 * Suspend and resume BTS or PEBS tracing
Markus Metzger93fa7632008-04-08 11:01:58 +0200117 *
Markus Metzgerde79f542009-04-03 16:43:40 +0200118 * Must be called with irq's enabled.
119 *
Markus Metzgerca0002a2008-11-25 09:01:25 +0100120 * tracer: the tracer handle returned from ds_request_~()
Markus Metzger93fa7632008-04-08 11:01:58 +0200121 */
Markus Metzgerc2724772008-12-11 13:49:59 +0100122extern void ds_release_bts(struct bts_tracer *tracer);
123extern void ds_suspend_bts(struct bts_tracer *tracer);
124extern void ds_resume_bts(struct bts_tracer *tracer);
125extern void ds_release_pebs(struct pebs_tracer *tracer);
126extern void ds_suspend_pebs(struct pebs_tracer *tracer);
127extern void ds_resume_pebs(struct pebs_tracer *tracer);
128
Markus Metzgerde79f542009-04-03 16:43:40 +0200129/*
130 * Release BTS or PEBS resources
131 * Suspend and resume BTS or PEBS tracing
132 *
133 * Cpu tracers must call this on the traced cpu.
134 * Task tracers must call ds_release_~_noirq() for themselves.
135 *
136 * May be called with irq's disabled.
137 *
138 * Returns 0 if successful;
139 * -EPERM if the cpu tracer does not trace the current cpu.
140 * -EPERM if the task tracer does not trace itself.
141 *
142 * tracer: the tracer handle returned from ds_request_~()
143 */
144extern int ds_release_bts_noirq(struct bts_tracer *tracer);
145extern int ds_suspend_bts_noirq(struct bts_tracer *tracer);
146extern int ds_resume_bts_noirq(struct bts_tracer *tracer);
147extern int ds_release_pebs_noirq(struct pebs_tracer *tracer);
148extern int ds_suspend_pebs_noirq(struct pebs_tracer *tracer);
149extern int ds_resume_pebs_noirq(struct pebs_tracer *tracer);
150
Markus Metzger93fa7632008-04-08 11:01:58 +0200151
152/*
Markus Metzgerc2724772008-12-11 13:49:59 +0100153 * The raw DS buffer state as it is used for BTS and PEBS recording.
Markus Metzger93fa7632008-04-08 11:01:58 +0200154 *
Markus Metzgerc2724772008-12-11 13:49:59 +0100155 * This is the low-level, arch-dependent interface for working
156 * directly on the raw trace data.
Markus Metzger93fa7632008-04-08 11:01:58 +0200157 */
Markus Metzgerc2724772008-12-11 13:49:59 +0100158struct ds_trace {
159 /* the number of bts/pebs records */
160 size_t n;
161 /* the size of a bts/pebs record in bytes */
162 size_t size;
163 /* pointers into the raw buffer:
164 - to the first entry */
165 void *begin;
166 /* - one beyond the last entry */
167 void *end;
168 /* - one beyond the newest entry */
169 void *top;
170 /* - the interrupt threshold */
171 void *ith;
172 /* flags given on ds_request() */
173 unsigned int flags;
174};
Markus Metzger93fa7632008-04-08 11:01:58 +0200175
176/*
Markus Metzgerc2724772008-12-11 13:49:59 +0100177 * An arch-independent view on branch trace data.
Markus Metzger93fa7632008-04-08 11:01:58 +0200178 */
Markus Metzgerc2724772008-12-11 13:49:59 +0100179enum bts_qualifier {
180 bts_invalid,
181#define BTS_INVALID bts_invalid
182
183 bts_branch,
184#define BTS_BRANCH bts_branch
185
186 bts_task_arrives,
187#define BTS_TASK_ARRIVES bts_task_arrives
188
189 bts_task_departs,
190#define BTS_TASK_DEPARTS bts_task_departs
191
192 bts_qual_bit_size = 4,
193 bts_qual_max = (1 << bts_qual_bit_size),
194};
195
196struct bts_struct {
197 __u64 qualifier;
198 union {
199 /* BTS_BRANCH */
200 struct {
201 __u64 from;
202 __u64 to;
203 } lbr;
204 /* BTS_TASK_ARRIVES or BTS_TASK_DEPARTS */
205 struct {
Markus Metzger15879d02009-04-03 16:43:38 +0200206 __u64 clock;
Markus Metzgerc2724772008-12-11 13:49:59 +0100207 pid_t pid;
Markus Metzger15879d02009-04-03 16:43:38 +0200208 } event;
Markus Metzgerc2724772008-12-11 13:49:59 +0100209 } variant;
210};
211
Markus Metzger93fa7632008-04-08 11:01:58 +0200212
213/*
Markus Metzgerc2724772008-12-11 13:49:59 +0100214 * The BTS state.
Markus Metzger93fa7632008-04-08 11:01:58 +0200215 *
Markus Metzgerc2724772008-12-11 13:49:59 +0100216 * This gives access to the raw DS state and adds functions to provide
217 * an arch-independent view of the BTS data.
Markus Metzger93fa7632008-04-08 11:01:58 +0200218 */
Markus Metzgerc2724772008-12-11 13:49:59 +0100219struct bts_trace {
220 struct ds_trace ds;
221
222 int (*read)(struct bts_tracer *tracer, const void *at,
223 struct bts_struct *out);
224 int (*write)(struct bts_tracer *tracer, const struct bts_struct *in);
225};
226
Markus Metzger93fa7632008-04-08 11:01:58 +0200227
228/*
Markus Metzgerc2724772008-12-11 13:49:59 +0100229 * The PEBS state.
Markus Metzger93fa7632008-04-08 11:01:58 +0200230 *
Markus Metzgerc2724772008-12-11 13:49:59 +0100231 * This gives access to the raw DS state and the PEBS-specific counter
232 * reset value.
233 */
234struct pebs_trace {
235 struct ds_trace ds;
236
Markus Metzger017bc612009-04-03 16:43:52 +0200237 /* the number of valid counters in the below array */
238 unsigned int counters;
239
240#define MAX_PEBS_COUNTERS 4
241 /* the counter reset value */
242 unsigned long long counter_reset[MAX_PEBS_COUNTERS];
Markus Metzgerc2724772008-12-11 13:49:59 +0100243};
244
245
246/*
247 * Read the BTS or PEBS trace.
Markus Metzger93fa7632008-04-08 11:01:58 +0200248 *
Markus Metzgerc2724772008-12-11 13:49:59 +0100249 * Returns a view on the trace collected for the parameter tracer.
Markus Metzger93fa7632008-04-08 11:01:58 +0200250 *
Markus Metzgerc2724772008-12-11 13:49:59 +0100251 * The view remains valid as long as the traced task is not running or
252 * the tracer is suspended.
253 * Writes into the trace buffer are not reflected.
Markus Metzger93fa7632008-04-08 11:01:58 +0200254 *
Markus Metzgerca0002a2008-11-25 09:01:25 +0100255 * tracer: the tracer handle returned from ds_request_~()
Markus Metzger93fa7632008-04-08 11:01:58 +0200256 */
Markus Metzgerc2724772008-12-11 13:49:59 +0100257extern const struct bts_trace *ds_read_bts(struct bts_tracer *tracer);
258extern const struct pebs_trace *ds_read_pebs(struct pebs_tracer *tracer);
259
Markus Metzger93fa7632008-04-08 11:01:58 +0200260
261/*
Markus Metzger93fa7632008-04-08 11:01:58 +0200262 * Reset the write pointer of the BTS/PEBS buffer.
263 *
264 * Returns 0 on success; -Eerrno on error
265 *
Markus Metzgerca0002a2008-11-25 09:01:25 +0100266 * tracer: the tracer handle returned from ds_request_~()
Markus Metzger93fa7632008-04-08 11:01:58 +0200267 */
Markus Metzgerca0002a2008-11-25 09:01:25 +0100268extern int ds_reset_bts(struct bts_tracer *tracer);
269extern int ds_reset_pebs(struct pebs_tracer *tracer);
Markus Metzger93fa7632008-04-08 11:01:58 +0200270
271/*
Markus Metzger93fa7632008-04-08 11:01:58 +0200272 * Set the PEBS counter reset value.
273 *
274 * Returns 0 on success; -Eerrno on error
275 *
Markus Metzgerca0002a2008-11-25 09:01:25 +0100276 * tracer: the tracer handle returned from ds_request_pebs()
Markus Metzger017bc612009-04-03 16:43:52 +0200277 * counter: the index of the counter
Markus Metzger93fa7632008-04-08 11:01:58 +0200278 * value: the new counter reset value
279 */
Markus Metzger017bc612009-04-03 16:43:52 +0200280extern int ds_set_pebs_reset(struct pebs_tracer *tracer,
281 unsigned int counter, u64 value);
Markus Metzger93fa7632008-04-08 11:01:58 +0200282
283/*
284 * Initialization
285 */
286struct cpuinfo_x86;
287extern void __cpuinit ds_init_intel(struct cpuinfo_x86 *);
288
Markus Metzger93fa7632008-04-08 11:01:58 +0200289/*
Markus Metzgerc2724772008-12-11 13:49:59 +0100290 * Context switch work
Markus Metzger93fa7632008-04-08 11:01:58 +0200291 */
Markus Metzgerc2724772008-12-11 13:49:59 +0100292extern void ds_switch_to(struct task_struct *prev, struct task_struct *next);
Markus Metzgereee3af42008-01-30 13:31:09 +0100293
Markus Metzger93fa7632008-04-08 11:01:58 +0200294#else /* CONFIG_X86_DS */
Markus Metzgereee3af42008-01-30 13:31:09 +0100295
Markus Metzgere5e8ca62008-11-25 08:47:19 +0100296struct cpuinfo_x86;
297static inline void __cpuinit ds_init_intel(struct cpuinfo_x86 *ignored) {}
Markus Metzgerc2724772008-12-11 13:49:59 +0100298static inline void ds_switch_to(struct task_struct *prev,
299 struct task_struct *next) {}
Markus Metzgereee3af42008-01-30 13:31:09 +0100300
Markus Metzger93fa7632008-04-08 11:01:58 +0200301#endif /* CONFIG_X86_DS */
H. Peter Anvin1965aae2008-10-22 22:26:29 -0700302#endif /* _ASM_X86_DS_H */