blob: a3481f86e2ae1aa049fc66afd39e413fa8bcc1d6 [file] [log] [blame]
Jiri Olsa7aef3bf2016-09-22 17:36:38 +02001#include <linux/compiler.h>
2#include <linux/kernel.h>
3#include "util.h"
4#include "debug.h"
5#include "builtin.h"
6#include <subcmd/parse-options.h>
Jiri Olsa39bcd4a2016-09-22 17:36:39 +02007#include "mem-events.h"
Jiri Olsa903a6f12016-09-22 17:36:40 +02008#include "session.h"
9#include "hist.h"
10#include "tool.h"
11#include "data.h"
Jiri Olsa8d3f9382016-09-22 17:36:42 +020012#include "sort.h"
Jiri Olsa903a6f12016-09-22 17:36:40 +020013
Jiri Olsac75540e2016-09-22 17:36:41 +020014struct c2c_hists {
15 struct hists hists;
16 struct perf_hpp_list list;
17};
18
Jiri Olsa903a6f12016-09-22 17:36:40 +020019struct perf_c2c {
Jiri Olsac75540e2016-09-22 17:36:41 +020020 struct perf_tool tool;
21 struct c2c_hists hists;
Jiri Olsa903a6f12016-09-22 17:36:40 +020022};
23
24static struct perf_c2c c2c;
Jiri Olsa7aef3bf2016-09-22 17:36:38 +020025
26static const char * const c2c_usage[] = {
Jiri Olsa903a6f12016-09-22 17:36:40 +020027 "perf c2c {record|report}",
Jiri Olsa7aef3bf2016-09-22 17:36:38 +020028 NULL
29};
30
Jiri Olsa903a6f12016-09-22 17:36:40 +020031static const char * const __usage_report[] = {
32 "perf c2c report",
33 NULL
34};
35
36static const char * const *report_c2c_usage = __usage_report;
37
Jiri Olsac75540e2016-09-22 17:36:41 +020038#define C2C_HEADER_MAX 2
39
40struct c2c_header {
41 struct {
42 const char *text;
43 int span;
44 } line[C2C_HEADER_MAX];
45};
46
47struct c2c_dimension {
48 struct c2c_header header;
49 const char *name;
50 int width;
Jiri Olsa8d3f9382016-09-22 17:36:42 +020051 struct sort_entry *se;
Jiri Olsac75540e2016-09-22 17:36:41 +020052
53 int64_t (*cmp)(struct perf_hpp_fmt *fmt,
54 struct hist_entry *, struct hist_entry *);
55 int (*entry)(struct perf_hpp_fmt *fmt, struct perf_hpp *hpp,
56 struct hist_entry *he);
57 int (*color)(struct perf_hpp_fmt *fmt, struct perf_hpp *hpp,
58 struct hist_entry *he);
59};
60
61struct c2c_fmt {
62 struct perf_hpp_fmt fmt;
63 struct c2c_dimension *dim;
64};
65
66static int c2c_width(struct perf_hpp_fmt *fmt,
67 struct perf_hpp *hpp __maybe_unused,
68 struct hists *hists __maybe_unused)
69{
70 struct c2c_fmt *c2c_fmt;
Jiri Olsac75540e2016-09-22 17:36:41 +020071 struct c2c_dimension *dim;
Jiri Olsac75540e2016-09-22 17:36:41 +020072
73 c2c_fmt = container_of(fmt, struct c2c_fmt, fmt);
74 dim = c2c_fmt->dim;
75
Jiri Olsa8d3f9382016-09-22 17:36:42 +020076 return dim->se ? hists__col_len(hists, dim->se->se_width_idx) :
77 c2c_fmt->dim->width;
78}
79
80static int c2c_header(struct perf_hpp_fmt *fmt, struct perf_hpp *hpp,
81 struct hists *hists, int line, int *span)
82{
83 struct perf_hpp_list *hpp_list = hists->hpp_list;
84 struct c2c_fmt *c2c_fmt;
85 struct c2c_dimension *dim;
86 const char *text = NULL;
87 int width = c2c_width(fmt, hpp, hists);
88
89 c2c_fmt = container_of(fmt, struct c2c_fmt, fmt);
90 dim = c2c_fmt->dim;
91
92 if (dim->se) {
93 text = dim->header.line[line].text;
94 /* Use the last line from sort_entry if not defined. */
95 if (!text && (line == hpp_list->nr_header_lines - 1))
96 text = dim->se->se_header;
97 } else {
98 text = dim->header.line[line].text;
99
100 if (*span) {
101 (*span)--;
102 return 0;
103 } else {
104 *span = dim->header.line[line].span;
105 }
106 }
107
Jiri Olsac75540e2016-09-22 17:36:41 +0200108 if (text == NULL)
109 text = "";
110
Jiri Olsa8d3f9382016-09-22 17:36:42 +0200111 return scnprintf(hpp->buf, hpp->size, "%*s", width, text);
Jiri Olsac75540e2016-09-22 17:36:41 +0200112}
113
114static struct c2c_dimension *dimensions[] = {
115 NULL,
116};
117
118static void fmt_free(struct perf_hpp_fmt *fmt)
119{
120 struct c2c_fmt *c2c_fmt;
121
122 c2c_fmt = container_of(fmt, struct c2c_fmt, fmt);
123 free(c2c_fmt);
124}
125
126static bool fmt_equal(struct perf_hpp_fmt *a, struct perf_hpp_fmt *b)
127{
128 struct c2c_fmt *c2c_a = container_of(a, struct c2c_fmt, fmt);
129 struct c2c_fmt *c2c_b = container_of(b, struct c2c_fmt, fmt);
130
131 return c2c_a->dim == c2c_b->dim;
132}
133
134static struct c2c_dimension *get_dimension(const char *name)
135{
136 unsigned int i;
137
138 for (i = 0; dimensions[i]; i++) {
139 struct c2c_dimension *dim = dimensions[i];
140
141 if (!strcmp(dim->name, name))
142 return dim;
143 };
144
145 return NULL;
146}
147
Jiri Olsa8d3f9382016-09-22 17:36:42 +0200148static int c2c_se_entry(struct perf_hpp_fmt *fmt, struct perf_hpp *hpp,
149 struct hist_entry *he)
150{
151 struct c2c_fmt *c2c_fmt = container_of(fmt, struct c2c_fmt, fmt);
152 struct c2c_dimension *dim = c2c_fmt->dim;
153 size_t len = fmt->user_len;
154
155 if (!len)
156 len = hists__col_len(he->hists, dim->se->se_width_idx);
157
158 return dim->se->se_snprintf(he, hpp->buf, hpp->size, len);
159}
160
161static int64_t c2c_se_cmp(struct perf_hpp_fmt *fmt,
162 struct hist_entry *a, struct hist_entry *b)
163{
164 struct c2c_fmt *c2c_fmt = container_of(fmt, struct c2c_fmt, fmt);
165 struct c2c_dimension *dim = c2c_fmt->dim;
166
167 return dim->se->se_cmp(a, b);
168}
169
170static int64_t c2c_se_collapse(struct perf_hpp_fmt *fmt,
171 struct hist_entry *a, struct hist_entry *b)
172{
173 struct c2c_fmt *c2c_fmt = container_of(fmt, struct c2c_fmt, fmt);
174 struct c2c_dimension *dim = c2c_fmt->dim;
175 int64_t (*collapse_fn)(struct hist_entry *, struct hist_entry *);
176
177 collapse_fn = dim->se->se_collapse ?: dim->se->se_cmp;
178 return collapse_fn(a, b);
179}
180
Jiri Olsac75540e2016-09-22 17:36:41 +0200181static struct c2c_fmt *get_format(const char *name)
182{
183 struct c2c_dimension *dim = get_dimension(name);
184 struct c2c_fmt *c2c_fmt;
185 struct perf_hpp_fmt *fmt;
186
187 if (!dim)
188 return NULL;
189
190 c2c_fmt = zalloc(sizeof(*c2c_fmt));
191 if (!c2c_fmt)
192 return NULL;
193
194 c2c_fmt->dim = dim;
195
196 fmt = &c2c_fmt->fmt;
197 INIT_LIST_HEAD(&fmt->list);
198 INIT_LIST_HEAD(&fmt->sort_list);
199
Jiri Olsa8d3f9382016-09-22 17:36:42 +0200200 fmt->cmp = dim->se ? c2c_se_cmp : dim->cmp;
201 fmt->sort = dim->se ? c2c_se_cmp : dim->cmp;
202 fmt->entry = dim->se ? c2c_se_entry : dim->entry;
Jiri Olsac75540e2016-09-22 17:36:41 +0200203 fmt->header = c2c_header;
204 fmt->width = c2c_width;
Jiri Olsa8d3f9382016-09-22 17:36:42 +0200205 fmt->collapse = dim->se ? c2c_se_collapse : dim->cmp;
Jiri Olsac75540e2016-09-22 17:36:41 +0200206 fmt->equal = fmt_equal;
207 fmt->free = fmt_free;
208
209 return c2c_fmt;
210}
211
212static int c2c_hists__init_output(struct perf_hpp_list *hpp_list, char *name)
213{
214 struct c2c_fmt *c2c_fmt = get_format(name);
215
Jiri Olsa5f2eca82016-09-22 17:36:43 +0200216 if (!c2c_fmt) {
217 reset_dimensions();
218 return output_field_add(hpp_list, name);
219 }
Jiri Olsac75540e2016-09-22 17:36:41 +0200220
221 perf_hpp_list__column_register(hpp_list, &c2c_fmt->fmt);
222 return 0;
223}
224
225static int c2c_hists__init_sort(struct perf_hpp_list *hpp_list, char *name)
226{
227 struct c2c_fmt *c2c_fmt = get_format(name);
228
Jiri Olsa5f2eca82016-09-22 17:36:43 +0200229 if (!c2c_fmt) {
230 reset_dimensions();
231 return sort_dimension__add(hpp_list, name, NULL, 0);
232 }
Jiri Olsac75540e2016-09-22 17:36:41 +0200233
234 perf_hpp_list__register_sort_field(hpp_list, &c2c_fmt->fmt);
235 return 0;
236}
237
238#define PARSE_LIST(_list, _fn) \
239 do { \
240 char *tmp, *tok; \
241 ret = 0; \
242 \
243 if (!_list) \
244 break; \
245 \
246 for (tok = strtok_r((char *)_list, ", ", &tmp); \
247 tok; tok = strtok_r(NULL, ", ", &tmp)) { \
248 ret = _fn(hpp_list, tok); \
249 if (ret == -EINVAL) { \
250 error("Invalid --fields key: `%s'", tok); \
251 break; \
252 } else if (ret == -ESRCH) { \
253 error("Unknown --fields key: `%s'", tok); \
254 break; \
255 } \
256 } \
257 } while (0)
258
259static int hpp_list__parse(struct perf_hpp_list *hpp_list,
260 const char *output_,
261 const char *sort_)
262{
263 char *output = output_ ? strdup(output_) : NULL;
264 char *sort = sort_ ? strdup(sort_) : NULL;
265 int ret;
266
267 PARSE_LIST(output, c2c_hists__init_output);
268 PARSE_LIST(sort, c2c_hists__init_sort);
269
270 /* copy sort keys to output fields */
271 perf_hpp__setup_output_field(hpp_list);
272
273 /*
274 * We dont need other sorting keys other than those
275 * we already specified. It also really slows down
276 * the processing a lot with big number of output
277 * fields, so switching this off for c2c.
278 */
279
280#if 0
281 /* and then copy output fields to sort keys */
282 perf_hpp__append_sort_keys(&hists->list);
283#endif
284
285 free(output);
286 free(sort);
287 return ret;
288}
289
290static int c2c_hists__init(struct c2c_hists *hists,
291 const char *sort)
292{
293 __hists__init(&hists->hists, &hists->list);
294
295 /*
296 * Initialize only with sort fields, we need to resort
297 * later anyway, and that's where we add output fields
298 * as well.
299 */
300 perf_hpp_list__init(&hists->list);
301
302 return hpp_list__parse(&hists->list, NULL, sort);
303}
304
305__maybe_unused
306static int c2c_hists__reinit(struct c2c_hists *c2c_hists,
307 const char *output,
308 const char *sort)
309{
310 perf_hpp__reset_output_field(&c2c_hists->list);
311 return hpp_list__parse(&c2c_hists->list, output, sort);
312}
313
Jiri Olsa903a6f12016-09-22 17:36:40 +0200314static int perf_c2c__report(int argc, const char **argv)
315{
316 struct perf_session *session;
317 struct perf_data_file file = {
318 .mode = PERF_DATA_MODE_READ,
319 };
320 const struct option c2c_options[] = {
321 OPT_STRING('k', "vmlinux", &symbol_conf.vmlinux_name,
322 "file", "vmlinux pathname"),
323 OPT_INCR('v', "verbose", &verbose,
324 "be more verbose (show counter open errors, etc)"),
325 OPT_STRING('i', "input", &input_name, "file",
326 "the input file to process"),
327 OPT_END()
328 };
329 int err = 0;
330
331 argc = parse_options(argc, argv, c2c_options, report_c2c_usage,
332 PARSE_OPT_STOP_AT_NON_OPTION);
333 if (!argc)
334 usage_with_options(report_c2c_usage, c2c_options);
335
336 file.path = input_name;
337
Jiri Olsac75540e2016-09-22 17:36:41 +0200338 err = c2c_hists__init(&c2c.hists, "dcacheline");
339 if (err) {
340 pr_debug("Failed to initialize hists\n");
341 goto out;
342 }
343
Jiri Olsa903a6f12016-09-22 17:36:40 +0200344 session = perf_session__new(&file, 0, &c2c.tool);
345 if (session == NULL) {
346 pr_debug("No memory for session\n");
347 goto out;
348 }
349
350 if (symbol__init(&session->header.env) < 0)
351 goto out_session;
352
353 /* No pipe support at the moment. */
354 if (perf_data_file__is_pipe(session->file)) {
355 pr_debug("No pipe support at the moment.\n");
356 goto out_session;
357 }
358
359out_session:
360 perf_session__delete(session);
361out:
362 return err;
363}
364
Jiri Olsa39bcd4a2016-09-22 17:36:39 +0200365static int parse_record_events(const struct option *opt __maybe_unused,
366 const char *str, int unset __maybe_unused)
367{
368 bool *event_set = (bool *) opt->value;
369
370 *event_set = true;
371 return perf_mem_events__parse(str);
372}
373
374
375static const char * const __usage_record[] = {
376 "perf c2c record [<options>] [<command>]",
377 "perf c2c record [<options>] -- <command> [<options>]",
378 NULL
379};
380
381static const char * const *record_mem_usage = __usage_record;
382
383static int perf_c2c__record(int argc, const char **argv)
384{
385 int rec_argc, i = 0, j;
386 const char **rec_argv;
387 int ret;
388 bool all_user = false, all_kernel = false;
389 bool event_set = false;
390 struct option options[] = {
391 OPT_CALLBACK('e', "event", &event_set, "event",
392 "event selector. Use 'perf mem record -e list' to list available events",
393 parse_record_events),
394 OPT_INCR('v', "verbose", &verbose,
395 "be more verbose (show counter open errors, etc)"),
396 OPT_BOOLEAN('u', "all-user", &all_user, "collect only user level data"),
397 OPT_BOOLEAN('k', "all-kernel", &all_kernel, "collect only kernel level data"),
398 OPT_UINTEGER('l', "ldlat", &perf_mem_events__loads_ldlat, "setup mem-loads latency"),
399 OPT_END()
400 };
401
402 if (perf_mem_events__init()) {
403 pr_err("failed: memory events not supported\n");
404 return -1;
405 }
406
407 argc = parse_options(argc, argv, options, record_mem_usage,
408 PARSE_OPT_KEEP_UNKNOWN);
409
410 rec_argc = argc + 10; /* max number of arguments */
411 rec_argv = calloc(rec_argc + 1, sizeof(char *));
412 if (!rec_argv)
413 return -1;
414
415 rec_argv[i++] = "record";
416
417 if (!event_set) {
418 perf_mem_events[PERF_MEM_EVENTS__LOAD].record = true;
419 perf_mem_events[PERF_MEM_EVENTS__STORE].record = true;
420 }
421
422 if (perf_mem_events[PERF_MEM_EVENTS__LOAD].record)
423 rec_argv[i++] = "-W";
424
425 rec_argv[i++] = "-d";
426 rec_argv[i++] = "--sample-cpu";
427
428 for (j = 0; j < PERF_MEM_EVENTS__MAX; j++) {
429 if (!perf_mem_events[j].record)
430 continue;
431
432 if (!perf_mem_events[j].supported) {
433 pr_err("failed: event '%s' not supported\n",
434 perf_mem_events[j].name);
435 return -1;
436 }
437
438 rec_argv[i++] = "-e";
439 rec_argv[i++] = perf_mem_events__name(j);
440 };
441
442 if (all_user)
443 rec_argv[i++] = "--all-user";
444
445 if (all_kernel)
446 rec_argv[i++] = "--all-kernel";
447
448 for (j = 0; j < argc; j++, i++)
449 rec_argv[i] = argv[j];
450
451 if (verbose > 0) {
452 pr_debug("calling: ");
453
454 j = 0;
455
456 while (rec_argv[j]) {
457 pr_debug("%s ", rec_argv[j]);
458 j++;
459 }
460 pr_debug("\n");
461 }
462
463 ret = cmd_record(i, rec_argv, NULL);
464 free(rec_argv);
465 return ret;
466}
467
Jiri Olsa7aef3bf2016-09-22 17:36:38 +0200468int cmd_c2c(int argc, const char **argv, const char *prefix __maybe_unused)
469{
470 const struct option c2c_options[] = {
471 OPT_INCR('v', "verbose", &verbose, "be more verbose"),
472 OPT_END()
473 };
474
475 argc = parse_options(argc, argv, c2c_options, c2c_usage,
476 PARSE_OPT_STOP_AT_NON_OPTION);
Jiri Olsa39bcd4a2016-09-22 17:36:39 +0200477
478 if (!argc)
479 usage_with_options(c2c_usage, c2c_options);
480
481 if (!strncmp(argv[0], "rec", 3)) {
482 return perf_c2c__record(argc, argv);
Jiri Olsa903a6f12016-09-22 17:36:40 +0200483 } else if (!strncmp(argv[0], "rep", 3)) {
484 return perf_c2c__report(argc, argv);
Jiri Olsa39bcd4a2016-09-22 17:36:39 +0200485 } else {
486 usage_with_options(c2c_usage, c2c_options);
487 }
488
Jiri Olsa7aef3bf2016-09-22 17:36:38 +0200489 return 0;
490}