blob: 56099e52970d26536ff879525347112a2c526014 [file] [log] [blame]
Thomas Gleixner0793a612008-12-04 20:12:29 +01001/*
2 * Performance counters:
3 *
4 * Copyright(C) 2008, Thomas Gleixner <tglx@linutronix.de>
5 * Copyright(C) 2008, Red Hat, Inc., Ingo Molnar
6 *
7 * Data type definitions, declarations, prototypes.
8 *
9 * Started by: Thomas Gleixner and Ingo Molnar
10 *
11 * For licencing details see kernel-base/COPYING
12 */
13#ifndef _LINUX_PERF_COUNTER_H
14#define _LINUX_PERF_COUNTER_H
15
Paul Mackerrasf3dfd262009-02-26 22:43:46 +110016#include <linux/types.h>
17#include <linux/ioctl.h>
Paul Mackerras9aaa1312009-03-21 15:31:47 +110018#include <asm/byteorder.h>
Thomas Gleixner0793a612008-12-04 20:12:29 +010019
20/*
Ingo Molnar9f66a382008-12-10 12:33:23 +010021 * User-space ABI bits:
22 */
23
24/*
Peter Zijlstrab8e83512009-03-19 20:26:18 +010025 * hw_event.type
26 */
27enum perf_event_types {
28 PERF_TYPE_HARDWARE = 0,
29 PERF_TYPE_SOFTWARE = 1,
30 PERF_TYPE_TRACEPOINT = 2,
31
32 /*
33 * available TYPE space, raw is the max value.
34 */
35
36 PERF_TYPE_RAW = 128,
37};
38
39/*
40 * Generalized performance counter event types, used by the hw_event.event_id
Ingo Molnar9f66a382008-12-10 12:33:23 +010041 * parameter of the sys_perf_counter_open() syscall:
Thomas Gleixner0793a612008-12-04 20:12:29 +010042 */
Peter Zijlstrab8e83512009-03-19 20:26:18 +010043enum hw_event_ids {
Thomas Gleixner0793a612008-12-04 20:12:29 +010044 /*
Ingo Molnar9f66a382008-12-10 12:33:23 +010045 * Common hardware events, generalized by the kernel:
Thomas Gleixner0793a612008-12-04 20:12:29 +010046 */
Peter Zijlstrab8e83512009-03-19 20:26:18 +010047 PERF_COUNT_CPU_CYCLES = 0,
48 PERF_COUNT_INSTRUCTIONS = 1,
49 PERF_COUNT_CACHE_REFERENCES = 2,
50 PERF_COUNT_CACHE_MISSES = 3,
51 PERF_COUNT_BRANCH_INSTRUCTIONS = 4,
52 PERF_COUNT_BRANCH_MISSES = 5,
53 PERF_COUNT_BUS_CYCLES = 6,
Ingo Molnar9f66a382008-12-10 12:33:23 +010054
Peter Zijlstrab8e83512009-03-19 20:26:18 +010055 PERF_HW_EVENTS_MAX = 7,
56};
Ingo Molnar6c594c22008-12-14 12:34:15 +010057
Peter Zijlstrab8e83512009-03-19 20:26:18 +010058/*
59 * Special "software" counters provided by the kernel, even if the hardware
60 * does not support performance counters. These counters measure various
61 * physical and sw events of the kernel (and allow the profiling of them as
62 * well):
63 */
64enum sw_event_ids {
65 PERF_COUNT_CPU_CLOCK = 0,
66 PERF_COUNT_TASK_CLOCK = 1,
67 PERF_COUNT_PAGE_FAULTS = 2,
68 PERF_COUNT_CONTEXT_SWITCHES = 3,
69 PERF_COUNT_CPU_MIGRATIONS = 4,
70 PERF_COUNT_PAGE_FAULTS_MIN = 5,
71 PERF_COUNT_PAGE_FAULTS_MAJ = 6,
Ingo Molnar6c594c22008-12-14 12:34:15 +010072
Peter Zijlstrab8e83512009-03-19 20:26:18 +010073 PERF_SW_EVENTS_MAX = 7,
Thomas Gleixner0793a612008-12-04 20:12:29 +010074};
75
76/*
77 * IRQ-notification data record type:
78 */
Ingo Molnar9f66a382008-12-10 12:33:23 +010079enum perf_counter_record_type {
Peter Zijlstrab8e83512009-03-19 20:26:18 +010080 PERF_RECORD_SIMPLE = 0,
81 PERF_RECORD_IRQ = 1,
82 PERF_RECORD_GROUP = 2,
Thomas Gleixner0793a612008-12-04 20:12:29 +010083};
84
Peter Zijlstraf4a2deb42009-03-23 18:22:06 +010085#define __PERF_COUNTER_MASK(name) \
86 (((1ULL << PERF_COUNTER_##name##_BITS) - 1) << \
87 PERF_COUNTER_##name##_SHIFT)
88
89#define PERF_COUNTER_RAW_BITS 1
90#define PERF_COUNTER_RAW_SHIFT 63
91#define PERF_COUNTER_RAW_MASK __PERF_COUNTER_MASK(RAW)
92
93#define PERF_COUNTER_CONFIG_BITS 63
94#define PERF_COUNTER_CONFIG_SHIFT 0
95#define PERF_COUNTER_CONFIG_MASK __PERF_COUNTER_MASK(CONFIG)
96
97#define PERF_COUNTER_TYPE_BITS 7
98#define PERF_COUNTER_TYPE_SHIFT 56
99#define PERF_COUNTER_TYPE_MASK __PERF_COUNTER_MASK(TYPE)
100
101#define PERF_COUNTER_EVENT_BITS 56
102#define PERF_COUNTER_EVENT_SHIFT 0
103#define PERF_COUNTER_EVENT_MASK __PERF_COUNTER_MASK(EVENT)
104
Ingo Molnar9f66a382008-12-10 12:33:23 +0100105/*
106 * Hardware event to monitor via a performance monitoring counter:
107 */
108struct perf_counter_hw_event {
Peter Zijlstraf4a2deb42009-03-23 18:22:06 +0100109 /*
110 * The MSB of the config word signifies if the rest contains cpu
111 * specific (raw) counter configuration data, if unset, the next
112 * 7 bits are an event type and the rest of the bits are the event
113 * identifier.
114 */
115 __u64 config;
Ingo Molnar9f66a382008-12-10 12:33:23 +0100116
Paul Mackerrasf3dfd262009-02-26 22:43:46 +1100117 __u64 irq_period;
Paul Mackerras2743a5b2009-03-04 20:36:51 +1100118 __u64 record_type;
119 __u64 read_format;
Ingo Molnar9f66a382008-12-10 12:33:23 +0100120
Paul Mackerras2743a5b2009-03-04 20:36:51 +1100121 __u64 disabled : 1, /* off by default */
Paul Mackerras0475f9e2009-02-11 14:35:35 +1100122 nmi : 1, /* NMI sampling */
Paul Mackerras0475f9e2009-02-11 14:35:35 +1100123 inherit : 1, /* children inherit it */
124 pinned : 1, /* must always be on PMU */
125 exclusive : 1, /* only group on PMU */
126 exclude_user : 1, /* don't count user */
127 exclude_kernel : 1, /* ditto kernel */
128 exclude_hv : 1, /* ditto hypervisor */
Paul Mackerras2743a5b2009-03-04 20:36:51 +1100129 exclude_idle : 1, /* don't count when idle */
Paul Mackerras3b6f9e52009-01-14 21:00:30 +1100130
Peter Zijlstrab8e83512009-03-19 20:26:18 +0100131 __reserved_1 : 55;
Paul Mackerras2743a5b2009-03-04 20:36:51 +1100132
133 __u32 extra_config_len;
134 __u32 __reserved_4;
Ingo Molnar9f66a382008-12-10 12:33:23 +0100135
Paul Mackerrasf3dfd262009-02-26 22:43:46 +1100136 __u64 __reserved_2;
Paul Mackerras2743a5b2009-03-04 20:36:51 +1100137 __u64 __reserved_3;
Thomas Gleixnereab656a2008-12-08 19:26:59 +0100138};
139
Ingo Molnar9f66a382008-12-10 12:33:23 +0100140/*
Paul Mackerrasd859e292009-01-17 18:10:22 +1100141 * Ioctls that can be done on a perf counter fd:
142 */
143#define PERF_COUNTER_IOC_ENABLE _IO('$', 0)
144#define PERF_COUNTER_IOC_DISABLE _IO('$', 1)
145
Paul Mackerrasf3dfd262009-02-26 22:43:46 +1100146#ifdef __KERNEL__
Paul Mackerrasd859e292009-01-17 18:10:22 +1100147/*
Paul Mackerrasf3dfd262009-02-26 22:43:46 +1100148 * Kernel-internal data types and definitions:
Ingo Molnar9f66a382008-12-10 12:33:23 +0100149 */
150
Paul Mackerrasf3dfd262009-02-26 22:43:46 +1100151#ifdef CONFIG_PERF_COUNTERS
152# include <asm/perf_counter.h>
153#endif
154
155#include <linux/list.h>
156#include <linux/mutex.h>
157#include <linux/rculist.h>
158#include <linux/rcupdate.h>
159#include <linux/spinlock.h>
Peter Zijlstrad6d020e2009-03-13 12:21:35 +0100160#include <linux/hrtimer.h>
Paul Mackerrasf3dfd262009-02-26 22:43:46 +1100161#include <asm/atomic.h>
162
163struct task_struct;
164
Peter Zijlstraf4a2deb42009-03-23 18:22:06 +0100165static inline u64 perf_event_raw(struct perf_counter_hw_event *hw_event)
166{
167 return hw_event->config & PERF_COUNTER_RAW_MASK;
168}
169
170static inline u64 perf_event_config(struct perf_counter_hw_event *hw_event)
171{
172 return hw_event->config & PERF_COUNTER_CONFIG_MASK;
173}
174
175static inline u64 perf_event_type(struct perf_counter_hw_event *hw_event)
176{
177 return (hw_event->config & PERF_COUNTER_TYPE_MASK) >>
178 PERF_COUNTER_TYPE_SHIFT;
179}
180
181static inline u64 perf_event_id(struct perf_counter_hw_event *hw_event)
182{
183 return hw_event->config & PERF_COUNTER_EVENT_MASK;
184}
185
Thomas Gleixner0793a612008-12-04 20:12:29 +0100186/**
Ingo Molnar9f66a382008-12-10 12:33:23 +0100187 * struct hw_perf_counter - performance counter hardware details:
Thomas Gleixner0793a612008-12-04 20:12:29 +0100188 */
189struct hw_perf_counter {
Ingo Molnaree060942008-12-13 09:00:03 +0100190#ifdef CONFIG_PERF_COUNTERS
Peter Zijlstrad6d020e2009-03-13 12:21:35 +0100191 union {
192 struct { /* hardware */
193 u64 config;
194 unsigned long config_base;
195 unsigned long counter_base;
196 int nmi;
197 unsigned int idx;
198 };
199 union { /* software */
200 atomic64_t count;
201 struct hrtimer hrtimer;
202 };
203 };
Ingo Molnaree060942008-12-13 09:00:03 +0100204 atomic64_t prev_count;
Ingo Molnar9f66a382008-12-10 12:33:23 +0100205 u64 irq_period;
Ingo Molnaree060942008-12-13 09:00:03 +0100206 atomic64_t period_left;
207#endif
Thomas Gleixner0793a612008-12-04 20:12:29 +0100208};
209
210/*
211 * Hardcoded buffer length limit for now, for IRQ-fed events:
212 */
Ingo Molnar9f66a382008-12-10 12:33:23 +0100213#define PERF_DATA_BUFLEN 2048
Thomas Gleixner0793a612008-12-04 20:12:29 +0100214
215/**
216 * struct perf_data - performance counter IRQ data sampling ...
217 */
218struct perf_data {
Ingo Molnar9f66a382008-12-10 12:33:23 +0100219 int len;
220 int rd_idx;
221 int overrun;
222 u8 data[PERF_DATA_BUFLEN];
Thomas Gleixner0793a612008-12-04 20:12:29 +0100223};
224
Ingo Molnar621a01e2008-12-11 12:46:46 +0100225struct perf_counter;
226
227/**
228 * struct hw_perf_counter_ops - performance counter hw ops
229 */
230struct hw_perf_counter_ops {
Ingo Molnar95cdd2e2008-12-21 13:50:42 +0100231 int (*enable) (struct perf_counter *counter);
Ingo Molnar76715812008-12-17 14:20:28 +0100232 void (*disable) (struct perf_counter *counter);
233 void (*read) (struct perf_counter *counter);
Ingo Molnar621a01e2008-12-11 12:46:46 +0100234};
235
Thomas Gleixner0793a612008-12-04 20:12:29 +0100236/**
Ingo Molnar6a930702008-12-11 15:17:03 +0100237 * enum perf_counter_active_state - the states of a counter
238 */
239enum perf_counter_active_state {
Paul Mackerras3b6f9e52009-01-14 21:00:30 +1100240 PERF_COUNTER_STATE_ERROR = -2,
Ingo Molnar6a930702008-12-11 15:17:03 +0100241 PERF_COUNTER_STATE_OFF = -1,
242 PERF_COUNTER_STATE_INACTIVE = 0,
243 PERF_COUNTER_STATE_ACTIVE = 1,
244};
245
Ingo Molnar9b51f662008-12-12 13:49:45 +0100246struct file;
247
Ingo Molnar6a930702008-12-11 15:17:03 +0100248/**
Thomas Gleixner0793a612008-12-04 20:12:29 +0100249 * struct perf_counter - performance counter kernel representation:
250 */
251struct perf_counter {
Ingo Molnaree060942008-12-13 09:00:03 +0100252#ifdef CONFIG_PERF_COUNTERS
Ingo Molnar04289bb2008-12-11 08:38:42 +0100253 struct list_head list_entry;
Peter Zijlstra592903c2009-03-13 12:21:36 +0100254 struct list_head event_entry;
Ingo Molnar04289bb2008-12-11 08:38:42 +0100255 struct list_head sibling_list;
256 struct perf_counter *group_leader;
Ingo Molnar5c92d122008-12-11 13:21:10 +0100257 const struct hw_perf_counter_ops *hw_ops;
Ingo Molnar04289bb2008-12-11 08:38:42 +0100258
Ingo Molnar6a930702008-12-11 15:17:03 +0100259 enum perf_counter_active_state state;
Paul Mackerrasc07c99b2009-02-13 22:10:34 +1100260 enum perf_counter_active_state prev_state;
Thomas Gleixner0793a612008-12-04 20:12:29 +0100261 atomic64_t count;
Ingo Molnaree060942008-12-13 09:00:03 +0100262
Ingo Molnar9f66a382008-12-10 12:33:23 +0100263 struct perf_counter_hw_event hw_event;
Thomas Gleixner0793a612008-12-04 20:12:29 +0100264 struct hw_perf_counter hw;
265
266 struct perf_counter_context *ctx;
267 struct task_struct *task;
Ingo Molnar9b51f662008-12-12 13:49:45 +0100268 struct file *filp;
Thomas Gleixner0793a612008-12-04 20:12:29 +0100269
Ingo Molnar9b51f662008-12-12 13:49:45 +0100270 struct perf_counter *parent;
Paul Mackerrasd859e292009-01-17 18:10:22 +1100271 struct list_head child_list;
272
Thomas Gleixner0793a612008-12-04 20:12:29 +0100273 /*
Paul Mackerrasd859e292009-01-17 18:10:22 +1100274 * Protect attach/detach and child_list:
Thomas Gleixner0793a612008-12-04 20:12:29 +0100275 */
276 struct mutex mutex;
277
278 int oncpu;
279 int cpu;
280
Thomas Gleixner0793a612008-12-04 20:12:29 +0100281 /* read() / irq related data */
282 wait_queue_head_t waitq;
283 /* optional: for NMIs */
284 int wakeup_pending;
285 struct perf_data *irqdata;
286 struct perf_data *usrdata;
287 struct perf_data data[2];
Peter Zijlstra592903c2009-03-13 12:21:36 +0100288
Peter Zijlstrae077df42009-03-19 20:26:17 +0100289 void (*destroy)(struct perf_counter *);
Peter Zijlstra592903c2009-03-13 12:21:36 +0100290 struct rcu_head rcu_head;
Ingo Molnaree060942008-12-13 09:00:03 +0100291#endif
Thomas Gleixner0793a612008-12-04 20:12:29 +0100292};
293
294/**
295 * struct perf_counter_context - counter context structure
296 *
297 * Used as a container for task counters and CPU counters as well:
298 */
299struct perf_counter_context {
300#ifdef CONFIG_PERF_COUNTERS
301 /*
Paul Mackerrasd859e292009-01-17 18:10:22 +1100302 * Protect the states of the counters in the list,
303 * nr_active, and the list:
Thomas Gleixner0793a612008-12-04 20:12:29 +0100304 */
305 spinlock_t lock;
Paul Mackerrasd859e292009-01-17 18:10:22 +1100306 /*
307 * Protect the list of counters. Locking either mutex or lock
308 * is sufficient to ensure the list doesn't change; to change
309 * the list you need to lock both the mutex and the spinlock.
310 */
311 struct mutex mutex;
Ingo Molnar04289bb2008-12-11 08:38:42 +0100312
313 struct list_head counter_list;
Peter Zijlstra592903c2009-03-13 12:21:36 +0100314 struct list_head event_list;
Thomas Gleixner0793a612008-12-04 20:12:29 +0100315 int nr_counters;
316 int nr_active;
Paul Mackerrasd859e292009-01-17 18:10:22 +1100317 int is_active;
Thomas Gleixner0793a612008-12-04 20:12:29 +0100318 struct task_struct *task;
319#endif
320};
321
322/**
323 * struct perf_counter_cpu_context - per cpu counter context structure
324 */
325struct perf_cpu_context {
326 struct perf_counter_context ctx;
327 struct perf_counter_context *task_ctx;
328 int active_oncpu;
329 int max_pertask;
Paul Mackerras3b6f9e52009-01-14 21:00:30 +1100330 int exclusive;
Thomas Gleixner0793a612008-12-04 20:12:29 +0100331};
332
333/*
334 * Set by architecture code:
335 */
336extern int perf_max_counters;
337
338#ifdef CONFIG_PERF_COUNTERS
Ingo Molnar5c92d122008-12-11 13:21:10 +0100339extern const struct hw_perf_counter_ops *
Ingo Molnar621a01e2008-12-11 12:46:46 +0100340hw_perf_counter_init(struct perf_counter *counter);
341
Thomas Gleixner0793a612008-12-04 20:12:29 +0100342extern void perf_counter_task_sched_in(struct task_struct *task, int cpu);
343extern void perf_counter_task_sched_out(struct task_struct *task, int cpu);
344extern void perf_counter_task_tick(struct task_struct *task, int cpu);
Ingo Molnar9b51f662008-12-12 13:49:45 +0100345extern void perf_counter_init_task(struct task_struct *child);
346extern void perf_counter_exit_task(struct task_struct *child);
Thomas Gleixner0793a612008-12-04 20:12:29 +0100347extern void perf_counter_notify(struct pt_regs *regs);
348extern void perf_counter_print_debug(void);
Mike Galbraith1b023a92009-01-23 10:13:01 +0100349extern void perf_counter_unthrottle(void);
Ingo Molnar01b28382008-12-11 13:45:51 +0100350extern u64 hw_perf_save_disable(void);
351extern void hw_perf_restore(u64 ctrl);
Ingo Molnar1d1c7dd2008-12-11 14:59:31 +0100352extern int perf_counter_task_disable(void);
353extern int perf_counter_task_enable(void);
Paul Mackerras3cbed422009-01-09 16:43:42 +1100354extern int hw_perf_group_sched_in(struct perf_counter *group_leader,
355 struct perf_cpu_context *cpuctx,
356 struct perf_counter_context *ctx, int cpu);
Ingo Molnar5c92d122008-12-11 13:21:10 +0100357
Peter Zijlstra0322cd62009-03-19 20:26:19 +0100358extern void perf_counter_output(struct perf_counter *counter,
359 int nmi, struct pt_regs *regs);
Paul Mackerras3b6f9e52009-01-14 21:00:30 +1100360/*
361 * Return 1 for a software counter, 0 for a hardware counter
362 */
363static inline int is_software_counter(struct perf_counter *counter)
364{
Peter Zijlstraf4a2deb42009-03-23 18:22:06 +0100365 return !perf_event_raw(&counter->hw_event) &&
366 perf_event_type(&counter->hw_event) != PERF_TYPE_HARDWARE;
Paul Mackerras3b6f9e52009-01-14 21:00:30 +1100367}
368
Peter Zijlstrab8e83512009-03-19 20:26:18 +0100369extern void perf_swcounter_event(u32, u64, int, struct pt_regs *);
Peter Zijlstra15dbf272009-03-13 12:21:32 +0100370
Thomas Gleixner0793a612008-12-04 20:12:29 +0100371#else
372static inline void
373perf_counter_task_sched_in(struct task_struct *task, int cpu) { }
374static inline void
375perf_counter_task_sched_out(struct task_struct *task, int cpu) { }
376static inline void
377perf_counter_task_tick(struct task_struct *task, int cpu) { }
Ingo Molnar9b51f662008-12-12 13:49:45 +0100378static inline void perf_counter_init_task(struct task_struct *child) { }
379static inline void perf_counter_exit_task(struct task_struct *child) { }
Thomas Gleixner0793a612008-12-04 20:12:29 +0100380static inline void perf_counter_notify(struct pt_regs *regs) { }
381static inline void perf_counter_print_debug(void) { }
Mike Galbraith1b023a92009-01-23 10:13:01 +0100382static inline void perf_counter_unthrottle(void) { }
Peter Zijlstra15dbf272009-03-13 12:21:32 +0100383static inline void hw_perf_restore(u64 ctrl) { }
Ingo Molnar01b28382008-12-11 13:45:51 +0100384static inline u64 hw_perf_save_disable(void) { return 0; }
Ingo Molnar1d1c7dd2008-12-11 14:59:31 +0100385static inline int perf_counter_task_disable(void) { return -EINVAL; }
386static inline int perf_counter_task_enable(void) { return -EINVAL; }
Peter Zijlstra15dbf272009-03-13 12:21:32 +0100387
Peter Zijlstrab8e83512009-03-19 20:26:18 +0100388static inline void perf_swcounter_event(u32 event, u64 nr,
Peter Zijlstra15dbf272009-03-13 12:21:32 +0100389 int nmi, struct pt_regs *regs) { }
Thomas Gleixner0793a612008-12-04 20:12:29 +0100390#endif
391
Paul Mackerrasf3dfd262009-02-26 22:43:46 +1100392#endif /* __KERNEL__ */
Thomas Gleixner0793a612008-12-04 20:12:29 +0100393#endif /* _LINUX_PERF_COUNTER_H */