]> git.proxmox.com Git - mirror_ubuntu-kernels.git/commitdiff
libbpf: Make perf_buffer__new() use OPTS-based interface
authorAndrii Nakryiko <andrii@kernel.org>
Thu, 11 Nov 2021 05:36:20 +0000 (21:36 -0800)
committerAlexei Starovoitov <ast@kernel.org>
Fri, 12 Nov 2021 00:54:05 +0000 (16:54 -0800)
Add new variants of perf_buffer__new() and perf_buffer__new_raw() that
use OPTS-based options for future extensibility ([0]). Given all the
currently used API names are best fits, re-use them and use
___libbpf_override() approach and symbol versioning to preserve ABI and
source code compatibility. struct perf_buffer_opts and struct
perf_buffer_raw_opts are kept as well, but they are restructured such
that they are OPTS-based when used with new APIs. For struct
perf_buffer_raw_opts we keep few fields intact, so we have to also
preserve the memory location of them both when used as OPTS and for
legacy API variants. This is achieved with anonymous padding for OPTS
"incarnation" of the struct.  These pads can be eventually used for new
options.

  [0] Closes: https://github.com/libbpf/libbpf/issues/311

Signed-off-by: Andrii Nakryiko <andrii@kernel.org>
Signed-off-by: Alexei Starovoitov <ast@kernel.org>
Link: https://lore.kernel.org/bpf/20211111053624.190580-6-andrii@kernel.org
tools/lib/bpf/libbpf.c
tools/lib/bpf/libbpf.h
tools/lib/bpf/libbpf.map

index a823b5ed705b1c5e9650850d054c96f5f6fcc44a..7db14859b27c3c04141996a00ec68fd5fa4cbb1f 100644 (file)
@@ -10578,11 +10578,18 @@ error:
 static struct perf_buffer *__perf_buffer__new(int map_fd, size_t page_cnt,
                                              struct perf_buffer_params *p);
 
