2016-01-05 01:41:48 +03:00
/*
* Test functionality of BPF filters for SO_REUSEPORT . The tests below will use
* a BPF program ( both classic and extended ) to read the first word from an
* incoming packet ( expected to be in network byte - order ) , calculate a modulus
* of that number , and then dispatch the packet to the Nth socket using the
* result . These tests are run for each supported address family and protocol .
* Additionally , a few edge cases in the implementation are tested .
*/
# include <errno.h>
# include <error.h>
2016-02-10 19:50:41 +03:00
# include <fcntl.h>
2016-01-05 01:41:48 +03:00
# include <linux/bpf.h>
# include <linux/filter.h>
# include <linux/unistd.h>
# include <netinet/in.h>
2016-02-10 19:50:41 +03:00
# include <netinet/tcp.h>
2016-01-05 01:41:48 +03:00
# include <stdio.h>
# include <stdlib.h>
# include <string.h>
# include <sys/epoll.h>
# include <sys/types.h>
# include <sys/socket.h>
2018-02-09 16:49:44 +03:00
# include <sys/resource.h>
2016-01-05 01:41:48 +03:00
# include <unistd.h>
# ifndef ARRAY_SIZE
# define ARRAY_SIZE(arr) (sizeof(arr) / sizeof((arr)[0]))
# endif
struct test_params {
int recv_family ;
int send_family ;
int protocol ;
size_t recv_socks ;
uint16_t recv_port ;
uint16_t send_port_min ;
} ;
static size_t sockaddr_size ( void )
{
return sizeof ( struct sockaddr_storage ) ;
}
static struct sockaddr * new_any_sockaddr ( int family , uint16_t port )
{
struct sockaddr_storage * addr ;
struct sockaddr_in * addr4 ;
struct sockaddr_in6 * addr6 ;
addr = malloc ( sizeof ( struct sockaddr_storage ) ) ;
memset ( addr , 0 , sizeof ( struct sockaddr_storage ) ) ;
switch ( family ) {
case AF_INET :
addr4 = ( struct sockaddr_in * ) addr ;
addr4 - > sin_family = AF_INET ;
addr4 - > sin_addr . s_addr = htonl ( INADDR_ANY ) ;
addr4 - > sin_port = htons ( port ) ;
break ;
case AF_INET6 :
addr6 = ( struct sockaddr_in6 * ) addr ;
addr6 - > sin6_family = AF_INET6 ;
addr6 - > sin6_addr = in6addr_any ;
addr6 - > sin6_port = htons ( port ) ;
break ;
default :
error ( 1 , 0 , " Unsupported family %d " , family ) ;
}
return ( struct sockaddr * ) addr ;
}
static struct sockaddr * new_loopback_sockaddr ( int family , uint16_t port )
{
struct sockaddr * addr = new_any_sockaddr ( family , port ) ;
struct sockaddr_in * addr4 ;
struct sockaddr_in6 * addr6 ;
switch ( family ) {
case AF_INET :
addr4 = ( struct sockaddr_in * ) addr ;
addr4 - > sin_addr . s_addr = htonl ( INADDR_LOOPBACK ) ;
break ;
case AF_INET6 :
addr6 = ( struct sockaddr_in6 * ) addr ;
addr6 - > sin6_addr = in6addr_loopback ;
break ;
default :
error ( 1 , 0 , " Unsupported family %d " , family ) ;
}
return addr ;
}
static void attach_ebpf ( int fd , uint16_t mod )
{
static char bpf_log_buf [ 65536 ] ;
static const char bpf_license [ ] = " GPL " ;
int bpf_fd ;
const struct bpf_insn prog [ ] = {
/* BPF_MOV64_REG(BPF_REG_6, BPF_REG_1) */
{ BPF_ALU64 | BPF_MOV | BPF_X , BPF_REG_6 , BPF_REG_1 , 0 , 0 } ,
/* BPF_LD_ABS(BPF_W, 0) R0 = (uint32_t)skb[0] */
{ BPF_LD | BPF_ABS | BPF_W , 0 , 0 , 0 , 0 } ,
/* BPF_ALU64_IMM(BPF_MOD, BPF_REG_0, mod) */
{ BPF_ALU64 | BPF_MOD | BPF_K , BPF_REG_0 , 0 , 0 , mod } ,
/* BPF_EXIT_INSN() */
{ BPF_JMP | BPF_EXIT , 0 , 0 , 0 , 0 }
} ;
union bpf_attr attr ;
memset ( & attr , 0 , sizeof ( attr ) ) ;
attr . prog_type = BPF_PROG_TYPE_SOCKET_FILTER ;
attr . insn_cnt = ARRAY_SIZE ( prog ) ;
2016-06-03 20:19:20 +03:00
attr . insns = ( unsigned long ) & prog ;
attr . license = ( unsigned long ) & bpf_license ;
attr . log_buf = ( unsigned long ) & bpf_log_buf ;
2016-01-05 01:41:48 +03:00
attr . log_size = sizeof ( bpf_log_buf ) ;
attr . log_level = 1 ;
attr . kern_version = 0 ;
bpf_fd = syscall ( __NR_bpf , BPF_PROG_LOAD , & attr , sizeof ( attr ) ) ;
if ( bpf_fd < 0 )
error ( 1 , errno , " ebpf error. log: \n %s \n " , bpf_log_buf ) ;
if ( setsockopt ( fd , SOL_SOCKET , SO_ATTACH_REUSEPORT_EBPF , & bpf_fd ,
sizeof ( bpf_fd ) ) )
error ( 1 , errno , " failed to set SO_ATTACH_REUSEPORT_EBPF " ) ;
2016-01-05 23:08:07 +03:00
close ( bpf_fd ) ;
2016-01-05 01:41:48 +03:00
}
static void attach_cbpf ( int fd , uint16_t mod )
{
struct sock_filter code [ ] = {
/* A = (uint32_t)skb[0] */
{ BPF_LD | BPF_W | BPF_ABS , 0 , 0 , 0 } ,
/* A = A % mod */
{ BPF_ALU | BPF_MOD , 0 , 0 , mod } ,
/* return A */
{ BPF_RET | BPF_A , 0 , 0 , 0 } ,
} ;
struct sock_fprog p = {
. len = ARRAY_SIZE ( code ) ,
. filter = code ,
} ;
if ( setsockopt ( fd , SOL_SOCKET , SO_ATTACH_REUSEPORT_CBPF , & p , sizeof ( p ) ) )
error ( 1 , errno , " failed to set SO_ATTACH_REUSEPORT_CBPF " ) ;
}
static void build_recv_group ( const struct test_params p , int fd [ ] , uint16_t mod ,
void ( * attach_bpf ) ( int , uint16_t ) )
{
struct sockaddr * const addr =
new_any_sockaddr ( p . recv_family , p . recv_port ) ;
int i , opt ;
for ( i = 0 ; i < p . recv_socks ; + + i ) {
fd [ i ] = socket ( p . recv_family , p . protocol , 0 ) ;
if ( fd [ i ] < 0 )
error ( 1 , errno , " failed to create recv %d " , i ) ;
opt = 1 ;
if ( setsockopt ( fd [ i ] , SOL_SOCKET , SO_REUSEPORT , & opt ,
sizeof ( opt ) ) )
error ( 1 , errno , " failed to set SO_REUSEPORT on %d " , i ) ;
if ( i = = 0 )
attach_bpf ( fd [ i ] , mod ) ;
if ( bind ( fd [ i ] , addr , sockaddr_size ( ) ) )
error ( 1 , errno , " failed to bind recv socket %d " , i ) ;
2016-02-10 19:50:41 +03:00
if ( p . protocol = = SOCK_STREAM ) {
opt = 4 ;
if ( setsockopt ( fd [ i ] , SOL_TCP , TCP_FASTOPEN , & opt ,
sizeof ( opt ) ) )
error ( 1 , errno ,
" failed to set TCP_FASTOPEN on %d " , i ) ;
2016-01-05 01:41:48 +03:00
if ( listen ( fd [ i ] , p . recv_socks * 10 ) )
error ( 1 , errno , " failed to listen on socket " ) ;
2016-02-10 19:50:41 +03:00
}
2016-01-05 01:41:48 +03:00
}
free ( addr ) ;
}
static void send_from ( struct test_params p , uint16_t sport , char * buf ,
size_t len )
{
struct sockaddr * const saddr = new_any_sockaddr ( p . send_family , sport ) ;
struct sockaddr * const daddr =
new_loopback_sockaddr ( p . send_family , p . recv_port ) ;
2018-02-09 16:49:44 +03:00
const int fd = socket ( p . send_family , p . protocol , 0 ) , one = 1 ;
2016-01-05 01:41:48 +03:00
if ( fd < 0 )
error ( 1 , errno , " failed to create send socket " ) ;
2018-02-09 16:49:44 +03:00
if ( setsockopt ( fd , SOL_SOCKET , SO_REUSEADDR , & one , sizeof ( one ) ) )
error ( 1 , errno , " failed to set reuseaddr " ) ;
2016-01-05 01:41:48 +03:00
if ( bind ( fd , saddr , sockaddr_size ( ) ) )
error ( 1 , errno , " failed to bind send socket " ) ;
2016-02-10 19:50:41 +03:00
if ( sendto ( fd , buf , len , MSG_FASTOPEN , daddr , sockaddr_size ( ) ) < 0 )
2016-01-05 01:41:48 +03:00
error ( 1 , errno , " failed to send message " ) ;
close ( fd ) ;
free ( saddr ) ;
free ( daddr ) ;
}
static void test_recv_order ( const struct test_params p , int fd [ ] , int mod )
{
char recv_buf [ 8 ] , send_buf [ 8 ] ;
struct msghdr msg ;
struct iovec recv_io = { recv_buf , 8 } ;
struct epoll_event ev ;
int epfd , conn , i , sport , expected ;
uint32_t data , ndata ;
epfd = epoll_create ( 1 ) ;
if ( epfd < 0 )
error ( 1 , errno , " failed to create epoll " ) ;
for ( i = 0 ; i < p . recv_socks ; + + i ) {
ev . events = EPOLLIN ;
ev . data . fd = fd [ i ] ;
if ( epoll_ctl ( epfd , EPOLL_CTL_ADD , fd [ i ] , & ev ) )
error ( 1 , errno , " failed to register sock %d epoll " , i ) ;
}
memset ( & msg , 0 , sizeof ( msg ) ) ;
msg . msg_iov = & recv_io ;
msg . msg_iovlen = 1 ;
for ( data = 0 ; data < p . recv_socks * 2 ; + + data ) {
sport = p . send_port_min + data ;
ndata = htonl ( data ) ;
memcpy ( send_buf , & ndata , sizeof ( ndata ) ) ;
send_from ( p , sport , send_buf , sizeof ( ndata ) ) ;
i = epoll_wait ( epfd , & ev , 1 , - 1 ) ;
if ( i < 0 )
error ( 1 , errno , " epoll wait failed " ) ;
if ( p . protocol = = SOCK_STREAM ) {
conn = accept ( ev . data . fd , NULL , NULL ) ;
if ( conn < 0 )
error ( 1 , errno , " error accepting " ) ;
i = recvmsg ( conn , & msg , 0 ) ;
close ( conn ) ;
} else {
i = recvmsg ( ev . data . fd , & msg , 0 ) ;
}
if ( i < 0 )
error ( 1 , errno , " recvmsg error " ) ;
if ( i ! = sizeof ( ndata ) )
error ( 1 , 0 , " expected size %zd got %d " ,
sizeof ( ndata ) , i ) ;
for ( i = 0 ; i < p . recv_socks ; + + i )
if ( ev . data . fd = = fd [ i ] )
break ;
memcpy ( & ndata , recv_buf , sizeof ( ndata ) ) ;
fprintf ( stderr , " Socket %d: %d \n " , i , ntohl ( ndata ) ) ;
expected = ( sport % mod ) ;
if ( i ! = expected )
error ( 1 , 0 , " expected socket %d " , expected ) ;
}
}
2016-02-10 19:50:41 +03:00
static void test_reuseport_ebpf ( struct test_params p )
2016-01-05 01:41:48 +03:00
{
int i , fd [ p . recv_socks ] ;
fprintf ( stderr , " Testing EBPF mod %zd... \n " , p . recv_socks ) ;
build_recv_group ( p , fd , p . recv_socks , attach_ebpf ) ;
test_recv_order ( p , fd , p . recv_socks ) ;
2016-02-10 19:50:41 +03:00
p . send_port_min + = p . recv_socks * 2 ;
2016-01-05 01:41:48 +03:00
fprintf ( stderr , " Reprograming, testing mod %zd... \n " , p . recv_socks / 2 ) ;
attach_ebpf ( fd [ 0 ] , p . recv_socks / 2 ) ;
test_recv_order ( p , fd , p . recv_socks / 2 ) ;
for ( i = 0 ; i < p . recv_socks ; + + i )
close ( fd [ i ] ) ;
}
2016-02-10 19:50:41 +03:00
static void test_reuseport_cbpf ( struct test_params p )
2016-01-05 01:41:48 +03:00
{
int i , fd [ p . recv_socks ] ;
fprintf ( stderr , " Testing CBPF mod %zd... \n " , p . recv_socks ) ;
build_recv_group ( p , fd , p . recv_socks , attach_cbpf ) ;
test_recv_order ( p , fd , p . recv_socks ) ;
2016-02-10 19:50:41 +03:00
p . send_port_min + = p . recv_socks * 2 ;
2016-01-05 01:41:48 +03:00
fprintf ( stderr , " Reprograming, testing mod %zd... \n " , p . recv_socks / 2 ) ;
attach_cbpf ( fd [ 0 ] , p . recv_socks / 2 ) ;
test_recv_order ( p , fd , p . recv_socks / 2 ) ;
for ( i = 0 ; i < p . recv_socks ; + + i )
close ( fd [ i ] ) ;
}
static void test_extra_filter ( const struct test_params p )
{
struct sockaddr * const addr =
new_any_sockaddr ( p . recv_family , p . recv_port ) ;
int fd1 , fd2 , opt ;
fprintf ( stderr , " Testing too many filters... \n " ) ;
fd1 = socket ( p . recv_family , p . protocol , 0 ) ;
if ( fd1 < 0 )
error ( 1 , errno , " failed to create socket 1 " ) ;
fd2 = socket ( p . recv_family , p . protocol , 0 ) ;
if ( fd2 < 0 )
error ( 1 , errno , " failed to create socket 2 " ) ;
opt = 1 ;
if ( setsockopt ( fd1 , SOL_SOCKET , SO_REUSEPORT , & opt , sizeof ( opt ) ) )
error ( 1 , errno , " failed to set SO_REUSEPORT on socket 1 " ) ;
if ( setsockopt ( fd2 , SOL_SOCKET , SO_REUSEPORT , & opt , sizeof ( opt ) ) )
error ( 1 , errno , " failed to set SO_REUSEPORT on socket 2 " ) ;
attach_ebpf ( fd1 , 10 ) ;
attach_ebpf ( fd2 , 10 ) ;
if ( bind ( fd1 , addr , sockaddr_size ( ) ) )
error ( 1 , errno , " failed to bind recv socket 1 " ) ;
if ( ! bind ( fd2 , addr , sockaddr_size ( ) ) & & errno ! = EADDRINUSE )
error ( 1 , errno , " bind socket 2 should fail with EADDRINUSE " ) ;
free ( addr ) ;
}
static void test_filter_no_reuseport ( const struct test_params p )
{
struct sockaddr * const addr =
new_any_sockaddr ( p . recv_family , p . recv_port ) ;
const char bpf_license [ ] = " GPL " ;
struct bpf_insn ecode [ ] = {
{ BPF_ALU64 | BPF_MOV | BPF_K , BPF_REG_0 , 0 , 0 , 10 } ,
{ BPF_JMP | BPF_EXIT , 0 , 0 , 0 , 0 }
} ;
struct sock_filter ccode [ ] = { { BPF_RET | BPF_A , 0 , 0 , 0 } } ;
union bpf_attr eprog ;
struct sock_fprog cprog ;
int fd , bpf_fd ;
fprintf ( stderr , " Testing filters on non-SO_REUSEPORT socket... \n " ) ;
memset ( & eprog , 0 , sizeof ( eprog ) ) ;
eprog . prog_type = BPF_PROG_TYPE_SOCKET_FILTER ;
eprog . insn_cnt = ARRAY_SIZE ( ecode ) ;
2016-06-03 20:19:20 +03:00
eprog . insns = ( unsigned long ) & ecode ;
eprog . license = ( unsigned long ) & bpf_license ;
2016-01-05 01:41:48 +03:00
eprog . kern_version = 0 ;
memset ( & cprog , 0 , sizeof ( cprog ) ) ;
cprog . len = ARRAY_SIZE ( ccode ) ;
cprog . filter = ccode ;
bpf_fd = syscall ( __NR_bpf , BPF_PROG_LOAD , & eprog , sizeof ( eprog ) ) ;
if ( bpf_fd < 0 )
error ( 1 , errno , " ebpf error " ) ;
fd = socket ( p . recv_family , p . protocol , 0 ) ;
if ( fd < 0 )
error ( 1 , errno , " failed to create socket 1 " ) ;
if ( bind ( fd , addr , sockaddr_size ( ) ) )
error ( 1 , errno , " failed to bind recv socket 1 " ) ;
errno = 0 ;
if ( ! setsockopt ( fd , SOL_SOCKET , SO_ATTACH_REUSEPORT_EBPF , & bpf_fd ,
sizeof ( bpf_fd ) ) | | errno ! = EINVAL )
error ( 1 , errno , " setsockopt should have returned EINVAL " ) ;
errno = 0 ;
if ( ! setsockopt ( fd , SOL_SOCKET , SO_ATTACH_REUSEPORT_CBPF , & cprog ,
sizeof ( cprog ) ) | | errno ! = EINVAL )
error ( 1 , errno , " setsockopt should have returned EINVAL " ) ;
free ( addr ) ;
}
static void test_filter_without_bind ( void )
{
2016-02-10 19:50:41 +03:00
int fd1 , fd2 , opt = 1 ;
2016-01-05 01:41:48 +03:00
fprintf ( stderr , " Testing filter add without bind... \n " ) ;
fd1 = socket ( AF_INET , SOCK_DGRAM , 0 ) ;
if ( fd1 < 0 )
error ( 1 , errno , " failed to create socket 1 " ) ;
fd2 = socket ( AF_INET , SOCK_DGRAM , 0 ) ;
if ( fd2 < 0 )
error ( 1 , errno , " failed to create socket 2 " ) ;
2016-02-10 19:50:41 +03:00
if ( setsockopt ( fd1 , SOL_SOCKET , SO_REUSEPORT , & opt , sizeof ( opt ) ) )
error ( 1 , errno , " failed to set SO_REUSEPORT on socket 1 " ) ;
if ( setsockopt ( fd2 , SOL_SOCKET , SO_REUSEPORT , & opt , sizeof ( opt ) ) )
error ( 1 , errno , " failed to set SO_REUSEPORT on socket 2 " ) ;
2016-01-05 01:41:48 +03:00
attach_ebpf ( fd1 , 10 ) ;
attach_cbpf ( fd2 , 10 ) ;
close ( fd1 ) ;
close ( fd2 ) ;
}
2016-02-10 19:50:41 +03:00
void enable_fastopen ( void )
{
int fd = open ( " /proc/sys/net/ipv4/tcp_fastopen " , 0 ) ;
int rw_mask = 3 ; /* bit 1: client side; bit-2 server side */
int val , size ;
char buf [ 16 ] ;
if ( fd < 0 )
error ( 1 , errno , " Unable to open tcp_fastopen sysctl " ) ;
if ( read ( fd , buf , sizeof ( buf ) ) < = 0 )
error ( 1 , errno , " Unable to read tcp_fastopen sysctl " ) ;
val = atoi ( buf ) ;
close ( fd ) ;
if ( ( val & rw_mask ) ! = rw_mask ) {
fd = open ( " /proc/sys/net/ipv4/tcp_fastopen " , O_RDWR ) ;
if ( fd < 0 )
error ( 1 , errno ,
" Unable to open tcp_fastopen sysctl for writing " ) ;
val | = rw_mask ;
size = snprintf ( buf , 16 , " %d " , val ) ;
if ( write ( fd , buf , size ) < = 0 )
error ( 1 , errno , " Unable to write tcp_fastopen sysctl " ) ;
close ( fd ) ;
}
}
2016-01-05 01:41:48 +03:00
2018-02-09 16:49:44 +03:00
static struct rlimit rlim_old , rlim_new ;
static __attribute__ ( ( constructor ) ) void main_ctor ( void )
{
getrlimit ( RLIMIT_MEMLOCK , & rlim_old ) ;
rlim_new . rlim_cur = rlim_old . rlim_cur + ( 1UL < < 20 ) ;
rlim_new . rlim_max = rlim_old . rlim_max + ( 1UL < < 20 ) ;
setrlimit ( RLIMIT_MEMLOCK , & rlim_new ) ;
}
static __attribute__ ( ( destructor ) ) void main_dtor ( void )
{
setrlimit ( RLIMIT_MEMLOCK , & rlim_old ) ;
}
2016-01-05 01:41:48 +03:00
int main ( void )
{
fprintf ( stderr , " ---- IPv4 UDP ---- \n " ) ;
2016-01-05 23:08:07 +03:00
/* NOTE: UDP socket lookups traverse a different code path when there
* are > 10 sockets in a group . Run the bpf test through both paths .
*/
2016-01-05 01:41:48 +03:00
test_reuseport_ebpf ( ( struct test_params ) {
. recv_family = AF_INET ,
. send_family = AF_INET ,
. protocol = SOCK_DGRAM ,
. recv_socks = 10 ,
. recv_port = 8000 ,
. send_port_min = 9000 } ) ;
2016-01-05 23:08:07 +03:00
test_reuseport_ebpf ( ( struct test_params ) {
. recv_family = AF_INET ,
. send_family = AF_INET ,
. protocol = SOCK_DGRAM ,
. recv_socks = 20 ,
. recv_port = 8000 ,
. send_port_min = 9000 } ) ;
2016-01-05 01:41:48 +03:00
test_reuseport_cbpf ( ( struct test_params ) {
. recv_family = AF_INET ,
. send_family = AF_INET ,
. protocol = SOCK_DGRAM ,
. recv_socks = 10 ,
. recv_port = 8001 ,
. send_port_min = 9020 } ) ;
2016-01-05 23:08:07 +03:00
test_reuseport_cbpf ( ( struct test_params ) {
. recv_family = AF_INET ,
. send_family = AF_INET ,
. protocol = SOCK_DGRAM ,
. recv_socks = 20 ,
. recv_port = 8001 ,
. send_port_min = 9020 } ) ;
2016-01-05 01:41:48 +03:00
test_extra_filter ( ( struct test_params ) {
. recv_family = AF_INET ,
. protocol = SOCK_DGRAM ,
. recv_port = 8002 } ) ;
test_filter_no_reuseport ( ( struct test_params ) {
. recv_family = AF_INET ,
. protocol = SOCK_DGRAM ,
. recv_port = 8008 } ) ;
fprintf ( stderr , " ---- IPv6 UDP ---- \n " ) ;
test_reuseport_ebpf ( ( struct test_params ) {
. recv_family = AF_INET6 ,
. send_family = AF_INET6 ,
. protocol = SOCK_DGRAM ,
. recv_socks = 10 ,
. recv_port = 8003 ,
. send_port_min = 9040 } ) ;
2016-01-05 23:08:07 +03:00
test_reuseport_ebpf ( ( struct test_params ) {
. recv_family = AF_INET6 ,
. send_family = AF_INET6 ,
. protocol = SOCK_DGRAM ,
. recv_socks = 20 ,
. recv_port = 8003 ,
. send_port_min = 9040 } ) ;
2016-01-05 01:41:48 +03:00
test_reuseport_cbpf ( ( struct test_params ) {
. recv_family = AF_INET6 ,
. send_family = AF_INET6 ,
. protocol = SOCK_DGRAM ,
. recv_socks = 10 ,
. recv_port = 8004 ,
. send_port_min = 9060 } ) ;
2016-01-05 23:08:07 +03:00
test_reuseport_cbpf ( ( struct test_params ) {
. recv_family = AF_INET6 ,
. send_family = AF_INET6 ,
. protocol = SOCK_DGRAM ,
. recv_socks = 20 ,
. recv_port = 8004 ,
. send_port_min = 9060 } ) ;
2016-01-05 01:41:48 +03:00
test_extra_filter ( ( struct test_params ) {
. recv_family = AF_INET6 ,
. protocol = SOCK_DGRAM ,
. recv_port = 8005 } ) ;
test_filter_no_reuseport ( ( struct test_params ) {
. recv_family = AF_INET6 ,
. protocol = SOCK_DGRAM ,
. recv_port = 8009 } ) ;
fprintf ( stderr , " ---- IPv6 UDP w/ mapped IPv4 ---- \n " ) ;
2016-01-05 23:08:07 +03:00
test_reuseport_ebpf ( ( struct test_params ) {
. recv_family = AF_INET6 ,
. send_family = AF_INET ,
. protocol = SOCK_DGRAM ,
. recv_socks = 20 ,
. recv_port = 8006 ,
. send_port_min = 9080 } ) ;
2016-01-05 01:41:48 +03:00
test_reuseport_ebpf ( ( struct test_params ) {
. recv_family = AF_INET6 ,
. send_family = AF_INET ,
. protocol = SOCK_DGRAM ,
. recv_socks = 10 ,
. recv_port = 8006 ,
. send_port_min = 9080 } ) ;
test_reuseport_cbpf ( ( struct test_params ) {
. recv_family = AF_INET6 ,
. send_family = AF_INET ,
. protocol = SOCK_DGRAM ,
. recv_socks = 10 ,
. recv_port = 8007 ,
. send_port_min = 9100 } ) ;
2016-01-05 23:08:07 +03:00
test_reuseport_cbpf ( ( struct test_params ) {
. recv_family = AF_INET6 ,
. send_family = AF_INET ,
. protocol = SOCK_DGRAM ,
. recv_socks = 20 ,
. recv_port = 8007 ,
. send_port_min = 9100 } ) ;
2016-01-05 01:41:48 +03:00
2016-02-10 19:50:41 +03:00
/* TCP fastopen is required for the TCP tests */
enable_fastopen ( ) ;
fprintf ( stderr , " ---- IPv4 TCP ---- \n " ) ;
test_reuseport_ebpf ( ( struct test_params ) {
. recv_family = AF_INET ,
. send_family = AF_INET ,
. protocol = SOCK_STREAM ,
. recv_socks = 10 ,
. recv_port = 8008 ,
. send_port_min = 9120 } ) ;
test_reuseport_cbpf ( ( struct test_params ) {
. recv_family = AF_INET ,
. send_family = AF_INET ,
. protocol = SOCK_STREAM ,
. recv_socks = 10 ,
. recv_port = 8009 ,
. send_port_min = 9160 } ) ;
test_extra_filter ( ( struct test_params ) {
. recv_family = AF_INET ,
. protocol = SOCK_STREAM ,
. recv_port = 8010 } ) ;
test_filter_no_reuseport ( ( struct test_params ) {
. recv_family = AF_INET ,
. protocol = SOCK_STREAM ,
. recv_port = 8011 } ) ;
fprintf ( stderr , " ---- IPv6 TCP ---- \n " ) ;
test_reuseport_ebpf ( ( struct test_params ) {
. recv_family = AF_INET6 ,
. send_family = AF_INET6 ,
. protocol = SOCK_STREAM ,
. recv_socks = 10 ,
. recv_port = 8012 ,
. send_port_min = 9200 } ) ;
test_reuseport_cbpf ( ( struct test_params ) {
. recv_family = AF_INET6 ,
. send_family = AF_INET6 ,
. protocol = SOCK_STREAM ,
. recv_socks = 10 ,
. recv_port = 8013 ,
. send_port_min = 9240 } ) ;
test_extra_filter ( ( struct test_params ) {
. recv_family = AF_INET6 ,
. protocol = SOCK_STREAM ,
. recv_port = 8014 } ) ;
test_filter_no_reuseport ( ( struct test_params ) {
. recv_family = AF_INET6 ,
. protocol = SOCK_STREAM ,
. recv_port = 8015 } ) ;
fprintf ( stderr , " ---- IPv6 TCP w/ mapped IPv4 ---- \n " ) ;
test_reuseport_ebpf ( ( struct test_params ) {
. recv_family = AF_INET6 ,
. send_family = AF_INET ,
. protocol = SOCK_STREAM ,
. recv_socks = 10 ,
. recv_port = 8016 ,
. send_port_min = 9320 } ) ;
test_reuseport_cbpf ( ( struct test_params ) {
. recv_family = AF_INET6 ,
. send_family = AF_INET ,
. protocol = SOCK_STREAM ,
. recv_socks = 10 ,
. recv_port = 8017 ,
. send_port_min = 9360 } ) ;
2016-01-05 01:41:48 +03:00
test_filter_without_bind ( ) ;
fprintf ( stderr , " SUCCESS \n " ) ;
return 0 ;
}