perf annotate-data: Update sample histogram for type
The annotated_data_type__update_samples() to get histogram for data type access. It'll be called by perf annotate to show which fields in the data type are accessed frequently. Signed-off-by: Namhyung Kim <namhyung@kernel.org> Cc: Adrian Hunter <adrian.hunter@intel.com> Cc: Ian Rogers <irogers@google.com> Cc: Ingo Molnar <mingo@kernel.org> Cc: Jiri Olsa <jolsa@kernel.org> Cc: Linus Torvalds <torvalds@linux-foundation.org> Cc: Masami Hiramatsu <mhiramat@kernel.org> Cc: Peter Zijlstra <peterz@infradead.org> Cc: Stephane Eranian <eranian@google.com> Cc: linux-toolchains@vger.kernel.org Cc: linux-trace-devel@vger.kernel.org Link: https://lore.kernel.org/r/20231213001323.718046-12-namhyung@kernel.org Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
This commit is contained in:
parent
4a111cadac
commit
9bd7ddd157
@ -13,6 +13,8 @@
|
|||||||
#include "debuginfo.h"
|
#include "debuginfo.h"
|
||||||
#include "debug.h"
|
#include "debug.h"
|
||||||
#include "dso.h"
|
#include "dso.h"
|
||||||
|
#include "evsel.h"
|
||||||
|
#include "evlist.h"
|
||||||
#include "map.h"
|
#include "map.h"
|
||||||
#include "map_symbol.h"
|
#include "map_symbol.h"
|
||||||
#include "strbuf.h"
|
#include "strbuf.h"
|
||||||
@ -302,6 +304,44 @@ out:
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int alloc_data_type_histograms(struct annotated_data_type *adt, int nr_entries)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
size_t sz = sizeof(struct type_hist);
|
||||||
|
|
||||||
|
sz += sizeof(struct type_hist_entry) * adt->self.size;
|
||||||
|
|
||||||
|
/* Allocate a table of pointers for each event */
|
||||||
|
adt->nr_histograms = nr_entries;
|
||||||
|
adt->histograms = calloc(nr_entries, sizeof(*adt->histograms));
|
||||||
|
if (adt->histograms == NULL)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Each histogram is allocated for the whole size of the type.
|
||||||
|
* TODO: Probably we can move the histogram to members.
|
||||||
|
*/
|
||||||
|
for (i = 0; i < nr_entries; i++) {
|
||||||
|
adt->histograms[i] = zalloc(sz);
|
||||||
|
if (adt->histograms[i] == NULL)
|
||||||
|
goto err;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
err:
|
||||||
|
while (--i >= 0)
|
||||||
|
free(adt->histograms[i]);
|
||||||
|
free(adt->histograms);
|
||||||
|
return -ENOMEM;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void delete_data_type_histograms(struct annotated_data_type *adt)
|
||||||
|
{
|
||||||
|
for (int i = 0; i < adt->nr_histograms; i++)
|
||||||
|
free(adt->histograms[i]);
|
||||||
|
free(adt->histograms);
|
||||||
|
}
|
||||||
|
|
||||||
void annotated_data_type__tree_delete(struct rb_root *root)
|
void annotated_data_type__tree_delete(struct rb_root *root)
|
||||||
{
|
{
|
||||||
struct annotated_data_type *pos;
|
struct annotated_data_type *pos;
|
||||||
@ -312,7 +352,48 @@ void annotated_data_type__tree_delete(struct rb_root *root)
|
|||||||
rb_erase(node, root);
|
rb_erase(node, root);
|
||||||
pos = rb_entry(node, struct annotated_data_type, node);
|
pos = rb_entry(node, struct annotated_data_type, node);
|
||||||
delete_members(&pos->self);
|
delete_members(&pos->self);
|
||||||
|
delete_data_type_histograms(pos);
|
||||||
free(pos->self.type_name);
|
free(pos->self.type_name);
|
||||||
free(pos);
|
free(pos);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* annotated_data_type__update_samples - Update histogram
|
||||||
|
* @adt: Data type to update
|
||||||
|
* @evsel: Event to update
|
||||||
|
* @offset: Offset in the type
|
||||||
|
* @nr_samples: Number of samples at this offset
|
||||||
|
* @period: Event count at this offset
|
||||||
|
*
|
||||||
|
* This function updates type histogram at @ofs for @evsel. Samples are
|
||||||
|
* aggregated before calling this function so it can be called with more
|
||||||
|
* than one samples at a certain offset.
|
||||||
|
*/
|
||||||
|
int annotated_data_type__update_samples(struct annotated_data_type *adt,
|
||||||
|
struct evsel *evsel, int offset,
|
||||||
|
int nr_samples, u64 period)
|
||||||
|
{
|
||||||
|
struct type_hist *h;
|
||||||
|
|
||||||
|
if (adt == NULL)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (adt->histograms == NULL) {
|
||||||
|
int nr = evsel->evlist->core.nr_entries;
|
||||||
|
|
||||||
|
if (alloc_data_type_histograms(adt, nr) < 0)
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (offset < 0 || offset >= adt->self.size)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
h = adt->histograms[evsel->core.idx];
|
||||||
|
|
||||||
|
h->nr_samples += nr_samples;
|
||||||
|
h->addr[offset].nr_samples += nr_samples;
|
||||||
|
h->period += period;
|
||||||
|
h->addr[offset].period += period;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
@ -7,6 +7,7 @@
|
|||||||
#include <linux/rbtree.h>
|
#include <linux/rbtree.h>
|
||||||
#include <linux/types.h>
|
#include <linux/types.h>
|
||||||
|
|
||||||
|
struct evsel;
|
||||||
struct map_symbol;
|
struct map_symbol;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -29,16 +30,42 @@ struct annotated_member {
|
|||||||
int size;
|
int size;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* struct type_hist_entry - Histogram entry per offset
|
||||||
|
* @nr_samples: Number of samples
|
||||||
|
* @period: Count of event
|
||||||
|
*/
|
||||||
|
struct type_hist_entry {
|
||||||
|
int nr_samples;
|
||||||
|
u64 period;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* struct type_hist - Type histogram for each event
|
||||||
|
* @nr_samples: Total number of samples in this data type
|
||||||
|
* @period: Total count of the event in this data type
|
||||||
|
* @offset: Array of histogram entry
|
||||||
|
*/
|
||||||
|
struct type_hist {
|
||||||
|
u64 nr_samples;
|
||||||
|
u64 period;
|
||||||
|
struct type_hist_entry addr[];
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* struct annotated_data_type - Data type to profile
|
* struct annotated_data_type - Data type to profile
|
||||||
* @node: RB-tree node for dso->type_tree
|
* @node: RB-tree node for dso->type_tree
|
||||||
* @self: Actual type information
|
* @self: Actual type information
|
||||||
|
* @nr_histogram: Number of histogram entries
|
||||||
|
* @histograms: An array of pointers to histograms
|
||||||
*
|
*
|
||||||
* This represents a data type accessed by samples in the profile data.
|
* This represents a data type accessed by samples in the profile data.
|
||||||
*/
|
*/
|
||||||
struct annotated_data_type {
|
struct annotated_data_type {
|
||||||
struct rb_node node;
|
struct rb_node node;
|
||||||
struct annotated_member self;
|
struct annotated_member self;
|
||||||
|
int nr_histograms;
|
||||||
|
struct type_hist **histograms;
|
||||||
};
|
};
|
||||||
|
|
||||||
extern struct annotated_data_type unknown_type;
|
extern struct annotated_data_type unknown_type;
|
||||||
@ -49,6 +76,11 @@ extern struct annotated_data_type unknown_type;
|
|||||||
struct annotated_data_type *find_data_type(struct map_symbol *ms, u64 ip,
|
struct annotated_data_type *find_data_type(struct map_symbol *ms, u64 ip,
|
||||||
int reg, int offset);
|
int reg, int offset);
|
||||||
|
|
||||||
|
/* Update type access histogram at the given offset */
|
||||||
|
int annotated_data_type__update_samples(struct annotated_data_type *adt,
|
||||||
|
struct evsel *evsel, int offset,
|
||||||
|
int nr_samples, u64 period);
|
||||||
|
|
||||||
/* Release all data type information in the tree */
|
/* Release all data type information in the tree */
|
||||||
void annotated_data_type__tree_delete(struct rb_root *root);
|
void annotated_data_type__tree_delete(struct rb_root *root);
|
||||||
|
|
||||||
@ -61,6 +93,16 @@ find_data_type(struct map_symbol *ms __maybe_unused, u64 ip __maybe_unused,
|
|||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline int
|
||||||
|
annotated_data_type__update_samples(struct annotated_data_type *adt __maybe_unused,
|
||||||
|
struct evsel *evsel __maybe_unused,
|
||||||
|
int offset __maybe_unused,
|
||||||
|
int nr_samples __maybe_unused,
|
||||||
|
u64 period __maybe_unused)
|
||||||
|
{
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
static inline void annotated_data_type__tree_delete(struct rb_root *root __maybe_unused)
|
static inline void annotated_data_type__tree_delete(struct rb_root *root __maybe_unused)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
@ -3679,6 +3679,7 @@ struct annotated_data_type *hist_entry__get_data_type(struct hist_entry *he)
|
|||||||
struct disasm_line *dl;
|
struct disasm_line *dl;
|
||||||
struct annotated_insn_loc loc;
|
struct annotated_insn_loc loc;
|
||||||
struct annotated_op_loc *op_loc;
|
struct annotated_op_loc *op_loc;
|
||||||
|
struct annotated_data_type *mem_type;
|
||||||
u64 ip = he->ip;
|
u64 ip = he->ip;
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
@ -3709,7 +3710,13 @@ struct annotated_data_type *hist_entry__get_data_type(struct hist_entry *he)
|
|||||||
if (!op_loc->mem_ref)
|
if (!op_loc->mem_ref)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
return find_data_type(ms, ip, op_loc->reg, op_loc->offset);
|
mem_type = find_data_type(ms, ip, op_loc->reg, op_loc->offset);
|
||||||
|
|
||||||
|
annotated_data_type__update_samples(mem_type, evsel,
|
||||||
|
op_loc->offset,
|
||||||
|
he->stat.nr_events,
|
||||||
|
he->stat.period);
|
||||||
|
return mem_type;
|
||||||
}
|
}
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user