diff options
-rw-r--r-- | tools/perf/builtin-kmem.c | 142 |
1 files changed, 108 insertions, 34 deletions
diff --git a/tools/perf/builtin-kmem.c b/tools/perf/builtin-kmem.c index 1ef43c212d9a..dc86f1e64b66 100644 --- a/tools/perf/builtin-kmem.c +++ b/tools/perf/builtin-kmem.c | |||
@@ -26,14 +26,13 @@ static u64 sample_type; | |||
26 | static int alloc_flag; | 26 | static int alloc_flag; |
27 | static int caller_flag; | 27 | static int caller_flag; |
28 | 28 | ||
29 | sort_fn_t alloc_sort_fn; | ||
30 | sort_fn_t caller_sort_fn; | ||
31 | |||
32 | static int alloc_lines = -1; | 29 | static int alloc_lines = -1; |
33 | static int caller_lines = -1; | 30 | static int caller_lines = -1; |
34 | 31 | ||
35 | static bool raw_ip; | 32 | static bool raw_ip; |
36 | 33 | ||
34 | static char default_sort_order[] = "frag,hit,bytes"; | ||
35 | |||
37 | static char *cwd; | 36 | static char *cwd; |
38 | static int cwdlen; | 37 | static int cwdlen; |
39 | 38 | ||
@@ -371,20 +370,34 @@ static void print_result(void) | |||
371 | print_summary(); | 370 | print_summary(); |
372 | } | 371 | } |
373 | 372 | ||
373 | struct sort_dimension { | ||
374 | const char name[20]; | ||
375 | sort_fn_t cmp; | ||
376 | struct list_head list; | ||
377 | }; | ||
378 | |||
379 | static LIST_HEAD(caller_sort); | ||
380 | static LIST_HEAD(alloc_sort); | ||
381 | |||
374 | static void sort_insert(struct rb_root *root, struct alloc_stat *data, | 382 | static void sort_insert(struct rb_root *root, struct alloc_stat *data, |
375 | sort_fn_t sort_fn) | 383 | struct list_head *sort_list) |
376 | { | 384 | { |
377 | struct rb_node **new = &(root->rb_node); | 385 | struct rb_node **new = &(root->rb_node); |
378 | struct rb_node *parent = NULL; | 386 | struct rb_node *parent = NULL; |
387 | struct sort_dimension *sort; | ||
379 | 388 | ||
380 | while (*new) { | 389 | while (*new) { |
381 | struct alloc_stat *this; | 390 | struct alloc_stat *this; |
382 | int cmp; | 391 | int cmp = 0; |
383 | 392 | ||
384 | this = rb_entry(*new, struct alloc_stat, node); | 393 | this = rb_entry(*new, struct alloc_stat, node); |
385 | parent = *new; | 394 | parent = *new; |
386 | 395 | ||
387 | cmp = sort_fn(data, this); | 396 | list_for_each_entry(sort, sort_list, list) { |
397 | cmp = sort->cmp(data, this); | ||
398 | if (cmp) | ||
399 | break; | ||
400 | } | ||
388 | 401 | ||
389 | if (cmp > 0) | 402 | if (cmp > 0) |
390 | new = &((*new)->rb_left); | 403 | new = &((*new)->rb_left); |
@@ -397,7 +410,7 @@ static void sort_insert(struct rb_root *root, struct alloc_stat *data, | |||
397 | } | 410 | } |
398 | 411 | ||
399 | static void __sort_result(struct rb_root *root, struct rb_root *root_sorted, | 412 | static void __sort_result(struct rb_root *root, struct rb_root *root_sorted, |
400 | sort_fn_t sort_fn) | 413 | struct list_head *sort_list) |
401 | { | 414 | { |
402 | struct rb_node *node; | 415 | struct rb_node *node; |
403 | struct alloc_stat *data; | 416 | struct alloc_stat *data; |
@@ -409,14 +422,14 @@ static void __sort_result(struct rb_root *root, struct rb_root *root_sorted, | |||
409 | 422 | ||
410 | rb_erase(node, root); | 423 | rb_erase(node, root); |
411 | data = rb_entry(node, struct alloc_stat, node); | 424 | data = rb_entry(node, struct alloc_stat, node); |
412 | sort_insert(root_sorted, data, sort_fn); | 425 | sort_insert(root_sorted, data, sort_list); |
413 | } | 426 | } |
414 | } | 427 | } |
415 | 428 | ||
416 | static void sort_result(void) | 429 | static void sort_result(void) |
417 | { | 430 | { |
418 | __sort_result(&root_alloc_stat, &root_alloc_sorted, alloc_sort_fn); | 431 | __sort_result(&root_alloc_stat, &root_alloc_sorted, &alloc_sort); |
419 | __sort_result(&root_caller_stat, &root_caller_sorted, caller_sort_fn); | 432 | __sort_result(&root_caller_stat, &root_caller_sorted, &caller_sort); |
420 | } | 433 | } |
421 | 434 | ||
422 | static int __cmd_kmem(void) | 435 | static int __cmd_kmem(void) |
@@ -434,7 +447,6 @@ static const char * const kmem_usage[] = { | |||
434 | NULL | 447 | NULL |
435 | }; | 448 | }; |
436 | 449 | ||
437 | |||
438 | static int ptr_cmp(struct alloc_stat *l, struct alloc_stat *r) | 450 | static int ptr_cmp(struct alloc_stat *l, struct alloc_stat *r) |
439 | { | 451 | { |
440 | if (l->ptr < r->ptr) | 452 | if (l->ptr < r->ptr) |
@@ -444,6 +456,11 @@ static int ptr_cmp(struct alloc_stat *l, struct alloc_stat *r) | |||
444 | return 0; | 456 | return 0; |
445 | } | 457 | } |
446 | 458 | ||
459 | static struct sort_dimension ptr_sort_dimension = { | ||
460 | .name = "ptr", | ||
461 | .cmp = ptr_cmp, | ||
462 | }; | ||
463 | |||
447 | static int callsite_cmp(struct alloc_stat *l, struct alloc_stat *r) | 464 | static int callsite_cmp(struct alloc_stat *l, struct alloc_stat *r) |
448 | { | 465 | { |
449 | if (l->call_site < r->call_site) | 466 | if (l->call_site < r->call_site) |
@@ -453,6 +470,11 @@ static int callsite_cmp(struct alloc_stat *l, struct alloc_stat *r) | |||
453 | return 0; | 470 | return 0; |
454 | } | 471 | } |
455 | 472 | ||
473 | static struct sort_dimension callsite_sort_dimension = { | ||
474 | .name = "callsite", | ||
475 | .cmp = callsite_cmp, | ||
476 | }; | ||
477 | |||
456 | static int hit_cmp(struct alloc_stat *l, struct alloc_stat *r) | 478 | static int hit_cmp(struct alloc_stat *l, struct alloc_stat *r) |
457 | { | 479 | { |
458 | if (l->hit < r->hit) | 480 | if (l->hit < r->hit) |
@@ -462,6 +484,11 @@ static int hit_cmp(struct alloc_stat *l, struct alloc_stat *r) | |||
462 | return 0; | 484 | return 0; |
463 | } | 485 | } |
464 | 486 | ||
487 | static struct sort_dimension hit_sort_dimension = { | ||
488 | .name = "hit", | ||
489 | .cmp = hit_cmp, | ||
490 | }; | ||
491 | |||
465 | static int bytes_cmp(struct alloc_stat *l, struct alloc_stat *r) | 492 | static int bytes_cmp(struct alloc_stat *l, struct alloc_stat *r) |
466 | { | 493 | { |
467 | if (l->bytes_alloc < r->bytes_alloc) | 494 | if (l->bytes_alloc < r->bytes_alloc) |
@@ -471,6 +498,11 @@ static int bytes_cmp(struct alloc_stat *l, struct alloc_stat *r) | |||
471 | return 0; | 498 | return 0; |
472 | } | 499 | } |
473 | 500 | ||
501 | static struct sort_dimension bytes_sort_dimension = { | ||
502 | .name = "bytes", | ||
503 | .cmp = bytes_cmp, | ||
504 | }; | ||
505 | |||
474 | static int frag_cmp(struct alloc_stat *l, struct alloc_stat *r) | 506 | static int frag_cmp(struct alloc_stat *l, struct alloc_stat *r) |
475 | { | 507 | { |
476 | double x, y; | 508 | double x, y; |
@@ -485,31 +517,73 @@ static int frag_cmp(struct alloc_stat *l, struct alloc_stat *r) | |||
485 | return 0; | 517 | return 0; |
486 | } | 518 | } |
487 | 519 | ||
520 | static struct sort_dimension frag_sort_dimension = { | ||
521 | .name = "frag", | ||
522 | .cmp = frag_cmp, | ||
523 | }; | ||
524 | |||
525 | static struct sort_dimension *avail_sorts[] = { | ||
526 | &ptr_sort_dimension, | ||
527 | &callsite_sort_dimension, | ||
528 | &hit_sort_dimension, | ||
529 | &bytes_sort_dimension, | ||
530 | &frag_sort_dimension, | ||
531 | }; | ||
532 | |||
533 | #define NUM_AVAIL_SORTS \ | ||
534 | (int)(sizeof(avail_sorts) / sizeof(struct sort_dimension *)) | ||
535 | |||
536 | static int sort_dimension__add(const char *tok, struct list_head *list) | ||
537 | { | ||
538 | struct sort_dimension *sort; | ||
539 | int i; | ||
540 | |||
541 | for (i = 0; i < NUM_AVAIL_SORTS; i++) { | ||
542 | if (!strcmp(avail_sorts[i]->name, tok)) { | ||
543 | sort = malloc(sizeof(*sort)); | ||
544 | if (!sort) | ||
545 | die("malloc"); | ||
546 | memcpy(sort, avail_sorts[i], sizeof(*sort)); | ||
547 | list_add_tail(&sort->list, list); | ||
548 | return 0; | ||
549 | } | ||
550 | } | ||
551 | |||
552 | return -1; | ||
553 | } | ||
554 | |||
555 | static int setup_sorting(struct list_head *sort_list, const char *arg) | ||
556 | { | ||
557 | char *tok; | ||
558 | char *str = strdup(arg); | ||
559 | |||
560 | if (!str) | ||
561 | die("strdup"); | ||
562 | |||
563 | while (true) { | ||
564 | tok = strsep(&str, ","); | ||
565 | if (!tok) | ||
566 | break; | ||
567 | if (sort_dimension__add(tok, sort_list) < 0) { | ||
568 | error("Unknown --sort key: '%s'", tok); | ||
569 | return -1; | ||
570 | } | ||
571 | } | ||
572 | |||
573 | free(str); | ||
574 | return 0; | ||
575 | } | ||
576 | |||
488 | static int parse_sort_opt(const struct option *opt __used, | 577 | static int parse_sort_opt(const struct option *opt __used, |
489 | const char *arg, int unset __used) | 578 | const char *arg, int unset __used) |
490 | { | 579 | { |
491 | sort_fn_t sort_fn; | ||
492 | |||
493 | if (!arg) | 580 | if (!arg) |
494 | return -1; | 581 | return -1; |
495 | 582 | ||
496 | if (strcmp(arg, "ptr") == 0) | ||
497 | sort_fn = ptr_cmp; | ||
498 | else if (strcmp(arg, "call_site") == 0) | ||
499 | sort_fn = callsite_cmp; | ||
500 | else if (strcmp(arg, "hit") == 0) | ||
501 | sort_fn = hit_cmp; | ||
502 | else if (strcmp(arg, "bytes") == 0) | ||
503 | sort_fn = bytes_cmp; | ||
504 | else if (strcmp(arg, "frag") == 0) | ||
505 | sort_fn = frag_cmp; | ||
506 | else | ||
507 | return -1; | ||
508 | |||
509 | if (caller_flag > alloc_flag) | 583 | if (caller_flag > alloc_flag) |
510 | caller_sort_fn = sort_fn; | 584 | return setup_sorting(&caller_sort, arg); |
511 | else | 585 | else |
512 | alloc_sort_fn = sort_fn; | 586 | return setup_sorting(&alloc_sort, arg); |
513 | 587 | ||
514 | return 0; | 588 | return 0; |
515 | } | 589 | } |
@@ -553,8 +627,8 @@ static const struct option kmem_options[] = { | |||
553 | OPT_CALLBACK(0, "stat", NULL, "<alloc>|<caller>", | 627 | OPT_CALLBACK(0, "stat", NULL, "<alloc>|<caller>", |
554 | "stat selector, Pass 'alloc' or 'caller'.", | 628 | "stat selector, Pass 'alloc' or 'caller'.", |
555 | parse_stat_opt), | 629 | parse_stat_opt), |
556 | OPT_CALLBACK('s', "sort", NULL, "key", | 630 | OPT_CALLBACK('s', "sort", NULL, "key[,key2...]", |
557 | "sort by key: ptr, call_site, hit, bytes, frag", | 631 | "sort by key(s): ptr, call_site, bytes, hit, frag", |
558 | parse_sort_opt), | 632 | parse_sort_opt), |
559 | OPT_CALLBACK('l', "line", NULL, "num", | 633 | OPT_CALLBACK('l', "line", NULL, "num", |
560 | "show n lins", | 634 | "show n lins", |
@@ -606,10 +680,10 @@ int cmd_kmem(int argc, const char **argv, const char *prefix __used) | |||
606 | else if (argc) | 680 | else if (argc) |
607 | usage_with_options(kmem_usage, kmem_options); | 681 | usage_with_options(kmem_usage, kmem_options); |
608 | 682 | ||
609 | if (!alloc_sort_fn) | 683 | if (list_empty(&caller_sort)) |
610 | alloc_sort_fn = bytes_cmp; | 684 | setup_sorting(&caller_sort, default_sort_order); |
611 | if (!caller_sort_fn) | 685 | if (list_empty(&alloc_sort)) |
612 | caller_sort_fn = bytes_cmp; | 686 | setup_sorting(&alloc_sort, default_sort_order); |
613 | 687 | ||
614 | return __cmd_kmem(); | 688 | return __cmd_kmem(); |
615 | } | 689 | } |