Skip to content

Commit

Permalink
bpf: Factor out bpf_link_by_id() helper.
Browse files Browse the repository at this point in the history
Refactor the code a bit to extract bpf_link_by_id() helper.
It's similar to existing bpf_prog_by_id().

Signed-off-by: Alexei Starovoitov <ast@kernel.org>
Signed-off-by: Daniel Borkmann <daniel@iogearbox.net>
Acked-by: Andrii Nakryiko <andriin@fb.com>
Acked-by: Song Liu <songliubraving@fb.com>
Link: https://lore.kernel.org/bpf/20200819042759.51280-2-alexei.starovoitov@gmail.com
  • Loading branch information
Alexei Starovoitov authored and Daniel Borkmann committed Aug 20, 2020
1 parent 6e9cab2 commit 005142b
Show file tree
Hide file tree
Showing 2 changed files with 29 additions and 18 deletions.
1 change: 1 addition & 0 deletions include/linux/bpf.h
Original file line number Diff line number Diff line change
Expand Up @@ -1358,6 +1358,7 @@ int btf_check_type_match(struct bpf_verifier_env *env, struct bpf_prog *prog,
struct btf *btf, const struct btf_type *t);

struct bpf_prog *bpf_prog_by_id(u32 id);
struct bpf_link *bpf_link_by_id(u32 id);

const struct bpf_func_proto *bpf_base_func_proto(enum bpf_func_id func_id);
#else /* !CONFIG_BPF_SYSCALL */
Expand Down
46 changes: 28 additions & 18 deletions kernel/bpf/syscall.c
Original file line number Diff line number Diff line change
Expand Up @@ -4014,40 +4014,50 @@ static int link_detach(union bpf_attr *attr)
return ret;
}

static int bpf_link_inc_not_zero(struct bpf_link *link)
static struct bpf_link *bpf_link_inc_not_zero(struct bpf_link *link)
{
return atomic64_fetch_add_unless(&link->refcnt, 1, 0) ? 0 : -ENOENT;
return atomic64_fetch_add_unless(&link->refcnt, 1, 0) ? link : ERR_PTR(-ENOENT);
}

#define BPF_LINK_GET_FD_BY_ID_LAST_FIELD link_id

static int bpf_link_get_fd_by_id(const union bpf_attr *attr)
struct bpf_link *bpf_link_by_id(u32 id)
{
struct bpf_link *link;
u32 id = attr->link_id;
int fd, err;

if (CHECK_ATTR(BPF_LINK_GET_FD_BY_ID))
return -EINVAL;

if (!capable(CAP_SYS_ADMIN))
return -EPERM;
if (!id)
return ERR_PTR(-ENOENT);

spin_lock_bh(&link_idr_lock);
link = idr_find(&link_idr, id);
/* before link is "settled", ID is 0, pretend it doesn't exist yet */
link = idr_find(&link_idr, id);
if (link) {
if (link->id)
err = bpf_link_inc_not_zero(link);
link = bpf_link_inc_not_zero(link);
else
err = -EAGAIN;
link = ERR_PTR(-EAGAIN);
} else {
err = -ENOENT;
link = ERR_PTR(-ENOENT);
}
spin_unlock_bh(&link_idr_lock);
return link;
}

if (err)
return err;
#define BPF_LINK_GET_FD_BY_ID_LAST_FIELD link_id

static int bpf_link_get_fd_by_id(const union bpf_attr *attr)
{
struct bpf_link *link;
u32 id = attr->link_id;
int fd;

if (CHECK_ATTR(BPF_LINK_GET_FD_BY_ID))
return -EINVAL;

if (!capable(CAP_SYS_ADMIN))
return -EPERM;

link = bpf_link_by_id(id);
if (IS_ERR(link))
return PTR_ERR(link);

fd = bpf_link_new_fd(link);
if (fd < 0)
Expand Down

0 comments on commit 005142b

Please sign in to comment.