soreuseport: setsockopt SO_ATTACH_REUSEPORT_[CE]BPF
Expose socket options for setting a classic or extended BPF program for use when selecting sockets in an SO_REUSEPORT group. These options can be used on the first socket to belong to a group before bind or on any socket in the group after bind. This change includes refactoring of the existing sk_filter code to allow reuse of the existing BPF filter validation checks. Signed-off-by: Craig Gallek <kraig@google.com> Acked-by: Alexei Starovoitov <ast@kernel.org> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
Родитель
e32ea7e747
Коммит
538950a1b7
|
@ -92,4 +92,7 @@
|
||||||
#define SO_ATTACH_BPF 50
|
#define SO_ATTACH_BPF 50
|
||||||
#define SO_DETACH_BPF SO_DETACH_FILTER
|
#define SO_DETACH_BPF SO_DETACH_FILTER
|
||||||
|
|
||||||
|
#define SO_ATTACH_REUSEPORT_CBPF 51
|
||||||
|
#define SO_ATTACH_REUSEPORT_EBPF 52
|
||||||
|
|
||||||
#endif /* _UAPI_ASM_SOCKET_H */
|
#endif /* _UAPI_ASM_SOCKET_H */
|
||||||
|
|
|
@ -85,4 +85,7 @@
|
||||||
#define SO_ATTACH_BPF 50
|
#define SO_ATTACH_BPF 50
|
||||||
#define SO_DETACH_BPF SO_DETACH_FILTER
|
#define SO_DETACH_BPF SO_DETACH_FILTER
|
||||||
|
|
||||||
|
#define SO_ATTACH_REUSEPORT_CBPF 51
|
||||||
|
#define SO_ATTACH_REUSEPORT_EBPF 52
|
||||||
|
|
||||||
#endif /* _UAPI__ASM_AVR32_SOCKET_H */
|
#endif /* _UAPI__ASM_AVR32_SOCKET_H */
|
||||||
|
|
|
@ -85,5 +85,8 @@
|
||||||
#define SO_ATTACH_BPF 50
|
#define SO_ATTACH_BPF 50
|
||||||
#define SO_DETACH_BPF SO_DETACH_FILTER
|
#define SO_DETACH_BPF SO_DETACH_FILTER
|
||||||
|
|
||||||
|
#define SO_ATTACH_REUSEPORT_CBPF 51
|
||||||
|
#define SO_ATTACH_REUSEPORT_EBPF 52
|
||||||
|
|
||||||
#endif /* _ASM_SOCKET_H */
|
#endif /* _ASM_SOCKET_H */
|
||||||
|
|
||||||
|
|
|
@ -94,4 +94,7 @@
|
||||||
#define SO_ATTACH_BPF 50
|
#define SO_ATTACH_BPF 50
|
||||||
#define SO_DETACH_BPF SO_DETACH_FILTER
|
#define SO_DETACH_BPF SO_DETACH_FILTER
|
||||||
|
|
||||||
|
#define SO_ATTACH_REUSEPORT_CBPF 51
|
||||||
|
#define SO_ATTACH_REUSEPORT_EBPF 52
|
||||||
|
|
||||||
#endif /* _ASM_IA64_SOCKET_H */
|
#endif /* _ASM_IA64_SOCKET_H */
|
||||||
|
|
|
@ -85,4 +85,7 @@
|
||||||
#define SO_ATTACH_BPF 50
|
#define SO_ATTACH_BPF 50
|
||||||
#define SO_DETACH_BPF SO_DETACH_FILTER
|
#define SO_DETACH_BPF SO_DETACH_FILTER
|
||||||
|
|
||||||
|
#define SO_ATTACH_REUSEPORT_CBPF 51
|
||||||
|
#define SO_ATTACH_REUSEPORT_EBPF 52
|
||||||
|
|
||||||
#endif /* _ASM_M32R_SOCKET_H */
|
#endif /* _ASM_M32R_SOCKET_H */
|
||||||
|
|
|
@ -103,4 +103,7 @@
|
||||||
#define SO_ATTACH_BPF 50
|
#define SO_ATTACH_BPF 50
|
||||||
#define SO_DETACH_BPF SO_DETACH_FILTER
|
#define SO_DETACH_BPF SO_DETACH_FILTER
|
||||||
|
|
||||||
|
#define SO_ATTACH_REUSEPORT_CBPF 51
|
||||||
|
#define SO_ATTACH_REUSEPORT_EBPF 52
|
||||||
|
|
||||||
#endif /* _UAPI_ASM_SOCKET_H */
|
#endif /* _UAPI_ASM_SOCKET_H */
|
||||||
|
|
|
@ -85,4 +85,7 @@
|
||||||
#define SO_ATTACH_BPF 50
|
#define SO_ATTACH_BPF 50
|
||||||
#define SO_DETACH_BPF SO_DETACH_FILTER
|
#define SO_DETACH_BPF SO_DETACH_FILTER
|
||||||
|
|
||||||
|
#define SO_ATTACH_REUSEPORT_CBPF 51
|
||||||
|
#define SO_ATTACH_REUSEPORT_EBPF 52
|
||||||
|
|
||||||
#endif /* _ASM_SOCKET_H */
|
#endif /* _ASM_SOCKET_H */
|
||||||
|
|
|
@ -84,4 +84,7 @@
|
||||||
#define SO_ATTACH_BPF 0x402B
|
#define SO_ATTACH_BPF 0x402B
|
||||||
#define SO_DETACH_BPF SO_DETACH_FILTER
|
#define SO_DETACH_BPF SO_DETACH_FILTER
|
||||||
|
|
||||||
|
#define SO_ATTACH_REUSEPORT_CBPF 0x402C
|
||||||
|
#define SO_ATTACH_REUSEPORT_EBPF 0x402D
|
||||||
|
|
||||||
#endif /* _UAPI_ASM_SOCKET_H */
|
#endif /* _UAPI_ASM_SOCKET_H */
|
||||||
|
|
|
@ -92,4 +92,7 @@
|
||||||
#define SO_ATTACH_BPF 50
|
#define SO_ATTACH_BPF 50
|
||||||
#define SO_DETACH_BPF SO_DETACH_FILTER
|
#define SO_DETACH_BPF SO_DETACH_FILTER
|
||||||
|
|
||||||
|
#define SO_ATTACH_REUSEPORT_CBPF 51
|
||||||
|
#define SO_ATTACH_REUSEPORT_EBPF 52
|
||||||
|
|
||||||
#endif /* _ASM_POWERPC_SOCKET_H */
|
#endif /* _ASM_POWERPC_SOCKET_H */
|
||||||
|
|
|
@ -91,4 +91,7 @@
|
||||||
#define SO_ATTACH_BPF 50
|
#define SO_ATTACH_BPF 50
|
||||||
#define SO_DETACH_BPF SO_DETACH_FILTER
|
#define SO_DETACH_BPF SO_DETACH_FILTER
|
||||||
|
|
||||||
|
#define SO_ATTACH_REUSEPORT_CBPF 51
|
||||||
|
#define SO_ATTACH_REUSEPORT_EBPF 52
|
||||||
|
|
||||||
#endif /* _ASM_SOCKET_H */
|
#endif /* _ASM_SOCKET_H */
|
||||||
|
|
|
@ -81,6 +81,9 @@
|
||||||
#define SO_ATTACH_BPF 0x0034
|
#define SO_ATTACH_BPF 0x0034
|
||||||
#define SO_DETACH_BPF SO_DETACH_FILTER
|
#define SO_DETACH_BPF SO_DETACH_FILTER
|
||||||
|
|
||||||
|
#define SO_ATTACH_REUSEPORT_CBPF 0x0035
|
||||||
|
#define SO_ATTACH_REUSEPORT_EBPF 0x0036
|
||||||
|
|
||||||
/* Security levels - as per NRL IPv6 - don't actually do anything */
|
/* Security levels - as per NRL IPv6 - don't actually do anything */
|
||||||
#define SO_SECURITY_AUTHENTICATION 0x5001
|
#define SO_SECURITY_AUTHENTICATION 0x5001
|
||||||
#define SO_SECURITY_ENCRYPTION_TRANSPORT 0x5002
|
#define SO_SECURITY_ENCRYPTION_TRANSPORT 0x5002
|
||||||
|
|
|
@ -96,4 +96,7 @@
|
||||||
#define SO_ATTACH_BPF 50
|
#define SO_ATTACH_BPF 50
|
||||||
#define SO_DETACH_BPF SO_DETACH_FILTER
|
#define SO_DETACH_BPF SO_DETACH_FILTER
|
||||||
|
|
||||||
|
#define SO_ATTACH_REUSEPORT_CBPF 51
|
||||||
|
#define SO_ATTACH_REUSEPORT_EBPF 52
|
||||||
|
|
||||||
#endif /* _XTENSA_SOCKET_H */
|
#endif /* _XTENSA_SOCKET_H */
|
||||||
|
|
|
@ -447,6 +447,8 @@ void bpf_prog_destroy(struct bpf_prog *fp);
|
||||||
|
|
||||||
int sk_attach_filter(struct sock_fprog *fprog, struct sock *sk);
|
int sk_attach_filter(struct sock_fprog *fprog, struct sock *sk);
|
||||||
int sk_attach_bpf(u32 ufd, struct sock *sk);
|
int sk_attach_bpf(u32 ufd, struct sock *sk);
|
||||||
|
int sk_reuseport_attach_filter(struct sock_fprog *fprog, struct sock *sk);
|
||||||
|
int sk_reuseport_attach_bpf(u32 ufd, struct sock *sk);
|
||||||
int sk_detach_filter(struct sock *sk);
|
int sk_detach_filter(struct sock *sk);
|
||||||
int sk_get_filter(struct sock *sk, struct sock_filter __user *filter,
|
int sk_get_filter(struct sock *sk, struct sock_filter __user *filter,
|
||||||
unsigned int len);
|
unsigned int len);
|
||||||
|
|
|
@ -1,6 +1,8 @@
|
||||||
#ifndef _SOCK_REUSEPORT_H
|
#ifndef _SOCK_REUSEPORT_H
|
||||||
#define _SOCK_REUSEPORT_H
|
#define _SOCK_REUSEPORT_H
|
||||||
|
|
||||||
|
#include <linux/filter.h>
|
||||||
|
#include <linux/skbuff.h>
|
||||||
#include <linux/types.h>
|
#include <linux/types.h>
|
||||||
#include <net/sock.h>
|
#include <net/sock.h>
|
||||||
|
|
||||||
|
@ -9,12 +11,18 @@ struct sock_reuseport {
|
||||||
|
|
||||||
u16 max_socks; /* length of socks */
|
u16 max_socks; /* length of socks */
|
||||||
u16 num_socks; /* elements in socks */
|
u16 num_socks; /* elements in socks */
|
||||||
|
struct bpf_prog __rcu *prog; /* optional BPF sock selector */
|
||||||
struct sock *socks[0]; /* array of sock pointers */
|
struct sock *socks[0]; /* array of sock pointers */
|
||||||
};
|
};
|
||||||
|
|
||||||
extern int reuseport_alloc(struct sock *sk);
|
extern int reuseport_alloc(struct sock *sk);
|
||||||
extern int reuseport_add_sock(struct sock *sk, const struct sock *sk2);
|
extern int reuseport_add_sock(struct sock *sk, const struct sock *sk2);
|
||||||
extern void reuseport_detach_sock(struct sock *sk);
|
extern void reuseport_detach_sock(struct sock *sk);
|
||||||
extern struct sock *reuseport_select_sock(struct sock *sk, u32 hash);
|
extern struct sock *reuseport_select_sock(struct sock *sk,
|
||||||
|
u32 hash,
|
||||||
|
struct sk_buff *skb,
|
||||||
|
int hdr_len);
|
||||||
|
extern struct bpf_prog *reuseport_attach_prog(struct sock *sk,
|
||||||
|
struct bpf_prog *prog);
|
||||||
|
|
||||||
#endif /* _SOCK_REUSEPORT_H */
|
#endif /* _SOCK_REUSEPORT_H */
|
||||||
|
|
|
@ -258,7 +258,7 @@ struct sock *udp4_lib_lookup(struct net *net, __be32 saddr, __be16 sport,
|
||||||
__be32 daddr, __be16 dport, int dif);
|
__be32 daddr, __be16 dport, int dif);
|
||||||
struct sock *__udp4_lib_lookup(struct net *net, __be32 saddr, __be16 sport,
|
struct sock *__udp4_lib_lookup(struct net *net, __be32 saddr, __be16 sport,
|
||||||
__be32 daddr, __be16 dport, int dif,
|
__be32 daddr, __be16 dport, int dif,
|
||||||
struct udp_table *tbl);
|
struct udp_table *tbl, struct sk_buff *skb);
|
||||||
struct sock *udp6_lib_lookup(struct net *net,
|
struct sock *udp6_lib_lookup(struct net *net,
|
||||||
const struct in6_addr *saddr, __be16 sport,
|
const struct in6_addr *saddr, __be16 sport,
|
||||||
const struct in6_addr *daddr, __be16 dport,
|
const struct in6_addr *daddr, __be16 dport,
|
||||||
|
@ -266,7 +266,8 @@ struct sock *udp6_lib_lookup(struct net *net,
|
||||||
struct sock *__udp6_lib_lookup(struct net *net,
|
struct sock *__udp6_lib_lookup(struct net *net,
|
||||||
const struct in6_addr *saddr, __be16 sport,
|
const struct in6_addr *saddr, __be16 sport,
|
||||||
const struct in6_addr *daddr, __be16 dport,
|
const struct in6_addr *daddr, __be16 dport,
|
||||||
int dif, struct udp_table *tbl);
|
int dif, struct udp_table *tbl,
|
||||||
|
struct sk_buff *skb);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* SNMP statistics for UDP and UDP-Lite
|
* SNMP statistics for UDP and UDP-Lite
|
||||||
|
|
|
@ -87,4 +87,7 @@
|
||||||
#define SO_ATTACH_BPF 50
|
#define SO_ATTACH_BPF 50
|
||||||
#define SO_DETACH_BPF SO_DETACH_FILTER
|
#define SO_DETACH_BPF SO_DETACH_FILTER
|
||||||
|
|
||||||
|
#define SO_ATTACH_REUSEPORT_CBPF 51
|
||||||
|
#define SO_ATTACH_REUSEPORT_EBPF 52
|
||||||
|
|
||||||
#endif /* __ASM_GENERIC_SOCKET_H */
|
#endif /* __ASM_GENERIC_SOCKET_H */
|
||||||
|
|
|
@ -50,6 +50,7 @@
|
||||||
#include <net/cls_cgroup.h>
|
#include <net/cls_cgroup.h>
|
||||||
#include <net/dst_metadata.h>
|
#include <net/dst_metadata.h>
|
||||||
#include <net/dst.h>
|
#include <net/dst.h>
|
||||||
|
#include <net/sock_reuseport.h>
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* sk_filter - run a packet through a socket filter
|
* sk_filter - run a packet through a socket filter
|
||||||
|
@ -1167,6 +1168,68 @@ static int __sk_attach_prog(struct bpf_prog *prog, struct sock *sk)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int __reuseport_attach_prog(struct bpf_prog *prog, struct sock *sk)
|
||||||
|
{
|
||||||
|
struct bpf_prog *old_prog;
|
||||||
|
int err;
|
||||||
|
|
||||||
|
if (bpf_prog_size(prog->len) > sysctl_optmem_max)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
if (sk_unhashed(sk)) {
|
||||||
|
err = reuseport_alloc(sk);
|
||||||
|
if (err)
|
||||||
|
return err;
|
||||||
|
} else if (!rcu_access_pointer(sk->sk_reuseport_cb)) {
|
||||||
|
/* The socket wasn't bound with SO_REUSEPORT */
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
old_prog = reuseport_attach_prog(sk, prog);
|
||||||
|
if (old_prog)
|
||||||
|
bpf_prog_destroy(old_prog);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static
|
||||||
|
struct bpf_prog *__get_filter(struct sock_fprog *fprog, struct sock *sk)
|
||||||
|
{
|
||||||
|
unsigned int fsize = bpf_classic_proglen(fprog);
|
||||||
|
unsigned int bpf_fsize = bpf_prog_size(fprog->len);
|
||||||
|
struct bpf_prog *prog;
|
||||||
|
int err;
|
||||||
|
|
||||||
|
if (sock_flag(sk, SOCK_FILTER_LOCKED))
|
||||||
|
return ERR_PTR(-EPERM);
|
||||||
|
|
||||||
|
/* Make sure new filter is there and in the right amounts. */
|
||||||
|
if (fprog->filter == NULL)
|
||||||
|
return ERR_PTR(-EINVAL);
|
||||||
|
|
||||||
|
prog = bpf_prog_alloc(bpf_fsize, 0);
|
||||||
|
if (!prog)
|
||||||
|
return ERR_PTR(-ENOMEM);
|
||||||
|
|
||||||
|
if (copy_from_user(prog->insns, fprog->filter, fsize)) {
|
||||||
|
__bpf_prog_free(prog);
|
||||||
|
return ERR_PTR(-EFAULT);
|
||||||
|
}
|
||||||
|
|
||||||
|
prog->len = fprog->len;
|
||||||
|
|
||||||
|
err = bpf_prog_store_orig_filter(prog, fprog);
|
||||||
|
if (err) {
|
||||||
|
__bpf_prog_free(prog);
|
||||||
|
return ERR_PTR(-ENOMEM);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* bpf_prepare_filter() already takes care of freeing
|
||||||
|
* memory in case something goes wrong.
|
||||||
|
*/
|
||||||
|
return bpf_prepare_filter(prog, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* sk_attach_filter - attach a socket filter
|
* sk_attach_filter - attach a socket filter
|
||||||
* @fprog: the filter program
|
* @fprog: the filter program
|
||||||
|
@ -1179,39 +1242,9 @@ static int __sk_attach_prog(struct bpf_prog *prog, struct sock *sk)
|
||||||
*/
|
*/
|
||||||
int sk_attach_filter(struct sock_fprog *fprog, struct sock *sk)
|
int sk_attach_filter(struct sock_fprog *fprog, struct sock *sk)
|
||||||
{
|
{
|
||||||
unsigned int fsize = bpf_classic_proglen(fprog);
|
struct bpf_prog *prog = __get_filter(fprog, sk);
|
||||||
unsigned int bpf_fsize = bpf_prog_size(fprog->len);
|
|
||||||
struct bpf_prog *prog;
|
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
if (sock_flag(sk, SOCK_FILTER_LOCKED))
|
|
||||||
return -EPERM;
|
|
||||||
|
|
||||||
/* Make sure new filter is there and in the right amounts. */
|
|
||||||
if (fprog->filter == NULL)
|
|
||||||
return -EINVAL;
|
|
||||||
|
|
||||||
prog = bpf_prog_alloc(bpf_fsize, 0);
|
|
||||||
if (!prog)
|
|
||||||
return -ENOMEM;
|
|
||||||
|
|
||||||
if (copy_from_user(prog->insns, fprog->filter, fsize)) {
|
|
||||||
__bpf_prog_free(prog);
|
|
||||||
return -EFAULT;
|
|
||||||
}
|
|
||||||
|
|
||||||
prog->len = fprog->len;
|
|
||||||
|
|
||||||
err = bpf_prog_store_orig_filter(prog, fprog);
|
|
||||||
if (err) {
|
|
||||||
__bpf_prog_free(prog);
|
|
||||||
return -ENOMEM;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* bpf_prepare_filter() already takes care of freeing
|
|
||||||
* memory in case something goes wrong.
|
|
||||||
*/
|
|
||||||
prog = bpf_prepare_filter(prog, NULL);
|
|
||||||
if (IS_ERR(prog))
|
if (IS_ERR(prog))
|
||||||
return PTR_ERR(prog);
|
return PTR_ERR(prog);
|
||||||
|
|
||||||
|
@ -1225,23 +1258,50 @@ int sk_attach_filter(struct sock_fprog *fprog, struct sock *sk)
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(sk_attach_filter);
|
EXPORT_SYMBOL_GPL(sk_attach_filter);
|
||||||
|
|
||||||
int sk_attach_bpf(u32 ufd, struct sock *sk)
|
int sk_reuseport_attach_filter(struct sock_fprog *fprog, struct sock *sk)
|
||||||
{
|
{
|
||||||
struct bpf_prog *prog;
|
struct bpf_prog *prog = __get_filter(fprog, sk);
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
if (sock_flag(sk, SOCK_FILTER_LOCKED))
|
|
||||||
return -EPERM;
|
|
||||||
|
|
||||||
prog = bpf_prog_get(ufd);
|
|
||||||
if (IS_ERR(prog))
|
if (IS_ERR(prog))
|
||||||
return PTR_ERR(prog);
|
return PTR_ERR(prog);
|
||||||
|
|
||||||
if (prog->type != BPF_PROG_TYPE_SOCKET_FILTER) {
|
err = __reuseport_attach_prog(prog, sk);
|
||||||
bpf_prog_put(prog);
|
if (err < 0) {
|
||||||
return -EINVAL;
|
__bpf_prog_release(prog);
|
||||||
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct bpf_prog *__get_bpf(u32 ufd, struct sock *sk)
|
||||||
|
{
|
||||||
|
struct bpf_prog *prog;
|
||||||
|
|
||||||
|
if (sock_flag(sk, SOCK_FILTER_LOCKED))
|
||||||
|
return ERR_PTR(-EPERM);
|
||||||
|
|
||||||
|
prog = bpf_prog_get(ufd);
|
||||||
|
if (IS_ERR(prog))
|
||||||
|
return prog;
|
||||||
|
|
||||||
|
if (prog->type != BPF_PROG_TYPE_SOCKET_FILTER) {
|
||||||
|
bpf_prog_put(prog);
|
||||||
|
return ERR_PTR(-EINVAL);
|
||||||
|
}
|
||||||
|
|
||||||
|
return prog;
|
||||||
|
}
|
||||||
|
|
||||||
|
int sk_attach_bpf(u32 ufd, struct sock *sk)
|
||||||
|
{
|
||||||
|
struct bpf_prog *prog = __get_bpf(ufd, sk);
|
||||||
|
int err;
|
||||||
|
|
||||||
|
if (IS_ERR(prog))
|
||||||
|
return PTR_ERR(prog);
|
||||||
|
|
||||||
err = __sk_attach_prog(prog, sk);
|
err = __sk_attach_prog(prog, sk);
|
||||||
if (err < 0) {
|
if (err < 0) {
|
||||||
bpf_prog_put(prog);
|
bpf_prog_put(prog);
|
||||||
|
@ -1251,6 +1311,23 @@ int sk_attach_bpf(u32 ufd, struct sock *sk)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int sk_reuseport_attach_bpf(u32 ufd, struct sock *sk)
|
||||||
|
{
|
||||||
|
struct bpf_prog *prog = __get_bpf(ufd, sk);
|
||||||
|
int err;
|
||||||
|
|
||||||
|
if (IS_ERR(prog))
|
||||||
|
return PTR_ERR(prog);
|
||||||
|
|
||||||
|
err = __reuseport_attach_prog(prog, sk);
|
||||||
|
if (err < 0) {
|
||||||
|
bpf_prog_put(prog);
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
#define BPF_RECOMPUTE_CSUM(flags) ((flags) & 1)
|
#define BPF_RECOMPUTE_CSUM(flags) ((flags) & 1)
|
||||||
#define BPF_LDST_LEN 16U
|
#define BPF_LDST_LEN 16U
|
||||||
|
|
||||||
|
|
|
@ -134,6 +134,7 @@
|
||||||
#include <linux/sock_diag.h>
|
#include <linux/sock_diag.h>
|
||||||
|
|
||||||
#include <linux/filter.h>
|
#include <linux/filter.h>
|
||||||
|
#include <net/sock_reuseport.h>
|
||||||
|
|
||||||
#include <trace/events/sock.h>
|
#include <trace/events/sock.h>
|
||||||
|
|
||||||
|
@ -932,6 +933,32 @@ set_rcvbuf:
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case SO_ATTACH_REUSEPORT_CBPF:
|
||||||
|
ret = -EINVAL;
|
||||||
|
if (optlen == sizeof(struct sock_fprog)) {
|
||||||
|
struct sock_fprog fprog;
|
||||||
|
|
||||||
|
ret = -EFAULT;
|
||||||
|
if (copy_from_user(&fprog, optval, sizeof(fprog)))
|
||||||
|
break;
|
||||||
|
|
||||||
|
ret = sk_reuseport_attach_filter(&fprog, sk);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case SO_ATTACH_REUSEPORT_EBPF:
|
||||||
|
ret = -EINVAL;
|
||||||
|
if (optlen == sizeof(u32)) {
|
||||||
|
u32 ufd;
|
||||||
|
|
||||||
|
ret = -EFAULT;
|
||||||
|
if (copy_from_user(&ufd, optval, sizeof(ufd)))
|
||||||
|
break;
|
||||||
|
|
||||||
|
ret = sk_reuseport_attach_bpf(ufd, sk);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
case SO_DETACH_FILTER:
|
case SO_DETACH_FILTER:
|
||||||
ret = sk_detach_filter(sk);
|
ret = sk_detach_filter(sk);
|
||||||
break;
|
break;
|
||||||
|
@ -1443,6 +1470,8 @@ void sk_destruct(struct sock *sk)
|
||||||
sk_filter_uncharge(sk, filter);
|
sk_filter_uncharge(sk, filter);
|
||||||
RCU_INIT_POINTER(sk->sk_filter, NULL);
|
RCU_INIT_POINTER(sk->sk_filter, NULL);
|
||||||
}
|
}
|
||||||
|
if (rcu_access_pointer(sk->sk_reuseport_cb))
|
||||||
|
reuseport_detach_sock(sk);
|
||||||
|
|
||||||
sock_disable_timestamp(sk, SK_FLAGS_TIMESTAMP);
|
sock_disable_timestamp(sk, SK_FLAGS_TIMESTAMP);
|
||||||
|
|
||||||
|
|
|
@ -1,10 +1,12 @@
|
||||||
/*
|
/*
|
||||||
* To speed up listener socket lookup, create an array to store all sockets
|
* To speed up listener socket lookup, create an array to store all sockets
|
||||||
* listening on the same port. This allows a decision to be made after finding
|
* listening on the same port. This allows a decision to be made after finding
|
||||||
* the first socket.
|
* the first socket. An optional BPF program can also be configured for
|
||||||
|
* selecting the socket index from the array of available sockets.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <net/sock_reuseport.h>
|
#include <net/sock_reuseport.h>
|
||||||
|
#include <linux/bpf.h>
|
||||||
#include <linux/rcupdate.h>
|
#include <linux/rcupdate.h>
|
||||||
|
|
||||||
#define INIT_SOCKS 128
|
#define INIT_SOCKS 128
|
||||||
|
@ -22,6 +24,7 @@ static struct sock_reuseport *__reuseport_alloc(u16 max_socks)
|
||||||
|
|
||||||
reuse->max_socks = max_socks;
|
reuse->max_socks = max_socks;
|
||||||
|
|
||||||
|
RCU_INIT_POINTER(reuse->prog, NULL);
|
||||||
return reuse;
|
return reuse;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -67,6 +70,7 @@ static struct sock_reuseport *reuseport_grow(struct sock_reuseport *reuse)
|
||||||
|
|
||||||
more_reuse->max_socks = more_socks_size;
|
more_reuse->max_socks = more_socks_size;
|
||||||
more_reuse->num_socks = reuse->num_socks;
|
more_reuse->num_socks = reuse->num_socks;
|
||||||
|
more_reuse->prog = reuse->prog;
|
||||||
|
|
||||||
memcpy(more_reuse->socks, reuse->socks,
|
memcpy(more_reuse->socks, reuse->socks,
|
||||||
reuse->num_socks * sizeof(struct sock *));
|
reuse->num_socks * sizeof(struct sock *));
|
||||||
|
@ -75,6 +79,10 @@ static struct sock_reuseport *reuseport_grow(struct sock_reuseport *reuse)
|
||||||
rcu_assign_pointer(reuse->socks[i]->sk_reuseport_cb,
|
rcu_assign_pointer(reuse->socks[i]->sk_reuseport_cb,
|
||||||
more_reuse);
|
more_reuse);
|
||||||
|
|
||||||
|
/* Note: we use kfree_rcu here instead of reuseport_free_rcu so
|
||||||
|
* that reuse and more_reuse can temporarily share a reference
|
||||||
|
* to prog.
|
||||||
|
*/
|
||||||
kfree_rcu(reuse, rcu);
|
kfree_rcu(reuse, rcu);
|
||||||
return more_reuse;
|
return more_reuse;
|
||||||
}
|
}
|
||||||
|
@ -116,6 +124,16 @@ int reuseport_add_sock(struct sock *sk, const struct sock *sk2)
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL(reuseport_add_sock);
|
EXPORT_SYMBOL(reuseport_add_sock);
|
||||||
|
|
||||||
|
static void reuseport_free_rcu(struct rcu_head *head)
|
||||||
|
{
|
||||||
|
struct sock_reuseport *reuse;
|
||||||
|
|
||||||
|
reuse = container_of(head, struct sock_reuseport, rcu);
|
||||||
|
if (reuse->prog)
|
||||||
|
bpf_prog_destroy(reuse->prog);
|
||||||
|
kfree(reuse);
|
||||||
|
}
|
||||||
|
|
||||||
void reuseport_detach_sock(struct sock *sk)
|
void reuseport_detach_sock(struct sock *sk)
|
||||||
{
|
{
|
||||||
struct sock_reuseport *reuse;
|
struct sock_reuseport *reuse;
|
||||||
|
@ -131,7 +149,7 @@ void reuseport_detach_sock(struct sock *sk)
|
||||||
reuse->socks[i] = reuse->socks[reuse->num_socks - 1];
|
reuse->socks[i] = reuse->socks[reuse->num_socks - 1];
|
||||||
reuse->num_socks--;
|
reuse->num_socks--;
|
||||||
if (reuse->num_socks == 0)
|
if (reuse->num_socks == 0)
|
||||||
kfree_rcu(reuse, rcu);
|
call_rcu(&reuse->rcu, reuseport_free_rcu);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -139,15 +157,53 @@ void reuseport_detach_sock(struct sock *sk)
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL(reuseport_detach_sock);
|
EXPORT_SYMBOL(reuseport_detach_sock);
|
||||||
|
|
||||||
|
static struct sock *run_bpf(struct sock_reuseport *reuse, u16 socks,
|
||||||
|
struct bpf_prog *prog, struct sk_buff *skb,
|
||||||
|
int hdr_len)
|
||||||
|
{
|
||||||
|
struct sk_buff *nskb = NULL;
|
||||||
|
u32 index;
|
||||||
|
|
||||||
|
if (skb_shared(skb)) {
|
||||||
|
nskb = skb_clone(skb, GFP_ATOMIC);
|
||||||
|
if (!nskb)
|
||||||
|
return NULL;
|
||||||
|
skb = nskb;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* temporarily advance data past protocol header */
|
||||||
|
if (!pskb_pull(skb, hdr_len)) {
|
||||||
|
consume_skb(nskb);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
index = bpf_prog_run_save_cb(prog, skb);
|
||||||
|
__skb_push(skb, hdr_len);
|
||||||
|
|
||||||
|
consume_skb(nskb);
|
||||||
|
|
||||||
|
if (index >= socks)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
return reuse->socks[index];
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* reuseport_select_sock - Select a socket from an SO_REUSEPORT group.
|
* reuseport_select_sock - Select a socket from an SO_REUSEPORT group.
|
||||||
* @sk: First socket in the group.
|
* @sk: First socket in the group.
|
||||||
* @hash: Use this hash to select.
|
* @hash: When no BPF filter is available, use this hash to select.
|
||||||
|
* @skb: skb to run through BPF filter.
|
||||||
|
* @hdr_len: BPF filter expects skb data pointer at payload data. If
|
||||||
|
* the skb does not yet point at the payload, this parameter represents
|
||||||
|
* how far the pointer needs to advance to reach the payload.
|
||||||
* Returns a socket that should receive the packet (or NULL on error).
|
* Returns a socket that should receive the packet (or NULL on error).
|
||||||
*/
|
*/
|
||||||
struct sock *reuseport_select_sock(struct sock *sk, u32 hash)
|
struct sock *reuseport_select_sock(struct sock *sk,
|
||||||
|
u32 hash,
|
||||||
|
struct sk_buff *skb,
|
||||||
|
int hdr_len)
|
||||||
{
|
{
|
||||||
struct sock_reuseport *reuse;
|
struct sock_reuseport *reuse;
|
||||||
|
struct bpf_prog *prog;
|
||||||
struct sock *sk2 = NULL;
|
struct sock *sk2 = NULL;
|
||||||
u16 socks;
|
u16 socks;
|
||||||
|
|
||||||
|
@ -158,12 +214,16 @@ struct sock *reuseport_select_sock(struct sock *sk, u32 hash)
|
||||||
if (!reuse)
|
if (!reuse)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
|
prog = rcu_dereference(reuse->prog);
|
||||||
socks = READ_ONCE(reuse->num_socks);
|
socks = READ_ONCE(reuse->num_socks);
|
||||||
if (likely(socks)) {
|
if (likely(socks)) {
|
||||||
/* paired with smp_wmb() in reuseport_add_sock() */
|
/* paired with smp_wmb() in reuseport_add_sock() */
|
||||||
smp_rmb();
|
smp_rmb();
|
||||||
|
|
||||||
sk2 = reuse->socks[reciprocal_scale(hash, socks)];
|
if (prog && skb)
|
||||||
|
sk2 = run_bpf(reuse, socks, prog, skb, hdr_len);
|
||||||
|
else
|
||||||
|
sk2 = reuse->socks[reciprocal_scale(hash, socks)];
|
||||||
}
|
}
|
||||||
|
|
||||||
out:
|
out:
|
||||||
|
@ -171,3 +231,21 @@ out:
|
||||||
return sk2;
|
return sk2;
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL(reuseport_select_sock);
|
EXPORT_SYMBOL(reuseport_select_sock);
|
||||||
|
|
||||||
|
struct bpf_prog *
|
||||||
|
reuseport_attach_prog(struct sock *sk, struct bpf_prog *prog)
|
||||||
|
{
|
||||||
|
struct sock_reuseport *reuse;
|
||||||
|
struct bpf_prog *old_prog;
|
||||||
|
|
||||||
|
spin_lock_bh(&reuseport_lock);
|
||||||
|
reuse = rcu_dereference_protected(sk->sk_reuseport_cb,
|
||||||
|
lockdep_is_held(&reuseport_lock));
|
||||||
|
old_prog = rcu_dereference_protected(reuse->prog,
|
||||||
|
lockdep_is_held(&reuseport_lock));
|
||||||
|
rcu_assign_pointer(reuse->prog, prog);
|
||||||
|
spin_unlock_bh(&reuseport_lock);
|
||||||
|
|
||||||
|
return old_prog;
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(reuseport_attach_prog);
|
||||||
|
|
|
@ -514,7 +514,7 @@ begin:
|
||||||
struct sock *sk2;
|
struct sock *sk2;
|
||||||
hash = udp_ehashfn(net, daddr, hnum,
|
hash = udp_ehashfn(net, daddr, hnum,
|
||||||
saddr, sport);
|
saddr, sport);
|
||||||
sk2 = reuseport_select_sock(sk, hash);
|
sk2 = reuseport_select_sock(sk, hash, NULL, 0);
|
||||||
if (sk2) {
|
if (sk2) {
|
||||||
result = sk2;
|
result = sk2;
|
||||||
goto found;
|
goto found;
|
||||||
|
@ -553,7 +553,7 @@ found:
|
||||||
*/
|
*/
|
||||||
struct sock *__udp4_lib_lookup(struct net *net, __be32 saddr,
|
struct sock *__udp4_lib_lookup(struct net *net, __be32 saddr,
|
||||||
__be16 sport, __be32 daddr, __be16 dport,
|
__be16 sport, __be32 daddr, __be16 dport,
|
||||||
int dif, struct udp_table *udptable)
|
int dif, struct udp_table *udptable, struct sk_buff *skb)
|
||||||
{
|
{
|
||||||
struct sock *sk, *result;
|
struct sock *sk, *result;
|
||||||
struct hlist_nulls_node *node;
|
struct hlist_nulls_node *node;
|
||||||
|
@ -602,7 +602,8 @@ begin:
|
||||||
struct sock *sk2;
|
struct sock *sk2;
|
||||||
hash = udp_ehashfn(net, daddr, hnum,
|
hash = udp_ehashfn(net, daddr, hnum,
|
||||||
saddr, sport);
|
saddr, sport);
|
||||||
sk2 = reuseport_select_sock(sk, hash);
|
sk2 = reuseport_select_sock(sk, hash, skb,
|
||||||
|
sizeof(struct udphdr));
|
||||||
if (sk2) {
|
if (sk2) {
|
||||||
result = sk2;
|
result = sk2;
|
||||||
goto found;
|
goto found;
|
||||||
|
@ -647,14 +648,14 @@ static inline struct sock *__udp4_lib_lookup_skb(struct sk_buff *skb,
|
||||||
|
|
||||||
return __udp4_lib_lookup(dev_net(skb_dst(skb)->dev), iph->saddr, sport,
|
return __udp4_lib_lookup(dev_net(skb_dst(skb)->dev), iph->saddr, sport,
|
||||||
iph->daddr, dport, inet_iif(skb),
|
iph->daddr, dport, inet_iif(skb),
|
||||||
udptable);
|
udptable, skb);
|
||||||
}
|
}
|
||||||
|
|
||||||
struct sock *udp4_lib_lookup(struct net *net, __be32 saddr, __be16 sport,
|
struct sock *udp4_lib_lookup(struct net *net, __be32 saddr, __be16 sport,
|
||||||
__be32 daddr, __be16 dport, int dif)
|
__be32 daddr, __be16 dport, int dif)
|
||||||
{
|
{
|
||||||
return __udp4_lib_lookup(net, saddr, sport, daddr, dport, dif,
|
return __udp4_lib_lookup(net, saddr, sport, daddr, dport, dif,
|
||||||
&udp_table);
|
&udp_table, NULL);
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(udp4_lib_lookup);
|
EXPORT_SYMBOL_GPL(udp4_lib_lookup);
|
||||||
|
|
||||||
|
@ -702,7 +703,8 @@ void __udp4_lib_err(struct sk_buff *skb, u32 info, struct udp_table *udptable)
|
||||||
struct net *net = dev_net(skb->dev);
|
struct net *net = dev_net(skb->dev);
|
||||||
|
|
||||||
sk = __udp4_lib_lookup(net, iph->daddr, uh->dest,
|
sk = __udp4_lib_lookup(net, iph->daddr, uh->dest,
|
||||||
iph->saddr, uh->source, skb->dev->ifindex, udptable);
|
iph->saddr, uh->source, skb->dev->ifindex, udptable,
|
||||||
|
NULL);
|
||||||
if (!sk) {
|
if (!sk) {
|
||||||
ICMP_INC_STATS_BH(net, ICMP_MIB_INERRORS);
|
ICMP_INC_STATS_BH(net, ICMP_MIB_INERRORS);
|
||||||
return; /* No socket for error */
|
return; /* No socket for error */
|
||||||
|
|
|
@ -44,7 +44,7 @@ static int udp_dump_one(struct udp_table *tbl, struct sk_buff *in_skb,
|
||||||
sk = __udp4_lib_lookup(net,
|
sk = __udp4_lib_lookup(net,
|
||||||
req->id.idiag_src[0], req->id.idiag_sport,
|
req->id.idiag_src[0], req->id.idiag_sport,
|
||||||
req->id.idiag_dst[0], req->id.idiag_dport,
|
req->id.idiag_dst[0], req->id.idiag_dport,
|
||||||
req->id.idiag_if, tbl);
|
req->id.idiag_if, tbl, NULL);
|
||||||
#if IS_ENABLED(CONFIG_IPV6)
|
#if IS_ENABLED(CONFIG_IPV6)
|
||||||
else if (req->sdiag_family == AF_INET6)
|
else if (req->sdiag_family == AF_INET6)
|
||||||
sk = __udp6_lib_lookup(net,
|
sk = __udp6_lib_lookup(net,
|
||||||
|
@ -52,7 +52,7 @@ static int udp_dump_one(struct udp_table *tbl, struct sk_buff *in_skb,
|
||||||
req->id.idiag_sport,
|
req->id.idiag_sport,
|
||||||
(struct in6_addr *)req->id.idiag_dst,
|
(struct in6_addr *)req->id.idiag_dst,
|
||||||
req->id.idiag_dport,
|
req->id.idiag_dport,
|
||||||
req->id.idiag_if, tbl);
|
req->id.idiag_if, tbl, NULL);
|
||||||
#endif
|
#endif
|
||||||
else
|
else
|
||||||
goto out_nosk;
|
goto out_nosk;
|
||||||
|
|
|
@ -272,7 +272,7 @@ begin:
|
||||||
struct sock *sk2;
|
struct sock *sk2;
|
||||||
hash = udp6_ehashfn(net, daddr, hnum,
|
hash = udp6_ehashfn(net, daddr, hnum,
|
||||||
saddr, sport);
|
saddr, sport);
|
||||||
sk2 = reuseport_select_sock(sk, hash);
|
sk2 = reuseport_select_sock(sk, hash, NULL, 0);
|
||||||
if (sk2) {
|
if (sk2) {
|
||||||
result = sk2;
|
result = sk2;
|
||||||
goto found;
|
goto found;
|
||||||
|
@ -310,7 +310,8 @@ found:
|
||||||
struct sock *__udp6_lib_lookup(struct net *net,
|
struct sock *__udp6_lib_lookup(struct net *net,
|
||||||
const struct in6_addr *saddr, __be16 sport,
|
const struct in6_addr *saddr, __be16 sport,
|
||||||
const struct in6_addr *daddr, __be16 dport,
|
const struct in6_addr *daddr, __be16 dport,
|
||||||
int dif, struct udp_table *udptable)
|
int dif, struct udp_table *udptable,
|
||||||
|
struct sk_buff *skb)
|
||||||
{
|
{
|
||||||
struct sock *sk, *result;
|
struct sock *sk, *result;
|
||||||
struct hlist_nulls_node *node;
|
struct hlist_nulls_node *node;
|
||||||
|
@ -358,7 +359,8 @@ begin:
|
||||||
struct sock *sk2;
|
struct sock *sk2;
|
||||||
hash = udp6_ehashfn(net, daddr, hnum,
|
hash = udp6_ehashfn(net, daddr, hnum,
|
||||||
saddr, sport);
|
saddr, sport);
|
||||||
sk2 = reuseport_select_sock(sk, hash);
|
sk2 = reuseport_select_sock(sk, hash, skb,
|
||||||
|
sizeof(struct udphdr));
|
||||||
if (sk2) {
|
if (sk2) {
|
||||||
result = sk2;
|
result = sk2;
|
||||||
goto found;
|
goto found;
|
||||||
|
@ -407,13 +409,13 @@ static struct sock *__udp6_lib_lookup_skb(struct sk_buff *skb,
|
||||||
return sk;
|
return sk;
|
||||||
return __udp6_lib_lookup(dev_net(skb_dst(skb)->dev), &iph->saddr, sport,
|
return __udp6_lib_lookup(dev_net(skb_dst(skb)->dev), &iph->saddr, sport,
|
||||||
&iph->daddr, dport, inet6_iif(skb),
|
&iph->daddr, dport, inet6_iif(skb),
|
||||||
udptable);
|
udptable, skb);
|
||||||
}
|
}
|
||||||
|
|
||||||
struct sock *udp6_lib_lookup(struct net *net, const struct in6_addr *saddr, __be16 sport,
|
struct sock *udp6_lib_lookup(struct net *net, const struct in6_addr *saddr, __be16 sport,
|
||||||
const struct in6_addr *daddr, __be16 dport, int dif)
|
const struct in6_addr *daddr, __be16 dport, int dif)
|
||||||
{
|
{
|
||||||
return __udp6_lib_lookup(net, saddr, sport, daddr, dport, dif, &udp_table);
|
return __udp6_lib_lookup(net, saddr, sport, daddr, dport, dif, &udp_table, NULL);
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(udp6_lib_lookup);
|
EXPORT_SYMBOL_GPL(udp6_lib_lookup);
|
||||||
|
|
||||||
|
@ -580,7 +582,7 @@ void __udp6_lib_err(struct sk_buff *skb, struct inet6_skb_parm *opt,
|
||||||
struct net *net = dev_net(skb->dev);
|
struct net *net = dev_net(skb->dev);
|
||||||
|
|
||||||
sk = __udp6_lib_lookup(net, daddr, uh->dest, saddr, uh->source,
|
sk = __udp6_lib_lookup(net, daddr, uh->dest, saddr, uh->source,
|
||||||
inet6_iif(skb), udptable);
|
inet6_iif(skb), udptable, skb);
|
||||||
if (!sk) {
|
if (!sk) {
|
||||||
ICMP6_INC_STATS_BH(net, __in6_dev_get(skb->dev),
|
ICMP6_INC_STATS_BH(net, __in6_dev_get(skb->dev),
|
||||||
ICMP6_MIB_INERRORS);
|
ICMP6_MIB_INERRORS);
|
||||||
|
|
Загрузка…
Ссылка в новой задаче