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-bzdckgv1zfp2y8up9l7oj...@git.kernel.org
> Signed-off-by: Jiri Olsa <jo...@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
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/

Reply via email to