-struct perf_buffer *perf_buffer__new(int map_fd, size_t page_cnt,
-                                    const struct perf_buffer_opts *opts)
+DEFAULT_VERSION(perf_buffer__new_v0_6_0, perf_buffer__new, LIBBPF_0.6.0)
+struct perf_buffer *perf_buffer__new_v0_6_0(int map_fd, size_t page_cnt,
+                                           perf_buffer_sample_fn sample_cb,
+                                           perf_buffer_lost_fn lost_cb,
+                                           void *ctx,
+                                           const struct perf_buffer_opts *opts)
 {
        struct perf_buffer_params p = {};
-       struct perf_event_attr attr = { 0, };
+       struct perf_event_attr attr = {};
+
+       if (!OPTS_VALID(opts, perf_buffer_opts))
+               return libbpf_err_ptr(-EINVAL);
 
        attr.config = PERF_COUNT_SW_BPF_OUTPUT;
        attr.type = PERF_TYPE_SOFTWARE;
@@ -10591,29 +10598,62 @@ struct perf_buffer *perf_buffer__new(int map_fd, size_t page_cnt,
        attr.wakeup_events = 1;
 
        p.attr = &attr;
-       p.sample_cb = opts ? opts->sample_cb : NULL;
-       p.lost_cb = opts ? opts->lost_cb : NULL;
-       p.ctx = opts ? opts->ctx : NULL;
+       p.sample_cb = sample_cb;
+       p.lost_cb = lost_cb;
+       p.ctx = ctx;
 
        return libbpf_ptr(__perf_buffer__new(map_fd, page_cnt, &p));
 }
 
-struct perf_buffer *
-perf_buffer__new_raw(int map_fd, size_t page_cnt,
-                    const struct perf_buffer_raw_opts *opts)
+COMPAT_VERSION(perf_buffer__new_deprecated, perf_buffer__new, LIBBPF_0.0.4)
+struct perf_buffer *perf_buffer__new_deprecated(int map_fd, size_t page_cnt,
+                                               const struct perf_buffer_opts *opts)
+{
+       return perf_buffer__new_v0_6_0(map_fd, page_cnt,
+                                      opts ? opts->sample_cb : NULL,
+                                      opts ? opts->lost_cb : NULL,
+                                      opts ? opts->ctx : NULL,
+                                      NULL);
+}
+
+DEFAULT_VERSION(perf_buffer__new_raw_v0_6_0, perf_buffer__new_raw, LIBBPF_0.6.0)
+struct perf_buffer *perf_buffer__new_raw_v0_6_0(int map_fd, size_t page_cnt,
+                                               struct perf_event_attr *attr,
+                                               perf_buffer_event_fn event_cb, void *ctx,
+                                               const struct perf_buffer_raw_opts *opts)
 {
        struct perf_buffer_params p = {};
 
-       p.attr = opts->attr;
-       p.event_cb = opts->event_cb;
-       p.ctx = opts->ctx;
-       p.cpu_cnt = opts->cpu_cnt;
-       p.cpus = opts->cpus;
-       p.map_keys = opts->map_keys;
+       if (page_cnt == 0 || !attr)
+               return libbpf_err_ptr(-EINVAL);
+
+       if (!OPTS_VALID(opts, perf_buffer_raw_opts))
+               return libbpf_err_ptr(-EINVAL);
+
+       p.attr = attr;
+       p.event_cb = event_cb;
+       p.ctx = ctx;
+       p.cpu_cnt = OPTS_GET(opts, cpu_cnt, 0);
+       p.cpus = OPTS_GET(opts, cpus, NULL);
+       p.map_keys = OPTS_GET(opts, map_keys, NULL);
 
        return libbpf_ptr(__perf_buffer__new(map_fd, page_cnt, &p));
 }
 
+COMPAT_VERSION(perf_buffer__new_raw_deprecated, perf_buffer__new_raw, LIBBPF_0.0.4)
+struct perf_buffer *perf_buffer__new_raw_deprecated(int map_fd, size_t page_cnt,
+                                                   const struct perf_buffer_raw_opts *opts)
+{
+       LIBBPF_OPTS(perf_buffer_raw_opts, inner_opts,
+               .cpu_cnt = opts->cpu_cnt,
+               .cpus = opts->cpus,
+               .map_keys = opts->map_keys,
+       );
+
+       return perf_buffer__new_raw_v0_6_0(map_fd, page_cnt, opts->attr,
+                                          opts->event_cb, opts->ctx, &inner_opts);
+}
+
 static struct perf_buffer *__perf_buffer__new(int map_fd, size_t page_cnt,
                                              struct perf_buffer_params *p)
 {
index f69512ae85055841b7a9a6bcd53fce71c8e83ad5..4ec69f22434260ce03a3989221ff59aa3643a7a1 100644 (file)
@@ -778,18 +778,52 @@ typedef void (*perf_buffer_lost_fn)(void *ctx, int cpu, __u64 cnt);
 
 /* common use perf buffer options */
 struct perf_buffer_opts {
-       /* if specified, sample_cb is called for each sample */
-       perf_buffer_sample_fn sample_cb;
-       /* if specified, lost_cb is called for each batch of lost samples */
-       perf_buffer_lost_fn lost_cb;
-       /* ctx is provided to sample_cb and lost_cb */
-       void *ctx;
+       union {
+               size_t sz;
+               struct { /* DEPRECATED: will be removed in v1.0 */
+                       /* if specified, sample_cb is called for each sample */
+                       perf_buffer_sample_fn sample_cb;
+                       /* if specified, lost_cb is called for each batch of lost samples */
+                       perf_buffer_lost_fn lost_cb;
+                       /* ctx is provided to sample_cb and lost_cb */
+                       void *ctx;
+               };
+       };
 };
+#define perf_buffer_opts__last_field sz
 
+/**
+ * @brief **perf_buffer__new()** creates BPF perfbuf manager for a specified
+ * BPF_PERF_EVENT_ARRAY map
+ * @param map_fd FD of BPF_PERF_EVENT_ARRAY BPF map that will be used by BPF
+ * code to send data over to user-space
+ * @param page_cnt number of memory pages allocated for each per-CPU buffer
+ * @param sample_cb function called on each received data record
+ * @param lost_cb function called when record loss has occurred
+ * @param ctx user-provided extra context passed into *sample_cb* and *lost_cb*
+ * @return a new instance of struct perf_buffer on success, NULL on error with
+ * *errno* containing an error code
+ */
 LIBBPF_API struct perf_buffer *
 perf_buffer__new(int map_fd, size_t page_cnt,
+                perf_buffer_sample_fn sample_cb, perf_buffer_lost_fn lost_cb, void *ctx,
                 const struct perf_buffer_opts *opts);
 
+LIBBPF_API struct perf_buffer *
+perf_buffer__new_v0_6_0(int map_fd, size_t page_cnt,
+                       perf_buffer_sample_fn sample_cb, perf_buffer_lost_fn lost_cb, void *ctx,
+                       const struct perf_buffer_opts *opts);
+
+LIBBPF_API LIBBPF_DEPRECATED_SINCE(0, 7, "use new variant of perf_buffer__new() instead")
+struct perf_buffer *perf_buffer__new_deprecated(int map_fd, size_t page_cnt,
+                                               const struct perf_buffer_opts *opts);
+
+#define perf_buffer__new(...) ___libbpf_overload(___perf_buffer_new, __VA_ARGS__)
+#define ___perf_buffer_new6(map_fd, page_cnt, sample_cb, lost_cb, ctx, opts) \
+       perf_buffer__new(map_fd, page_cnt, sample_cb, lost_cb, ctx, opts)
+#define ___perf_buffer_new3(map_fd, page_cnt, opts) \
+       perf_buffer__new_deprecated(map_fd, page_cnt, opts)
+
 enum bpf_perf_event_ret {
        LIBBPF_PERF_EVENT_DONE  = 0,
        LIBBPF_PERF_EVENT_ERROR = -1,
@@ -803,12 +837,21 @@ typedef enum bpf_perf_event_ret
 
 /* raw perf buffer options, giving most power and control */
 struct perf_buffer_raw_opts {
-       /* perf event attrs passed directly into perf_event_open() */
-       struct perf_event_attr *attr;
-       /* raw event callback */
-       perf_buffer_event_fn event_cb;
-       /* ctx is provided to event_cb */
-       void *ctx;
+       union {
+               struct {
+                       size_t sz;
+                       long :0;
+                       long :0;
+               };
+               struct { /* DEPRECATED: will be removed in v1.0 */
+                       /* perf event attrs passed directly into perf_event_open() */
+                       struct perf_event_attr *attr;
+                       /* raw event callback */
+                       perf_buffer_event_fn event_cb;
+                       /* ctx is provided to event_cb */
+                       void *ctx;
+               };
+       };
        /* if cpu_cnt == 0, open all on all possible CPUs (up to the number of
         * max_entries of given PERF_EVENT_ARRAY map)
         */
@@ -818,11 +861,28 @@ struct perf_buffer_raw_opts {
        /* if cpu_cnt > 0, map_keys specify map keys to set per-CPU FDs for */
        int *map_keys;
 };
+#define perf_buffer_raw_opts__last_field map_keys
 
 LIBBPF_API struct perf_buffer *
-perf_buffer__new_raw(int map_fd, size_t page_cnt,
+perf_buffer__new_raw(int map_fd, size_t page_cnt, struct perf_event_attr *attr,
+                    perf_buffer_event_fn event_cb, void *ctx,
                     const struct perf_buffer_raw_opts *opts);
 
+LIBBPF_API struct perf_buffer *
+perf_buffer__new_raw_v0_6_0(int map_fd, size_t page_cnt, struct perf_event_attr *attr,
+                           perf_buffer_event_fn event_cb, void *ctx,
+                           const struct perf_buffer_raw_opts *opts);
+
+LIBBPF_API LIBBPF_DEPRECATED_SINCE(0, 7, "use new variant of perf_buffer__new_raw() instead")
+struct perf_buffer *perf_buffer__new_raw_deprecated(int map_fd, size_t page_cnt,
+                                                   const struct perf_buffer_raw_opts *opts);
+
+#define perf_buffer__new_raw(...) ___libbpf_overload(___perf_buffer_new_raw, __VA_ARGS__)
+#define ___perf_buffer_new_raw6(map_fd, page_cnt, attr, event_cb, ctx, opts) \
+       perf_buffer__new_raw(map_fd, page_cnt, attr, event_cb, ctx, opts)
+#define ___perf_buffer_new_raw3(map_fd, page_cnt, opts) \
+       perf_buffer__new_raw_deprecated(map_fd, page_cnt, opts)
+
 LIBBPF_API void perf_buffer__free(struct perf_buffer *pb);
 LIBBPF_API int perf_buffer__epoll_fd(const struct perf_buffer *pb);
 LIBBPF_API int perf_buffer__poll(struct perf_buffer *pb, int timeout_ms);
index 07762d6723ef04726dc8f9582304fea3b29d05de..9e91aa8da303d0e9dd6c237901d1523af62de588 100644 (file)
@@ -409,4 +409,8 @@ LIBBPF_0.6.0 {
                btf__type_cnt;
                btf_dump__new;
                btf_dump__new_deprecated;
+               perf_buffer__new;
+               perf_buffer__new_deprecated;
+               perf_buffer__new_raw;
+               perf_buffer__new_raw_deprecated;
 } LIBBPF_0.5.0;