Check for requried sample attributes using evsel rather than sample_type in
the session header. If the attribute for a default field is not present
for the event type (e.g., new command operating on file from older kernel)
the field is removed from the output list.
Expected event types must exist. For example, if a user specifies
-f trace:time,trace -f sw:time,cpu,sym
the perf.data file must contain both tracepoints and software events
(ie., it is an error if either does not exist in the file).
Attribute checking is done once at the beginning of perf-script rather than
for each sample.
Signed-off-by: David Ahern <[email protected]>
---
tools/perf/builtin-script.c | 109 ++++++++++++++++++++++++++++++++++++-------
tools/perf/util/session.c | 12 +++++
tools/perf/util/session.h | 3 +
3 files changed, 107 insertions(+), 17 deletions(-)
diff --git a/tools/perf/builtin-script.c b/tools/perf/builtin-script.c
index 6cf811a..6debb9c 100644
--- a/tools/perf/builtin-script.c
+++ b/tools/perf/builtin-script.c
@@ -51,6 +51,7 @@ struct output_option {
/* default set to maintain compatibility with current format */
static struct {
bool user_set;
+ bool wildcard_set;
u64 fields;
u64 invalid_fields;
} output[PERF_TYPE_MAX] = {
@@ -94,41 +95,111 @@ static bool output_set_by_user(void)
return false;
}
+static const char *output_field2str(enum perf_output_field field)
+{
+ int i, imax = sizeof(all_output_options) / sizeof(struct output_option);
+ const char *str = "";
+
+ for (i = 0; i < imax; ++i) {
+ if (all_output_options[i].field == field) {
+ str = all_output_options[i].str;
+ break;
+ }
+ }
+ return str;
+}
+
#define PRINT_FIELD(x) (output[attr->type].fields & PERF_OUTPUT_##x)
-static int perf_session__check_attr(struct perf_session *session,
- struct perf_event_attr *attr)
+static int perf_attr__check_stype(struct perf_event_attr *attr,
+ u64 sample_type, const char *sample_msg,
+ enum perf_output_field field)
{
+ int type = attr->type;
+ const char *evname;
+
+ if (attr->sample_type & sample_type)
+ return 0;
+
+ if (output[type].user_set) {
+ evname = __event_name(attr->type, attr->config);
+ pr_err("Samples for '%s' event do not have %s attribute set. "
+ "Cannot print '%s' field.\n",
+ evname, sample_msg, output_field2str(field));
+ return -1;
+ }
+
+ /* user did not ask for it explicitly so remove from the default list */
+ output[type].fields &= ~field;
+ evname = __event_name(attr->type, attr->config);
+ pr_debug("Samples for '%s' event do not have %s attribute set. "
+ "Skipping '%s' field.\n",
+ evname, sample_msg, output_field2str(field));
+
+ return 0;
+}
+
+static int perf_evsel__check_attr(struct perf_session *session,
+ struct perf_evsel *evsel)
+{
+ struct perf_event_attr *attr = &evsel->attr;
+
if (PRINT_FIELD(TRACE) &&
!perf_session__has_traces(session, "record -R"))
return -EINVAL;
if (PRINT_FIELD(SYM)) {
- if (!(session->sample_type & PERF_SAMPLE_IP)) {
- pr_err("Samples do not contain IP data.\n");
+ if (perf_attr__check_stype(attr, PERF_SAMPLE_IP, "IP",
+ PERF_OUTPUT_SYM))
return -EINVAL;
- }
+
if (!no_callchain &&
- !(session->sample_type & PERF_SAMPLE_CALLCHAIN))
+ !(attr->sample_type & PERF_SAMPLE_CALLCHAIN))
symbol_conf.use_callchain = false;
}
if ((PRINT_FIELD(PID) || PRINT_FIELD(TID)) &&
- !(session->sample_type & PERF_SAMPLE_TID)) {
- pr_err("Samples do not contain TID/PID data.\n");
+ perf_attr__check_stype(attr, PERF_SAMPLE_TID, "TID",
+ PERF_OUTPUT_TID|PERF_OUTPUT_PID))
return -EINVAL;
- }
if (PRINT_FIELD(TIME) &&
- !(session->sample_type & PERF_SAMPLE_TIME)) {
- pr_err("Samples do not contain timestamps.\n");
+ perf_attr__check_stype(attr, PERF_SAMPLE_TIME, "TIME",
+ PERF_OUTPUT_TIME))
return -EINVAL;
- }
if (PRINT_FIELD(CPU) &&
- !(session->sample_type & PERF_SAMPLE_CPU)) {
- pr_err("Samples do not contain cpu.\n");
+ perf_attr__check_stype(attr, PERF_SAMPLE_CPU, "CPU",
+ PERF_OUTPUT_CPU))
return -EINVAL;
+
+ return 0;
+}
+
+/* verify all user requested events exist and the samples
+ * have the expected data
+ */
+static int perf_session__check_output_opt(struct perf_session *session)
+{
+ int j;
+ struct perf_evsel *evsel;
+
+ for (j = 0; j < PERF_TYPE_MAX; ++j) {
+ evsel = perf_session__find_event(session, j);
+
+ /* even if fields is set to 0 (ie., show nothing) event must
+ * exist if user explicitly includes it on the command line
+ */
+ if (!evsel && output[j].user_set && !output[j].wildcard_set) {
+ pr_err("%s events do not exist. "
+ "Remove corresponding -f option to proceed.\n",
+ event_type(j));
+ return -1;
+ }
+
+ if (evsel && output[j].fields &&
+ perf_evsel__check_attr(session, evsel))
+ return -1;
}
return 0;
@@ -200,9 +271,6 @@ static void process_event(union perf_event *event __unused,
if (output[attr->type].fields == 0)
return;
- if (perf_session__check_attr(session, attr) < 0)
- return;
-
print_sample_start(sample, thread, attr);
if (PRINT_FIELD(TRACE))
@@ -513,6 +581,7 @@ static int parse_output_fields(const struct option *opt __used,
output[type].fields = 0;
output[type].user_set = true;
+ output[type].wildcard_set = false;
} else {
tok = str;
@@ -529,6 +598,7 @@ static int parse_output_fields(const struct option *opt __used,
for (j = 0; j < PERF_TYPE_MAX; ++j) {
output[j].fields = 0;
output[j].user_set = true;
+ output[j].wildcard_set = true;
}
}
@@ -1133,6 +1203,11 @@ int cmd_script(int argc, const char **argv, const char *prefix __used)
pr_debug("perf script started with script %s\n\n", script_name);
}
+
+ err = perf_session__check_output_opt(session);
+ if (err < 0)
+ goto out;
+
err = __cmd_script(session);
perf_session__delete(session);
diff --git a/tools/perf/util/session.c b/tools/perf/util/session.c
index caa2245..f0b79ab 100644
--- a/tools/perf/util/session.c
+++ b/tools/perf/util/session.c
@@ -1156,6 +1156,18 @@ size_t perf_session__fprintf_nr_events(struct perf_session *session, FILE *fp)
return ret;
}
+struct perf_evsel *perf_session__find_event(struct perf_session *session,
+ unsigned int type)
+{
+ struct perf_evsel *pos;
+
+ list_for_each_entry(pos, &session->evlist->entries, node) {
+ if (pos->attr.type == type)
+ return pos;
+ }
+ return NULL;
+}
+
void perf_session__print_symbols(union perf_event *event,
struct perf_sample *sample,
struct perf_session *session)
diff --git a/tools/perf/util/session.h b/tools/perf/util/session.h
index 1ac481f..6bfc001 100644
--- a/tools/perf/util/session.h
+++ b/tools/perf/util/session.h
@@ -162,6 +162,9 @@ static inline int perf_session__parse_sample(struct perf_session *session,
session->sample_id_all, sample);
}
+struct perf_evsel *perf_session__find_event(struct perf_session *session,
+ unsigned int type);
+
void perf_session__print_symbols(union perf_event *event,
struct perf_sample *sample,
struct perf_session *session);
--
1.7.4
Em Sun, Apr 03, 2011 at 08:35:00AM -0600, David Ahern escreveu:
> +++ b/tools/perf/builtin-script.c
> @@ -94,41 +95,111 @@ static bool output_set_by_user(void)
> +static const char *output_field2str(enum perf_output_field field)
> +{
> + int i, imax = sizeof(all_output_options) / sizeof(struct output_option);
Use ARRAY_SIZE(all_output_options)
> + for (i = 0; i < imax; ++i) {
> + if (all_output_options[i].field == field) {
> + str = all_output_options[i].str;
> + break;
> +static int perf_attr__check_stype(struct perf_event_attr *attr,
> + u64 sample_type, const char *sample_msg,
> + enum perf_output_field field)
s/perf_attr__check_stype/perf_event_attr__check_stype/g
> + /* user did not ask for it explicitly so remove from the default list */
> + output[type].fields &= ~field;
> + evname = __event_name(attr->type, attr->config);
> + pr_debug("Samples for '%s' event do not have %s attribute set. "
> + "Skipping '%s' field.\n",
> + evname, sample_msg, output_field2str(field));
pr_warning?
> +static int perf_evsel__check_attr(struct perf_session *session,
> + struct perf_evsel *evsel)
for consistency, please make evsel the first parameter.
> +/* verify all user requested events exist and the samples
> + * have the expected data
> + */
> +static int perf_session__check_output_opt(struct perf_session *session)
> +{
> + int j;
> + struct perf_evsel *evsel;
> +
> + for (j = 0; j < PERF_TYPE_MAX; ++j) {
> + evsel = perf_session__find_event(session, j);
You're nog finding an specific event, you're looking for the first event
of type j, so I think perf_session__find_first_evtype is clearer.
> +
> + /* even if fields is set to 0 (ie., show nothing) event must
> + * exist if user explicitly includes it on the command line
> + */
Please use:
/*
* even if fields is set to 0 (ie., show nothing) event must
* exist if user explicitly includes it on the command line
*/
There are other places like this, please fix those too.
- Arnaldo
On 04/06/11 12:58, Arnaldo Carvalho de Melo wrote:
> Em Sun, Apr 03, 2011 at 08:35:00AM -0600, David Ahern escreveu:
>> +++ b/tools/perf/builtin-script.c
>> @@ -94,41 +95,111 @@ static bool output_set_by_user(void)
>> +static const char *output_field2str(enum perf_output_field field)
>> +{
>> + int i, imax = sizeof(all_output_options) / sizeof(struct output_option);
>
> Use ARRAY_SIZE(all_output_options)
>
>> + for (i = 0; i < imax; ++i) {
>> + if (all_output_options[i].field == field) {
>> + str = all_output_options[i].str;
>> + break;
>
>> +static int perf_attr__check_stype(struct perf_event_attr *attr,
>> + u64 sample_type, const char *sample_msg,
>> + enum perf_output_field field)
>
> s/perf_attr__check_stype/perf_event_attr__check_stype/g
>
>> + /* user did not ask for it explicitly so remove from the default list */
>> + output[type].fields &= ~field;
>> + evname = __event_name(attr->type, attr->config);
>> + pr_debug("Samples for '%s' event do not have %s attribute set. "
>> + "Skipping '%s' field.\n",
>> + evname, sample_msg, output_field2str(field));
>
> pr_warning?
In this case, it seems more like a debug. You have default options which
are not available for the specific perf.data file.
>
>> +static int perf_evsel__check_attr(struct perf_session *session,
>> + struct perf_evsel *evsel)
>
> for consistency, please make evsel the first parameter.
>
>> +/* verify all user requested events exist and the samples
>> + * have the expected data
>> + */
>> +static int perf_session__check_output_opt(struct perf_session *session)
>> +{
>> + int j;
>> + struct perf_evsel *evsel;
>> +
>> + for (j = 0; j < PERF_TYPE_MAX; ++j) {
>> + evsel = perf_session__find_event(session, j);
>
> You're nog finding an specific event, you're looking for the first event
> of type j, so I think perf_session__find_first_evtype is clearer.
>
>> +
>> + /* even if fields is set to 0 (ie., show nothing) event must
>> + * exist if user explicitly includes it on the command line
>> + */
>
> Please use:
>
> /*
> * even if fields is set to 0 (ie., show nothing) event must
> * exist if user explicitly includes it on the command line
> */
>
> There are other places like this, please fix those too.
Ok, I'll make the changes and resend.
David
>
> - Arnaldo