2009-09-24 23:02:49 +07:00
|
|
|
#include "sort.h"
|
|
|
|
|
|
|
|
regex_t parent_regex;
|
|
|
|
char default_parent_pattern[] = "^sys_|^do_page_fault";
|
|
|
|
char *parent_pattern = default_parent_pattern;
|
|
|
|
char default_sort_order[] = "comm,dso,symbol";
|
|
|
|
char *sort_order = default_sort_order;
|
2009-10-23 04:23:22 +07:00
|
|
|
int sort__need_collapse = 0;
|
|
|
|
int sort__has_parent = 0;
|
perf tools: Bind callchains to the first sort dimension column
Currently, the callchains are displayed using a constant left
margin. So depending on the current sort dimension
configuration, callchains may appear to be well attached to the
first sort dimension column field which is mostly the case,
except when the first dimension of sorting is done by comm,
because these are right aligned.
This patch binds the callchain to the first letter in the first
column, whatever type of column it is (dso, comm, symbol).
Before:
0.80% perf [k] __lock_acquire
__lock_acquire
lock_acquire
|
|--58.33%-- _spin_lock
| |
| |--28.57%-- inotify_should_send_event
| | fsnotify
| | __fsnotify_parent
After:
0.80% perf [k] __lock_acquire
__lock_acquire
lock_acquire
|
|--58.33%-- _spin_lock
| |
| |--28.57%-- inotify_should_send_event
| | fsnotify
| | __fsnotify_parent
Also, for clarity, we don't put anymore the callchain as is but:
- If we have a top level ancestor in the callchain, start it
with a first ascii hook.
Before:
0.80% perf [kernel] [k] __lock_acquire
__lock_acquire
lock_acquire
|
|--58.33%-- _spin_lock
| |
| |--28.57%-- inotify_should_send_event
| | fsnotify
[..] [..]
After:
0.80% perf [kernel] [k] __lock_acquire
|
--- __lock_acquire
lock_acquire
|
|--58.33%-- _spin_lock
| |
| |--28.57%-- inotify_should_send_event
| | fsnotify
[..] [..]
- Otherwise, if we have several top level ancestors, then
display these like we did before:
1.69% Xorg
|
|--21.21%-- vread_hpet
| 0x7fffd85b46fc
| 0x7fffd85b494d
| 0x7f4fafb4e54d
|
|--15.15%-- exaOffscreenAlloc
|
|--9.09%-- I830WaitLpRing
Signed-off-by: Frederic Weisbecker <fweisbec@gmail.com>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: Arnaldo Carvalho de Melo <acme@redhat.com>
Cc: Mike Galbraith <efault@gmx.de>
Cc: Paul Mackerras <paulus@samba.org>
Cc: Anton Blanchard <anton@samba.org>
LKML-Reference: <1256246604-17156-2-git-send-email-fweisbec@gmail.com>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2009-10-23 04:23:23 +07:00
|
|
|
|
|
|
|
enum sort_type sort__first_dimension;
|
2009-09-24 23:02:49 +07:00
|
|
|
|
|
|
|
unsigned int dsos__col_width;
|
|
|
|
unsigned int comms__col_width;
|
|
|
|
unsigned int threads__col_width;
|
|
|
|
static unsigned int parent_symbol__col_width;
|
|
|
|
char * field_sep;
|
|
|
|
|
|
|
|
LIST_HEAD(hist_entry__sort_list);
|
|
|
|
|
|
|
|
struct sort_entry sort_thread = {
|
|
|
|
.header = "Command: Pid",
|
|
|
|
.cmp = sort__thread_cmp,
|
|
|
|
.print = sort__thread_print,
|
|
|
|
.width = &threads__col_width,
|
|
|
|
};
|
|
|
|
|
|
|
|
struct sort_entry sort_comm = {
|
|
|
|
.header = "Command",
|
|
|
|
.cmp = sort__comm_cmp,
|
|
|
|
.collapse = sort__comm_collapse,
|
|
|
|
.print = sort__comm_print,
|
|
|
|
.width = &comms__col_width,
|
|
|
|
};
|
|
|
|
|
|
|
|
struct sort_entry sort_dso = {
|
|
|
|
.header = "Shared Object",
|
|
|
|
.cmp = sort__dso_cmp,
|
|
|
|
.print = sort__dso_print,
|
|
|
|
.width = &dsos__col_width,
|
|
|
|
};
|
|
|
|
|
|
|
|
struct sort_entry sort_sym = {
|
|
|
|
.header = "Symbol",
|
|
|
|
.cmp = sort__sym_cmp,
|
|
|
|
.print = sort__sym_print,
|
|
|
|
};
|
|
|
|
|
|
|
|
struct sort_entry sort_parent = {
|
|
|
|
.header = "Parent symbol",
|
|
|
|
.cmp = sort__parent_cmp,
|
|
|
|
.print = sort__parent_print,
|
|
|
|
.width = &parent_symbol__col_width,
|
|
|
|
};
|
|
|
|
|
|
|
|
struct sort_dimension {
|
|
|
|
const char *name;
|
|
|
|
struct sort_entry *entry;
|
|
|
|
int taken;
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct sort_dimension sort_dimensions[] = {
|
|
|
|
{ .name = "pid", .entry = &sort_thread, },
|
|
|
|
{ .name = "comm", .entry = &sort_comm, },
|
|
|
|
{ .name = "dso", .entry = &sort_dso, },
|
|
|
|
{ .name = "symbol", .entry = &sort_sym, },
|
|
|
|
{ .name = "parent", .entry = &sort_parent, },
|
|
|
|
};
|
|
|
|
|
|
|
|
int64_t cmp_null(void *l, void *r)
|
|
|
|
{
|
|
|
|
if (!l && !r)
|
|
|
|
return 0;
|
|
|
|
else if (!l)
|
|
|
|
return -1;
|
|
|
|
else
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --sort pid */
|
|
|
|
|
|
|
|
int64_t
|
|
|
|
sort__thread_cmp(struct hist_entry *left, struct hist_entry *right)
|
|
|
|
{
|
|
|
|
return right->thread->pid - left->thread->pid;
|
|
|
|
}
|
|
|
|
|
|
|
|
int repsep_fprintf(FILE *fp, const char *fmt, ...)
|
|
|
|
{
|
|
|
|
int n;
|
|
|
|
va_list ap;
|
|
|
|
|
|
|
|
va_start(ap, fmt);
|
|
|
|
if (!field_sep)
|
|
|
|
n = vfprintf(fp, fmt, ap);
|
|
|
|
else {
|
|
|
|
char *bf = NULL;
|
|
|
|
n = vasprintf(&bf, fmt, ap);
|
|
|
|
if (n > 0) {
|
|
|
|
char *sep = bf;
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
sep = strchr(sep, *field_sep);
|
|
|
|
if (sep == NULL)
|
|
|
|
break;
|
|
|
|
*sep = '.';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fputs(bf, fp);
|
|
|
|
free(bf);
|
|
|
|
}
|
|
|
|
va_end(ap);
|
|
|
|
return n;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t
|
|
|
|
sort__thread_print(FILE *fp, struct hist_entry *self, unsigned int width)
|
|
|
|
{
|
|
|
|
return repsep_fprintf(fp, "%*s:%5d", width - 6,
|
|
|
|
self->thread->comm ?: "", self->thread->pid);
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t
|
|
|
|
sort__comm_print(FILE *fp, struct hist_entry *self, unsigned int width)
|
|
|
|
{
|
|
|
|
return repsep_fprintf(fp, "%*s", width, self->thread->comm);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --sort dso */
|
|
|
|
|
|
|
|
int64_t
|
|
|
|
sort__dso_cmp(struct hist_entry *left, struct hist_entry *right)
|
|
|
|
{
|
2009-10-02 13:29:58 +07:00
|
|
|
struct dso *dso_l = left->map ? left->map->dso : NULL;
|
|
|
|
struct dso *dso_r = right->map ? right->map->dso : NULL;
|
|
|
|
const char *dso_name_l, *dso_name_r;
|
2009-09-24 23:02:49 +07:00
|
|
|
|
|
|
|
if (!dso_l || !dso_r)
|
|
|
|
return cmp_null(dso_l, dso_r);
|
|
|
|
|
2009-10-02 13:29:58 +07:00
|
|
|
if (verbose) {
|
|
|
|
dso_name_l = dso_l->long_name;
|
|
|
|
dso_name_r = dso_r->long_name;
|
|
|
|
} else {
|
|
|
|
dso_name_l = dso_l->short_name;
|
|
|
|
dso_name_r = dso_r->short_name;
|
|
|
|
}
|
|
|
|
|
|
|
|
return strcmp(dso_name_l, dso_name_r);
|
2009-09-24 23:02:49 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
size_t
|
|
|
|
sort__dso_print(FILE *fp, struct hist_entry *self, unsigned int width)
|
|
|
|
{
|
2009-10-02 13:29:58 +07:00
|
|
|
if (self->map && self->map->dso) {
|
|
|
|
const char *dso_name = !verbose ? self->map->dso->short_name :
|
|
|
|
self->map->dso->long_name;
|
|
|
|
return repsep_fprintf(fp, "%-*s", width, dso_name);
|
|
|
|
}
|
2009-09-24 23:02:49 +07:00
|
|
|
|
|
|
|
return repsep_fprintf(fp, "%*llx", width, (u64)self->ip);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --sort symbol */
|
|
|
|
|
|
|
|
int64_t
|
|
|
|
sort__sym_cmp(struct hist_entry *left, struct hist_entry *right)
|
|
|
|
{
|
|
|
|
u64 ip_l, ip_r;
|
|
|
|
|
|
|
|
if (left->sym == right->sym)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
ip_l = left->sym ? left->sym->start : left->ip;
|
|
|
|
ip_r = right->sym ? right->sym->start : right->ip;
|
|
|
|
|
|
|
|
return (int64_t)(ip_r - ip_l);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
size_t
|
|
|
|
sort__sym_print(FILE *fp, struct hist_entry *self, unsigned int width __used)
|
|
|
|
{
|
|
|
|
size_t ret = 0;
|
|
|
|
|
2009-10-02 13:29:58 +07:00
|
|
|
if (verbose) {
|
|
|
|
char o = self->map ? dso__symtab_origin(self->map->dso) : '!';
|
|
|
|
ret += repsep_fprintf(fp, "%#018llx %c ", (u64)self->ip, o);
|
|
|
|
}
|
2009-09-24 23:02:49 +07:00
|
|
|
|
|
|
|
ret += repsep_fprintf(fp, "[%c] ", self->level);
|
2009-10-02 13:29:58 +07:00
|
|
|
if (self->sym)
|
2009-09-24 23:02:49 +07:00
|
|
|
ret += repsep_fprintf(fp, "%s", self->sym->name);
|
2009-10-02 13:29:58 +07:00
|
|
|
else
|
2009-09-24 23:02:49 +07:00
|
|
|
ret += repsep_fprintf(fp, "%#016llx", (u64)self->ip);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --sort comm */
|
|
|
|
|
|
|
|
int64_t
|
|
|
|
sort__comm_cmp(struct hist_entry *left, struct hist_entry *right)
|
|
|
|
{
|
|
|
|
return right->thread->pid - left->thread->pid;
|
|
|
|
}
|
|
|
|
|
|
|
|
int64_t
|
|
|
|
sort__comm_collapse(struct hist_entry *left, struct hist_entry *right)
|
|
|
|
{
|
|
|
|
char *comm_l = left->thread->comm;
|
|
|
|
char *comm_r = right->thread->comm;
|
|
|
|
|
|
|
|
if (!comm_l || !comm_r)
|
|
|
|
return cmp_null(comm_l, comm_r);
|
|
|
|
|
|
|
|
return strcmp(comm_l, comm_r);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --sort parent */
|
|
|
|
|
|
|
|
int64_t
|
|
|
|
sort__parent_cmp(struct hist_entry *left, struct hist_entry *right)
|
|
|
|
{
|
|
|
|
struct symbol *sym_l = left->parent;
|
|
|
|
struct symbol *sym_r = right->parent;
|
|
|
|
|
|
|
|
if (!sym_l || !sym_r)
|
|
|
|
return cmp_null(sym_l, sym_r);
|
|
|
|
|
|
|
|
return strcmp(sym_l->name, sym_r->name);
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t
|
|
|
|
sort__parent_print(FILE *fp, struct hist_entry *self, unsigned int width)
|
|
|
|
{
|
|
|
|
return repsep_fprintf(fp, "%-*s", width,
|
|
|
|
self->parent ? self->parent->name : "[other]");
|
|
|
|
}
|
|
|
|
|
|
|
|
int sort_dimension__add(const char *tok)
|
|
|
|
{
|
|
|
|
unsigned int i;
|
|
|
|
|
|
|
|
for (i = 0; i < ARRAY_SIZE(sort_dimensions); i++) {
|
|
|
|
struct sort_dimension *sd = &sort_dimensions[i];
|
|
|
|
|
|
|
|
if (sd->taken)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (strncasecmp(tok, sd->name, strlen(tok)))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (sd->entry->collapse)
|
|
|
|
sort__need_collapse = 1;
|
|
|
|
|
|
|
|
if (sd->entry == &sort_parent) {
|
|
|
|
int ret = regcomp(&parent_regex, parent_pattern, REG_EXTENDED);
|
|
|
|
if (ret) {
|
|
|
|
char err[BUFSIZ];
|
|
|
|
|
|
|
|
regerror(ret, &parent_regex, err, sizeof(err));
|
|
|
|
fprintf(stderr, "Invalid regex: %s\n%s",
|
|
|
|
parent_pattern, err);
|
|
|
|
exit(-1);
|
|
|
|
}
|
|
|
|
sort__has_parent = 1;
|
|
|
|
}
|
|
|
|
|
perf tools: Bind callchains to the first sort dimension column
Currently, the callchains are displayed using a constant left
margin. So depending on the current sort dimension
configuration, callchains may appear to be well attached to the
first sort dimension column field which is mostly the case,
except when the first dimension of sorting is done by comm,
because these are right aligned.
This patch binds the callchain to the first letter in the first
column, whatever type of column it is (dso, comm, symbol).
Before:
0.80% perf [k] __lock_acquire
__lock_acquire
lock_acquire
|
|--58.33%-- _spin_lock
| |
| |--28.57%-- inotify_should_send_event
| | fsnotify
| | __fsnotify_parent
After:
0.80% perf [k] __lock_acquire
__lock_acquire
lock_acquire
|
|--58.33%-- _spin_lock
| |
| |--28.57%-- inotify_should_send_event
| | fsnotify
| | __fsnotify_parent
Also, for clarity, we don't put anymore the callchain as is but:
- If we have a top level ancestor in the callchain, start it
with a first ascii hook.
Before:
0.80% perf [kernel] [k] __lock_acquire
__lock_acquire
lock_acquire
|
|--58.33%-- _spin_lock
| |
| |--28.57%-- inotify_should_send_event
| | fsnotify
[..] [..]
After:
0.80% perf [kernel] [k] __lock_acquire
|
--- __lock_acquire
lock_acquire
|
|--58.33%-- _spin_lock
| |
| |--28.57%-- inotify_should_send_event
| | fsnotify
[..] [..]
- Otherwise, if we have several top level ancestors, then
display these like we did before:
1.69% Xorg
|
|--21.21%-- vread_hpet
| 0x7fffd85b46fc
| 0x7fffd85b494d
| 0x7f4fafb4e54d
|
|--15.15%-- exaOffscreenAlloc
|
|--9.09%-- I830WaitLpRing
Signed-off-by: Frederic Weisbecker <fweisbec@gmail.com>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: Arnaldo Carvalho de Melo <acme@redhat.com>
Cc: Mike Galbraith <efault@gmx.de>
Cc: Paul Mackerras <paulus@samba.org>
Cc: Anton Blanchard <anton@samba.org>
LKML-Reference: <1256246604-17156-2-git-send-email-fweisbec@gmail.com>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2009-10-23 04:23:23 +07:00
|
|
|
if (list_empty(&hist_entry__sort_list)) {
|
|
|
|
if (!strcmp(sd->name, "pid"))
|
|
|
|
sort__first_dimension = SORT_PID;
|
|
|
|
else if (!strcmp(sd->name, "comm"))
|
|
|
|
sort__first_dimension = SORT_COMM;
|
|
|
|
else if (!strcmp(sd->name, "dso"))
|
|
|
|
sort__first_dimension = SORT_DSO;
|
|
|
|
else if (!strcmp(sd->name, "symbol"))
|
|
|
|
sort__first_dimension = SORT_SYM;
|
|
|
|
else if (!strcmp(sd->name, "parent"))
|
|
|
|
sort__first_dimension = SORT_PARENT;
|
|
|
|
}
|
2009-10-23 04:23:22 +07:00
|
|
|
|
2009-09-24 23:02:49 +07:00
|
|
|
list_add_tail(&sd->entry->list, &hist_entry__sort_list);
|
|
|
|
sd->taken = 1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return -ESRCH;
|
|
|
|
}
|
2009-12-15 05:09:29 +07:00
|
|
|
|
|
|
|
void setup_sorting(const char * const usagestr[], const struct option *opts)
|
|
|
|
{
|
|
|
|
char *tmp, *tok, *str = strdup(sort_order);
|
|
|
|
|
|
|
|
for (tok = strtok_r(str, ", ", &tmp);
|
|
|
|
tok; tok = strtok_r(NULL, ", ", &tmp)) {
|
|
|
|
if (sort_dimension__add(tok) < 0) {
|
|
|
|
error("Unknown --sort key: `%s'", tok);
|
|
|
|
usage_with_options(usagestr, opts);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
free(str);
|
|
|
|
}
|