Commit b4ff7ad7 authored by Alexei Starovoitov's avatar Alexei Starovoitov

Merge branch 'libbpf-support-bpf-to-bpf-calls'

Andrii Nakryiko says:

====================
Currently, libbpf supports a limited form of BPF-to-BPF subprogram calls. The
restriction is that entry-point BPF program should use *all* of defined
sub-programs in BPF .o file. If any of the subprograms is not used, such
entry-point BPF program will be rejected by verifier as containing unreachable
dead code. This is not a big limitation for cases with single entry-point BPF
programs, but is quite a heavy restriction for multi-programs that use only
partially overlapping set of subprograms.

This patch set removes all such restrictions and adds complete support for
using BPF sub-program calls on BPF side. This is achieved through libbpf
tracking subprograms individually and detecting which subprograms are used by
any given entry-point BPF program, and subsequently only appending and
relocating code for just those used subprograms.

In addition, libbpf now also supports multiple entry-point BPF programs within
the same ELF section. This allows to structure code so that there are few
variants of BPF programs of the same type and attaching to the same target
(e.g., for tracepoints and kprobes) without the need to worry about ELF
section name clashes.

This patch set opens way for more wider adoption of BPF subprogram calls,
especially for real-world production use-cases with complicated net of
subprograms. This will allow to further scale BPF verification process through
good use of global functions, which can be verified independently. This is
also important prerequisite for static linking which allows static BPF
libraries to not worry about naming clashes for section names, as well as use
static non-inlined functions (subprograms) without worries of verifier
rejecting program due to dead code.

Patch set is structured as follows:
- patched 1-6 contain all the libbpf changes necessary to support multi-prog
  sections and bpf2bpf subcalls;
- patch 7 adds dedicated selftests validating all combinations of possible
  sub-calls (within and across sections, static vs global functions);
- patch 8 deprecated bpf_program__title() in favor of
  bpf_program__section_name(). The intent was to also deprecate
  bpf_object__find_program_by_title() as it's now non-sensical with multiple
  programs per section. But there were too many selftests uses of this and
  I didn't want to delay this patches further and make it even bigger, so left
  it for a follow up cleanup;
- patches 9-10 remove uses for title-related APIs from bpftool and
  bpf_program__title() use from selftests;
- patch 11 is converting fexit_bpf2bpf to have explicit subtest (it does
  contain 4 subtests, which are not handled as sub-tests);
- patches 12-14 convert few complicated BPF selftests to use __noinline
  functions to further validate correctness of libbpf's bpf2bpf processing
  logic.

v2->v3:
  - explained subprog relocation algorithm in more details (Alexei);
  - pyperf, strobelight and cls_redirect got new subprog variants, leaving
    other modes intact (Alexei);
v1->v2:
  - rename DEPRECATED to LIBBPF_DEPRECATED to avoid name clashes;
  - fix test_subprogs build;
  - convert a bunch of complicated selftests to __noinline (Alexei).
====================
Signed-off-by: default avatarAlexei Starovoitov <ast@kernel.org>
parents 83cf5c68 ee333df5
...@@ -1304,7 +1304,7 @@ static int load_with_options(int argc, char **argv, bool first_prog_only) ...@@ -1304,7 +1304,7 @@ static int load_with_options(int argc, char **argv, bool first_prog_only)
enum bpf_prog_type prog_type = common_prog_type; enum bpf_prog_type prog_type = common_prog_type;
if (prog_type == BPF_PROG_TYPE_UNSPEC) { if (prog_type == BPF_PROG_TYPE_UNSPEC) {
const char *sec_name = bpf_program__title(pos, false); const char *sec_name = bpf_program__section_name(pos);
err = get_prog_type_by_name(sec_name, &prog_type, err = get_prog_type_by_name(sec_name, &prog_type,
&expected_attach_type); &expected_attach_type);
...@@ -1398,7 +1398,7 @@ static int load_with_options(int argc, char **argv, bool first_prog_only) ...@@ -1398,7 +1398,7 @@ static int load_with_options(int argc, char **argv, bool first_prog_only)
err = bpf_obj_pin(bpf_program__fd(prog), pinfile); err = bpf_obj_pin(bpf_program__fd(prog), pinfile);
if (err) { if (err) {
p_err("failed to pin program %s", p_err("failed to pin program %s",
bpf_program__title(prog, false)); bpf_program__section_name(prog));
goto err_close_obj; goto err_close_obj;
} }
} else { } else {
......
...@@ -57,14 +57,16 @@ LIBBPF_API struct btf_ext *btf_ext__new(__u8 *data, __u32 size); ...@@ -57,14 +57,16 @@ LIBBPF_API struct btf_ext *btf_ext__new(__u8 *data, __u32 size);
LIBBPF_API void btf_ext__free(struct btf_ext *btf_ext); LIBBPF_API void btf_ext__free(struct btf_ext *btf_ext);
LIBBPF_API const void *btf_ext__get_raw_data(const struct btf_ext *btf_ext, LIBBPF_API const void *btf_ext__get_raw_data(const struct btf_ext *btf_ext,
__u32 *size); __u32 *size);
LIBBPF_API int btf_ext__reloc_func_info(const struct btf *btf, LIBBPF_API LIBBPF_DEPRECATED("btf_ext__reloc_func_info was never meant as a public API and has wrong assumptions embedded in it; it will be removed in the future libbpf versions")
const struct btf_ext *btf_ext, int btf_ext__reloc_func_info(const struct btf *btf,
const char *sec_name, __u32 insns_cnt, const struct btf_ext *btf_ext,
void **func_info, __u32 *cnt); const char *sec_name, __u32 insns_cnt,
LIBBPF_API int btf_ext__reloc_line_info(const struct btf *btf, void **func_info, __u32 *cnt);
const struct btf_ext *btf_ext, LIBBPF_API LIBBPF_DEPRECATED("btf_ext__reloc_line_info was never meant as a public API and has wrong assumptions embedded in it; it will be removed in the future libbpf versions")
const char *sec_name, __u32 insns_cnt, int btf_ext__reloc_line_info(const struct btf *btf,
void **line_info, __u32 *cnt); const struct btf_ext *btf_ext,
const char *sec_name, __u32 insns_cnt,
void **line_info, __u32 *cnt);
LIBBPF_API __u32 btf_ext__func_info_rec_size(const struct btf_ext *btf_ext); LIBBPF_API __u32 btf_ext__func_info_rec_size(const struct btf_ext *btf_ext);
LIBBPF_API __u32 btf_ext__line_info_rec_size(const struct btf_ext *btf_ext); LIBBPF_API __u32 btf_ext__line_info_rec_size(const struct btf_ext *btf_ext);
......
This diff is collapsed.
...@@ -198,8 +198,9 @@ LIBBPF_API void bpf_program__set_ifindex(struct bpf_program *prog, ...@@ -198,8 +198,9 @@ LIBBPF_API void bpf_program__set_ifindex(struct bpf_program *prog,
__u32 ifindex); __u32 ifindex);
LIBBPF_API const char *bpf_program__name(const struct bpf_program *prog); LIBBPF_API const char *bpf_program__name(const struct bpf_program *prog);
LIBBPF_API const char *bpf_program__title(const struct bpf_program *prog, LIBBPF_API const char *bpf_program__section_name(const struct bpf_program *prog);
bool needs_copy); LIBBPF_API LIBBPF_DEPRECATED("BPF program title is confusing term; please use bpf_program__section_name() instead")
const char *bpf_program__title(const struct bpf_program *prog, bool needs_copy);
LIBBPF_API bool bpf_program__autoload(const struct bpf_program *prog); LIBBPF_API bool bpf_program__autoload(const struct bpf_program *prog);
LIBBPF_API int bpf_program__set_autoload(struct bpf_program *prog, bool autoload); LIBBPF_API int bpf_program__set_autoload(struct bpf_program *prog, bool autoload);
......
...@@ -302,6 +302,7 @@ LIBBPF_0.1.0 { ...@@ -302,6 +302,7 @@ LIBBPF_0.1.0 {
LIBBPF_0.2.0 { LIBBPF_0.2.0 {
global: global:
bpf_program__section_name;
perf_buffer__buffer_cnt; perf_buffer__buffer_cnt;
perf_buffer__buffer_fd; perf_buffer__buffer_fd;
perf_buffer__epoll_fd; perf_buffer__epoll_fd;
......
...@@ -15,6 +15,8 @@ ...@@ -15,6 +15,8 @@
#define LIBBPF_API __attribute__((visibility("default"))) #define LIBBPF_API __attribute__((visibility("default")))
#endif #endif
#define LIBBPF_DEPRECATED(msg) __attribute__((deprecated(msg)))
/* Helper macro to declare and initialize libbpf options struct /* Helper macro to declare and initialize libbpf options struct
* *
* This dance with uninitialized declaration, followed by memset to zero, * This dance with uninitialized declaration, followed by memset to zero,
......
...@@ -23,7 +23,13 @@ static inline int bpf_flow_load(struct bpf_object **obj, ...@@ -23,7 +23,13 @@ static inline int bpf_flow_load(struct bpf_object **obj,
if (ret) if (ret)
return ret; return ret;
main_prog = bpf_object__find_program_by_title(*obj, section_name); main_prog = NULL;
bpf_object__for_each_program(prog, *obj) {
if (strcmp(section_name, bpf_program__section_name(prog)) == 0) {
main_prog = prog;
break;
}
}
if (!main_prog) if (!main_prog)
return -1; return -1;
......
...@@ -49,6 +49,7 @@ void test_bpf_verif_scale(void) ...@@ -49,6 +49,7 @@ void test_bpf_verif_scale(void)
{ "test_verif_scale3.o", BPF_PROG_TYPE_SCHED_CLS }, { "test_verif_scale3.o", BPF_PROG_TYPE_SCHED_CLS },
{ "pyperf_global.o", BPF_PROG_TYPE_RAW_TRACEPOINT }, { "pyperf_global.o", BPF_PROG_TYPE_RAW_TRACEPOINT },
{ "pyperf_subprogs.o", BPF_PROG_TYPE_RAW_TRACEPOINT },
/* full unroll by llvm */ /* full unroll by llvm */
{ "pyperf50.o", BPF_PROG_TYPE_RAW_TRACEPOINT }, { "pyperf50.o", BPF_PROG_TYPE_RAW_TRACEPOINT },
...@@ -86,6 +87,9 @@ void test_bpf_verif_scale(void) ...@@ -86,6 +87,9 @@ void test_bpf_verif_scale(void)
{ "strobemeta_nounroll1.o", BPF_PROG_TYPE_RAW_TRACEPOINT }, { "strobemeta_nounroll1.o", BPF_PROG_TYPE_RAW_TRACEPOINT },
{ "strobemeta_nounroll2.o", BPF_PROG_TYPE_RAW_TRACEPOINT }, { "strobemeta_nounroll2.o", BPF_PROG_TYPE_RAW_TRACEPOINT },
/* non-inlined subprogs */
{ "strobemeta_subprogs.o", BPF_PROG_TYPE_RAW_TRACEPOINT },
{ "test_sysctl_loop1.o", BPF_PROG_TYPE_CGROUP_SYSCTL }, { "test_sysctl_loop1.o", BPF_PROG_TYPE_CGROUP_SYSCTL },
{ "test_sysctl_loop2.o", BPF_PROG_TYPE_CGROUP_SYSCTL }, { "test_sysctl_loop2.o", BPF_PROG_TYPE_CGROUP_SYSCTL },
......
...@@ -12,10 +12,13 @@ ...@@ -12,10 +12,13 @@
#include "progs/test_cls_redirect.h" #include "progs/test_cls_redirect.h"
#include "test_cls_redirect.skel.h" #include "test_cls_redirect.skel.h"
#include "test_cls_redirect_subprogs.skel.h"
#define ENCAP_IP INADDR_LOOPBACK #define ENCAP_IP INADDR_LOOPBACK
#define ENCAP_PORT (1234) #define ENCAP_PORT (1234)
static int duration = 0;
struct addr_port { struct addr_port {
in_port_t port; in_port_t port;
union { union {
...@@ -361,30 +364,18 @@ static void close_fds(int *fds, int n) ...@@ -361,30 +364,18 @@ static void close_fds(int *fds, int n)
close(fds[i]); close(fds[i]);
} }
void test_cls_redirect(void) static void test_cls_redirect_common(struct bpf_program *prog)
{ {
struct test_cls_redirect *skel = NULL;
struct bpf_prog_test_run_attr tattr = {}; struct bpf_prog_test_run_attr tattr = {};
int families[] = { AF_INET, AF_INET6 }; int families[] = { AF_INET, AF_INET6 };
struct sockaddr_storage ss; struct sockaddr_storage ss;
struct sockaddr *addr; struct sockaddr *addr;
socklen_t slen; socklen_t slen;
int i, j, err; int i, j, err;
int servers[__NR_KIND][ARRAY_SIZE(families)] = {}; int servers[__NR_KIND][ARRAY_SIZE(families)] = {};
int conns[__NR_KIND][ARRAY_SIZE(families)] = {}; int conns[__NR_KIND][ARRAY_SIZE(families)] = {};
struct tuple tuples[__NR_KIND][ARRAY_SIZE(families)]; struct tuple tuples[__NR_KIND][ARRAY_SIZE(families)];
skel = test_cls_redirect__open();
if (CHECK_FAIL(!skel))
return;
skel->rodata->ENCAPSULATION_IP = htonl(ENCAP_IP);
skel->rodata->ENCAPSULATION_PORT = htons(ENCAP_PORT);
if (CHECK_FAIL(test_cls_redirect__load(skel)))
goto cleanup;
addr = (struct sockaddr *)&ss; addr = (struct sockaddr *)&ss;
for (i = 0; i < ARRAY_SIZE(families); i++) { for (i = 0; i < ARRAY_SIZE(families); i++) {
slen = prepare_addr(&ss, families[i]); slen = prepare_addr(&ss, families[i]);
...@@ -402,7 +393,7 @@ void test_cls_redirect(void) ...@@ -402,7 +393,7 @@ void test_cls_redirect(void)
goto cleanup; goto cleanup;
} }
tattr.prog_fd = bpf_program__fd(skel->progs.cls_redirect); tattr.prog_fd = bpf_program__fd(prog);
for (i = 0; i < ARRAY_SIZE(tests); i++) { for (i = 0; i < ARRAY_SIZE(tests); i++) {
struct test_cfg *test = &tests[i]; struct test_cfg *test = &tests[i];
...@@ -450,7 +441,58 @@ void test_cls_redirect(void) ...@@ -450,7 +441,58 @@ void test_cls_redirect(void)
} }
cleanup: cleanup:
test_cls_redirect__destroy(skel);
close_fds((int *)servers, sizeof(servers) / sizeof(servers[0][0])); close_fds((int *)servers, sizeof(servers) / sizeof(servers[0][0]));
close_fds((int *)conns, sizeof(conns) / sizeof(conns[0][0])); close_fds((int *)conns, sizeof(conns) / sizeof(conns[0][0]));
} }
static void test_cls_redirect_inlined(void)
{
struct test_cls_redirect *skel;
int err;
skel = test_cls_redirect__open();
if (CHECK(!skel, "skel_open", "failed\n"))
return;
skel->rodata->ENCAPSULATION_IP = htonl(ENCAP_IP);
skel->rodata->ENCAPSULATION_PORT = htons(ENCAP_PORT);
err = test_cls_redirect__load(skel);
if (CHECK(err, "skel_load", "failed: %d\n", err))
goto cleanup;
test_cls_redirect_common(skel->progs.cls_redirect);
cleanup:
test_cls_redirect__destroy(skel);
}
static void test_cls_redirect_subprogs(void)
{
struct test_cls_redirect_subprogs *skel;
int err;
skel = test_cls_redirect_subprogs__open();
if (CHECK(!skel, "skel_open", "failed\n"))
return;
skel->rodata->ENCAPSULATION_IP = htonl(ENCAP_IP);
skel->rodata->ENCAPSULATION_PORT = htons(ENCAP_PORT);
err = test_cls_redirect_subprogs__load(skel);
if (CHECK(err, "skel_load", "failed: %d\n", err))
goto cleanup;
test_cls_redirect_common(skel->progs.cls_redirect);
cleanup:
test_cls_redirect_subprogs__destroy(skel);
}
void test_cls_redirect(void)
{
if (test__start_subtest("cls_redirect_inlined"))
test_cls_redirect_inlined();
if (test__start_subtest("cls_redirect_subprogs"))
test_cls_redirect_subprogs();
}
...@@ -208,11 +208,18 @@ static void test_func_map_prog_compatibility(void) ...@@ -208,11 +208,18 @@ static void test_func_map_prog_compatibility(void)
void test_fexit_bpf2bpf(void) void test_fexit_bpf2bpf(void)
{ {
test_target_no_callees(); if (test__start_subtest("target_no_callees"))
test_target_yes_callees(); test_target_no_callees();
test_func_replace(); if (test__start_subtest("target_yes_callees"))
test_func_replace_verify(); test_target_yes_callees();
test_func_sockmap_update(); if (test__start_subtest("func_replace"))
test_func_replace_return_code(); test_func_replace();
test_func_map_prog_compatibility(); if (test__start_subtest("func_replace_verify"))
test_func_replace_verify();
if (test__start_subtest("func_sockmap_update"))
test_func_sockmap_update();
if (test__start_subtest("func_replace_return_code"))
test_func_replace_return_code();
if (test__start_subtest("func_map_prog_compatibility"))
test_func_map_prog_compatibility();
} }
...@@ -80,9 +80,8 @@ static void test_l4lb(const char *file) ...@@ -80,9 +80,8 @@ static void test_l4lb(const char *file)
void test_l4lb_all(void) void test_l4lb_all(void)
{ {
const char *file1 = "./test_l4lb.o"; if (test__start_subtest("l4lb_inline"))
const char *file2 = "./test_l4lb_noinline.o"; test_l4lb("test_l4lb.o");
if (test__start_subtest("l4lb_noinline"))
test_l4lb(file1); test_l4lb("test_l4lb_noinline.o");
test_l4lb(file2);
} }
...@@ -27,7 +27,7 @@ void test_reference_tracking(void) ...@@ -27,7 +27,7 @@ void test_reference_tracking(void)
const char *title; const char *title;
/* Ignore .text sections */ /* Ignore .text sections */
title = bpf_program__title(prog, false); title = bpf_program__section_name(prog);
if (strstr(title, ".text") != NULL) if (strstr(title, ".text") != NULL)
continue; continue;
......
// SPDX-License-Identifier: GPL-2.0
/* Copyright (c) 2020 Facebook */
#include <test_progs.h>
#include <time.h>
#include "test_subprogs.skel.h"
static int duration;
void test_subprogs(void)
{
struct test_subprogs *skel;
int err;
skel = test_subprogs__open_and_load();
if (CHECK(!skel, "skel_open", "failed to open skeleton\n"))
return;
err = test_subprogs__attach(skel);
if (CHECK(err, "skel_attach", "failed to attach skeleton: %d\n", err))
goto cleanup;
usleep(1);
CHECK(skel->bss->res1 != 12, "res1", "got %d, exp %d\n", skel->bss->res1, 12);
CHECK(skel->bss->res2 != 17, "res2", "got %d, exp %d\n", skel->bss->res2, 17);
CHECK(skel->bss->res3 != 19, "res3", "got %d, exp %d\n", skel->bss->res3, 19);
CHECK(skel->bss->res4 != 36, "res4", "got %d, exp %d\n", skel->bss->res4, 36);
cleanup:
test_subprogs__destroy(skel);
}
// SPDX-License-Identifier: GPL-2.0 // SPDX-License-Identifier: GPL-2.0
#include <test_progs.h> #include <test_progs.h>
#include <network_helpers.h> #include <network_helpers.h>
#include "test_xdp_noinline.skel.h"
void test_xdp_noinline(void) void test_xdp_noinline(void)
{ {
const char *file = "./test_xdp_noinline.o";
unsigned int nr_cpus = bpf_num_possible_cpus(); unsigned int nr_cpus = bpf_num_possible_cpus();
struct test_xdp_noinline *skel;
struct vip key = {.protocol = 6}; struct vip key = {.protocol = 6};
struct vip_meta { struct vip_meta {
__u32 flags; __u32 flags;
...@@ -25,58 +26,42 @@ void test_xdp_noinline(void) ...@@ -25,58 +26,42 @@ void test_xdp_noinline(void)
} real_def = {.dst = MAGIC_VAL}; } real_def = {.dst = MAGIC_VAL};
__u32 ch_key = 11, real_num = 3; __u32 ch_key = 11, real_num = 3;
__u32 duration, retval, size; __u32 duration, retval, size;
int err, i, prog_fd, map_fd; int err, i;
__u64 bytes = 0, pkts = 0; __u64 bytes = 0, pkts = 0;
struct bpf_object *obj;
char buf[128]; char buf[128];
u32 *magic = (u32 *)buf; u32 *magic = (u32 *)buf;
err = bpf_prog_load(file, BPF_PROG_TYPE_XDP, &obj, &prog_fd); skel = test_xdp_noinline__open_and_load();
if (CHECK_FAIL(err)) if (CHECK(!skel, "skel_open_and_load", "failed\n"))
return; return;
map_fd = bpf_find_map(__func__, obj, "vip_map"); bpf_map_update_elem(bpf_map__fd(skel->maps.vip_map), &key, &value, 0);
if (map_fd < 0) bpf_map_update_elem(bpf_map__fd(skel->maps.ch_rings), &ch_key, &real_num, 0);
goto out; bpf_map_update_elem(bpf_map__fd(skel->maps.reals), &real_num, &real_def, 0);
bpf_map_update_elem(map_fd, &key, &value, 0);
map_fd = bpf_find_map(__func__, obj, "ch_rings"); err = bpf_prog_test_run(bpf_program__fd(skel->progs.balancer_ingress_v4),
if (map_fd < 0) NUM_ITER, &pkt_v4, sizeof(pkt_v4),
goto out;
bpf_map_update_elem(map_fd, &ch_key, &real_num, 0);
map_fd = bpf_find_map(__func__, obj, "reals");
if (map_fd < 0)
goto out;
bpf_map_update_elem(map_fd, &real_num, &real_def, 0);
err = bpf_prog_test_run(prog_fd, NUM_ITER, &pkt_v4, sizeof(pkt_v4),
buf, &size, &retval, &duration); buf, &size, &retval, &duration);
CHECK(err || retval != 1 || size != 54 || CHECK(err || retval != 1 || size != 54 ||
*magic != MAGIC_VAL, "ipv4", *magic != MAGIC_VAL, "ipv4",
"err %d errno %d retval %d size %d magic %x\n", "err %d errno %d retval %d size %d magic %x\n",
err, errno, retval, size, *magic); err, errno, retval, size, *magic);
err = bpf_prog_test_run(prog_fd, NUM_ITER, &pkt_v6, sizeof(pkt_v6), err = bpf_prog_test_run(bpf_program__fd(skel->progs.balancer_ingress_v6),
NUM_ITER, &pkt_v6, sizeof(pkt_v6),
buf, &size, &retval, &duration); buf, &size, &retval, &duration);
CHECK(err || retval != 1 || size != 74 || CHECK(err || retval != 1 || size != 74 ||
*magic != MAGIC_VAL, "ipv6", *magic != MAGIC_VAL, "ipv6",
"err %d errno %d retval %d size %d magic %x\n", "err %d errno %d retval %d size %d magic %x\n",
err, errno, retval, size, *magic); err, errno, retval, size, *magic);
map_fd = bpf_find_map(__func__, obj, "stats"); bpf_map_lookup_elem(bpf_map__fd(skel->maps.stats), &stats_key, stats);
if (map_fd < 0)
goto out;
bpf_map_lookup_elem(map_fd, &stats_key, stats);
for (i = 0; i < nr_cpus; i++) { for (i = 0; i < nr_cpus; i++) {
bytes += stats[i].bytes; bytes += stats[i].bytes;
pkts += stats[i].pkts; pkts += stats[i].pkts;
} }
if (CHECK_FAIL(bytes != MAGIC_BYTES * NUM_ITER * 2 || CHECK(bytes != MAGIC_BYTES * NUM_ITER * 2 || pkts != NUM_ITER * 2,
pkts != NUM_ITER * 2)) { "stats", "bytes %lld pkts %lld\n",
printf("test_xdp_noinline:FAIL:stats %lld %lld\n", (unsigned long long)bytes, (unsigned long long)pkts);
bytes, pkts); test_xdp_noinline__destroy(skel);
}
out:
bpf_object__close(obj);
} }
...@@ -67,7 +67,12 @@ typedef struct { ...@@ -67,7 +67,12 @@ typedef struct {
void* co_name; // PyCodeObject.co_name void* co_name; // PyCodeObject.co_name
} FrameData; } FrameData;
static __always_inline void *get_thread_state(void *tls_base, PidData *pidData) #ifdef SUBPROGS
__noinline
#else
__always_inline
#endif
static void *get_thread_state(void *tls_base, PidData *pidData)
{ {
void* thread_state; void* thread_state;
int key; int key;
...@@ -155,7 +160,9 @@ struct { ...@@ -155,7 +160,9 @@ struct {
} stackmap SEC(".maps"); } stackmap SEC(".maps");
#ifdef GLOBAL_FUNC #ifdef GLOBAL_FUNC
__attribute__((noinline)) __noinline
#elif defined(SUBPROGS)
static __noinline
#else #else
static __always_inline static __always_inline
#endif #endif
......
// SPDX-License-Identifier: GPL-2.0
/* Copyright (c) 2020 Facebook */
#define STACK_MAX_LEN 50
#define SUBPROGS
#include "pyperf.h"
...@@ -266,8 +266,12 @@ struct tls_index { ...@@ -266,8 +266,12 @@ struct tls_index {
uint64_t offset; uint64_t offset;
}; };
static __always_inline void *calc_location(struct strobe_value_loc *loc, #ifdef SUBPROGS
void *tls_base) __noinline
#else
__always_inline
#endif
static void *calc_location(struct strobe_value_loc *loc, void *tls_base)
{ {
/* /*
* tls_mode value is: * tls_mode value is:
...@@ -327,10 +331,15 @@ static __always_inline void *calc_location(struct strobe_value_loc *loc, ...@@ -327,10 +331,15 @@ static __always_inline void *calc_location(struct strobe_value_loc *loc,
: NULL; : NULL;
} }
static __always_inline void read_int_var(struct strobemeta_cfg *cfg, #ifdef SUBPROGS
size_t idx, void *tls_base, __noinline
struct strobe_value_generic *value, #else
struct strobemeta_payload *data) __always_inline
#endif
static void read_int_var(struct strobemeta_cfg *cfg,
size_t idx, void *tls_base,
struct strobe_value_generic *value,
struct strobemeta_payload *data)
{ {
void *location = calc_location(&cfg->int_locs[idx], tls_base); void *location = calc_location(&cfg->int_locs[idx], tls_base);
if (!location) if (!location)
...@@ -440,8 +449,13 @@ static __always_inline void *read_map_var(struct strobemeta_cfg *cfg, ...@@ -440,8 +449,13 @@ static __always_inline void *read_map_var(struct strobemeta_cfg *cfg,
* read_strobe_meta returns NULL, if no metadata was read; otherwise returns * read_strobe_meta returns NULL, if no metadata was read; otherwise returns
* pointer to *right after* payload ends * pointer to *right after* payload ends
*/ */
static __always_inline void *read_strobe_meta(struct task_struct *task, #ifdef SUBPROGS
struct strobemeta_payload *data) __noinline
#else
__always_inline
#endif
static void *read_strobe_meta(struct task_struct *task,
struct strobemeta_payload *data)
{ {
pid_t pid = bpf_get_current_pid_tgid() >> 32; pid_t pid = bpf_get_current_pid_tgid() >> 32;
struct strobe_value_generic value = {0}; struct strobe_value_generic value = {0};
......
// SPDX-License-Identifier: (LGPL-2.1 OR BSD-2-Clause)
// Copyright (c) 2019 Facebook
#define STROBE_MAX_INTS 2
#define STROBE_MAX_STRS 25
#define STROBE_MAX_MAPS 13
#define STROBE_MAX_MAP_ENTRIES 20
#define NO_UNROLL
#define SUBPROGS
#include "strobemeta.h"
#define SUBPROGS
#include "test_cls_redirect.c"
...@@ -17,9 +17,7 @@ ...@@ -17,9 +17,7 @@
#include "test_iptunnel_common.h" #include "test_iptunnel_common.h"
#include <bpf/bpf_endian.h> #include <bpf/bpf_endian.h>
int _version SEC("version") = 1; static __always_inline __u32 rol32(__u32 word, unsigned int shift)
static __u32 rol32(__u32 word, unsigned int shift)
{ {
return (word << shift) | (word >> ((-shift) & 31)); return (word << shift) | (word >> ((-shift) & 31));
} }
...@@ -52,7 +50,7 @@ static __u32 rol32(__u32 word, unsigned int shift) ...@@ -52,7 +50,7 @@ static __u32 rol32(__u32 word, unsigned int shift)
typedef unsigned int u32; typedef unsigned int u32;
static u32 jhash(const void *key, u32 length, u32 initval) static __noinline u32 jhash(const void *key, u32 length, u32 initval)
{ {
u32 a, b, c; u32 a, b, c;
const unsigned char *k = key; const unsigned char *k = key;
...@@ -88,7 +86,7 @@ static u32 jhash(const void *key, u32 length, u32 initval) ...@@ -88,7 +86,7 @@ static u32 jhash(const void *key, u32 length, u32 initval)
return c; return c;
} }
static u32 __jhash_nwords(u32 a, u32 b, u32 c, u32 initval) static __noinline u32 __jhash_nwords(u32 a, u32 b, u32 c, u32 initval)
{ {
a += initval; a += initval;
b += initval; b += initval;
...@@ -97,7 +95,7 @@ static u32 __jhash_nwords(u32 a, u32 b, u32 c, u32 initval) ...@@ -97,7 +95,7 @@ static u32 __jhash_nwords(u32 a, u32 b, u32 c, u32 initval)
return c; return c;
} }
static u32 jhash_2words(u32 a, u32 b, u32 initval) static __noinline u32 jhash_2words(u32 a, u32 b, u32 initval)
{ {
return __jhash_nwords(a, b, 0, initval + JHASH_INITVAL + (2 << 2)); return __jhash_nwords(a, b, 0, initval + JHASH_INITVAL + (2 << 2));
} }
...@@ -200,8 +198,7 @@ struct { ...@@ -200,8 +198,7 @@ struct {
__type(value, struct ctl_value); __type(value, struct ctl_value);
} ctl_array SEC(".maps"); } ctl_array SEC(".maps");
static __u32 get_packet_hash(struct packet_description *pckt, static __noinline __u32 get_packet_hash(struct packet_description *pckt, bool ipv6)
bool ipv6)
{ {
if (ipv6) if (ipv6)
return jhash_2words(jhash(pckt->srcv6, 16, MAX_VIPS), return jhash_2words(jhash(pckt->srcv6, 16, MAX_VIPS),
...@@ -210,10 +207,10 @@ static __u32 get_packet_hash(struct packet_description *pckt, ...@@ -210,10 +207,10 @@ static __u32 get_packet_hash(struct packet_description *pckt,
return jhash_2words(pckt->src, pckt->ports, CH_RINGS_SIZE); return jhash_2words(pckt->src, pckt->ports, CH_RINGS_SIZE);
} }
static bool get_packet_dst(struct real_definition **real, static __noinline bool get_packet_dst(struct real_definition **real,
struct packet_description *pckt, struct packet_description *pckt,
struct vip_meta *vip_info, struct vip_meta *vip_info,
bool is_ipv6) bool is_ipv6)
{ {
__u32 hash = get_packet_hash(pckt, is_ipv6); __u32 hash = get_packet_hash(pckt, is_ipv6);
__u32 key = RING_SIZE * vip_info->vip_num + hash % RING_SIZE; __u32 key = RING_SIZE * vip_info->vip_num + hash % RING_SIZE;
...@@ -233,8 +230,8 @@ static bool get_packet_dst(struct real_definition **real, ...@@ -233,8 +230,8 @@ static bool get_packet_dst(struct real_definition **real,
return true; return true;
} }
static int parse_icmpv6(void *data, void *data_end, __u64 off, static __noinline int parse_icmpv6(void *data, void *data_end, __u64 off,
struct packet_description *pckt) struct packet_description *pckt)
{ {
struct icmp6hdr *icmp_hdr; struct icmp6hdr *icmp_hdr;
struct ipv6hdr *ip6h; struct ipv6hdr *ip6h;
...@@ -255,8 +252,8 @@ static int parse_icmpv6(void *data, void *data_end, __u64 off, ...@@ -255,8 +252,8 @@ static int parse_icmpv6(void *data, void *data_end, __u64 off,
return TC_ACT_UNSPEC; return TC_ACT_UNSPEC;
} }
static int parse_icmp(void *data, void *data_end, __u64 off, static __noinline int parse_icmp(void *data, void *data_end, __u64 off,
struct packet_description *pckt) struct packet_description *pckt)
{ {
struct icmphdr *icmp_hdr; struct icmphdr *icmp_hdr;
struct iphdr *iph; struct iphdr *iph;
...@@ -280,8 +277,8 @@ static int parse_icmp(void *data, void *data_end, __u64 off, ...@@ -280,8 +277,8 @@ static int parse_icmp(void *data, void *data_end, __u64 off,
return TC_ACT_UNSPEC; return TC_ACT_UNSPEC;
} }
static bool parse_udp(void *data, __u64 off, void *data_end, static __noinline bool parse_udp(void *data, __u64 off, void *data_end,
struct packet_description *pckt) struct packet_description *pckt)
{ {
struct udphdr *udp; struct udphdr *udp;
udp = data + off; udp = data + off;
...@@ -299,8 +296,8 @@ static bool parse_udp(void *data, __u64 off, void *data_end, ...@@ -299,8 +296,8 @@ static bool parse_udp(void *data, __u64 off, void *data_end,
return true; return true;
} }
static bool parse_tcp(void *data, __u64 off, void *data_end, static __noinline bool parse_tcp(void *data, __u64 off, void *data_end,
struct packet_description *pckt) struct packet_description *pckt)
{ {
struct tcphdr *tcp; struct tcphdr *tcp;
...@@ -321,8 +318,8 @@ static bool parse_tcp(void *data, __u64 off, void *data_end, ...@@ -321,8 +318,8 @@ static bool parse_tcp(void *data, __u64 off, void *data_end,
return true; return true;
} }
static int process_packet(void *data, __u64 off, void *data_end, static __noinline int process_packet(void *data, __u64 off, void *data_end,
bool is_ipv6, struct __sk_buff *skb) bool is_ipv6, struct __sk_buff *skb)
{ {
void *pkt_start = (void *)(long)skb->data; void *pkt_start = (void *)(long)skb->data;
struct packet_description pckt = {}; struct packet_description pckt = {};
......
#include "vmlinux.h"
#include <bpf/bpf_helpers.h>
#include <bpf/bpf_core_read.h>
const char LICENSE[] SEC("license") = "GPL";
__noinline int sub1(int x)
{
return x + 1;
}
static __noinline int sub5(int v);
__noinline int sub2(int y)
{
return sub5(y + 2);
}
static __noinline int sub3(int z)
{
return z + 3 + sub1(4);
}
static __noinline int sub4(int w)
{
return w + sub3(5) + sub1(6);
}
/* sub5() is an identitify function, just to test weirder functions layout and
* call patterns
*/
static __noinline int sub5(int v)
{
return sub1(v) - 1; /* compensates sub1()'s + 1 */
}
/* unfortunately verifier rejects `struct task_struct *t` as an unkown pointer
* type, so we need to accept pointer as integer and then cast it inside the
* function
*/
__noinline int get_task_tgid(uintptr_t t)
{
/* this ensures that CO-RE relocs work in multi-subprogs .text */
return BPF_CORE_READ((struct task_struct *)(void *)t, tgid);
}
int res1 = 0;
int res2 = 0;
int res3 = 0;
int res4 = 0;
SEC("raw_tp/sys_enter")
int prog1(void *ctx)
{
/* perform some CO-RE relocations to ensure they work with multi-prog
* sections correctly
*/
struct task_struct *t = (void *)bpf_get_current_task();
if (!BPF_CORE_READ(t, pid) || !get_task_tgid((uintptr_t)t))
return 1;
res1 = sub1(1) + sub3(2); /* (1 + 1) + (2 + 3 + (4 + 1)) = 12 */
return 0;
}
SEC("raw_tp/sys_exit")
int prog2(void *ctx)
{
struct task_struct *t = (void *)bpf_get_current_task();
if (!BPF_CORE_READ(t, pid) || !get_task_tgid((uintptr_t)t))
return 1;
res2 = sub2(3) + sub3(4); /* (3 + 2) + (4 + 3 + (4 + 1)) = 17 */
return 0;
}
/* prog3 has the same section name as prog1 */
SEC("raw_tp/sys_enter")
int prog3(void *ctx)
{
struct task_struct *t = (void *)bpf_get_current_task();
if (!BPF_CORE_READ(t, pid) || !get_task_tgid((uintptr_t)t))
return 1;
res3 = sub3(5) + 6; /* (5 + 3 + (4 + 1)) + 6 = 19 */
return 0;
}
/* prog4 has the same section name as prog2 */
SEC("raw_tp/sys_exit")
int prog4(void *ctx)
{
struct task_struct *t = (void *)bpf_get_current_task();
if (!BPF_CORE_READ(t, pid) || !get_task_tgid((uintptr_t)t))
return 1;
res4 = sub4(7) + sub1(8); /* (7 + (5 + 3 + (4 + 1)) + (6 + 1)) + (8 + 1) = 36 */
return 0;
}
...@@ -16,7 +16,7 @@ ...@@ -16,7 +16,7 @@
#include <bpf/bpf_helpers.h> #include <bpf/bpf_helpers.h>
#include <bpf/bpf_endian.h> #include <bpf/bpf_endian.h>
static __u32 rol32(__u32 word, unsigned int shift) static __always_inline __u32 rol32(__u32 word, unsigned int shift)
{ {
return (word << shift) | (word >> ((-shift) & 31)); return (word << shift) | (word >> ((-shift) & 31));
} }
...@@ -49,7 +49,7 @@ static __u32 rol32(__u32 word, unsigned int shift) ...@@ -49,7 +49,7 @@ static __u32 rol32(__u32 word, unsigned int shift)
typedef unsigned int u32; typedef unsigned int u32;
static __attribute__ ((noinline)) static __noinline
u32 jhash(const void *key, u32 length, u32 initval) u32 jhash(const void *key, u32 length, u32 initval)
{ {
u32 a, b, c; u32 a, b, c;
...@@ -86,7 +86,7 @@ u32 jhash(const void *key, u32 length, u32 initval) ...@@ -86,7 +86,7 @@ u32 jhash(const void *key, u32 length, u32 initval)
return c; return c;
} }
__attribute__ ((noinline)) __noinline
u32 __jhash_nwords(u32 a, u32 b, u32 c, u32 initval) u32 __jhash_nwords(u32 a, u32 b, u32 c, u32 initval)
{ {
a += initval; a += initval;
...@@ -96,7 +96,7 @@ u32 __jhash_nwords(u32 a, u32 b, u32 c, u32 initval) ...@@ -96,7 +96,7 @@ u32 __jhash_nwords(u32 a, u32 b, u32 c, u32 initval)
return c; return c;
} }
__attribute__ ((noinline)) __noinline
u32 jhash_2words(u32 a, u32 b, u32 initval) u32 jhash_2words(u32 a, u32 b, u32 initval)
{ {
return __jhash_nwords(a, b, 0, initval + JHASH_INITVAL + (2 << 2)); return __jhash_nwords(a, b, 0, initval + JHASH_INITVAL + (2 << 2));
...@@ -213,7 +213,7 @@ struct eth_hdr { ...@@ -213,7 +213,7 @@ struct eth_hdr {
unsigned short eth_proto; unsigned short eth_proto;
}; };
static inline __u64 calc_offset(bool is_ipv6, bool is_icmp) static __noinline __u64 calc_offset(bool is_ipv6, bool is_icmp)
{ {
__u64 off = sizeof(struct eth_hdr); __u64 off = sizeof(struct eth_hdr);
if (is_ipv6) { if (is_ipv6) {
...@@ -797,8 +797,8 @@ static int process_packet(void *data, __u64 off, void *data_end, ...@@ -797,8 +797,8 @@ static int process_packet(void *data, __u64 off, void *data_end,
return XDP_DROP; return XDP_DROP;
} }
__attribute__ ((section("xdp-test"), used)) SEC("xdp-test-v4")
int balancer_ingress(struct xdp_md *ctx) int balancer_ingress_v4(struct xdp_md *ctx)
{ {
void *data = (void *)(long)ctx->data; void *data = (void *)(long)ctx->data;
void *data_end = (void *)(long)ctx->data_end; void *data_end = (void *)(long)ctx->data_end;
...@@ -812,11 +812,27 @@ int balancer_ingress(struct xdp_md *ctx) ...@@ -812,11 +812,27 @@ int balancer_ingress(struct xdp_md *ctx)
eth_proto = bpf_ntohs(eth->eth_proto); eth_proto = bpf_ntohs(eth->eth_proto);
if (eth_proto == ETH_P_IP) if (eth_proto == ETH_P_IP)
return process_packet(data, nh_off, data_end, 0, ctx); return process_packet(data, nh_off, data_end, 0, ctx);
else if (eth_proto == ETH_P_IPV6) else
return XDP_DROP;
}
SEC("xdp-test-v6")
int balancer_ingress_v6(struct xdp_md *ctx)
{
void *data = (void *)(long)ctx->data;
void *data_end = (void *)(long)ctx->data_end;
struct eth_hdr *eth = data;
__u32 eth_proto;
__u32 nh_off;
nh_off = sizeof(struct eth_hdr);
if (data + nh_off > data_end)
return XDP_DROP;
eth_proto = bpf_ntohs(eth->eth_proto);
if (eth_proto == ETH_P_IPV6)
return process_packet(data, nh_off, data_end, 1, ctx); return process_packet(data, nh_off, data_end, 1, ctx);
else else
return XDP_DROP; return XDP_DROP;
} }
char _license[] __attribute__ ((section("license"), used)) = "GPL"; char _license[] SEC("license") = "GPL";
int _version __attribute__ ((section("version"), used)) = 1;
...@@ -151,7 +151,7 @@ static int run_test(int cgfd) ...@@ -151,7 +151,7 @@ static int run_test(int cgfd)
} }
bpf_object__for_each_program(prog, pobj) { bpf_object__for_each_program(prog, pobj) {
prog_name = bpf_program__title(prog, /*needs_copy*/ false); prog_name = bpf_program__section_name(prog);
if (libbpf_attach_type_by_name(prog_name, &attach_type)) if (libbpf_attach_type_by_name(prog_name, &attach_type))
goto err; goto err;
......
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