From: Arnaldo Carvalho de Melo <acme@kernel.org>
To: Jiri Olsa <jolsa@kernel.org>
Cc: lkml <linux-kernel@vger.kernel.org>,
"David Ahern" <dsahern@gmail.com>,
"Ingo Molnar" <mingo@kernel.org>,
"Namhyung Kim" <namhyung@kernel.org>,
"Peter Zijlstra" <a.p.zijlstra@chello.nl>,
"Matt Fleming" <matt@codeblueprint.co.uk>,
"Raphaël Beamonte" <raphael.beamonte@gmail.com>
Subject: Re: [PATCH 4/5] perf tools: Propagate error info from tp_format
Date: Wed, 9 Sep 2015 17:58:13 -0300 [thread overview]
Message-ID: <20150909205813.GV3475@kernel.org> (raw)
In-Reply-To: <1441615087-13886-5-git-send-email-jolsa@kernel.org>
Em Mon, Sep 07, 2015 at 10:38:06AM +0200, Jiri Olsa escreveu:
> Propagate error info from tp_format via ERR_PTR to get
> it all the way down to the parse-event.c tracepoint adding
> routines. Following functions now return pointer with
> encoded error:
> - tp_format
> - trace_event__tp_format
> - perf_evsel__newtp_idx
> - perf_evsel__newtp
>
> This affects several other places in perf, that cannot use
> pointer check anymore, but must utilize the err.h interface,
> when getting error information from above functions list.
Right, so this is tricky and we must be careful, see below...
> Link: http://lkml.kernel.org/n/tip-bzdckgv1zfp2y8up9l7ojt7y@git.kernel.org
> Signed-off-by: Jiri Olsa <jolsa@kernel.org>
> ---
> tools/perf/builtin-trace.c | 19 +++++++++++--------
> tools/perf/tests/evsel-tp-sched.c | 10 ++++++++--
> tools/perf/tests/openat-syscall-all-cpus.c | 3 ++-
> tools/perf/tests/openat-syscall-tp-fields.c | 3 ++-
> tools/perf/tests/openat-syscall.c | 3 ++-
> tools/perf/util/evlist.c | 3 ++-
> tools/perf/util/evsel.c | 11 +++++++++--
> tools/perf/util/evsel.h | 3 +++
> tools/perf/util/parse-events.c | 6 +++---
> tools/perf/util/trace-event.c | 13 +++++++++++--
> 10 files changed, 53 insertions(+), 21 deletions(-)
>
> diff --git a/tools/perf/builtin-trace.c b/tools/perf/builtin-trace.c
> index 215653274102..93b80f12f35e 100644
> --- a/tools/perf/builtin-trace.c
> +++ b/tools/perf/builtin-trace.c
> @@ -38,6 +38,7 @@
> #include <stdlib.h>
> #include <sys/mman.h>
> #include <linux/futex.h>
> +#include <linux/err.h>
>
> /* For older distros: */
> #ifndef MAP_STACK
> @@ -245,13 +246,14 @@ static struct perf_evsel *perf_evsel__syscall_newtp(const char *direction, void
> struct perf_evsel *evsel = perf_evsel__newtp("raw_syscalls", direction);
>
> /* older kernel (e.g., RHEL6) use syscalls:{enter,exit} */
> - if (evsel == NULL)
> + if (IS_ERR(evsel))
> evsel = perf_evsel__newtp("syscalls", direction);
>
> - if (evsel) {
> - if (perf_evsel__init_syscall_tp(evsel, handler))
> - goto out_delete;
> - }
> + if (IS_ERR(evsel))
> + return NULL;
> +
> + if (perf_evsel__init_syscall_tp(evsel, handler))
> + goto out_delete;
This kind of stuff is ok, as evsel is a local variable and you kept the
interface for perf_evsel__syscall_newtp(), i.e. it returns NULL if a new
evsel can't be instantiated.
Ok, but that is a different interface than the one used by
perf_evsel__newtp(), that also instantiates a new evsel.
So when one thinks about "foo__new()" we now need to check which one of
the two interfaces it uses, if err.h or if the old NULL based failure
reporting one.
Double tricky if it is foo__new() and foo__new_variant(), as
perf_evsel__syscall_newtp() and perf_evsel__newtp(), i.e. both will
return a "struct perf_evsel" instance, but one using err.h, the other
use NULL.
Ok, you marked the ones using a comment, wonder if we couldn't use
'sparse' somehow here, is it used to check IS_ERR() usage in the kernel?
Ah, but what about this in trace__event_handler() in builtin-trace.c?
if (evsel->tp_format) {
event_format__fprintf(evsel->tp_format, sample->cpu,
sample->raw_data, sample->raw_size,
trace->output);
}
Don't we have to use IS_ERR() here? Ok, no, because if setting up
evsel->tp_format fails, then that evsel will be destroyed and
perf_evsel__newtp() will return ERR_PTR(), so it is ok not no use
ERR_PTR(evsel->tp_format) because it will only be != NULL when it was
successfully set up.
But then, in perf_evsel__newtp_idx if zalloc() fails we will not return
ERR_PTR(), but instead NULL, a-ha, this one seems to be a real bug, no?
/*
* Returns pointer with encoded error via <linux/err.h> interface.
*/
struct perf_evsel *perf_evsel__newtp_idx(const char *sys, const char *name, int idx)
{
struct perf_evsel *evsel = zalloc(perf_evsel__object.size);
int err = -ENOMEM;
if (evsel != NULL) {
struct perf_event_attr attr = {
.type = PERF_TYPE_TRACEPOINT,
.sample_type = (PERF_SAMPLE_RAW | PERF_SAMPLE_TIME |
PERF_SAMPLE_CPU | PERF_SAMPLE_PERIOD),
};
if (asprintf(&evsel->name, "%s:%s", sys, name) < 0)
goto out_free;
evsel->tp_format = trace_event__tp_format(sys, name);
if (IS_ERR(evsel->tp_format)) {
err = PTR_ERR(evsel->tp_format);
goto out_free;
}
event_attr_init(&attr);
attr.config = evsel->tp_format->id;
attr.sample_period = 1;
perf_evsel__init(evsel, &attr, idx);
}
return evsel;
out_free:
zfree(&evsel->name);
free(evsel);
return ERR_PTR(err);
}
>
> return evsel;
>
> @@ -1705,12 +1707,12 @@ static int trace__read_syscall_info(struct trace *trace, int id)
> snprintf(tp_name, sizeof(tp_name), "sys_enter_%s", sc->name);
> sc->tp_format = trace_event__tp_format("syscalls", tp_name);
>
> - if (sc->tp_format == NULL && sc->fmt && sc->fmt->alias) {
> + if (IS_ERR(sc->tp_format) && sc->fmt && sc->fmt->alias) {
> snprintf(tp_name, sizeof(tp_name), "sys_enter_%s", sc->fmt->alias);
> sc->tp_format = trace_event__tp_format("syscalls", tp_name);
> }
>
> - if (sc->tp_format == NULL)
> + if (IS_ERR(sc->tp_format))
> return -1;
>
> sc->args = sc->tp_format->format.fields;
> @@ -2390,7 +2392,8 @@ static size_t trace__fprintf_thread_summary(struct trace *trace, FILE *fp);
> static bool perf_evlist__add_vfs_getname(struct perf_evlist *evlist)
> {
> struct perf_evsel *evsel = perf_evsel__newtp("probe", "vfs_getname");
> - if (evsel == NULL)
> +
> + if (IS_ERR(evsel))
> return false;
>
> if (perf_evsel__field(evsel, "pathname") == NULL) {
> diff --git a/tools/perf/tests/evsel-tp-sched.c b/tools/perf/tests/evsel-tp-sched.c
> index 52162425c969..790e413d9a1f 100644
> --- a/tools/perf/tests/evsel-tp-sched.c
> +++ b/tools/perf/tests/evsel-tp-sched.c
> @@ -1,3 +1,4 @@
> +#include <linux/err.h>
> #include <traceevent/event-parse.h>
> #include "evsel.h"
> #include "tests.h"
> @@ -36,8 +37,8 @@ int test__perf_evsel__tp_sched_test(void)
> struct perf_evsel *evsel = perf_evsel__newtp("sched", "sched_switch");
> int ret = 0;
>
> - if (evsel == NULL) {
> - pr_debug("perf_evsel__new\n");
> + if (IS_ERR(evsel)) {
> + pr_debug("perf_evsel__newtp failed with %ld\n", PTR_ERR(evsel));
> return -1;
> }
>
> @@ -66,6 +67,11 @@ int test__perf_evsel__tp_sched_test(void)
>
> evsel = perf_evsel__newtp("sched", "sched_wakeup");
>
> + if (IS_ERR(evsel)) {
> + pr_debug("perf_evsel__newtp failed with %ld\n", PTR_ERR(evsel));
> + return -1;
> + }
> +
> if (perf_evsel__test_field(evsel, "comm", 16, true))
> ret = -1;
>
> diff --git a/tools/perf/tests/openat-syscall-all-cpus.c b/tools/perf/tests/openat-syscall-all-cpus.c
> index 495d8126b722..9e104a2e973d 100644
> --- a/tools/perf/tests/openat-syscall-all-cpus.c
> +++ b/tools/perf/tests/openat-syscall-all-cpus.c
> @@ -1,4 +1,5 @@
> #include <api/fs/fs.h>
> +#include <linux/err.h>
> #include "evsel.h"
> #include "tests.h"
> #include "thread_map.h"
> @@ -31,7 +32,7 @@ int test__openat_syscall_event_on_all_cpus(void)
> CPU_ZERO(&cpu_set);
>
> evsel = perf_evsel__newtp("syscalls", "sys_enter_openat");
> - if (evsel == NULL) {
> + if (IS_ERR(evsel)) {
> tracing_path__strerror_open_tp(errno, errbuf, sizeof(errbuf), "syscalls", "sys_enter_openat");
> pr_err("%s\n", errbuf);
> goto out_thread_map_delete;
> diff --git a/tools/perf/tests/openat-syscall-tp-fields.c b/tools/perf/tests/openat-syscall-tp-fields.c
> index 01a19626c846..473d3869727e 100644
> --- a/tools/perf/tests/openat-syscall-tp-fields.c
> +++ b/tools/perf/tests/openat-syscall-tp-fields.c
> @@ -1,3 +1,4 @@
> +#include <linux/err.h>
> #include "perf.h"
> #include "evlist.h"
> #include "evsel.h"
> @@ -30,7 +31,7 @@ int test__syscall_openat_tp_fields(void)
> }
>
> evsel = perf_evsel__newtp("syscalls", "sys_enter_openat");
> - if (evsel == NULL) {
> + if (IS_ERR(evsel)) {
> pr_debug("%s: perf_evsel__newtp\n", __func__);
> goto out_delete_evlist;
> }
> diff --git a/tools/perf/tests/openat-syscall.c b/tools/perf/tests/openat-syscall.c
> index 08ac9d94a050..7b1db8306098 100644
> --- a/tools/perf/tests/openat-syscall.c
> +++ b/tools/perf/tests/openat-syscall.c
> @@ -1,4 +1,5 @@
> #include <api/fs/tracing_path.h>
> +#include <linux/err.h>
> #include "thread_map.h"
> #include "evsel.h"
> #include "debug.h"
> @@ -19,7 +20,7 @@ int test__openat_syscall_event(void)
> }
>
> evsel = perf_evsel__newtp("syscalls", "sys_enter_openat");
> - if (evsel == NULL) {
> + if (IS_ERR(evsel)) {
> tracing_path__strerror_open_tp(errno, errbuf, sizeof(errbuf), "syscalls", "sys_enter_openat");
> pr_err("%s\n", errbuf);
> goto out_thread_map_delete;
> diff --git a/tools/perf/util/evlist.c b/tools/perf/util/evlist.c
> index d51a5200c8af..3cb2bf9bd4bd 100644
> --- a/tools/perf/util/evlist.c
> +++ b/tools/perf/util/evlist.c
> @@ -25,6 +25,7 @@
> #include <linux/bitops.h>
> #include <linux/hash.h>
> #include <linux/log2.h>
> +#include <linux/err.h>
>
> static void perf_evlist__mmap_put(struct perf_evlist *evlist, int idx);
> static void __perf_evlist__munmap(struct perf_evlist *evlist, int idx);
> @@ -265,7 +266,7 @@ int perf_evlist__add_newtp(struct perf_evlist *evlist,
> {
> struct perf_evsel *evsel = perf_evsel__newtp(sys, name);
>
> - if (evsel == NULL)
> + if (IS_ERR(evsel))
> return -1;
>
> evsel->handler = handler;
> diff --git a/tools/perf/util/evsel.c b/tools/perf/util/evsel.c
> index 771ade4d5966..08c20ee4e27d 100644
> --- a/tools/perf/util/evsel.c
> +++ b/tools/perf/util/evsel.c
> @@ -13,6 +13,7 @@
> #include <traceevent/event-parse.h>
> #include <linux/hw_breakpoint.h>
> #include <linux/perf_event.h>
> +#include <linux/err.h>
> #include <sys/resource.h>
> #include "asm/bug.h"
> #include "callchain.h"
> @@ -225,9 +226,13 @@ struct perf_evsel *perf_evsel__new_idx(struct perf_event_attr *attr, int idx)
> return evsel;
> }
>
> +/*
> + * Returns pointer with encoded error via <linux/err.h> interface.
> + */
> struct perf_evsel *perf_evsel__newtp_idx(const char *sys, const char *name, int idx)
> {
> struct perf_evsel *evsel = zalloc(perf_evsel__object.size);
> + int err = -ENOMEM;
>
> if (evsel != NULL) {
> struct perf_event_attr attr = {
> @@ -240,8 +245,10 @@ struct perf_evsel *perf_evsel__newtp_idx(const char *sys, const char *name, int
> goto out_free;
>
> evsel->tp_format = trace_event__tp_format(sys, name);
> - if (evsel->tp_format == NULL)
> + if (IS_ERR(evsel->tp_format)) {
> + err = PTR_ERR(evsel->tp_format);
> goto out_free;
> + }
>
> event_attr_init(&attr);
> attr.config = evsel->tp_format->id;
> @@ -254,7 +261,7 @@ struct perf_evsel *perf_evsel__newtp_idx(const char *sys, const char *name, int
> out_free:
> zfree(&evsel->name);
> free(evsel);
> - return NULL;
> + return ERR_PTR(err);
> }
>
> const char *perf_evsel__hw_names[PERF_COUNT_HW_MAX] = {
> diff --git a/tools/perf/util/evsel.h b/tools/perf/util/evsel.h
> index 298e6bbca200..b6e8ff876f17 100644
> --- a/tools/perf/util/evsel.h
> +++ b/tools/perf/util/evsel.h
> @@ -161,6 +161,9 @@ static inline struct perf_evsel *perf_evsel__new(struct perf_event_attr *attr)
>
> struct perf_evsel *perf_evsel__newtp_idx(const char *sys, const char *name, int idx);
>
> +/*
> + * Returns pointer with encoded error via <linux/err.h> interface.
> + */
> static inline struct perf_evsel *perf_evsel__newtp(const char *sys, const char *name)
> {
> return perf_evsel__newtp_idx(sys, name, 0);
> diff --git a/tools/perf/util/parse-events.c b/tools/perf/util/parse-events.c
> index 1b284b8ad243..c47831c47220 100644
> --- a/tools/perf/util/parse-events.c
> +++ b/tools/perf/util/parse-events.c
> @@ -1,4 +1,5 @@
> #include <linux/hw_breakpoint.h>
> +#include <linux/err.h>
> #include "util.h"
> #include "../perf.h"
> #include "evlist.h"
> @@ -393,11 +394,10 @@ static int add_tracepoint(struct list_head *list, int *idx,
> struct perf_evsel *evsel;
>
> evsel = perf_evsel__newtp_idx(sys_name, evt_name, (*idx)++);
> - if (!evsel)
> - return -ENOMEM;
> + if (IS_ERR(evsel))
> + return PTR_ERR(evsel);
>
> list_add_tail(&evsel->node, list);
> -
> return 0;
> }
>
> diff --git a/tools/perf/util/trace-event.c b/tools/perf/util/trace-event.c
> index 2f4996ab313d..8e3a60e3e15f 100644
> --- a/tools/perf/util/trace-event.c
> +++ b/tools/perf/util/trace-event.c
> @@ -7,6 +7,7 @@
> #include <sys/stat.h>
> #include <fcntl.h>
> #include <linux/kernel.h>
> +#include <linux/err.h>
> #include <traceevent/event-parse.h>
> #include <api/fs/tracing_path.h>
> #include "trace-event.h"
> @@ -66,6 +67,9 @@ void trace_event__cleanup(struct trace_event *t)
> pevent_free(t->pevent);
> }
>
> +/*
> + * Returns pointer with encoded error via <linux/err.h> interface.
> + */
> static struct event_format*
> tp_format(const char *sys, const char *name)
> {
> @@ -74,12 +78,14 @@ tp_format(const char *sys, const char *name)
> char path[PATH_MAX];
> size_t size;
> char *data;
> + int err;
>
> scnprintf(path, PATH_MAX, "%s/%s/%s/format",
> tracing_events_path, sys, name);
>
> - if (filename__read_str(path, &data, &size))
> - return NULL;
> + err = filename__read_str(path, &data, &size);
> + if (err)
> + return ERR_PTR(err);
>
> pevent_parse_format(pevent, &event, data, size, sys);
>
> @@ -87,6 +93,9 @@ tp_format(const char *sys, const char *name)
> return event;
> }
>
> +/*
> + * Returns pointer with encoded error via <linux/err.h> interface.
> + */
> struct event_format*
> trace_event__tp_format(const char *sys, const char *name)
> {
> --
> 2.4.3
next prev parent reply other threads:[~2015-09-09 20:58 UTC|newest]
Thread overview: 49+ messages / expand[flat|nested] mbox.gz Atom feed top
2015-09-07 8:38 [PATCHv2 0/5] perf tools: Enhance parsing events tracepoint error output Jiri Olsa
2015-09-07 8:38 ` [PATCH 1/5] tools: Add err.h with ERR_PTR PTR_ERR interface Jiri Olsa
2015-09-08 20:22 ` Raphaël Beamonte
2015-09-08 20:24 ` Raphaël Beamonte
2015-09-08 21:06 ` Arnaldo Carvalho de Melo
2015-09-08 21:28 ` Raphaël Beamonte
2015-09-08 21:29 ` Raphaël Beamonte
2015-09-16 7:28 ` [tip:perf/core] " tip-bot for Jiri Olsa
2015-09-21 23:41 ` Vinson Lee
2015-09-29 6:35 ` Vinson Lee
2015-09-29 7:14 ` Jiri Olsa
2015-09-29 7:20 ` Jiri Olsa
2015-09-29 7:52 ` He Kuang
2015-09-29 7:57 ` Jiri Olsa
2015-09-29 8:15 ` He Kuang
2015-09-29 10:41 ` Jiri Olsa
2015-09-29 14:52 ` Arnaldo Carvalho de Melo
2015-09-29 15:05 ` [PATCH] perf tool: Fix shadowed declaration in parse-events.c Jiri Olsa
2015-10-01 7:10 ` [tip:perf/core] perf tools: " tip-bot for Jiri Olsa
2015-09-07 8:38 ` [PATCH 2/5] perf tools: Add tools/include into tags directories Jiri Olsa
2015-09-15 7:02 ` [tip:perf/core] perf tools: Add tools/ include " tip-bot for Jiri Olsa
2015-09-07 8:38 ` [PATCH 3/5] perf tools: Propagate error info for the tracepoint parsing Jiri Olsa
2015-09-08 21:42 ` Raphaël Beamonte
2015-09-09 7:50 ` Jiri Olsa
2015-09-12 10:54 ` Matt Fleming
2015-09-16 7:29 ` [tip:perf/core] " tip-bot for Jiri Olsa
2015-09-07 8:38 ` [PATCH 4/5] perf tools: Propagate error info from tp_format Jiri Olsa
2015-09-09 20:58 ` Arnaldo Carvalho de Melo [this message]
2015-09-10 8:24 ` Jiri Olsa
2015-09-10 14:16 ` Arnaldo Carvalho de Melo
2015-09-14 20:53 ` Arnaldo Carvalho de Melo
2015-09-14 20:59 ` Raphaël Beamonte
2015-09-14 21:36 ` Arnaldo Carvalho de Melo
2015-09-14 22:05 ` Raphaël Beamonte
2015-09-15 2:35 ` Arnaldo Carvalho de Melo
2015-09-14 21:02 ` Arnaldo Carvalho de Melo
2015-09-16 7:29 ` [tip:perf/core] perf evsel: " tip-bot for Jiri Olsa
2015-09-07 8:38 ` [PATCH 5/5] perf tools: Enhance parsing events tracepoint error output Jiri Olsa
2015-09-10 7:00 ` Namhyung Kim
2015-09-10 8:05 ` Jiri Olsa
2015-09-11 16:09 ` Namhyung Kim
2015-09-11 16:16 ` Jiri Olsa
2015-09-11 17:50 ` Raphaël Beamonte
2015-09-11 18:55 ` Arnaldo Carvalho de Melo
2015-09-11 19:56 ` Raphaël Beamonte
2015-09-11 20:22 ` Arnaldo Carvalho de Melo
2015-09-11 22:01 ` Raphaël Beamonte
2015-09-14 20:59 ` Arnaldo Carvalho de Melo
2015-09-16 7:29 ` [tip:perf/core] " tip-bot for Jiri Olsa
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=20150909205813.GV3475@kernel.org \
--to=acme@kernel.org \
--cc=a.p.zijlstra@chello.nl \
--cc=dsahern@gmail.com \
--cc=jolsa@kernel.org \
--cc=linux-kernel@vger.kernel.org \
--cc=matt@codeblueprint.co.uk \
--cc=mingo@kernel.org \
--cc=namhyung@kernel.org \
--cc=raphael.beamonte@gmail.com \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).