perf hists: Introduce hist_entry_ops

Introducing allocation callbacks, that allows to extend current
hist_entry object into objects with special needs without polluting the
current hist_entry object.

Signed-off-by: Jiri Olsa <jolsa@kernel.org>
Cc: David Ahern <dsahern@gmail.com>
Cc: Namhyung Kim <namhyung@kernel.org>
Cc: Peter Zijlstra <a.p.zijlstra@chello.nl>
Link: http://lkml.kernel.org/r/1467701765-26194-3-git-send-email-jolsa@kernel.org
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
This commit is contained in:
Jiri Olsa 2016-07-05 08:56:04 +02:00 committed by Arnaldo Carvalho de Melo
parent 0a269a6bb3
commit f542e7670e
2 changed files with 33 additions and 4 deletions

View file

@ -424,21 +424,42 @@ static int hist_entry__init(struct hist_entry *he,
return 0; return 0;
} }
static void *hist_entry__zalloc(size_t size)
{
return zalloc(size + sizeof(struct hist_entry));
}
static void hist_entry__free(void *ptr)
{
free(ptr);
}
static struct hist_entry_ops default_ops = {
.new = hist_entry__zalloc,
.free = hist_entry__free,
};
static struct hist_entry *hist_entry__new(struct hist_entry *template, static struct hist_entry *hist_entry__new(struct hist_entry *template,
bool sample_self) bool sample_self)
{ {
struct hist_entry_ops *ops = template->ops;
size_t callchain_size = 0; size_t callchain_size = 0;
struct hist_entry *he; struct hist_entry *he;
int err = 0; int err = 0;
if (!ops)
ops = template->ops = &default_ops;
if (symbol_conf.use_callchain) if (symbol_conf.use_callchain)
callchain_size = sizeof(struct callchain_root); callchain_size = sizeof(struct callchain_root);
he = zalloc(sizeof(*he) + callchain_size); he = ops->new(callchain_size);
if (he) { if (he) {
err = hist_entry__init(he, template, sample_self); err = hist_entry__init(he, template, sample_self);
if (err) if (err) {
zfree(&he); ops->free(he);
he = NULL;
}
} }
return he; return he;
@ -1050,6 +1071,8 @@ hist_entry__collapse(struct hist_entry *left, struct hist_entry *right)
void hist_entry__delete(struct hist_entry *he) void hist_entry__delete(struct hist_entry *he)
{ {
struct hist_entry_ops *ops = he->ops;
thread__zput(he->thread); thread__zput(he->thread);
map__zput(he->ms.map); map__zput(he->ms.map);
@ -1074,7 +1097,7 @@ void hist_entry__delete(struct hist_entry *he)
free_callchain(he->callchain); free_callchain(he->callchain);
free(he->trace_output); free(he->trace_output);
free(he->raw_data); free(he->raw_data);
free(he); ops->free(he);
} }
/* /*

View file

@ -67,6 +67,11 @@ struct hist_entry_diff {
}; };
}; };
struct hist_entry_ops {
void *(*new)(size_t size);
void (*free)(void *ptr);
};
/** /**
* struct hist_entry - histogram entry * struct hist_entry - histogram entry
* *
@ -125,6 +130,7 @@ struct hist_entry {
void *trace_output; void *trace_output;
struct perf_hpp_list *hpp_list; struct perf_hpp_list *hpp_list;
struct hist_entry *parent_he; struct hist_entry *parent_he;
struct hist_entry_ops *ops;
union { union {
/* this is for hierarchical entry structure */ /* this is for hierarchical entry structure */
struct { struct {