Commit f0bb7ee8 authored by Adrian Hunter's avatar Adrian Hunter Committed by Arnaldo Carvalho de Melo

perf auxtrace: Add support for AUX area sample recording

Add support for parsing and validating AUX area sample options. At
present, the only option is the sample size, but it is also necessary to
ensure that events are in a group with an AUX area event as the leader.

Committer note:

Add missing 'static inline' in front of auxtrace_parse_sample_options()
for when we don't HAVE_AUXTRACE_SUPPORT.
Signed-off-by: default avatarAdrian Hunter <adrian.hunter@intel.com>
Cc: Jiri Olsa <jolsa@redhat.com>
Link: http://lore.kernel.org/lkml/20191115124225.5247-6-adrian.hunter@intel.comSigned-off-by: default avatarArnaldo Carvalho de Melo <acme@redhat.com>
parent f306de27
...@@ -69,6 +69,13 @@ static struct perf_pmu *perf_evsel__find_pmu(struct evsel *evsel) ...@@ -69,6 +69,13 @@ static struct perf_pmu *perf_evsel__find_pmu(struct evsel *evsel)
return pmu; return pmu;
} }
static bool perf_evsel__is_aux_event(struct evsel *evsel)
{
struct perf_pmu *pmu = perf_evsel__find_pmu(evsel);
return pmu && pmu->auxtrace;
}
static bool auxtrace__dont_decode(struct perf_session *session) static bool auxtrace__dont_decode(struct perf_session *session)
{ {
return !session->itrace_synth_opts || return !session->itrace_synth_opts ||
...@@ -609,6 +616,106 @@ int auxtrace_parse_snapshot_options(struct auxtrace_record *itr, ...@@ -609,6 +616,106 @@ int auxtrace_parse_snapshot_options(struct auxtrace_record *itr,
return -EINVAL; return -EINVAL;
} }
/*
* Event record size is 16-bit which results in a maximum size of about 64KiB.
* Allow about 4KiB for the rest of the sample record, to give a maximum
* AUX area sample size of 60KiB.
*/
#define MAX_AUX_SAMPLE_SIZE (60 * 1024)
/* Arbitrary default size if no other default provided */
#define DEFAULT_AUX_SAMPLE_SIZE (4 * 1024)
static int auxtrace_validate_aux_sample_size(struct evlist *evlist,
struct record_opts *opts)
{
struct evsel *evsel;
bool has_aux_leader = false;
u32 sz;
evlist__for_each_entry(evlist, evsel) {
sz = evsel->core.attr.aux_sample_size;
if (perf_evsel__is_group_leader(evsel)) {
has_aux_leader = perf_evsel__is_aux_event(evsel);
if (sz) {
if (has_aux_leader)
pr_err("Cannot add AUX area sampling to an AUX area event\n");
else
pr_err("Cannot add AUX area sampling to a group leader\n");
return -EINVAL;
}
}
if (sz > MAX_AUX_SAMPLE_SIZE) {
pr_err("AUX area sample size %u too big, max. %d\n",
sz, MAX_AUX_SAMPLE_SIZE);
return -EINVAL;
}
if (sz) {
if (!has_aux_leader) {
pr_err("Cannot add AUX area sampling because group leader is not an AUX area event\n");
return -EINVAL;
}
perf_evsel__set_sample_bit(evsel, AUX);
opts->auxtrace_sample_mode = true;
} else {
perf_evsel__reset_sample_bit(evsel, AUX);
}
}
if (!opts->auxtrace_sample_mode) {
pr_err("AUX area sampling requires an AUX area event group leader plus other events to which to add samples\n");
return -EINVAL;
}
if (!perf_can_aux_sample()) {
pr_err("AUX area sampling is not supported by kernel\n");
return -EINVAL;
}
return 0;
}
int auxtrace_parse_sample_options(struct auxtrace_record *itr,
struct evlist *evlist,
struct record_opts *opts, const char *str)
{
bool has_aux_leader = false;
struct evsel *evsel;
char *endptr;
unsigned long sz;
if (!str)
return 0;
if (!itr) {
pr_err("No AUX area event to sample\n");
return -EINVAL;
}
sz = strtoul(str, &endptr, 0);
if (*endptr || sz > UINT_MAX) {
pr_err("Bad AUX area sampling option: '%s'\n", str);
return -EINVAL;
}
if (!sz)
sz = itr->default_aux_sample_size;
if (!sz)
sz = DEFAULT_AUX_SAMPLE_SIZE;
/* Set aux_sample_size based on --aux-sample option */
evlist__for_each_entry(evlist, evsel) {
if (perf_evsel__is_group_leader(evsel)) {
has_aux_leader = perf_evsel__is_aux_event(evsel);
} else if (has_aux_leader) {
evsel->core.attr.aux_sample_size = sz;
}
}
return auxtrace_validate_aux_sample_size(evlist, opts);
}
struct auxtrace_record *__weak struct auxtrace_record *__weak
auxtrace_record__init(struct evlist *evlist __maybe_unused, int *err) auxtrace_record__init(struct evlist *evlist __maybe_unused, int *err)
{ {
......
...@@ -313,6 +313,7 @@ struct auxtrace_mmap_params { ...@@ -313,6 +313,7 @@ struct auxtrace_mmap_params {
* @reference: provide a 64-bit reference number for auxtrace_event * @reference: provide a 64-bit reference number for auxtrace_event
* @read_finish: called after reading from an auxtrace mmap * @read_finish: called after reading from an auxtrace mmap
* @alignment: alignment (if any) for AUX area data * @alignment: alignment (if any) for AUX area data
* @default_aux_sample_size: default sample size for --aux sample option
*/ */
struct auxtrace_record { struct auxtrace_record {
int (*recording_options)(struct auxtrace_record *itr, int (*recording_options)(struct auxtrace_record *itr,
...@@ -336,6 +337,7 @@ struct auxtrace_record { ...@@ -336,6 +337,7 @@ struct auxtrace_record {
u64 (*reference)(struct auxtrace_record *itr); u64 (*reference)(struct auxtrace_record *itr);
int (*read_finish)(struct auxtrace_record *itr, int idx); int (*read_finish)(struct auxtrace_record *itr, int idx);
unsigned int alignment; unsigned int alignment;
unsigned int default_aux_sample_size;
}; };
/** /**
...@@ -498,6 +500,9 @@ struct auxtrace_record *auxtrace_record__init(struct evlist *evlist, ...@@ -498,6 +500,9 @@ struct auxtrace_record *auxtrace_record__init(struct evlist *evlist,
int auxtrace_parse_snapshot_options(struct auxtrace_record *itr, int auxtrace_parse_snapshot_options(struct auxtrace_record *itr,
struct record_opts *opts, struct record_opts *opts,
const char *str); const char *str);
int auxtrace_parse_sample_options(struct auxtrace_record *itr,
struct evlist *evlist,
struct record_opts *opts, const char *str);
int auxtrace_record__options(struct auxtrace_record *itr, int auxtrace_record__options(struct auxtrace_record *itr,
struct evlist *evlist, struct evlist *evlist,
struct record_opts *opts); struct record_opts *opts);
...@@ -648,6 +653,18 @@ int auxtrace_parse_snapshot_options(struct auxtrace_record *itr __maybe_unused, ...@@ -648,6 +653,18 @@ int auxtrace_parse_snapshot_options(struct auxtrace_record *itr __maybe_unused,
return -EINVAL; return -EINVAL;
} }
static inline
int auxtrace_parse_sample_options(struct auxtrace_record *itr __maybe_unused,
struct evlist *evlist __maybe_unused,
struct record_opts *opts __maybe_unused,
const char *str)
{
if (!str)
return 0;
pr_err("AUX area tracing not supported\n");
return -EINVAL;
}
static inline static inline
int auxtrace__process_event(struct perf_session *session __maybe_unused, int auxtrace__process_event(struct perf_session *session __maybe_unused,
union perf_event *event __maybe_unused, union perf_event *event __maybe_unused,
......
...@@ -26,6 +26,7 @@ struct perf_pmu { ...@@ -26,6 +26,7 @@ struct perf_pmu {
__u32 type; __u32 type;
bool selectable; bool selectable;
bool is_uncore; bool is_uncore;
bool auxtrace;
int max_precise; int max_precise;
struct perf_event_attr *default_config; struct perf_event_attr *default_config;
struct perf_cpu_map *cpus; struct perf_cpu_map *cpus;
......
...@@ -32,6 +32,7 @@ struct record_opts { ...@@ -32,6 +32,7 @@ struct record_opts {
bool full_auxtrace; bool full_auxtrace;
bool auxtrace_snapshot_mode; bool auxtrace_snapshot_mode;
bool auxtrace_snapshot_on_exit; bool auxtrace_snapshot_on_exit;
bool auxtrace_sample_mode;
bool record_namespaces; bool record_namespaces;
bool record_switch_events; bool record_switch_events;
bool all_kernel; bool all_kernel;
...@@ -56,6 +57,7 @@ struct record_opts { ...@@ -56,6 +57,7 @@ struct record_opts {
u64 user_interval; u64 user_interval;
size_t auxtrace_snapshot_size; size_t auxtrace_snapshot_size;
const char *auxtrace_snapshot_opts; const char *auxtrace_snapshot_opts;
const char *auxtrace_sample_opts;
bool sample_transaction; bool sample_transaction;
unsigned initial_delay; unsigned initial_delay;
bool use_clockid; bool use_clockid;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment