mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-01-24 17:23:25 -05:00
Daniel Borkmann says: ==================== pull-request: bpf-next 2020-12-14 1) Expose bpf_sk_storage_*() helpers to iterator programs, from Florent Revest. 2) Add AF_XDP selftests based on veth devs to BPF selftests, from Weqaar Janjua. 3) Support for finding BTF based kernel attach targets through libbpf's bpf_program__set_attach_target() API, from Andrii Nakryiko. 4) Permit pointers on stack for helper calls in the verifier, from Yonghong Song. 5) Fix overflows in hash map elem size after rlimit removal, from Eric Dumazet. 6) Get rid of direct invocation of llc in BPF selftests, from Andrew Delgadillo. 7) Fix xsk_recvmsg() to reorder socket state check before access, from Björn Töpel. 8) Add new libbpf API helper to retrieve ring buffer epoll fd, from Brendan Jackman. 9) Batch of minor BPF selftest improvements all over the place, from Florian Lehner, KP Singh, Jiri Olsa and various others. * https://git.kernel.org/pub/scm/linux/kernel/git/bpf/bpf-next: (31 commits) selftests/bpf: Add a test for ptr_to_map_value on stack for helper access bpf: Permits pointers on stack for helper calls libbpf: Expose libbpf ring_buffer epoll_fd selftests/bpf: Add set_attach_target() API selftest for module target libbpf: Support modules in bpf_program__set_attach_target() API selftests/bpf: Silence ima_setup.sh when not running in verbose mode. selftests/bpf: Drop the need for LLVM's llc selftests/bpf: fix bpf_testmod.ko recompilation logic samples/bpf: Fix possible hang in xdpsock with multiple threads selftests/bpf: Make selftest compilation work on clang 11 selftests/bpf: Xsk selftests - adding xdpxceiver to .gitignore selftests/bpf: Drop tcp-{client,server}.py from Makefile selftests/bpf: Xsk selftests - Bi-directional Sockets - SKB, DRV selftests/bpf: Xsk selftests - Socket Teardown - SKB, DRV selftests/bpf: Xsk selftests - DRV POLL, NOPOLL selftests/bpf: Xsk selftests - SKB POLL, NOPOLL selftests/bpf: Xsk selftests framework bpf: Only provide bpf_sock_from_file with CONFIG_NET bpf: Return -ENOTSUPP when attaching to non-kernel BTF xsk: Validate socket state in xsk_recvmsg, prior touching socket members ... ==================== Link: https://lore.kernel.org/r/20201214214316.20642-1-daniel@iogearbox.net Signed-off-by: Jakub Kicinski <kuba@kernel.org>
This commit is contained in:
commit
a6b5e026e6
40 changed files with 2063 additions and 114 deletions
|
@ -416,12 +416,11 @@ static inline void ep_set_busy_poll_napi_id(struct epitem *epi)
|
|||
unsigned int napi_id;
|
||||
struct socket *sock;
|
||||
struct sock *sk;
|
||||
int err;
|
||||
|
||||
if (!net_busy_loop_on())
|
||||
return;
|
||||
|
||||
sock = sock_from_file(epi->ffd.file, &err);
|
||||
sock = sock_from_file(epi->ffd.file);
|
||||
if (!sock)
|
||||
return;
|
||||
|
||||
|
|
|
@ -4356,9 +4356,9 @@ static int io_sendmsg(struct io_kiocb *req, bool force_nonblock,
|
|||
unsigned flags;
|
||||
int ret;
|
||||
|
||||
sock = sock_from_file(req->file, &ret);
|
||||
sock = sock_from_file(req->file);
|
||||
if (unlikely(!sock))
|
||||
return ret;
|
||||
return -ENOTSOCK;
|
||||
|
||||
if (req->async_data) {
|
||||
kmsg = req->async_data;
|
||||
|
@ -4405,9 +4405,9 @@ static int io_send(struct io_kiocb *req, bool force_nonblock,
|
|||
unsigned flags;
|
||||
int ret;
|
||||
|
||||
sock = sock_from_file(req->file, &ret);
|
||||
sock = sock_from_file(req->file);
|
||||
if (unlikely(!sock))
|
||||
return ret;
|
||||
return -ENOTSOCK;
|
||||
|
||||
ret = import_single_range(WRITE, sr->buf, sr->len, &iov, &msg.msg_iter);
|
||||
if (unlikely(ret))
|
||||
|
@ -4585,9 +4585,9 @@ static int io_recvmsg(struct io_kiocb *req, bool force_nonblock,
|
|||
unsigned flags;
|
||||
int ret, cflags = 0;
|
||||
|
||||
sock = sock_from_file(req->file, &ret);
|
||||
sock = sock_from_file(req->file);
|
||||
if (unlikely(!sock))
|
||||
return ret;
|
||||
return -ENOTSOCK;
|
||||
|
||||
if (req->async_data) {
|
||||
kmsg = req->async_data;
|
||||
|
@ -4648,9 +4648,9 @@ static int io_recv(struct io_kiocb *req, bool force_nonblock,
|
|||
unsigned flags;
|
||||
int ret, cflags = 0;
|
||||
|
||||
sock = sock_from_file(req->file, &ret);
|
||||
sock = sock_from_file(req->file);
|
||||
if (unlikely(!sock))
|
||||
return ret;
|
||||
return -ENOTSOCK;
|
||||
|
||||
if (req->flags & REQ_F_BUFFER_SELECT) {
|
||||
kbuf = io_recv_buffer_select(req, !force_nonblock);
|
||||
|
|
|
@ -1859,6 +1859,7 @@ extern const struct bpf_func_proto bpf_snprintf_btf_proto;
|
|||
extern const struct bpf_func_proto bpf_per_cpu_ptr_proto;
|
||||
extern const struct bpf_func_proto bpf_this_cpu_ptr_proto;
|
||||
extern const struct bpf_func_proto bpf_ktime_get_coarse_ns_proto;
|
||||
extern const struct bpf_func_proto bpf_sock_from_file_proto;
|
||||
|
||||
const struct bpf_func_proto *bpf_tracing_func_proto(
|
||||
enum bpf_func_id func_id, const struct bpf_prog *prog);
|
||||
|
|
|
@ -240,7 +240,7 @@ int sock_sendmsg(struct socket *sock, struct msghdr *msg);
|
|||
int sock_recvmsg(struct socket *sock, struct msghdr *msg, int flags);
|
||||
struct file *sock_alloc_file(struct socket *sock, int flags, const char *dname);
|
||||
struct socket *sockfd_lookup(int fd, int *err);
|
||||
struct socket *sock_from_file(struct file *file, int *err);
|
||||
struct socket *sock_from_file(struct file *file);
|
||||
#define sockfd_put(sock) fput(sock->file)
|
||||
int net_ratelimit(void);
|
||||
|
||||
|
|
|
@ -145,17 +145,17 @@ DEFINE_EVENT(xdp_redirect_template, xdp_redirect_err,
|
|||
TP_ARGS(dev, xdp, tgt, err, map, index)
|
||||
);
|
||||
|
||||
#define _trace_xdp_redirect(dev, xdp, to) \
|
||||
trace_xdp_redirect(dev, xdp, NULL, 0, NULL, to);
|
||||
#define _trace_xdp_redirect(dev, xdp, to) \
|
||||
trace_xdp_redirect(dev, xdp, NULL, 0, NULL, to)
|
||||
|
||||
#define _trace_xdp_redirect_err(dev, xdp, to, err) \
|
||||
trace_xdp_redirect_err(dev, xdp, NULL, err, NULL, to);
|
||||
#define _trace_xdp_redirect_err(dev, xdp, to, err) \
|
||||
trace_xdp_redirect_err(dev, xdp, NULL, err, NULL, to)
|
||||
|
||||
#define _trace_xdp_redirect_map(dev, xdp, to, map, index) \
|
||||
trace_xdp_redirect(dev, xdp, to, 0, map, index);
|
||||
trace_xdp_redirect(dev, xdp, to, 0, map, index)
|
||||
|
||||
#define _trace_xdp_redirect_map_err(dev, xdp, to, map, index, err) \
|
||||
trace_xdp_redirect_err(dev, xdp, to, err, map, index);
|
||||
trace_xdp_redirect_err(dev, xdp, to, err, map, index)
|
||||
|
||||
/* not used anymore, but kept around so as not to break old programs */
|
||||
DEFINE_EVENT(xdp_redirect_template, xdp_redirect_map,
|
||||
|
|
|
@ -3822,6 +3822,14 @@ union bpf_attr {
|
|||
* The **hash_algo** is returned on success,
|
||||
* **-EOPNOTSUP** if IMA is disabled or **-EINVAL** if
|
||||
* invalid arguments are passed.
|
||||
*
|
||||
* struct socket *bpf_sock_from_file(struct file *file)
|
||||
* Description
|
||||
* If the given file represents a socket, returns the associated
|
||||
* socket.
|
||||
* Return
|
||||
* A pointer to a struct socket on success or NULL if the file is
|
||||
* not a socket.
|
||||
*/
|
||||
#define __BPF_FUNC_MAPPER(FN) \
|
||||
FN(unspec), \
|
||||
|
@ -3986,6 +3994,7 @@ union bpf_attr {
|
|||
FN(bprm_opts_set), \
|
||||
FN(ktime_get_coarse_ns), \
|
||||
FN(ima_inode_hash), \
|
||||
FN(sock_from_file), \
|
||||
/* */
|
||||
|
||||
/* integer value in 'imm' field of BPF_CALL instruction selects which helper
|
||||
|
|
|
@ -224,7 +224,7 @@ static void *fd_htab_map_get_ptr(const struct bpf_map *map, struct htab_elem *l)
|
|||
|
||||
static struct htab_elem *get_htab_elem(struct bpf_htab *htab, int i)
|
||||
{
|
||||
return (struct htab_elem *) (htab->elems + i * htab->elem_size);
|
||||
return (struct htab_elem *) (htab->elems + i * (u64)htab->elem_size);
|
||||
}
|
||||
|
||||
static void htab_free_elems(struct bpf_htab *htab)
|
||||
|
@ -280,7 +280,7 @@ static int prealloc_init(struct bpf_htab *htab)
|
|||
if (!htab_is_percpu(htab) && !htab_is_lru(htab))
|
||||
num_entries += num_possible_cpus();
|
||||
|
||||
htab->elems = bpf_map_area_alloc(htab->elem_size * num_entries,
|
||||
htab->elems = bpf_map_area_alloc((u64)htab->elem_size * num_entries,
|
||||
htab->map.numa_node);
|
||||
if (!htab->elems)
|
||||
return -ENOMEM;
|
||||
|
@ -1412,7 +1412,7 @@ __htab_map_lookup_and_delete_batch(struct bpf_map *map,
|
|||
void *keys = NULL, *values = NULL, *value, *dst_key, *dst_val;
|
||||
void __user *uvalues = u64_to_user_ptr(attr->batch.values);
|
||||
void __user *ukeys = u64_to_user_ptr(attr->batch.keys);
|
||||
void *ubatch = u64_to_user_ptr(attr->batch.in_batch);
|
||||
void __user *ubatch = u64_to_user_ptr(attr->batch.in_batch);
|
||||
u32 batch, max_count, size, bucket_size;
|
||||
struct htab_elem *node_to_free = NULL;
|
||||
u64 elem_map_flags, map_flags;
|
||||
|
|
|
@ -2121,8 +2121,11 @@ static int bpf_prog_load(union bpf_attr *attr, union bpf_attr __user *uattr)
|
|||
if (IS_ERR(attach_btf))
|
||||
return -EINVAL;
|
||||
if (!btf_is_kernel(attach_btf)) {
|
||||
/* attaching through specifying bpf_prog's BTF
|
||||
* objects directly might be supported eventually
|
||||
*/
|
||||
btf_put(attach_btf);
|
||||
return -EINVAL;
|
||||
return -ENOTSUPP;
|
||||
}
|
||||
}
|
||||
} else if (attr->attach_btf_id) {
|
||||
|
|
|
@ -3767,7 +3767,8 @@ static int check_stack_boundary(struct bpf_verifier_env *env, int regno,
|
|||
goto mark;
|
||||
|
||||
if (state->stack[spi].slot_type[0] == STACK_SPILL &&
|
||||
state->stack[spi].spilled_ptr.type == SCALAR_VALUE) {
|
||||
(state->stack[spi].spilled_ptr.type == SCALAR_VALUE ||
|
||||
env->allow_ptr_leaks)) {
|
||||
__mark_reg_unknown(env, &state->stack[spi].spilled_ptr);
|
||||
for (j = 0; j < BPF_REG_SIZE; j++)
|
||||
state->stack[spi].slot_type[j] = STACK_MISC;
|
||||
|
|
|
@ -1758,6 +1758,8 @@ tracing_prog_func_proto(enum bpf_func_id func_id, const struct bpf_prog *prog)
|
|||
return &bpf_sk_storage_get_tracing_proto;
|
||||
case BPF_FUNC_sk_storage_delete:
|
||||
return &bpf_sk_storage_delete_tracing_proto;
|
||||
case BPF_FUNC_sock_from_file:
|
||||
return &bpf_sock_from_file_proto;
|
||||
#endif
|
||||
case BPF_FUNC_seq_printf:
|
||||
return prog->expected_attach_type == BPF_TRACE_ITER ?
|
||||
|
|
|
@ -394,6 +394,7 @@ static bool bpf_sk_storage_tracing_allowed(const struct bpf_prog *prog)
|
|||
* use the bpf_sk_storage_(get|delete) helper.
|
||||
*/
|
||||
switch (prog->expected_attach_type) {
|
||||
case BPF_TRACE_ITER:
|
||||
case BPF_TRACE_RAW_TP:
|
||||
/* bpf_sk_storage has no trace point */
|
||||
return true;
|
||||
|
|
|
@ -10413,6 +10413,24 @@ const struct bpf_func_proto bpf_skc_to_udp6_sock_proto = {
|
|||
.ret_btf_id = &btf_sock_ids[BTF_SOCK_TYPE_UDP6],
|
||||
};
|
||||
|
||||
BPF_CALL_1(bpf_sock_from_file, struct file *, file)
|
||||
{
|
||||
return (unsigned long)sock_from_file(file);
|
||||
}
|
||||
|
||||
BTF_ID_LIST(bpf_sock_from_file_btf_ids)
|
||||
BTF_ID(struct, socket)
|
||||
BTF_ID(struct, file)
|
||||
|
||||
const struct bpf_func_proto bpf_sock_from_file_proto = {
|
||||
.func = bpf_sock_from_file,
|
||||
.gpl_only = false,
|
||||
.ret_type = RET_PTR_TO_BTF_ID_OR_NULL,
|
||||
.ret_btf_id = &bpf_sock_from_file_btf_ids[0],
|
||||
.arg1_type = ARG_PTR_TO_BTF_ID,
|
||||
.arg1_btf_id = &bpf_sock_from_file_btf_ids[1],
|
||||
};
|
||||
|
||||
static const struct bpf_func_proto *
|
||||
bpf_sk_base_func_proto(enum bpf_func_id func_id)
|
||||
{
|
||||
|
|
|
@ -68,9 +68,8 @@ struct update_classid_context {
|
|||
|
||||
static int update_classid_sock(const void *v, struct file *file, unsigned n)
|
||||
{
|
||||
int err;
|
||||
struct update_classid_context *ctx = (void *)v;
|
||||
struct socket *sock = sock_from_file(file, &err);
|
||||
struct socket *sock = sock_from_file(file);
|
||||
|
||||
if (sock) {
|
||||
spin_lock(&cgroup_sk_update_lock);
|
||||
|
|
|
@ -220,8 +220,7 @@ static ssize_t write_priomap(struct kernfs_open_file *of,
|
|||
|
||||
static int update_netprio(const void *v, struct file *file, unsigned n)
|
||||
{
|
||||
int err;
|
||||
struct socket *sock = sock_from_file(file, &err);
|
||||
struct socket *sock = sock_from_file(file);
|
||||
if (sock) {
|
||||
spin_lock(&cgroup_sk_update_lock);
|
||||
sock_cgroup_set_prioidx(&sock->sk->sk_cgrp_data,
|
||||
|
|
|
@ -2827,14 +2827,8 @@ EXPORT_SYMBOL(sock_no_mmap);
|
|||
void __receive_sock(struct file *file)
|
||||
{
|
||||
struct socket *sock;
|
||||
int error;
|
||||
|
||||
/*
|
||||
* The resulting value of "error" is ignored here since we only
|
||||
* need to take action when the file is a socket and testing
|
||||
* "sock" for NULL is sufficient.
|
||||
*/
|
||||
sock = sock_from_file(file, &error);
|
||||
sock = sock_from_file(file);
|
||||
if (sock) {
|
||||
sock_update_netprioidx(&sock->sk->sk_cgrp_data);
|
||||
sock_update_classid(&sock->sk->sk_cgrp_data);
|
||||
|
|
27
net/socket.c
27
net/socket.c
|
@ -445,17 +445,15 @@ static int sock_map_fd(struct socket *sock, int flags)
|
|||
/**
|
||||
* sock_from_file - Return the &socket bounded to @file.
|
||||
* @file: file
|
||||
* @err: pointer to an error code return
|
||||
*
|
||||
* On failure returns %NULL and assigns -ENOTSOCK to @err.
|
||||
* On failure returns %NULL.
|
||||
*/
|
||||
|
||||
struct socket *sock_from_file(struct file *file, int *err)
|
||||
struct socket *sock_from_file(struct file *file)
|
||||
{
|
||||
if (file->f_op == &socket_file_ops)
|
||||
return file->private_data; /* set in sock_map_fd */
|
||||
|
||||
*err = -ENOTSOCK;
|
||||
return NULL;
|
||||
}
|
||||
EXPORT_SYMBOL(sock_from_file);
|
||||
|
@ -484,9 +482,11 @@ struct socket *sockfd_lookup(int fd, int *err)
|
|||
return NULL;
|
||||
}
|
||||
|
||||
sock = sock_from_file(file, err);
|
||||
if (!sock)
|
||||
sock = sock_from_file(file);
|
||||
if (!sock) {
|
||||
*err = -ENOTSOCK;
|
||||
fput(file);
|
||||
}
|
||||
return sock;
|
||||
}
|
||||
EXPORT_SYMBOL(sockfd_lookup);
|
||||
|
@ -498,11 +498,12 @@ static struct socket *sockfd_lookup_light(int fd, int *err, int *fput_needed)
|
|||
|
||||
*err = -EBADF;
|
||||
if (f.file) {
|
||||
sock = sock_from_file(f.file, err);
|
||||
sock = sock_from_file(f.file);
|
||||
if (likely(sock)) {
|
||||
*fput_needed = f.flags & FDPUT_FPUT;
|
||||
return sock;
|
||||
}
|
||||
*err = -ENOTSOCK;
|
||||
fdput(f);
|
||||
}
|
||||
return NULL;
|
||||
|
@ -1693,9 +1694,11 @@ int __sys_accept4_file(struct file *file, unsigned file_flags,
|
|||
if (SOCK_NONBLOCK != O_NONBLOCK && (flags & SOCK_NONBLOCK))
|
||||
flags = (flags & ~SOCK_NONBLOCK) | O_NONBLOCK;
|
||||
|
||||
sock = sock_from_file(file, &err);
|
||||
if (!sock)
|
||||
sock = sock_from_file(file);
|
||||
if (!sock) {
|
||||
err = -ENOTSOCK;
|
||||
goto out;
|
||||
}
|
||||
|
||||
err = -ENFILE;
|
||||
newsock = sock_alloc();
|
||||
|
@ -1818,9 +1821,11 @@ int __sys_connect_file(struct file *file, struct sockaddr_storage *address,
|
|||
struct socket *sock;
|
||||
int err;
|
||||
|
||||
sock = sock_from_file(file, &err);
|
||||
if (!sock)
|
||||
sock = sock_from_file(file);
|
||||
if (!sock) {
|
||||
err = -ENOTSOCK;
|
||||
goto out;
|
||||
}
|
||||
|
||||
err =
|
||||
security_socket_connect(sock, (struct sockaddr *)address, addrlen);
|
||||
|
|
|
@ -564,12 +564,12 @@ static int xsk_recvmsg(struct socket *sock, struct msghdr *m, size_t len, int fl
|
|||
struct sock *sk = sock->sk;
|
||||
struct xdp_sock *xs = xdp_sk(sk);
|
||||
|
||||
if (unlikely(!xsk_is_bound(xs)))
|
||||
return -ENXIO;
|
||||
if (unlikely(!(xs->dev->flags & IFF_UP)))
|
||||
return -ENETDOWN;
|
||||
if (unlikely(!xs->rx))
|
||||
return -ENOBUFS;
|
||||
if (unlikely(!xsk_is_bound(xs)))
|
||||
return -ENXIO;
|
||||
if (unlikely(need_wait))
|
||||
return -EOPNOTSUPP;
|
||||
|
||||
|
|
|
@ -1275,6 +1275,8 @@ static void tx_only(struct xsk_socket_info *xsk, u32 *frame_nb, int batch_size)
|
|||
while (xsk_ring_prod__reserve(&xsk->tx, batch_size, &idx) <
|
||||
batch_size) {
|
||||
complete_tx_only(xsk, batch_size);
|
||||
if (benchmark_done)
|
||||
return;
|
||||
}
|
||||
|
||||
for (i = 0; i < batch_size; i++) {
|
||||
|
|
|
@ -437,6 +437,8 @@ class PrinterHelpers(Printer):
|
|||
'struct path',
|
||||
'struct btf_ptr',
|
||||
'struct inode',
|
||||
'struct socket',
|
||||
'struct file',
|
||||
]
|
||||
known_types = {
|
||||
'...',
|
||||
|
@ -482,6 +484,8 @@ class PrinterHelpers(Printer):
|
|||
'struct path',
|
||||
'struct btf_ptr',
|
||||
'struct inode',
|
||||
'struct socket',
|
||||
'struct file',
|
||||
}
|
||||
mapped_types = {
|
||||
'u8': '__u8',
|
||||
|
|
|
@ -3822,6 +3822,14 @@ union bpf_attr {
|
|||
* The **hash_algo** is returned on success,
|
||||
* **-EOPNOTSUP** if IMA is disabled or **-EINVAL** if
|
||||
* invalid arguments are passed.
|
||||
*
|
||||
* struct socket *bpf_sock_from_file(struct file *file)
|
||||
* Description
|
||||
* If the given file represents a socket, returns the associated
|
||||
* socket.
|
||||
* Return
|
||||
* A pointer to a struct socket on success or NULL if the file is
|
||||
* not a socket.
|
||||
*/
|
||||
#define __BPF_FUNC_MAPPER(FN) \
|
||||
FN(unspec), \
|
||||
|
@ -3986,6 +3994,7 @@ union bpf_attr {
|
|||
FN(bprm_opts_set), \
|
||||
FN(ktime_get_coarse_ns), \
|
||||
FN(ima_inode_hash), \
|
||||
FN(sock_from_file), \
|
||||
/* */
|
||||
|
||||
/* integer value in 'imm' field of BPF_CALL instruction selects which helper
|
||||
|
|
|
@ -2518,7 +2518,7 @@ static int bpf_object__finalize_btf(struct bpf_object *obj)
|
|||
return 0;
|
||||
}
|
||||
|
||||
static inline bool libbpf_prog_needs_vmlinux_btf(struct bpf_program *prog)
|
||||
static bool prog_needs_vmlinux_btf(struct bpf_program *prog)
|
||||
{
|
||||
if (prog->type == BPF_PROG_TYPE_STRUCT_OPS ||
|
||||
prog->type == BPF_PROG_TYPE_LSM)
|
||||
|
@ -2533,37 +2533,43 @@ static inline bool libbpf_prog_needs_vmlinux_btf(struct bpf_program *prog)
|
|||
return false;
|
||||
}
|
||||
|
||||
static int bpf_object__load_vmlinux_btf(struct bpf_object *obj)
|
||||
static bool obj_needs_vmlinux_btf(const struct bpf_object *obj)
|
||||
{
|
||||
bool need_vmlinux_btf = false;
|
||||
struct bpf_program *prog;
|
||||
int i, err;
|
||||
int i;
|
||||
|
||||
/* CO-RE relocations need kernel BTF */
|
||||
if (obj->btf_ext && obj->btf_ext->core_relo_info.len)
|
||||
need_vmlinux_btf = true;
|
||||
return true;
|
||||
|
||||
/* Support for typed ksyms needs kernel BTF */
|
||||
for (i = 0; i < obj->nr_extern; i++) {
|
||||
const struct extern_desc *ext;
|
||||
|
||||
ext = &obj->externs[i];
|
||||
if (ext->type == EXT_KSYM && ext->ksym.type_id) {
|
||||
need_vmlinux_btf = true;
|
||||
break;
|
||||
}
|
||||
if (ext->type == EXT_KSYM && ext->ksym.type_id)
|
||||
return true;
|
||||
}
|
||||
|
||||
bpf_object__for_each_program(prog, obj) {
|
||||
if (!prog->load)
|
||||
continue;
|
||||
if (libbpf_prog_needs_vmlinux_btf(prog)) {
|
||||
need_vmlinux_btf = true;
|
||||
break;
|
||||
}
|
||||
if (prog_needs_vmlinux_btf(prog))
|
||||
return true;
|
||||
}
|
||||
|
||||
if (!need_vmlinux_btf)
|
||||
return false;
|
||||
}
|
||||
|
||||
static int bpf_object__load_vmlinux_btf(struct bpf_object *obj, bool force)
|
||||
{
|
||||
int err;
|
||||
|
||||
/* btf_vmlinux could be loaded earlier */
|
||||
if (obj->btf_vmlinux)
|
||||
return 0;
|
||||
|
||||
if (!force && !obj_needs_vmlinux_btf(obj))
|
||||
return 0;
|
||||
|
||||
obj->btf_vmlinux = libbpf_find_kernel_btf();
|
||||
|
@ -7475,7 +7481,7 @@ int bpf_object__load_xattr(struct bpf_object_load_attr *attr)
|
|||
}
|
||||
|
||||
err = bpf_object__probe_loading(obj);
|
||||
err = err ? : bpf_object__load_vmlinux_btf(obj);
|
||||
err = err ? : bpf_object__load_vmlinux_btf(obj, false);
|
||||
err = err ? : bpf_object__resolve_externs(obj, obj->kconfig);
|
||||
err = err ? : bpf_object__sanitize_and_load_btf(obj);
|
||||
err = err ? : bpf_object__sanitize_maps(obj);
|
||||
|
@ -10870,23 +10876,33 @@ int bpf_program__set_attach_target(struct bpf_program *prog,
|
|||
int attach_prog_fd,
|
||||
const char *attach_func_name)
|
||||
{
|
||||
int btf_id;
|
||||
int btf_obj_fd = 0, btf_id = 0, err;
|
||||
|
||||
if (!prog || attach_prog_fd < 0 || !attach_func_name)
|
||||
return -EINVAL;
|
||||
|
||||
if (attach_prog_fd)
|
||||
if (prog->obj->loaded)
|
||||
return -EINVAL;
|
||||
|
||||
if (attach_prog_fd) {
|
||||
btf_id = libbpf_find_prog_btf_id(attach_func_name,
|
||||
attach_prog_fd);
|
||||
else
|
||||
btf_id = libbpf_find_vmlinux_btf_id(attach_func_name,
|
||||
prog->expected_attach_type);
|
||||
|
||||
if (btf_id < 0)
|
||||
return btf_id;
|
||||
if (btf_id < 0)
|
||||
return btf_id;
|
||||
} else {
|
||||
/* load btf_vmlinux, if not yet */
|
||||
err = bpf_object__load_vmlinux_btf(prog->obj, true);
|
||||
if (err)
|
||||
return err;
|
||||
err = find_kernel_btf_id(prog->obj, attach_func_name,
|
||||
prog->expected_attach_type,
|
||||
&btf_obj_fd, &btf_id);
|
||||
if (err)
|
||||
return err;
|
||||
}
|
||||
|
||||
prog->attach_btf_id = btf_id;
|
||||
prog->attach_btf_obj_fd = 0;
|
||||
prog->attach_btf_obj_fd = btf_obj_fd;
|
||||
prog->attach_prog_fd = attach_prog_fd;
|
||||
return 0;
|
||||
}
|
||||
|
|
|
@ -536,6 +536,7 @@ LIBBPF_API int ring_buffer__add(struct ring_buffer *rb, int map_fd,
|
|||
ring_buffer_sample_fn sample_cb, void *ctx);
|
||||
LIBBPF_API int ring_buffer__poll(struct ring_buffer *rb, int timeout_ms);
|
||||
LIBBPF_API int ring_buffer__consume(struct ring_buffer *rb);
|
||||
LIBBPF_API int ring_buffer__epoll_fd(const struct ring_buffer *rb);
|
||||
|
||||
/* Perf buffer APIs */
|
||||
struct perf_buffer;
|
||||
|
|
|
@ -346,6 +346,7 @@ LIBBPF_0.3.0 {
|
|||
btf__parse_split;
|
||||
btf__new_empty_split;
|
||||
btf__new_split;
|
||||
ring_buffer__epoll_fd;
|
||||
xsk_setup_xdp_prog;
|
||||
xsk_socket__update_xskmap;
|
||||
} LIBBPF_0.2.0;
|
||||
|
|
|
@ -282,3 +282,9 @@ int ring_buffer__poll(struct ring_buffer *rb, int timeout_ms)
|
|||
}
|
||||
return cnt < 0 ? -errno : res;
|
||||
}
|
||||
|
||||
/* Get an fd that can be used to sleep until data is available in the ring(s) */
|
||||
int ring_buffer__epoll_fd(const struct ring_buffer *rb)
|
||||
{
|
||||
return rb->epoll_fd;
|
||||
}
|
||||
|
|
1
tools/testing/selftests/bpf/.gitignore
vendored
1
tools/testing/selftests/bpf/.gitignore
vendored
|
@ -36,3 +36,4 @@ test_cpp
|
|||
/runqslower
|
||||
/bench
|
||||
*.ko
|
||||
xdpxceiver
|
||||
|
|
|
@ -19,7 +19,6 @@ ifneq ($(wildcard $(GENHDR)),)
|
|||
endif
|
||||
|
||||
CLANG ?= clang
|
||||
LLC ?= llc
|
||||
LLVM_OBJCOPY ?= llvm-objcopy
|
||||
BPF_GCC ?= $(shell command -v bpf-gcc;)
|
||||
SAN_CFLAGS ?=
|
||||
|
@ -46,7 +45,8 @@ endif
|
|||
|
||||
TEST_GEN_FILES =
|
||||
TEST_FILES = test_lwt_ip_encap.o \
|
||||
test_tc_edt.o
|
||||
test_tc_edt.o \
|
||||
xsk_prereqs.sh
|
||||
|
||||
# Order correspond to 'make run_tests' order
|
||||
TEST_PROGS := test_kmod.sh \
|
||||
|
@ -70,17 +70,17 @@ TEST_PROGS := test_kmod.sh \
|
|||
test_bpftool_build.sh \
|
||||
test_bpftool.sh \
|
||||
test_bpftool_metadata.sh \
|
||||
test_xsk.sh
|
||||
|
||||
TEST_PROGS_EXTENDED := with_addr.sh \
|
||||
with_tunnels.sh \
|
||||
tcp_client.py \
|
||||
tcp_server.py \
|
||||
test_xdp_vlan.sh
|
||||
|
||||
# Compile but not part of 'make run_tests'
|
||||
TEST_GEN_PROGS_EXTENDED = test_sock_addr test_skb_cgroup_id_user \
|
||||
flow_dissector_load test_flow_dissector test_tcp_check_syncookie_user \
|
||||
test_lirc_mode2_user xdping test_cpp runqslower bench bpf_testmod.ko
|
||||
test_lirc_mode2_user xdping test_cpp runqslower bench bpf_testmod.ko \
|
||||
xdpxceiver
|
||||
|
||||
TEST_CUSTOM_PROGS = urandom_read
|
||||
|
||||
|
@ -115,6 +115,13 @@ INCLUDE_DIR := $(SCRATCH_DIR)/include
|
|||
BPFOBJ := $(BUILD_DIR)/libbpf/libbpf.a
|
||||
RESOLVE_BTFIDS := $(BUILD_DIR)/resolve_btfids/resolve_btfids
|
||||
|
||||
VMLINUX_BTF_PATHS ?= $(if $(O),$(O)/vmlinux) \
|
||||
$(if $(KBUILD_OUTPUT),$(KBUILD_OUTPUT)/vmlinux) \
|
||||
../../../../vmlinux \
|
||||
/sys/kernel/btf/vmlinux \
|
||||
/boot/vmlinux-$(shell uname -r)
|
||||
VMLINUX_BTF ?= $(abspath $(firstword $(wildcard $(VMLINUX_BTF_PATHS))))
|
||||
|
||||
# Define simple and short `make test_progs`, `make test_sysctl`, etc targets
|
||||
# to build individual tests.
|
||||
# NOTE: Semicolon at the end is critical to override lib.mk's default static
|
||||
|
@ -139,6 +146,7 @@ $(OUTPUT)/urandom_read: urandom_read.c
|
|||
|
||||
$(OUTPUT)/bpf_testmod.ko: $(VMLINUX_BTF) $(wildcard bpf_testmod/Makefile bpf_testmod/*.[ch])
|
||||
$(call msg,MOD,,$@)
|
||||
$(Q)$(RM) bpf_testmod/bpf_testmod.ko # force re-compilation
|
||||
$(Q)$(MAKE) $(submake_extras) -C bpf_testmod
|
||||
$(Q)cp bpf_testmod/bpf_testmod.ko $@
|
||||
|
||||
|
@ -146,13 +154,6 @@ $(OUTPUT)/test_stub.o: test_stub.c $(BPFOBJ)
|
|||
$(call msg,CC,,$@)
|
||||
$(Q)$(CC) -c $(CFLAGS) -o $@ $<
|
||||
|
||||
VMLINUX_BTF_PATHS ?= $(if $(O),$(O)/vmlinux) \
|
||||
$(if $(KBUILD_OUTPUT),$(KBUILD_OUTPUT)/vmlinux) \
|
||||
../../../../vmlinux \
|
||||
/sys/kernel/btf/vmlinux \
|
||||
/boot/vmlinux-$(shell uname -r)
|
||||
VMLINUX_BTF ?= $(abspath $(firstword $(wildcard $(VMLINUX_BTF_PATHS))))
|
||||
|
||||
DEFAULT_BPFTOOL := $(SCRATCH_DIR)/sbin/bpftool
|
||||
|
||||
$(OUTPUT)/runqslower: $(BPFOBJ) | $(DEFAULT_BPFTOOL)
|
||||
|
@ -251,31 +252,19 @@ $(OUTPUT)/flow_dissector_load.o: flow_dissector_load.h
|
|||
# $1 - input .c file
|
||||
# $2 - output .o file
|
||||
# $3 - CFLAGS
|
||||
# $4 - LDFLAGS
|
||||
define CLANG_BPF_BUILD_RULE
|
||||
$(call msg,CLNG-LLC,$(TRUNNER_BINARY),$2)
|
||||
$(Q)($(CLANG) $3 -O2 -target bpf -emit-llvm \
|
||||
-c $1 -o - || echo "BPF obj compilation failed") | \
|
||||
$(LLC) -mattr=dwarfris -march=bpf -mcpu=v3 $4 -filetype=obj -o $2
|
||||
$(call msg,CLNG-BPF,$(TRUNNER_BINARY),$2)
|
||||
$(Q)$(CLANG) $3 -O2 -target bpf -c $1 -o $2 -mcpu=v3
|
||||
endef
|
||||
# Similar to CLANG_BPF_BUILD_RULE, but with disabled alu32
|
||||
define CLANG_NOALU32_BPF_BUILD_RULE
|
||||
$(call msg,CLNG-LLC,$(TRUNNER_BINARY),$2)
|
||||
$(Q)($(CLANG) $3 -O2 -target bpf -emit-llvm \
|
||||
-c $1 -o - || echo "BPF obj compilation failed") | \
|
||||
$(LLC) -march=bpf -mcpu=v2 $4 -filetype=obj -o $2
|
||||
endef
|
||||
# Similar to CLANG_BPF_BUILD_RULE, but using native Clang and bpf LLC
|
||||
define CLANG_NATIVE_BPF_BUILD_RULE
|
||||
$(call msg,CLNG-BPF,$(TRUNNER_BINARY),$2)
|
||||
$(Q)($(CLANG) $3 -O2 -emit-llvm \
|
||||
-c $1 -o - || echo "BPF obj compilation failed") | \
|
||||
$(LLC) -march=bpf -mcpu=v3 $4 -filetype=obj -o $2
|
||||
$(Q)$(CLANG) $3 -O2 -target bpf -c $1 -o $2 -mcpu=v2
|
||||
endef
|
||||
# Build BPF object using GCC
|
||||
define GCC_BPF_BUILD_RULE
|
||||
$(call msg,GCC-BPF,$(TRUNNER_BINARY),$2)
|
||||
$(Q)$(BPF_GCC) $3 $4 -O2 -c $1 -o $2
|
||||
$(Q)$(BPF_GCC) $3 -O2 -c $1 -o $2
|
||||
endef
|
||||
|
||||
SKEL_BLACKLIST := btf__% test_pinning_invalid.c test_sk_assign.c
|
||||
|
@ -330,8 +319,7 @@ $(TRUNNER_BPF_OBJS): $(TRUNNER_OUTPUT)/%.o: \
|
|||
$$(INCLUDE_DIR)/vmlinux.h \
|
||||
$(wildcard $(BPFDIR)/bpf_*.h) | $(TRUNNER_OUTPUT)
|
||||
$$(call $(TRUNNER_BPF_BUILD_RULE),$$<,$$@, \
|
||||
$(TRUNNER_BPF_CFLAGS), \
|
||||
$(TRUNNER_BPF_LDFLAGS))
|
||||
$(TRUNNER_BPF_CFLAGS))
|
||||
|
||||
$(TRUNNER_BPF_SKELS): $(TRUNNER_OUTPUT)/%.skel.h: \
|
||||
$(TRUNNER_OUTPUT)/%.o \
|
||||
|
@ -399,19 +387,16 @@ TRUNNER_EXTRA_FILES := $(OUTPUT)/urandom_read $(OUTPUT)/bpf_testmod.ko \
|
|||
$(wildcard progs/btf_dump_test_case_*.c)
|
||||
TRUNNER_BPF_BUILD_RULE := CLANG_BPF_BUILD_RULE
|
||||
TRUNNER_BPF_CFLAGS := $(BPF_CFLAGS) $(CLANG_CFLAGS)
|
||||
TRUNNER_BPF_LDFLAGS := -mattr=+alu32
|
||||
$(eval $(call DEFINE_TEST_RUNNER,test_progs))
|
||||
|
||||
# Define test_progs-no_alu32 test runner.
|
||||
TRUNNER_BPF_BUILD_RULE := CLANG_NOALU32_BPF_BUILD_RULE
|
||||
TRUNNER_BPF_LDFLAGS :=
|
||||
$(eval $(call DEFINE_TEST_RUNNER,test_progs,no_alu32))
|
||||
|
||||
# Define test_progs BPF-GCC-flavored test runner.
|
||||
ifneq ($(BPF_GCC),)
|
||||
TRUNNER_BPF_BUILD_RULE := GCC_BPF_BUILD_RULE
|
||||
TRUNNER_BPF_CFLAGS := $(BPF_CFLAGS) $(call get_sys_includes,gcc)
|
||||
TRUNNER_BPF_LDFLAGS :=
|
||||
$(eval $(call DEFINE_TEST_RUNNER,test_progs,bpf_gcc))
|
||||
endif
|
||||
|
||||
|
@ -422,7 +407,6 @@ TRUNNER_EXTRA_SOURCES := test_maps.c
|
|||
TRUNNER_EXTRA_FILES :=
|
||||
TRUNNER_BPF_BUILD_RULE := $$(error no BPF objects should be built)
|
||||
TRUNNER_BPF_CFLAGS :=
|
||||
TRUNNER_BPF_LDFLAGS :=
|
||||
$(eval $(call DEFINE_TEST_RUNNER,test_maps))
|
||||
|
||||
# Define test_verifier test runner.
|
||||
|
|
|
@ -7,6 +7,8 @@ set -o pipefail
|
|||
|
||||
IMA_POLICY_FILE="/sys/kernel/security/ima/policy"
|
||||
TEST_BINARY="/bin/true"
|
||||
VERBOSE="${SELFTESTS_VERBOSE:=0}"
|
||||
LOG_FILE="$(mktemp /tmp/ima_setup.XXXX.log)"
|
||||
|
||||
usage()
|
||||
{
|
||||
|
@ -75,6 +77,19 @@ run()
|
|||
exec "${copied_bin_path}"
|
||||
}
|
||||
|
||||
catch()
|
||||
{
|
||||
local exit_code="$1"
|
||||
local log_file="$2"
|
||||
|
||||
if [[ "${exit_code}" -ne 0 ]]; then
|
||||
cat "${log_file}" >&3
|
||||
fi
|
||||
|
||||
rm -f "${log_file}"
|
||||
exit ${exit_code}
|
||||
}
|
||||
|
||||
main()
|
||||
{
|
||||
[[ $# -ne 2 ]] && usage
|
||||
|
@ -96,4 +111,13 @@ main()
|
|||
fi
|
||||
}
|
||||
|
||||
trap 'catch "$?" "${LOG_FILE}"' EXIT
|
||||
|
||||
if [[ "${VERBOSE}" -eq 0 ]]; then
|
||||
# Save the stderr to 3 so that we can output back to
|
||||
# it incase of an error.
|
||||
exec 3>&2 1>"${LOG_FILE}" 2>&1
|
||||
fi
|
||||
|
||||
main "$@"
|
||||
rm -f "${LOG_FILE}"
|
||||
|
|
|
@ -20,6 +20,7 @@
|
|||
#include "bpf_iter_bpf_percpu_hash_map.skel.h"
|
||||
#include "bpf_iter_bpf_array_map.skel.h"
|
||||
#include "bpf_iter_bpf_percpu_array_map.skel.h"
|
||||
#include "bpf_iter_bpf_sk_storage_helpers.skel.h"
|
||||
#include "bpf_iter_bpf_sk_storage_map.skel.h"
|
||||
#include "bpf_iter_test_kern5.skel.h"
|
||||
#include "bpf_iter_test_kern6.skel.h"
|
||||
|
@ -913,6 +914,119 @@ out:
|
|||
bpf_iter_bpf_percpu_array_map__destroy(skel);
|
||||
}
|
||||
|
||||
/* An iterator program deletes all local storage in a map. */
|
||||
static void test_bpf_sk_storage_delete(void)
|
||||
{
|
||||
DECLARE_LIBBPF_OPTS(bpf_iter_attach_opts, opts);
|
||||
struct bpf_iter_bpf_sk_storage_helpers *skel;
|
||||
union bpf_iter_link_info linfo;
|
||||
int err, len, map_fd, iter_fd;
|
||||
struct bpf_link *link;
|
||||
int sock_fd = -1;
|
||||
__u32 val = 42;
|
||||
char buf[64];
|
||||
|
||||
skel = bpf_iter_bpf_sk_storage_helpers__open_and_load();
|
||||
if (CHECK(!skel, "bpf_iter_bpf_sk_storage_helpers__open_and_load",
|
||||
"skeleton open_and_load failed\n"))
|
||||
return;
|
||||
|
||||
map_fd = bpf_map__fd(skel->maps.sk_stg_map);
|
||||
|
||||
sock_fd = socket(AF_INET6, SOCK_STREAM, 0);
|
||||
if (CHECK(sock_fd < 0, "socket", "errno: %d\n", errno))
|
||||
goto out;
|
||||
err = bpf_map_update_elem(map_fd, &sock_fd, &val, BPF_NOEXIST);
|
||||
if (CHECK(err, "map_update", "map_update failed\n"))
|
||||
goto out;
|
||||
|
||||
memset(&linfo, 0, sizeof(linfo));
|
||||
linfo.map.map_fd = map_fd;
|
||||
opts.link_info = &linfo;
|
||||
opts.link_info_len = sizeof(linfo);
|
||||
link = bpf_program__attach_iter(skel->progs.delete_bpf_sk_storage_map,
|
||||
&opts);
|
||||
if (CHECK(IS_ERR(link), "attach_iter", "attach_iter failed\n"))
|
||||
goto out;
|
||||
|
||||
iter_fd = bpf_iter_create(bpf_link__fd(link));
|
||||
if (CHECK(iter_fd < 0, "create_iter", "create_iter failed\n"))
|
||||
goto free_link;
|
||||
|
||||
/* do some tests */
|
||||
while ((len = read(iter_fd, buf, sizeof(buf))) > 0)
|
||||
;
|
||||
if (CHECK(len < 0, "read", "read failed: %s\n", strerror(errno)))
|
||||
goto close_iter;
|
||||
|
||||
/* test results */
|
||||
err = bpf_map_lookup_elem(map_fd, &sock_fd, &val);
|
||||
if (CHECK(!err || errno != ENOENT, "bpf_map_lookup_elem",
|
||||
"map value wasn't deleted (err=%d, errno=%d)\n", err, errno))
|
||||
goto close_iter;
|
||||
|
||||
close_iter:
|
||||
close(iter_fd);
|
||||
free_link:
|
||||
bpf_link__destroy(link);
|
||||
out:
|
||||
if (sock_fd >= 0)
|
||||
close(sock_fd);
|
||||
bpf_iter_bpf_sk_storage_helpers__destroy(skel);
|
||||
}
|
||||
|
||||
/* This creates a socket and its local storage. It then runs a task_iter BPF
|
||||
* program that replaces the existing socket local storage with the tgid of the
|
||||
* only task owning a file descriptor to this socket, this process, prog_tests.
|
||||
* It then runs a tcp socket iterator that negates the value in the existing
|
||||
* socket local storage, the test verifies that the resulting value is -pid.
|
||||
*/
|
||||
static void test_bpf_sk_storage_get(void)
|
||||
{
|
||||
struct bpf_iter_bpf_sk_storage_helpers *skel;
|
||||
int err, map_fd, val = -1;
|
||||
int sock_fd = -1;
|
||||
|
||||
skel = bpf_iter_bpf_sk_storage_helpers__open_and_load();
|
||||
if (CHECK(!skel, "bpf_iter_bpf_sk_storage_helpers__open_and_load",
|
||||
"skeleton open_and_load failed\n"))
|
||||
return;
|
||||
|
||||
sock_fd = socket(AF_INET6, SOCK_STREAM, 0);
|
||||
if (CHECK(sock_fd < 0, "socket", "errno: %d\n", errno))
|
||||
goto out;
|
||||
|
||||
err = listen(sock_fd, 1);
|
||||
if (CHECK(err != 0, "listen", "errno: %d\n", errno))
|
||||
goto close_socket;
|
||||
|
||||
map_fd = bpf_map__fd(skel->maps.sk_stg_map);
|
||||
|
||||
err = bpf_map_update_elem(map_fd, &sock_fd, &val, BPF_NOEXIST);
|
||||
if (CHECK(err, "bpf_map_update_elem", "map_update_failed\n"))
|
||||
goto close_socket;
|
||||
|
||||
do_dummy_read(skel->progs.fill_socket_owner);
|
||||
|
||||
err = bpf_map_lookup_elem(map_fd, &sock_fd, &val);
|
||||
if (CHECK(err || val != getpid(), "bpf_map_lookup_elem",
|
||||
"map value wasn't set correctly (expected %d, got %d, err=%d)\n",
|
||||
getpid(), val, err))
|
||||
goto close_socket;
|
||||
|
||||
do_dummy_read(skel->progs.negate_socket_local_storage);
|
||||
|
||||
err = bpf_map_lookup_elem(map_fd, &sock_fd, &val);
|
||||
CHECK(err || val != -getpid(), "bpf_map_lookup_elem",
|
||||
"map value wasn't set correctly (expected %d, got %d, err=%d)\n",
|
||||
-getpid(), val, err);
|
||||
|
||||
close_socket:
|
||||
close(sock_fd);
|
||||
out:
|
||||
bpf_iter_bpf_sk_storage_helpers__destroy(skel);
|
||||
}
|
||||
|
||||
static void test_bpf_sk_storage_map(void)
|
||||
{
|
||||
DECLARE_LIBBPF_OPTS(bpf_iter_attach_opts, opts);
|
||||
|
@ -1067,6 +1181,10 @@ void test_bpf_iter(void)
|
|||
test_bpf_percpu_array_map();
|
||||
if (test__start_subtest("bpf_sk_storage_map"))
|
||||
test_bpf_sk_storage_map();
|
||||
if (test__start_subtest("bpf_sk_storage_delete"))
|
||||
test_bpf_sk_storage_delete();
|
||||
if (test__start_subtest("bpf_sk_storage_get"))
|
||||
test_bpf_sk_storage_get();
|
||||
if (test__start_subtest("rdonly-buf-out-of-bound"))
|
||||
test_rdonly_buf_out_of_bound();
|
||||
if (test__start_subtest("buf-neg-offset"))
|
||||
|
|
|
@ -28,10 +28,18 @@ void test_module_attach(void)
|
|||
struct test_module_attach__bss *bss;
|
||||
int err;
|
||||
|
||||
skel = test_module_attach__open_and_load();
|
||||
skel = test_module_attach__open();
|
||||
if (CHECK(!skel, "skel_open", "failed to open skeleton\n"))
|
||||
return;
|
||||
|
||||
err = bpf_program__set_attach_target(skel->progs.handle_fentry_manual,
|
||||
0, "bpf_testmod_test_read");
|
||||
ASSERT_OK(err, "set_attach_target");
|
||||
|
||||
err = test_module_attach__load(skel);
|
||||
if (CHECK(err, "skel_load", "failed to load skeleton\n"))
|
||||
return;
|
||||
|
||||
bss = skel->bss;
|
||||
|
||||
err = test_module_attach__attach(skel);
|
||||
|
@ -44,6 +52,7 @@ void test_module_attach(void)
|
|||
ASSERT_EQ(bss->raw_tp_read_sz, READ_SZ, "raw_tp");
|
||||
ASSERT_EQ(bss->tp_btf_read_sz, READ_SZ, "tp_btf");
|
||||
ASSERT_EQ(bss->fentry_read_sz, READ_SZ, "fentry");
|
||||
ASSERT_EQ(bss->fentry_manual_read_sz, READ_SZ, "fentry_manual");
|
||||
ASSERT_EQ(bss->fexit_read_sz, READ_SZ, "fexit");
|
||||
ASSERT_EQ(bss->fexit_ret, -EIO, "fexit_tet");
|
||||
ASSERT_EQ(bss->fmod_ret_read_sz, READ_SZ, "fmod_ret");
|
||||
|
|
|
@ -0,0 +1,65 @@
|
|||
// SPDX-License-Identifier: GPL-2.0
|
||||
/* Copyright (c) 2020 Google LLC. */
|
||||
#include "bpf_iter.h"
|
||||
#include <bpf/bpf_helpers.h>
|
||||
#include <bpf/bpf_tracing.h>
|
||||
|
||||
char _license[] SEC("license") = "GPL";
|
||||
|
||||
struct {
|
||||
__uint(type, BPF_MAP_TYPE_SK_STORAGE);
|
||||
__uint(map_flags, BPF_F_NO_PREALLOC);
|
||||
__type(key, int);
|
||||
__type(value, int);
|
||||
} sk_stg_map SEC(".maps");
|
||||
|
||||
SEC("iter/bpf_sk_storage_map")
|
||||
int delete_bpf_sk_storage_map(struct bpf_iter__bpf_sk_storage_map *ctx)
|
||||
{
|
||||
if (ctx->sk)
|
||||
bpf_sk_storage_delete(&sk_stg_map, ctx->sk);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
SEC("iter/task_file")
|
||||
int fill_socket_owner(struct bpf_iter__task_file *ctx)
|
||||
{
|
||||
struct task_struct *task = ctx->task;
|
||||
struct file *file = ctx->file;
|
||||
struct socket *sock;
|
||||
int *sock_tgid;
|
||||
|
||||
if (!task || !file)
|
||||
return 0;
|
||||
|
||||
sock = bpf_sock_from_file(file);
|
||||
if (!sock)
|
||||
return 0;
|
||||
|
||||
sock_tgid = bpf_sk_storage_get(&sk_stg_map, sock->sk, 0, 0);
|
||||
if (!sock_tgid)
|
||||
return 0;
|
||||
|
||||
*sock_tgid = task->tgid;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
SEC("iter/tcp")
|
||||
int negate_socket_local_storage(struct bpf_iter__tcp *ctx)
|
||||
{
|
||||
struct sock_common *sk_common = ctx->sk_common;
|
||||
int *sock_tgid;
|
||||
|
||||
if (!sk_common)
|
||||
return 0;
|
||||
|
||||
sock_tgid = bpf_sk_storage_get(&sk_stg_map, sk_common, 0, 0);
|
||||
if (!sock_tgid)
|
||||
return 0;
|
||||
|
||||
*sock_tgid = -*sock_tgid;
|
||||
|
||||
return 0;
|
||||
}
|
|
@ -11,9 +11,10 @@ int dump_task(struct bpf_iter__task *ctx)
|
|||
{
|
||||
struct seq_file *seq = ctx->meta->seq;
|
||||
struct task_struct *task = ctx->task;
|
||||
static char info[] = " === END ===";
|
||||
|
||||
if (task == (void *)0) {
|
||||
BPF_SEQ_PRINTF(seq, " === END ===\n");
|
||||
BPF_SEQ_PRINTF(seq, "%s\n", info);
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
|
|
@ -40,6 +40,7 @@ int BPF_PROG(test_core_module_probed,
|
|||
struct task_struct *task,
|
||||
struct bpf_testmod_test_read_ctx *read_ctx)
|
||||
{
|
||||
#if __has_builtin(__builtin_preserve_enum_value)
|
||||
struct core_reloc_module_output *out = (void *)&data.out;
|
||||
__u64 pid_tgid = bpf_get_current_pid_tgid();
|
||||
__u32 real_tgid = (__u32)(pid_tgid >> 32);
|
||||
|
@ -61,6 +62,9 @@ int BPF_PROG(test_core_module_probed,
|
|||
out->len_exists = bpf_core_field_exists(read_ctx->len);
|
||||
|
||||
out->comm_len = BPF_CORE_READ_STR_INTO(&out->comm, task, comm);
|
||||
#else
|
||||
data.skip = true;
|
||||
#endif
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
@ -70,6 +74,7 @@ int BPF_PROG(test_core_module_direct,
|
|||
struct task_struct *task,
|
||||
struct bpf_testmod_test_read_ctx *read_ctx)
|
||||
{
|
||||
#if __has_builtin(__builtin_preserve_enum_value)
|
||||
struct core_reloc_module_output *out = (void *)&data.out;
|
||||
__u64 pid_tgid = bpf_get_current_pid_tgid();
|
||||
__u32 real_tgid = (__u32)(pid_tgid >> 32);
|
||||
|
@ -91,6 +96,9 @@ int BPF_PROG(test_core_module_direct,
|
|||
out->len_exists = bpf_core_field_exists(read_ctx->len);
|
||||
|
||||
out->comm_len = BPF_CORE_READ_STR_INTO(&out->comm, task, comm);
|
||||
#else
|
||||
data.skip = true;
|
||||
#endif
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
|
|
@ -38,6 +38,17 @@ int BPF_PROG(handle_fentry,
|
|||
return 0;
|
||||
}
|
||||
|
||||
__u32 fentry_manual_read_sz = 0;
|
||||
|
||||
SEC("fentry/placeholder")
|
||||
int BPF_PROG(handle_fentry_manual,
|
||||
struct file *file, struct kobject *kobj,
|
||||
struct bin_attribute *bin_attr, char *buf, loff_t off, size_t len)
|
||||
{
|
||||
fentry_manual_read_sz = len;
|
||||
return 0;
|
||||
}
|
||||
|
||||
__u32 fexit_read_sz = 0;
|
||||
int fexit_ret = 0;
|
||||
|
||||
|
|
|
@ -587,6 +587,16 @@ static error_t parse_arg(int key, char *arg, struct argp_state *state)
|
|||
return -EINVAL;
|
||||
}
|
||||
}
|
||||
|
||||
if (env->verbosity > VERBOSE_NONE) {
|
||||
if (setenv("SELFTESTS_VERBOSE", "1", 1) == -1) {
|
||||
fprintf(stderr,
|
||||
"Unable to setenv SELFTESTS_VERBOSE=1 (errno=%d)",
|
||||
errno);
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
|
||||
break;
|
||||
case ARG_GET_TEST_CNT:
|
||||
env->get_test_cnt = true;
|
||||
|
|
|
@ -875,19 +875,36 @@ static int do_prog_test_run(int fd_prog, bool unpriv, uint32_t expected_val,
|
|||
__u8 tmp[TEST_DATA_LEN << 2];
|
||||
__u32 size_tmp = sizeof(tmp);
|
||||
uint32_t retval;
|
||||
int err;
|
||||
int err, saved_errno;
|
||||
|
||||
if (unpriv)
|
||||
set_admin(true);
|
||||
err = bpf_prog_test_run(fd_prog, 1, data, size_data,
|
||||
tmp, &size_tmp, &retval, NULL);
|
||||
saved_errno = errno;
|
||||
|
||||
if (unpriv)
|
||||
set_admin(false);
|
||||
if (err && errno != 524/*ENOTSUPP*/ && errno != EPERM) {
|
||||
printf("Unexpected bpf_prog_test_run error ");
|
||||
return err;
|
||||
|
||||
if (err) {
|
||||
switch (saved_errno) {
|
||||
case 524/*ENOTSUPP*/:
|
||||
printf("Did not run the program (not supported) ");
|
||||
return 0;
|
||||
case EPERM:
|
||||
if (unpriv) {
|
||||
printf("Did not run the program (no permission) ");
|
||||
return 0;
|
||||
}
|
||||
/* fallthrough; */
|
||||
default:
|
||||
printf("FAIL: Unexpected bpf_prog_test_run error (%s) ",
|
||||
strerror(saved_errno));
|
||||
return err;
|
||||
}
|
||||
}
|
||||
if (!err && retval != expected_val &&
|
||||
|
||||
if (retval != expected_val &&
|
||||
expected_val != POINTER_VALUE) {
|
||||
printf("FAIL retval %d != %d ", retval, expected_val);
|
||||
return 1;
|
||||
|
@ -936,6 +953,7 @@ static void do_test_single(struct bpf_test *test, bool unpriv,
|
|||
int run_errs, run_successes;
|
||||
int map_fds[MAX_NR_MAPS];
|
||||
const char *expected_err;
|
||||
int saved_errno;
|
||||
int fixup_skips;
|
||||
__u32 pflags;
|
||||
int i, err;
|
||||
|
@ -997,6 +1015,7 @@ static void do_test_single(struct bpf_test *test, bool unpriv,
|
|||
}
|
||||
|
||||
fd_prog = bpf_load_program_xattr(&attr, bpf_vlog, sizeof(bpf_vlog));
|
||||
saved_errno = errno;
|
||||
|
||||
/* BPF_PROG_TYPE_TRACING requires more setup and
|
||||
* bpf_probe_prog_type won't give correct answer
|
||||
|
@ -1013,7 +1032,7 @@ static void do_test_single(struct bpf_test *test, bool unpriv,
|
|||
if (expected_ret == ACCEPT || expected_ret == VERBOSE_ACCEPT) {
|
||||
if (fd_prog < 0) {
|
||||
printf("FAIL\nFailed to load prog '%s'!\n",
|
||||
strerror(errno));
|
||||
strerror(saved_errno));
|
||||
goto fail_log;
|
||||
}
|
||||
#ifndef CONFIG_HAVE_EFFICIENT_UNALIGNED_ACCESS
|
||||
|
|
259
tools/testing/selftests/bpf/test_xsk.sh
Executable file
259
tools/testing/selftests/bpf/test_xsk.sh
Executable file
|
@ -0,0 +1,259 @@
|
|||
#!/bin/bash
|
||||
# SPDX-License-Identifier: GPL-2.0
|
||||
# Copyright(c) 2020 Intel Corporation, Weqaar Janjua <weqaar.a.janjua@intel.com>
|
||||
|
||||
# AF_XDP selftests based on veth
|
||||
#
|
||||
# End-to-end AF_XDP over Veth test
|
||||
#
|
||||
# Topology:
|
||||
# ---------
|
||||
# -----------
|
||||
# _ | Process | _
|
||||
# / ----------- \
|
||||
# / | \
|
||||
# / | \
|
||||
# ----------- | -----------
|
||||
# | Thread1 | | | Thread2 |
|
||||
# ----------- | -----------
|
||||
# | | |
|
||||
# ----------- | -----------
|
||||
# | xskX | | | xskY |
|
||||
# ----------- | -----------
|
||||
# | | |
|
||||
# ----------- | ----------
|
||||
# | vethX | --------- | vethY |
|
||||
# ----------- peer ----------
|
||||
# | | |
|
||||
# namespaceX | namespaceY
|
||||
#
|
||||
# AF_XDP is an address family optimized for high performance packet processing,
|
||||
# it is XDP’s user-space interface.
|
||||
#
|
||||
# An AF_XDP socket is linked to a single UMEM which is a region of virtual
|
||||
# contiguous memory, divided into equal-sized frames.
|
||||
#
|
||||
# Refer to AF_XDP Kernel Documentation for detailed information:
|
||||
# https://www.kernel.org/doc/html/latest/networking/af_xdp.html
|
||||
#
|
||||
# Prerequisites setup by script:
|
||||
#
|
||||
# Set up veth interfaces as per the topology shown ^^:
|
||||
# * setup two veth interfaces and one namespace
|
||||
# ** veth<xxxx> in root namespace
|
||||
# ** veth<yyyy> in af_xdp<xxxx> namespace
|
||||
# ** namespace af_xdp<xxxx>
|
||||
# * create a spec file veth.spec that includes this run-time configuration
|
||||
# *** xxxx and yyyy are randomly generated 4 digit numbers used to avoid
|
||||
# conflict with any existing interface
|
||||
# * tests the veth and xsk layers of the topology
|
||||
#
|
||||
# See the source xdpxceiver.c for information on each test
|
||||
#
|
||||
# Kernel configuration:
|
||||
# ---------------------
|
||||
# See "config" file for recommended kernel config options.
|
||||
#
|
||||
# Turn on XDP sockets and veth support when compiling i.e.
|
||||
# Networking support -->
|
||||
# Networking options -->
|
||||
# [ * ] XDP sockets
|
||||
#
|
||||
# Executing Tests:
|
||||
# ----------------
|
||||
# Must run with CAP_NET_ADMIN capability.
|
||||
#
|
||||
# Run (full color-coded output):
|
||||
# sudo ./test_xsk.sh -c
|
||||
#
|
||||
# If running from kselftests:
|
||||
# sudo make colorconsole=1 run_tests
|
||||
#
|
||||
# Run (full output without color-coding):
|
||||
# sudo ./test_xsk.sh
|
||||
|
||||
. xsk_prereqs.sh
|
||||
|
||||
while getopts c flag
|
||||
do
|
||||
case "${flag}" in
|
||||
c) colorconsole=1;;
|
||||
esac
|
||||
done
|
||||
|
||||
TEST_NAME="PREREQUISITES"
|
||||
|
||||
URANDOM=/dev/urandom
|
||||
[ ! -e "${URANDOM}" ] && { echo "${URANDOM} not found. Skipping tests."; test_exit 1 1; }
|
||||
|
||||
VETH0_POSTFIX=$(cat ${URANDOM} | tr -dc '0-9' | fold -w 256 | head -n 1 | head --bytes 4)
|
||||
VETH0=ve${VETH0_POSTFIX}
|
||||
VETH1_POSTFIX=$(cat ${URANDOM} | tr -dc '0-9' | fold -w 256 | head -n 1 | head --bytes 4)
|
||||
VETH1=ve${VETH1_POSTFIX}
|
||||
NS0=root
|
||||
NS1=af_xdp${VETH1_POSTFIX}
|
||||
MTU=1500
|
||||
|
||||
setup_vethPairs() {
|
||||
echo "setting up ${VETH0}: namespace: ${NS0}"
|
||||
ip netns add ${NS1}
|
||||
ip link add ${VETH0} type veth peer name ${VETH1}
|
||||
if [ -f /proc/net/if_inet6 ]; then
|
||||
echo 1 > /proc/sys/net/ipv6/conf/${VETH0}/disable_ipv6
|
||||
fi
|
||||
echo "setting up ${VETH1}: namespace: ${NS1}"
|
||||
ip link set ${VETH1} netns ${NS1}
|
||||
ip netns exec ${NS1} ip link set ${VETH1} mtu ${MTU}
|
||||
ip link set ${VETH0} mtu ${MTU}
|
||||
ip netns exec ${NS1} ip link set ${VETH1} up
|
||||
ip link set ${VETH0} up
|
||||
}
|
||||
|
||||
validate_root_exec
|
||||
validate_veth_support ${VETH0}
|
||||
validate_ip_utility
|
||||
setup_vethPairs
|
||||
|
||||
retval=$?
|
||||
if [ $retval -ne 0 ]; then
|
||||
test_status $retval "${TEST_NAME}"
|
||||
cleanup_exit ${VETH0} ${VETH1} ${NS1}
|
||||
exit $retval
|
||||
fi
|
||||
|
||||
echo "${VETH0}:${VETH1},${NS1}" > ${SPECFILE}
|
||||
|
||||
validate_veth_spec_file
|
||||
|
||||
echo "Spec file created: ${SPECFILE}"
|
||||
|
||||
test_status $retval "${TEST_NAME}"
|
||||
|
||||
## START TESTS
|
||||
|
||||
statusList=()
|
||||
|
||||
### TEST 1
|
||||
TEST_NAME="XSK KSELFTEST FRAMEWORK"
|
||||
|
||||
echo "Switching interfaces [${VETH0}, ${VETH1}] to XDP Generic mode"
|
||||
vethXDPgeneric ${VETH0} ${VETH1} ${NS1}
|
||||
|
||||
retval=$?
|
||||
if [ $retval -eq 0 ]; then
|
||||
echo "Switching interfaces [${VETH0}, ${VETH1}] to XDP Native mode"
|
||||
vethXDPnative ${VETH0} ${VETH1} ${NS1}
|
||||
fi
|
||||
|
||||
retval=$?
|
||||
test_status $retval "${TEST_NAME}"
|
||||
statusList+=($retval)
|
||||
|
||||
### TEST 2
|
||||
TEST_NAME="SKB NOPOLL"
|
||||
|
||||
vethXDPgeneric ${VETH0} ${VETH1} ${NS1}
|
||||
|
||||
params=("-S")
|
||||
execxdpxceiver params
|
||||
|
||||
retval=$?
|
||||
test_status $retval "${TEST_NAME}"
|
||||
statusList+=($retval)
|
||||
|
||||
### TEST 3
|
||||
TEST_NAME="SKB POLL"
|
||||
|
||||
vethXDPgeneric ${VETH0} ${VETH1} ${NS1}
|
||||
|
||||
params=("-S" "-p")
|
||||
execxdpxceiver params
|
||||
|
||||
retval=$?
|
||||
test_status $retval "${TEST_NAME}"
|
||||
statusList+=($retval)
|
||||
|
||||
### TEST 4
|
||||
TEST_NAME="DRV NOPOLL"
|
||||
|
||||
vethXDPnative ${VETH0} ${VETH1} ${NS1}
|
||||
|
||||
params=("-N")
|
||||
execxdpxceiver params
|
||||
|
||||
retval=$?
|
||||
test_status $retval "${TEST_NAME}"
|
||||
statusList+=($retval)
|
||||
|
||||
### TEST 5
|
||||
TEST_NAME="DRV POLL"
|
||||
|
||||
vethXDPnative ${VETH0} ${VETH1} ${NS1}
|
||||
|
||||
params=("-N" "-p")
|
||||
execxdpxceiver params
|
||||
|
||||
retval=$?
|
||||
test_status $retval "${TEST_NAME}"
|
||||
statusList+=($retval)
|
||||
|
||||
### TEST 6
|
||||
TEST_NAME="SKB SOCKET TEARDOWN"
|
||||
|
||||
vethXDPgeneric ${VETH0} ${VETH1} ${NS1}
|
||||
|
||||
params=("-S" "-T")
|
||||
execxdpxceiver params
|
||||
|
||||
retval=$?
|
||||
test_status $retval "${TEST_NAME}"
|
||||
statusList+=($retval)
|
||||
|
||||
### TEST 7
|
||||
TEST_NAME="DRV SOCKET TEARDOWN"
|
||||
|
||||
vethXDPnative ${VETH0} ${VETH1} ${NS1}
|
||||
|
||||
params=("-N" "-T")
|
||||
execxdpxceiver params
|
||||
|
||||
retval=$?
|
||||
test_status $retval "${TEST_NAME}"
|
||||
statusList+=($retval)
|
||||
|
||||
### TEST 8
|
||||
TEST_NAME="SKB BIDIRECTIONAL SOCKETS"
|
||||
|
||||
vethXDPgeneric ${VETH0} ${VETH1} ${NS1}
|
||||
|
||||
params=("-S" "-B")
|
||||
execxdpxceiver params
|
||||
|
||||
retval=$?
|
||||
test_status $retval "${TEST_NAME}"
|
||||
statusList+=($retval)
|
||||
|
||||
### TEST 9
|
||||
TEST_NAME="DRV BIDIRECTIONAL SOCKETS"
|
||||
|
||||
vethXDPnative ${VETH0} ${VETH1} ${NS1}
|
||||
|
||||
params=("-N" "-B")
|
||||
execxdpxceiver params
|
||||
|
||||
retval=$?
|
||||
test_status $retval "${TEST_NAME}"
|
||||
statusList+=($retval)
|
||||
|
||||
## END TESTS
|
||||
|
||||
cleanup_exit ${VETH0} ${VETH1} ${NS1}
|
||||
|
||||
for _status in "${statusList[@]}"
|
||||
do
|
||||
if [ $_status -ne 0 ]; then
|
||||
test_exit $ksft_fail 0
|
||||
fi
|
||||
done
|
||||
|
||||
test_exit $ksft_pass 0
|
|
@ -108,8 +108,9 @@
|
|||
BPF_EXIT_INSN(),
|
||||
},
|
||||
.fixup_map_hash_8b = { 3 },
|
||||
.errstr = "invalid indirect read from stack off -8+0 size 8",
|
||||
.result = REJECT,
|
||||
.errstr_unpriv = "invalid indirect read from stack off -8+0 size 8",
|
||||
.result_unpriv = REJECT,
|
||||
.result = ACCEPT,
|
||||
},
|
||||
{
|
||||
"unpriv: mangle pointer on stack 1",
|
||||
|
|
1074
tools/testing/selftests/bpf/xdpxceiver.c
Normal file
1074
tools/testing/selftests/bpf/xdpxceiver.c
Normal file
File diff suppressed because it is too large
Load diff
160
tools/testing/selftests/bpf/xdpxceiver.h
Normal file
160
tools/testing/selftests/bpf/xdpxceiver.h
Normal file
|
@ -0,0 +1,160 @@
|
|||
/* SPDX-License-Identifier: GPL-2.0
|
||||
* Copyright(c) 2020 Intel Corporation.
|
||||
*/
|
||||
|
||||
#ifndef XDPXCEIVER_H_
|
||||
#define XDPXCEIVER_H_
|
||||
|
||||
#ifndef SOL_XDP
|
||||
#define SOL_XDP 283
|
||||
#endif
|
||||
|
||||
#ifndef AF_XDP
|
||||
#define AF_XDP 44
|
||||
#endif
|
||||
|
||||
#ifndef PF_XDP
|
||||
#define PF_XDP AF_XDP
|
||||
#endif
|
||||
|
||||
#define MAX_INTERFACES 2
|
||||
#define MAX_INTERFACE_NAME_CHARS 7
|
||||
#define MAX_INTERFACES_NAMESPACE_CHARS 10
|
||||
#define MAX_SOCKS 1
|
||||
#define MAX_TEARDOWN_ITER 10
|
||||
#define MAX_BIDI_ITER 2
|
||||
#define PKT_HDR_SIZE (sizeof(struct ethhdr) + sizeof(struct iphdr) + \
|
||||
sizeof(struct udphdr))
|
||||
#define MIN_PKT_SIZE 64
|
||||
#define ETH_FCS_SIZE 4
|
||||
#define PKT_SIZE (MIN_PKT_SIZE - ETH_FCS_SIZE)
|
||||
#define IP_PKT_SIZE (PKT_SIZE - sizeof(struct ethhdr))
|
||||
#define IP_PKT_VER 0x4
|
||||
#define IP_PKT_TOS 0x9
|
||||
#define UDP_PKT_SIZE (IP_PKT_SIZE - sizeof(struct iphdr))
|
||||
#define UDP_PKT_DATA_SIZE (UDP_PKT_SIZE - sizeof(struct udphdr))
|
||||
#define TMOUT_SEC (3)
|
||||
#define EOT (-1)
|
||||
#define USLEEP_MAX 200000
|
||||
#define THREAD_STACK 60000000
|
||||
#define SOCK_RECONF_CTR 10
|
||||
#define BATCH_SIZE 64
|
||||
#define POLL_TMOUT 1000
|
||||
#define NEED_WAKEUP true
|
||||
|
||||
typedef __u32 u32;
|
||||
typedef __u16 u16;
|
||||
typedef __u8 u8;
|
||||
|
||||
enum TESTS {
|
||||
ORDER_CONTENT_VALIDATE_XDP_SKB = 0,
|
||||
ORDER_CONTENT_VALIDATE_XDP_DRV = 1,
|
||||
};
|
||||
|
||||
u8 uut;
|
||||
u8 debug_pkt_dump;
|
||||
u32 num_frames;
|
||||
u8 switching_notify;
|
||||
u8 bidi_pass;
|
||||
|
||||
static u32 opt_xdp_flags = XDP_FLAGS_UPDATE_IF_NOEXIST;
|
||||
static int opt_queue;
|
||||
static int opt_pkt_count;
|
||||
static int opt_poll;
|
||||
static int opt_teardown;
|
||||
static int opt_bidi;
|
||||
static u32 opt_xdp_bind_flags = XDP_USE_NEED_WAKEUP;
|
||||
static u8 pkt_data[XSK_UMEM__DEFAULT_FRAME_SIZE];
|
||||
static u32 pkt_counter;
|
||||
static u32 prev_pkt = -1;
|
||||
static int sigvar;
|
||||
|
||||
struct xsk_umem_info {
|
||||
struct xsk_ring_prod fq;
|
||||
struct xsk_ring_cons cq;
|
||||
struct xsk_umem *umem;
|
||||
void *buffer;
|
||||
};
|
||||
|
||||
struct xsk_socket_info {
|
||||
struct xsk_ring_cons rx;
|
||||
struct xsk_ring_prod tx;
|
||||
struct xsk_umem_info *umem;
|
||||
struct xsk_socket *xsk;
|
||||
unsigned long rx_npkts;
|
||||
unsigned long tx_npkts;
|
||||
unsigned long prev_rx_npkts;
|
||||
unsigned long prev_tx_npkts;
|
||||
u32 outstanding_tx;
|
||||
};
|
||||
|
||||
struct flow_vector {
|
||||
enum fvector {
|
||||
tx,
|
||||
rx,
|
||||
bidi,
|
||||
undef,
|
||||
} vector;
|
||||
};
|
||||
|
||||
struct generic_data {
|
||||
u32 seqnum;
|
||||
};
|
||||
|
||||
struct ifaceconfigobj {
|
||||
u8 dst_mac[ETH_ALEN];
|
||||
u8 src_mac[ETH_ALEN];
|
||||
struct in_addr dst_ip;
|
||||
struct in_addr src_ip;
|
||||
u16 src_port;
|
||||
u16 dst_port;
|
||||
} *ifaceconfig;
|
||||
|
||||
struct ifobject {
|
||||
int ifindex;
|
||||
int ifdict_index;
|
||||
char ifname[MAX_INTERFACE_NAME_CHARS];
|
||||
char nsname[MAX_INTERFACES_NAMESPACE_CHARS];
|
||||
struct flow_vector fv;
|
||||
struct xsk_socket_info *xsk;
|
||||
struct xsk_umem_info *umem;
|
||||
u8 dst_mac[ETH_ALEN];
|
||||
u8 src_mac[ETH_ALEN];
|
||||
u32 dst_ip;
|
||||
u32 src_ip;
|
||||
u16 src_port;
|
||||
u16 dst_port;
|
||||
};
|
||||
|
||||
static struct ifobject *ifdict[MAX_INTERFACES];
|
||||
|
||||
/*threads*/
|
||||
atomic_int spinning_tx;
|
||||
atomic_int spinning_rx;
|
||||
pthread_mutex_t sync_mutex;
|
||||
pthread_mutex_t sync_mutex_tx;
|
||||
pthread_cond_t signal_rx_condition;
|
||||
pthread_cond_t signal_tx_condition;
|
||||
pthread_t t0, t1, ns_thread;
|
||||
pthread_attr_t attr;
|
||||
|
||||
struct targs {
|
||||
bool retptr;
|
||||
int idx;
|
||||
};
|
||||
|
||||
TAILQ_HEAD(head_s, pkt) head = TAILQ_HEAD_INITIALIZER(head);
|
||||
struct head_s *head_p;
|
||||
struct pkt {
|
||||
char *pkt_frame;
|
||||
|
||||
TAILQ_ENTRY(pkt) pkt_nodes;
|
||||
} *pkt_node_rx, *pkt_node_rx_q;
|
||||
|
||||
struct pkt_frame {
|
||||
char *payload;
|
||||
} *pkt_obj;
|
||||
|
||||
struct pkt_frame **pkt_buf;
|
||||
|
||||
#endif /* XDPXCEIVER_H */
|
135
tools/testing/selftests/bpf/xsk_prereqs.sh
Executable file
135
tools/testing/selftests/bpf/xsk_prereqs.sh
Executable file
|
@ -0,0 +1,135 @@
|
|||
#!/bin/bash
|
||||
# SPDX-License-Identifier: GPL-2.0
|
||||
# Copyright(c) 2020 Intel Corporation.
|
||||
|
||||
ksft_pass=0
|
||||
ksft_fail=1
|
||||
ksft_xfail=2
|
||||
ksft_xpass=3
|
||||
ksft_skip=4
|
||||
|
||||
GREEN='\033[0;92m'
|
||||
YELLOW='\033[0;93m'
|
||||
RED='\033[0;31m'
|
||||
NC='\033[0m'
|
||||
STACK_LIM=131072
|
||||
SPECFILE=veth.spec
|
||||
XSKOBJ=xdpxceiver
|
||||
NUMPKTS=10000
|
||||
|
||||
validate_root_exec()
|
||||
{
|
||||
msg="skip all tests:"
|
||||
if [ $UID != 0 ]; then
|
||||
echo $msg must be run as root >&2
|
||||
test_exit $ksft_fail 2
|
||||
else
|
||||
return $ksft_pass
|
||||
fi
|
||||
}
|
||||
|
||||
validate_veth_support()
|
||||
{
|
||||
msg="skip all tests:"
|
||||
if [ $(ip link add $1 type veth 2>/dev/null; echo $?;) != 0 ]; then
|
||||
echo $msg veth kernel support not available >&2
|
||||
test_exit $ksft_skip 1
|
||||
else
|
||||
ip link del $1
|
||||
return $ksft_pass
|
||||
fi
|
||||
}
|
||||
|
||||
validate_veth_spec_file()
|
||||
{
|
||||
if [ ! -f ${SPECFILE} ]; then
|
||||
test_exit $ksft_skip 1
|
||||
fi
|
||||
}
|
||||
|
||||
test_status()
|
||||
{
|
||||
statusval=$1
|
||||
if [ -n "${colorconsole+set}" ]; then
|
||||
if [ $statusval -eq 2 ]; then
|
||||
echo -e "${YELLOW}$2${NC}: [ ${RED}FAIL${NC} ]"
|
||||
elif [ $statusval -eq 1 ]; then
|
||||
echo -e "${YELLOW}$2${NC}: [ ${RED}SKIPPED${NC} ]"
|
||||
elif [ $statusval -eq 0 ]; then
|
||||
echo -e "${YELLOW}$2${NC}: [ ${GREEN}PASS${NC} ]"
|
||||
fi
|
||||
else
|
||||
if [ $statusval -eq 2 ]; then
|
||||
echo -e "$2: [ FAIL ]"
|
||||
elif [ $statusval -eq 1 ]; then
|
||||
echo -e "$2: [ SKIPPED ]"
|
||||
elif [ $statusval -eq 0 ]; then
|
||||
echo -e "$2: [ PASS ]"
|
||||
fi
|
||||
fi
|
||||
}
|
||||
|
||||
test_exit()
|
||||
{
|
||||
retval=$1
|
||||
if [ $2 -ne 0 ]; then
|
||||
test_status $2 $(basename $0)
|
||||
fi
|
||||
exit $retval
|
||||
}
|
||||
|
||||
clear_configs()
|
||||
{
|
||||
if [ $(ip netns show | grep $3 &>/dev/null; echo $?;) == 0 ]; then
|
||||
[ $(ip netns exec $3 ip link show $2 &>/dev/null; echo $?;) == 0 ] &&
|
||||
{ echo "removing link $1:$2"; ip netns exec $3 ip link del $2; }
|
||||
echo "removing ns $3"
|
||||
ip netns del $3
|
||||
fi
|
||||
#Once we delete a veth pair node, the entire veth pair is removed,
|
||||
#this is just to be cautious just incase the NS does not exist then
|
||||
#veth node inside NS won't get removed so we explicitly remove it
|
||||
[ $(ip link show $1 &>/dev/null; echo $?;) == 0 ] &&
|
||||
{ echo "removing link $1"; ip link del $1; }
|
||||
if [ -f ${SPECFILE} ]; then
|
||||
echo "removing spec file:" ${SPECFILE}
|
||||
rm -f ${SPECFILE}
|
||||
fi
|
||||
}
|
||||
|
||||
cleanup_exit()
|
||||
{
|
||||
echo "cleaning up..."
|
||||
clear_configs $1 $2 $3
|
||||
}
|
||||
|
||||
validate_ip_utility()
|
||||
{
|
||||
[ ! $(type -P ip) ] && { echo "'ip' not found. Skipping tests."; test_exit $ksft_skip 1; }
|
||||
}
|
||||
|
||||
vethXDPgeneric()
|
||||
{
|
||||
ip link set dev $1 xdpdrv off
|
||||
ip netns exec $3 ip link set dev $2 xdpdrv off
|
||||
}
|
||||
|
||||
vethXDPnative()
|
||||
{
|
||||
ip link set dev $1 xdpgeneric off
|
||||
ip netns exec $3 ip link set dev $2 xdpgeneric off
|
||||
}
|
||||
|
||||
execxdpxceiver()
|
||||
{
|
||||
local -a 'paramkeys=("${!'"$1"'[@]}")' copy
|
||||
paramkeysstr=${paramkeys[*]}
|
||||
|
||||
for index in $paramkeysstr;
|
||||
do
|
||||
current=$1"[$index]"
|
||||
copy[$index]=${!current}
|
||||
done
|
||||
|
||||
./${XSKOBJ} -i ${VETH0} -i ${VETH1},${NS1} ${copy[*]} -C ${NUMPKTS}
|
||||
}
|
Loading…
Add table
Reference in a new issue