blob: 5f5e6df0260df49f249fa36e2bf92292e5fa322a [file] [log] [blame]
Peter Zijlstrade9ac072009-04-08 15:01:31 +02001
2
Thomas Gleixner6eda5832009-05-01 18:29:57 +02003#include "util/util.h"
4
Peter Zijlstrade9ac072009-04-08 15:01:31 +02005#include <sys/types.h>
6#include <sys/stat.h>
7#include <sys/time.h>
8#include <unistd.h>
9#include <stdint.h>
10#include <stdlib.h>
11#include <string.h>
12#include <limits.h>
13#include <getopt.h>
14#include <assert.h>
15#include <fcntl.h>
16#include <stdio.h>
17#include <errno.h>
Peter Zijlstrade9ac072009-04-08 15:01:31 +020018#include <time.h>
19#include <sched.h>
20#include <pthread.h>
21
22#include <sys/syscall.h>
23#include <sys/ioctl.h>
24#include <sys/poll.h>
25#include <sys/prctl.h>
26#include <sys/wait.h>
27#include <sys/uio.h>
28#include <sys/mman.h>
29
30#include <linux/unistd.h>
31#include <linux/types.h>
32
33#include "../../include/linux/perf_counter.h"
34
Thomas Gleixner6eda5832009-05-01 18:29:57 +020035#include "perf.h"
Peter Zijlstrade9ac072009-04-08 15:01:31 +020036
37static int nr_counters = 0;
38static __u64 event_id[MAX_COUNTERS] = { };
39static int default_interval = 100000;
40static int event_count[MAX_COUNTERS];
41static int fd[MAX_NR_CPUS][MAX_COUNTERS];
42static int nr_cpus = 0;
43static unsigned int page_size;
44static unsigned int mmap_pages = 16;
45static int output;
46static char *output_name = "output.perf";
47static int group = 0;
Peter Zijlstra16c8a102009-05-05 17:50:27 +020048static unsigned int realtime_prio = 0;
49static int system_wide = 0;
50static int inherit = 1;
51static int nmi = 1;
Peter Zijlstrade9ac072009-04-08 15:01:31 +020052
53const unsigned int default_count[] = {
54 1000000,
55 1000000,
56 10000,
57 10000,
58 1000000,
59 10000,
60};
61
Peter Zijlstrade9ac072009-04-08 15:01:31 +020062struct event_symbol {
63 __u64 event;
64 char *symbol;
65};
66
67static struct event_symbol event_symbols[] = {
68 {EID(PERF_TYPE_HARDWARE, PERF_COUNT_CPU_CYCLES), "cpu-cycles", },
69 {EID(PERF_TYPE_HARDWARE, PERF_COUNT_CPU_CYCLES), "cycles", },
70 {EID(PERF_TYPE_HARDWARE, PERF_COUNT_INSTRUCTIONS), "instructions", },
71 {EID(PERF_TYPE_HARDWARE, PERF_COUNT_CACHE_REFERENCES), "cache-references", },
72 {EID(PERF_TYPE_HARDWARE, PERF_COUNT_CACHE_MISSES), "cache-misses", },
73 {EID(PERF_TYPE_HARDWARE, PERF_COUNT_BRANCH_INSTRUCTIONS), "branch-instructions", },
74 {EID(PERF_TYPE_HARDWARE, PERF_COUNT_BRANCH_INSTRUCTIONS), "branches", },
75 {EID(PERF_TYPE_HARDWARE, PERF_COUNT_BRANCH_MISSES), "branch-misses", },
76 {EID(PERF_TYPE_HARDWARE, PERF_COUNT_BUS_CYCLES), "bus-cycles", },
77
78 {EID(PERF_TYPE_SOFTWARE, PERF_COUNT_CPU_CLOCK), "cpu-clock", },
79 {EID(PERF_TYPE_SOFTWARE, PERF_COUNT_TASK_CLOCK), "task-clock", },
80 {EID(PERF_TYPE_SOFTWARE, PERF_COUNT_PAGE_FAULTS), "page-faults", },
81 {EID(PERF_TYPE_SOFTWARE, PERF_COUNT_PAGE_FAULTS), "faults", },
82 {EID(PERF_TYPE_SOFTWARE, PERF_COUNT_PAGE_FAULTS_MIN), "minor-faults", },
83 {EID(PERF_TYPE_SOFTWARE, PERF_COUNT_PAGE_FAULTS_MAJ), "major-faults", },
84 {EID(PERF_TYPE_SOFTWARE, PERF_COUNT_CONTEXT_SWITCHES), "context-switches", },
85 {EID(PERF_TYPE_SOFTWARE, PERF_COUNT_CONTEXT_SWITCHES), "cs", },
86 {EID(PERF_TYPE_SOFTWARE, PERF_COUNT_CPU_MIGRATIONS), "cpu-migrations", },
87 {EID(PERF_TYPE_SOFTWARE, PERF_COUNT_CPU_MIGRATIONS), "migrations", },
88};
89
90/*
91 * Each event can have multiple symbolic names.
92 * Symbolic names are (almost) exactly matched.
93 */
94static __u64 match_event_symbols(char *str)
95{
96 __u64 config, id;
97 int type;
98 unsigned int i;
99
100 if (sscanf(str, "r%llx", &config) == 1)
101 return config | PERF_COUNTER_RAW_MASK;
102
103 if (sscanf(str, "%d:%llu", &type, &id) == 2)
104 return EID(type, id);
105
106 for (i = 0; i < ARRAY_SIZE(event_symbols); i++) {
107 if (!strncmp(str, event_symbols[i].symbol,
108 strlen(event_symbols[i].symbol)))
109 return event_symbols[i].event;
110 }
111
112 return ~0ULL;
113}
114
115static int parse_events(char *str)
116{
117 __u64 config;
118
119again:
120 if (nr_counters == MAX_COUNTERS)
121 return -1;
122
123 config = match_event_symbols(str);
124 if (config == ~0ULL)
125 return -1;
126
127 event_id[nr_counters] = config;
128 nr_counters++;
129
130 str = strstr(str, ",");
131 if (str) {
132 str++;
133 goto again;
134 }
135
136 return 0;
137}
138
139#define __PERF_COUNTER_FIELD(config, name) \
140 ((config & PERF_COUNTER_##name##_MASK) >> PERF_COUNTER_##name##_SHIFT)
141
142#define PERF_COUNTER_RAW(config) __PERF_COUNTER_FIELD(config, RAW)
143#define PERF_COUNTER_CONFIG(config) __PERF_COUNTER_FIELD(config, CONFIG)
144#define PERF_COUNTER_TYPE(config) __PERF_COUNTER_FIELD(config, TYPE)
145#define PERF_COUNTER_ID(config) __PERF_COUNTER_FIELD(config, EVENT)
146
147static void display_events_help(void)
148{
149 unsigned int i;
150 __u64 e;
151
152 printf(
153 " -e EVENT --event=EVENT # symbolic-name abbreviations");
154
155 for (i = 0; i < ARRAY_SIZE(event_symbols); i++) {
156 int type, id;
157
158 e = event_symbols[i].event;
159 type = PERF_COUNTER_TYPE(e);
160 id = PERF_COUNTER_ID(e);
161
162 printf("\n %d:%d: %-20s",
163 type, id, event_symbols[i].symbol);
164 }
165
166 printf("\n"
167 " rNNN: raw PMU events (eventsel+umask)\n\n");
168}
169
170static void display_help(void)
171{
172 printf(
Peter Zijlstra16c8a102009-05-05 17:50:27 +0200173 "Usage: perf-record [<options>] <cmd>\n"
Peter Zijlstrade9ac072009-04-08 15:01:31 +0200174 "perf-record Options (up to %d event types can be specified at once):\n\n",
175 MAX_COUNTERS);
176
177 display_events_help();
178
179 printf(
180 " -c CNT --count=CNT # event period to sample\n"
181 " -m pages --mmap_pages=<pages> # number of mmap data pages\n"
182 " -o file --output=<file> # output file\n"
183 " -r prio --realtime=<prio> # use RT prio\n"
Peter Zijlstra16c8a102009-05-05 17:50:27 +0200184 " -s --system # system wide profiling\n"
Peter Zijlstrade9ac072009-04-08 15:01:31 +0200185 );
186
187 exit(0);
188}
189
Peter Zijlstra16c8a102009-05-05 17:50:27 +0200190static void process_options(int argc, const char *argv[])
Peter Zijlstrade9ac072009-04-08 15:01:31 +0200191{
192 int error = 0, counter;
193
194 for (;;) {
195 int option_index = 0;
196 /** Options for getopt */
197 static struct option long_options[] = {
198 {"count", required_argument, NULL, 'c'},
199 {"event", required_argument, NULL, 'e'},
200 {"mmap_pages", required_argument, NULL, 'm'},
201 {"output", required_argument, NULL, 'o'},
202 {"realtime", required_argument, NULL, 'r'},
Peter Zijlstra16c8a102009-05-05 17:50:27 +0200203 {"system", no_argument, NULL, 's'},
204 {"inherit", no_argument, NULL, 'i'},
205 {"nmi", no_argument, NULL, 'n'},
Peter Zijlstrade9ac072009-04-08 15:01:31 +0200206 {NULL, 0, NULL, 0 }
207 };
Peter Zijlstra16c8a102009-05-05 17:50:27 +0200208 int c = getopt_long(argc, argv, "+:c:e:m:o:r:sin",
Peter Zijlstrade9ac072009-04-08 15:01:31 +0200209 long_options, &option_index);
210 if (c == -1)
211 break;
212
213 switch (c) {
214 case 'c': default_interval = atoi(optarg); break;
215 case 'e': error = parse_events(optarg); break;
216 case 'm': mmap_pages = atoi(optarg); break;
217 case 'o': output_name = strdup(optarg); break;
218 case 'r': realtime_prio = atoi(optarg); break;
Peter Zijlstra16c8a102009-05-05 17:50:27 +0200219 case 's': system_wide ^= 1; break;
220 case 'i': inherit ^= 1; break;
221 case 'n': nmi ^= 1; break;
Peter Zijlstrade9ac072009-04-08 15:01:31 +0200222 default: error = 1; break;
223 }
224 }
Peter Zijlstra16c8a102009-05-05 17:50:27 +0200225
226 if (argc - optind == 0)
227 error = 1;
228
Peter Zijlstrade9ac072009-04-08 15:01:31 +0200229 if (error)
230 display_help();
231
232 if (!nr_counters) {
233 nr_counters = 1;
234 event_id[0] = 0;
235 }
236
237 for (counter = 0; counter < nr_counters; counter++) {
238 if (event_count[counter])
239 continue;
240
241 event_count[counter] = default_interval;
242 }
243}
244
245struct mmap_data {
246 int counter;
247 void *base;
248 unsigned int mask;
249 unsigned int prev;
250};
251
252static unsigned int mmap_read_head(struct mmap_data *md)
253{
254 struct perf_counter_mmap_page *pc = md->base;
255 int head;
256
257 head = pc->data_head;
258 rmb();
259
260 return head;
261}
262
263static long events;
264static struct timeval last_read, this_read;
265
266static void mmap_read(struct mmap_data *md)
267{
268 unsigned int head = mmap_read_head(md);
269 unsigned int old = md->prev;
270 unsigned char *data = md->base + page_size;
271 unsigned long size;
272 void *buf;
273 int diff;
274
275 gettimeofday(&this_read, NULL);
276
277 /*
278 * If we're further behind than half the buffer, there's a chance
279 * the writer will bite our tail and screw up the events under us.
280 *
281 * If we somehow ended up ahead of the head, we got messed up.
282 *
283 * In either case, truncate and restart at head.
284 */
285 diff = head - old;
286 if (diff > md->mask / 2 || diff < 0) {
287 struct timeval iv;
288 unsigned long msecs;
289
290 timersub(&this_read, &last_read, &iv);
291 msecs = iv.tv_sec*1000 + iv.tv_usec/1000;
292
293 fprintf(stderr, "WARNING: failed to keep up with mmap data."
294 " Last read %lu msecs ago.\n", msecs);
295
296 /*
297 * head points to a known good entry, start there.
298 */
299 old = head;
300 }
301
302 last_read = this_read;
303
304 if (old != head)
305 events++;
306
307 size = head - old;
308
309 if ((old & md->mask) + size != (head & md->mask)) {
310 buf = &data[old & md->mask];
311 size = md->mask + 1 - (old & md->mask);
312 old += size;
313 while (size) {
314 int ret = write(output, buf, size);
315 if (ret < 0) {
316 perror("failed to write");
317 exit(-1);
318 }
319 size -= ret;
320 buf += ret;
321 }
322 }
323
324 buf = &data[old & md->mask];
325 size = head - old;
326 old += size;
327 while (size) {
328 int ret = write(output, buf, size);
329 if (ret < 0) {
330 perror("failed to write");
331 exit(-1);
332 }
333 size -= ret;
334 buf += ret;
335 }
336
337 md->prev = old;
338}
339
340static volatile int done = 0;
341
Peter Zijlstra16c8a102009-05-05 17:50:27 +0200342static void sig_handler(int sig)
Peter Zijlstrade9ac072009-04-08 15:01:31 +0200343{
Peter Zijlstra16c8a102009-05-05 17:50:27 +0200344 done = 1;
Peter Zijlstrade9ac072009-04-08 15:01:31 +0200345}
346
Peter Zijlstra16c8a102009-05-05 17:50:27 +0200347static struct pollfd event_array[MAX_NR_CPUS * MAX_COUNTERS];
348static struct mmap_data mmap_array[MAX_NR_CPUS][MAX_COUNTERS];
349
350static int nr_poll;
351static int nr_cpu;
352
353static void open_counters(int cpu)
Peter Zijlstrade9ac072009-04-08 15:01:31 +0200354{
Peter Zijlstrade9ac072009-04-08 15:01:31 +0200355 struct perf_counter_hw_event hw_event;
Peter Zijlstra16c8a102009-05-05 17:50:27 +0200356 int counter, group_fd;
357 int track = 1;
358 pid_t pid = -1;
359
360 if (cpu < 0)
361 pid = 0;
362
363 group_fd = -1;
364 for (counter = 0; counter < nr_counters; counter++) {
365
366 memset(&hw_event, 0, sizeof(hw_event));
367 hw_event.config = event_id[counter];
368 hw_event.irq_period = event_count[counter];
369 hw_event.record_type = PERF_RECORD_IP | PERF_RECORD_TID;
370 hw_event.nmi = nmi;
371 hw_event.mmap = track;
372 hw_event.comm = track;
373 hw_event.inherit = (cpu < 0) && inherit;
374
375 track = 0; // only the first counter needs these
376
377 fd[nr_cpu][counter] =
378 sys_perf_counter_open(&hw_event, pid, cpu, group_fd, 0);
379
380 if (fd[nr_cpu][counter] < 0) {
381 int err = errno;
382 printf("kerneltop error: syscall returned with %d (%s)\n",
383 fd[nr_cpu][counter], strerror(err));
384 if (err == EPERM)
385 printf("Are you root?\n");
386 exit(-1);
387 }
388 assert(fd[nr_cpu][counter] >= 0);
389 fcntl(fd[nr_cpu][counter], F_SETFL, O_NONBLOCK);
390
391 /*
392 * First counter acts as the group leader:
393 */
394 if (group && group_fd == -1)
395 group_fd = fd[nr_cpu][counter];
396
397 event_array[nr_poll].fd = fd[nr_cpu][counter];
398 event_array[nr_poll].events = POLLIN;
399 nr_poll++;
400
401 mmap_array[nr_cpu][counter].counter = counter;
402 mmap_array[nr_cpu][counter].prev = 0;
403 mmap_array[nr_cpu][counter].mask = mmap_pages*page_size - 1;
404 mmap_array[nr_cpu][counter].base = mmap(NULL, (mmap_pages+1)*page_size,
405 PROT_READ, MAP_SHARED, fd[nr_cpu][counter], 0);
406 if (mmap_array[nr_cpu][counter].base == MAP_FAILED) {
407 printf("kerneltop error: failed to mmap with %d (%s)\n",
408 errno, strerror(errno));
409 exit(-1);
410 }
411 }
412 nr_cpu++;
413}
414
415int cmd_record(int argc, const char **argv)
416{
417 int i, counter;
Peter Zijlstrade9ac072009-04-08 15:01:31 +0200418 pid_t pid;
419 int ret;
420
421 page_size = sysconf(_SC_PAGE_SIZE);
422
423 process_options(argc, argv);
424
425 nr_cpus = sysconf(_SC_NPROCESSORS_ONLN);
426 assert(nr_cpus <= MAX_NR_CPUS);
427 assert(nr_cpus >= 0);
428
429 output = open(output_name, O_CREAT|O_RDWR, S_IRWXU);
430 if (output < 0) {
431 perror("failed to create output file");
432 exit(-1);
433 }
434
435 argc -= optind;
436 argv += optind;
437
Peter Zijlstra16c8a102009-05-05 17:50:27 +0200438 if (!system_wide)
439 open_counters(-1);
440 else for (i = 0; i < nr_cpus; i++)
441 open_counters(i);
Peter Zijlstrade9ac072009-04-08 15:01:31 +0200442
Peter Zijlstra16c8a102009-05-05 17:50:27 +0200443 signal(SIGCHLD, sig_handler);
444 signal(SIGINT, sig_handler);
Peter Zijlstrade9ac072009-04-08 15:01:31 +0200445
446 pid = fork();
447 if (pid < 0)
448 perror("failed to fork");
449
450 if (!pid) {
451 if (execvp(argv[0], argv)) {
452 perror(argv[0]);
453 exit(-1);
454 }
455 }
456
457 if (realtime_prio) {
458 struct sched_param param;
459
460 param.sched_priority = realtime_prio;
461 if (sched_setscheduler(0, SCHED_FIFO, &param)) {
462 printf("Could not set realtime priority.\n");
463 exit(-1);
464 }
465 }
466
467 /*
468 * TODO: store the current /proc/$/maps information somewhere
469 */
470
471 while (!done) {
472 int hits = events;
473
Peter Zijlstra16c8a102009-05-05 17:50:27 +0200474 for (i = 0; i < nr_cpu; i++) {
Peter Zijlstrade9ac072009-04-08 15:01:31 +0200475 for (counter = 0; counter < nr_counters; counter++)
476 mmap_read(&mmap_array[i][counter]);
477 }
478
479 if (hits == events)
480 ret = poll(event_array, nr_poll, 100);
481 }
482
483 return 0;
484}