Commit 910b543d authored by Daniel Borkmann's avatar Daniel Borkmann Committed by Stephen Hemminger

{f,m}_bpf: make tail calls working

Now that we have the possibility of sharing maps, it's time we get the
ELF loader fully working with regards to tail calls. Since program array
maps are pinned, we can keep them finally alive. I've noticed two bugs
that are being fixed in bpf_fill_prog_arrays() with this patch. Example
code comes as follow-up.
Signed-off-by: default avatarDaniel Borkmann <daniel@iogearbox.net>
Acked-by: default avatarAlexei Starovoitov <ast@kernel.org>
parent fece33c1
...@@ -1139,11 +1139,22 @@ static int bpf_fetch_prog_sec(struct bpf_elf_ctx *ctx, const char *section) ...@@ -1139,11 +1139,22 @@ static int bpf_fetch_prog_sec(struct bpf_elf_ctx *ctx, const char *section)
return ret; return ret;
} }
static int bpf_find_map_by_id(struct bpf_elf_ctx *ctx, uint32_t id)
{
int i;
for (i = 0; i < ARRAY_SIZE(ctx->map_fds); i++)
if (ctx->map_fds[i] && ctx->maps[i].id == id &&
ctx->maps[i].type == BPF_MAP_TYPE_PROG_ARRAY)
return i;
return -1;
}
static int bpf_fill_prog_arrays(struct bpf_elf_ctx *ctx) static int bpf_fill_prog_arrays(struct bpf_elf_ctx *ctx)
{ {
struct bpf_elf_sec_data data; struct bpf_elf_sec_data data;
uint32_t map_id, key_id; uint32_t map_id, key_id;
int fd, i, ret; int fd, i, ret, idx;
for (i = 1; i < ctx->elf_hdr.e_shnum; i++) { for (i = 1; i < ctx->elf_hdr.e_shnum; i++) {
if (ctx->sec_done[i]) if (ctx->sec_done[i])
...@@ -1153,20 +1164,20 @@ static int bpf_fill_prog_arrays(struct bpf_elf_ctx *ctx) ...@@ -1153,20 +1164,20 @@ static int bpf_fill_prog_arrays(struct bpf_elf_ctx *ctx)
if (ret < 0) if (ret < 0)
continue; continue;
ret = sscanf(data.sec_name, "%u/%u", &map_id, &key_id); ret = sscanf(data.sec_name, "%i/%i", &map_id, &key_id);
if (ret != 2 || map_id >= ARRAY_SIZE(ctx->map_fds) || if (ret != 2)
!ctx->map_fds[map_id])
continue; continue;
if (ctx->maps[map_id].type != BPF_MAP_TYPE_PROG_ARRAY ||
ctx->maps[map_id].max_elem <= key_id) idx = bpf_find_map_by_id(ctx, map_id);
if (idx < 0)
continue; continue;
fd = bpf_fetch_prog_sec(ctx, data.sec_name); fd = bpf_fetch_prog_sec(ctx, data.sec_name);
if (fd < 0) if (fd < 0)
return -EIO; return -EIO;
ret = bpf_map_update(ctx->map_fds[map_id], &key_id, ret = bpf_map_update(ctx->map_fds[idx], &key_id,
&fd, BPF_NOEXIST); &fd, BPF_ANY);
if (ret < 0) if (ret < 0)
return -ENOENT; return -ENOENT;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment