1999-12-16 13:06:13 +00:00
/*
* BIRD Internet Routing Daemon - - Linux Multicasting and Network Includes
*
2000-04-20 23:05:41 +00:00
* ( c ) 1998 - - 2000 Martin Mares < mj @ ucw . cz >
1999-12-16 13:06:13 +00:00
*
* Can be freely distributed and used under the terms of the GNU GPL .
*/
2024-02-05 13:43:46 +00:00
# include "sysdep/linux/tcp-ao.h"
2014-05-18 09:42:26 +00:00
# ifndef IPV6_MINHOPCOUNT
# define IPV6_MINHOPCOUNT 73
# endif
1999-12-16 13:06:13 +00:00
2022-01-08 17:31:56 +00:00
# ifndef IPV6_FREEBIND
# define IPV6_FREEBIND 78
# endif
2020-02-27 15:16:48 +00:00
# ifndef TCP_MD5SIG_EXT
# define TCP_MD5SIG_EXT 32
# endif
2009-09-04 09:06:51 +00:00
2020-02-27 15:16:48 +00:00
# ifndef TCP_MD5SIG_FLAG_PREFIX
# define TCP_MD5SIG_FLAG_PREFIX 1
# endif
2009-09-04 09:06:51 +00:00
2024-02-05 13:43:46 +00:00
2020-02-27 15:16:48 +00:00
/* We redefine the tcp_md5sig structure with different name to avoid collision with older headers */
struct tcp_md5sig_ext {
struct sockaddr_storage tcpm_addr ; /* Address associated */
u8 tcpm_flags ; /* Extension flags */
u8 tcpm_prefixlen ; /* Address prefix */
u16 tcpm_keylen ; /* Key length */
u32 __tcpm_pad2 ; /* Zero */
u8 tcpm_key [ TCP_MD5SIG_MAXKEYLEN ] ; /* Key (binary) */
2014-05-18 09:42:26 +00:00
} ;
2009-09-04 09:06:51 +00:00
2014-05-18 09:42:26 +00:00
/* Linux does not care if sa_len is larger than needed */
# define SA_LEN(x) sizeof(sockaddr)
2009-09-04 09:06:51 +00:00
2014-05-18 09:42:26 +00:00
/*
* Linux IPv4 multicast syscalls
*/
2009-09-04 09:06:51 +00:00
2014-05-18 09:42:26 +00:00
# define INIT_MREQ4(maddr,ifa) \
{ . imr_multiaddr = ipa_to_in4 ( maddr ) , . imr_ifindex = ifa - > index }
1999-12-16 13:06:13 +00:00
2014-05-18 09:42:26 +00:00
static inline int
sk_setup_multicast4 ( sock * s )
{
struct ip_mreqn mr = { . imr_ifindex = s - > iface - > index } ;
int ttl = s - > ttl ;
int n = 0 ;
2008-10-26 21:42:39 +00:00
2014-05-18 09:42:26 +00:00
/* This defines where should we send _outgoing_ multicasts */
if ( setsockopt ( s - > fd , SOL_IP , IP_MULTICAST_IF , & mr , sizeof ( mr ) ) < 0 )
ERR ( " IP_MULTICAST_IF " ) ;
2009-09-04 09:06:51 +00:00
2014-05-18 09:42:26 +00:00
if ( setsockopt ( s - > fd , SOL_IP , IP_MULTICAST_TTL , & ttl , sizeof ( ttl ) ) < 0 )
ERR ( " IP_MULTICAST_TTL " ) ;
2008-10-26 21:42:39 +00:00
2014-05-18 09:42:26 +00:00
if ( setsockopt ( s - > fd , SOL_IP , IP_MULTICAST_LOOP , & n , sizeof ( n ) ) < 0 )
ERR ( " IP_MULTICAST_LOOP " ) ;
2008-10-26 21:42:39 +00:00
2014-05-18 09:42:26 +00:00
return 0 ;
}
2008-10-26 21:42:39 +00:00
2014-05-18 09:42:26 +00:00
static inline int
sk_join_group4 ( sock * s , ip_addr maddr )
{
struct ip_mreqn mr = INIT_MREQ4 ( maddr , s - > iface ) ;
2014-03-31 11:21:13 +00:00
2014-05-18 09:42:26 +00:00
if ( setsockopt ( s - > fd , SOL_IP , IP_ADD_MEMBERSHIP , & mr , sizeof ( mr ) ) < 0 )
ERR ( " IP_ADD_MEMBERSHIP " ) ;
2008-10-26 21:42:39 +00:00
2014-05-18 09:42:26 +00:00
return 0 ;
}
2009-05-04 16:17:46 +00:00
2014-05-18 09:42:26 +00:00
static inline int
sk_leave_group4 ( sock * s , ip_addr maddr )
2009-05-04 16:17:46 +00:00
{
2014-05-18 09:42:26 +00:00
struct ip_mreqn mr = INIT_MREQ4 ( maddr , s - > iface ) ;
2009-05-04 16:17:46 +00:00
2014-05-18 09:42:26 +00:00
if ( setsockopt ( s - > fd , SOL_IP , IP_DROP_MEMBERSHIP , & mr , sizeof ( mr ) ) < 0 )
ERR ( " IP_DROP_MEMBERSHIP " ) ;
2009-05-04 16:17:46 +00:00
2014-05-18 09:42:26 +00:00
return 0 ;
2009-05-04 16:17:46 +00:00
}
2010-02-11 09:23:35 +00:00
2014-05-18 09:42:26 +00:00
/*
* Linux IPv4 packet control messages
*/
2010-02-11 09:23:35 +00:00
/* Mostly similar to standardized IPv6 code */
2014-05-18 09:42:26 +00:00
# define CMSG4_SPACE_PKTINFO CMSG_SPACE(sizeof(struct in_pktinfo))
# define CMSG4_SPACE_TTL CMSG_SPACE(sizeof(int))
2010-02-11 09:23:35 +00:00
2014-05-18 09:42:26 +00:00
static inline int
sk_request_cmsg4_pktinfo ( sock * s )
2010-02-11 09:23:35 +00:00
{
2014-05-18 09:42:26 +00:00
int y = 1 ;
2013-06-25 13:33:00 +00:00
2014-05-18 09:42:26 +00:00
if ( setsockopt ( s - > fd , SOL_IP , IP_PKTINFO , & y , sizeof ( y ) ) < 0 )
ERR ( " IP_PKTINFO " ) ;
2010-02-11 09:23:35 +00:00
2014-05-18 09:42:26 +00:00
return 0 ;
2010-02-11 09:23:35 +00:00
}
2014-05-18 09:42:26 +00:00
static inline int
sk_request_cmsg4_ttl ( sock * s )
2010-02-11 09:23:35 +00:00
{
2014-05-18 09:42:26 +00:00
int y = 1 ;
2010-02-11 09:23:35 +00:00
2014-05-18 09:42:26 +00:00
if ( setsockopt ( s - > fd , SOL_IP , IP_RECVTTL , & y , sizeof ( y ) ) < 0 )
ERR ( " IP_RECVTTL " ) ;
2013-06-25 13:33:00 +00:00
2014-05-18 09:42:26 +00:00
return 0 ;
}
2013-06-25 13:33:00 +00:00
2014-05-18 09:42:26 +00:00
static inline void
sk_process_cmsg4_pktinfo ( sock * s , struct cmsghdr * cm )
{
if ( cm - > cmsg_type = = IP_PKTINFO )
2013-06-25 13:33:00 +00:00
{
2014-05-18 09:42:26 +00:00
struct in_pktinfo * pi = ( struct in_pktinfo * ) CMSG_DATA ( cm ) ;
s - > laddr = ipa_from_in4 ( pi - > ipi_addr ) ;
s - > lifindex = pi - > ipi_ifindex ;
2013-06-25 13:33:00 +00:00
}
2014-05-18 09:42:26 +00:00
}
2013-06-25 13:33:00 +00:00
2014-05-18 09:42:26 +00:00
static inline void
sk_process_cmsg4_ttl ( sock * s , struct cmsghdr * cm )
{
if ( cm - > cmsg_type = = IP_TTL )
s - > rcv_ttl = * ( int * ) CMSG_DATA ( cm ) ;
2010-02-11 09:23:35 +00:00
}
2014-05-18 09:42:26 +00:00
static inline void
sk_prepare_cmsgs4 ( sock * s , struct msghdr * msg , void * cbuf , size_t cbuflen )
2010-02-11 09:23:35 +00:00
{
struct cmsghdr * cm ;
struct in_pktinfo * pi ;
2014-06-26 11:30:27 +00:00
int controllen = 0 ;
2010-02-11 09:23:35 +00:00
2010-04-02 09:31:20 +00:00
msg - > msg_control = cbuf ;
msg - > msg_controllen = cbuflen ;
2010-02-11 09:23:35 +00:00
cm = CMSG_FIRSTHDR ( msg ) ;
2014-02-06 16:46:01 +00:00
cm - > cmsg_level = SOL_IP ;
2010-02-11 09:23:35 +00:00
cm - > cmsg_type = IP_PKTINFO ;
cm - > cmsg_len = CMSG_LEN ( sizeof ( * pi ) ) ;
2014-06-26 11:30:27 +00:00
controllen + = CMSG_SPACE ( sizeof ( * pi ) ) ;
2010-02-11 09:23:35 +00:00
pi = ( struct in_pktinfo * ) CMSG_DATA ( cm ) ;
pi - > ipi_ifindex = s - > iface ? s - > iface - > index : 0 ;
2014-05-18 09:42:26 +00:00
pi - > ipi_spec_dst = ipa_to_in4 ( s - > saddr ) ;
pi - > ipi_addr = ipa_to_in4 ( IPA_NONE ) ;
2010-02-11 09:23:35 +00:00
2014-06-26 11:30:27 +00:00
msg - > msg_controllen = controllen ;
2010-02-11 09:23:35 +00:00
}
2014-02-06 16:46:01 +00:00
2010-04-07 21:15:56 +00:00
2014-05-18 09:42:26 +00:00
/*
* Miscellaneous Linux socket syscalls
*/
2024-03-07 11:25:58 +00:00
2014-05-18 09:42:26 +00:00
int
2024-02-14 12:48:08 +00:00
sk_set_md5_auth ( sock * s , ip_addr local UNUSED , ip_addr remote , int pxlen , struct iface * ifa , const char * passwd , int setkey UNUSED )
2014-05-18 09:42:26 +00:00
{
2020-02-27 15:16:48 +00:00
struct tcp_md5sig_ext md5 ;
2011-08-16 21:05:35 +00:00
2014-05-18 09:42:26 +00:00
memset ( & md5 , 0 , sizeof ( md5 ) ) ;
2016-04-13 12:30:28 +00:00
sockaddr_fill ( ( sockaddr * ) & md5 . tcpm_addr , s - > af , remote , ifa , 0 ) ;
2011-08-16 21:05:35 +00:00
2014-05-18 09:42:26 +00:00
if ( passwd )
{
int len = strlen ( passwd ) ;
if ( len > TCP_MD5SIG_MAXKEYLEN )
2016-04-13 12:30:28 +00:00
ERR_MSG ( " The password for TCP MD5 Signature is too long " ) ;
2011-08-16 21:05:35 +00:00
2014-05-18 09:42:26 +00:00
md5 . tcpm_keylen = len ;
memcpy ( & md5 . tcpm_key , passwd , len ) ;
}
2024-02-05 13:43:46 +00:00
if ( pxlen < 0 )
{
2024-03-07 11:25:58 +00:00
if ( setsockopt ( s - > fd , SOL_TCP , TCP_MD5SIG , & md5 , sizeof ( md5 ) ) < 0 )
2024-02-05 13:43:46 +00:00
if ( errno = = ENOPROTOOPT )
ERR_MSG ( " Kernel does not support TCP MD5 signatures " ) ;
else
2024-02-14 12:48:08 +00:00
ERR ( " TCP_MD5SIG " ) ;
2024-02-05 13:43:46 +00:00
}
else
{
md5 . tcpm_flags = TCP_MD5SIG_FLAG_PREFIX ;
md5 . tcpm_prefixlen = pxlen ;
if ( setsockopt ( s - > fd , SOL_TCP , TCP_MD5SIG_EXT , & md5 , sizeof ( md5 ) ) < 0 )
{
if ( errno = = ENOPROTOOPT )
ERR_MSG ( " Kernel does not support extended TCP MD5 signatures " ) ;
else
ERR ( " TCP_MD5SIG_EXT " ) ;
}
}
return 0 ;
}
2024-02-14 12:48:08 +00:00
void log_tcp_ao_info ( int sock_fd )
2024-02-05 13:43:46 +00:00
{
2024-02-14 12:48:08 +00:00
struct tcp_ao_info_opt_ext tmp ;
memset ( & tmp , 0 , sizeof ( struct tcp_ao_info_opt_ext ) ) ;
socklen_t len = sizeof ( tmp ) ;
log ( " socket: fd %i " , sock_fd ) ;
2024-02-05 13:43:46 +00:00
2024-02-14 12:48:08 +00:00
if ( getsockopt ( sock_fd , IPPROTO_TCP , TCP_AO_INFO , & tmp , & len ) )
2024-02-05 13:43:46 +00:00
{
2024-03-07 11:25:58 +00:00
log ( L_WARN " TCP AO: log tcp ao info failed with err code %i " , errno ) ;
return ;
2024-02-05 13:43:46 +00:00
}
2024-02-14 12:48:08 +00:00
else
2024-03-07 11:25:58 +00:00
log ( L_INFO " TCP AO: \n current key id %i (rem), next key %i (loc), \n set current %i, is ao required %i \n good packets %i, bad packets %i " ,
2024-02-19 11:09:36 +00:00
tmp . current_key , tmp . rnext , tmp . set_current , tmp . ao_required , tmp . pkt_good , tmp . pkt_bad ) ;
}
2024-02-23 08:33:08 +00:00
int get_current_key_id ( int sock_fd )
{
struct tcp_ao_info_opt_ext tmp ;
memset ( & tmp , 0 , sizeof ( struct tcp_ao_info_opt_ext ) ) ;
socklen_t len = sizeof ( tmp ) ;
if ( getsockopt ( sock_fd , IPPROTO_TCP , TCP_AO_INFO , & tmp , & len ) )
{
2024-03-07 11:25:58 +00:00
log ( L_WARN " TCP AO: Getting current ao key for socket file descriptor %i failed with errno %i " , sock_fd , errno ) ;
2024-02-23 08:33:08 +00:00
return - 1 ;
}
else
return tmp . current_key ;
}
2024-02-26 13:58:25 +00:00
int get_rnext_key_id ( int sock_fd )
{
struct tcp_ao_info_opt_ext tmp ;
memset ( & tmp , 0 , sizeof ( struct tcp_ao_info_opt_ext ) ) ;
socklen_t len = sizeof ( tmp ) ;
if ( getsockopt ( sock_fd , IPPROTO_TCP , TCP_AO_INFO , & tmp , & len ) )
{
2024-03-07 11:25:58 +00:00
log ( L_WARN " TCP AO: Getting rnext ao key for socket file descriptor %i failed with errno %i " , sock_fd , errno ) ;
return - 1 ;
2024-02-26 13:58:25 +00:00
}
else
return tmp . rnext ;
}
2024-02-28 15:15:28 +00:00
int get_num_ao_keys ( int sock_fd )
2024-02-19 11:09:36 +00:00
{
struct tcp_ao_getsockopt_ext tmp ;
memset ( & tmp , 0 , sizeof ( struct tcp_ao_getsockopt_ext ) ) ;
socklen_t len = sizeof ( tmp ) ;
tmp . nkeys = 1 ;
tmp . get_all = 1 ;
if ( getsockopt ( sock_fd , IPPROTO_TCP , TCP_AO_GET_KEYS , & tmp , & len ) )
{
2024-03-07 11:25:58 +00:00
log ( L_WARN " TCP AO: get keys on socket fd %i failed with err code %i " , sock_fd , errno ) ;
2024-02-28 15:15:28 +00:00
return - 1 ;
2024-02-19 11:09:36 +00:00
}
2024-02-28 15:15:28 +00:00
return tmp . nkeys ;
}
void
log_tcp_ao_get_key ( int sock_fd )
2024-03-07 11:25:58 +00:00
{
2024-02-28 15:15:28 +00:00
int nkeys = get_num_ao_keys ( sock_fd ) ;
if ( nkeys < 0 )
return ;
2024-02-23 08:33:08 +00:00
struct tcp_ao_getsockopt_ext tm_all [ nkeys ] ;
2024-02-28 15:15:28 +00:00
socklen_t len = sizeof ( struct tcp_ao_getsockopt_ext ) ;
2024-02-23 08:33:08 +00:00
memset ( tm_all , 0 , sizeof ( struct tcp_ao_getsockopt_ext ) * nkeys ) ;
tm_all [ 0 ] . nkeys = nkeys ;
tm_all [ 0 ] . get_all = 1 ;
if ( getsockopt ( sock_fd , IPPROTO_TCP , TCP_AO_GET_KEYS , tm_all , & len ) ) // len should be still size of one struct. Because kernel net/ipv4/tcp_ao.c line 2165
{
2024-03-07 11:25:58 +00:00
log ( L_WARN " TCP AO: getting keys on socket fd %i failed with err code %i " , sock_fd , errno ) ;
2024-02-23 08:33:08 +00:00
return ;
}
2024-03-07 11:25:58 +00:00
log ( L_INFO " TCP AO on socket fd %i has %i keys " , tm_all [ 0 ] . nkeys ) ;
2024-02-23 08:33:08 +00:00
for ( int i = 0 ; i < nkeys ; i + + )
{
2024-03-07 11:25:58 +00:00
log ( L_INFO " sndid %i rcvid %i, %s %s, cipher %s key %s (%i/%i) " , tm_all [ i ] . sndid , tm_all [ i ] . rcvid , tm_all [ i ] . is_current ? " current " : " " , tm_all [ i ] . is_rnext ? " rnext " : " " , tm_all [ i ] . alg_name , tm_all [ i ] . key , i + 1 , tm_all [ 0 ] . nkeys ) ;
}
}
void
tcp_ao_get_info ( int sock_fd , int key_info [ 4 ] )
{
struct tcp_ao_info_opt_ext tmp ;
memset ( & tmp , 0 , sizeof ( struct tcp_ao_info_opt_ext ) ) ;
socklen_t len = sizeof ( tmp ) ;
if ( getsockopt ( sock_fd , IPPROTO_TCP , TCP_AO_INFO , & tmp , & len ) )
{
log ( L_WARN " TCP AO: log tcp ao info failed with err code %i " , errno ) ;
return ;
2024-02-23 08:33:08 +00:00
}
2024-03-07 11:25:58 +00:00
key_info [ 0 ] = tmp . current_key ;
key_info [ 1 ] = tmp . rnext ;
key_info [ 2 ] = tmp . pkt_good ;
key_info [ 3 ] = tmp . pkt_bad ;
2024-02-14 12:48:08 +00:00
}
2024-02-05 13:43:46 +00:00
2024-02-14 12:48:08 +00:00
int
2024-02-28 15:15:28 +00:00
sk_set_ao_auth ( sock * s , ip_addr local UNUSED , ip_addr remote , int pxlen , struct iface * ifa , const char * passwd , int passwd_id_loc , int passwd_id_rem , const char * cipher , int set_current )
2024-02-14 12:48:08 +00:00
{
struct tcp_ao_add_ext ao ;
memset ( & ao , 0 , sizeof ( struct tcp_ao_add_ext ) ) ;
2024-03-07 11:25:58 +00:00
log ( L_DEBUG " tcp ao: socket sets ao, password %s socket fd %i " , passwd , s - > fd ) ;
2024-02-14 12:48:08 +00:00
sockaddr_fill ( ( sockaddr * ) & ao . addr , s - > af , remote , ifa , 0 ) ;
2024-02-23 08:33:08 +00:00
if ( set_current )
{
ao . set_rnext = 1 ;
ao . set_current = 1 ;
}
2024-02-14 12:48:08 +00:00
if ( pxlen > = 0 )
2024-02-23 08:33:08 +00:00
ao . prefix = pxlen ;
2024-02-14 12:48:08 +00:00
else if ( s - > af = = AF_INET )
ao . prefix = 32 ;
else
ao . prefix = 128 ;
ao . sndid = passwd_id_rem ;
ao . rcvid = passwd_id_loc ;
ao . maclen = 0 ;
ao . keyflags = 0 ;
ao . ifindex = 0 ;
2024-02-19 11:09:36 +00:00
strncpy ( ao . alg_name , ( cipher ) ? cipher : DEFAULT_TEST_ALGO , 64 ) ;
2024-03-07 11:25:58 +00:00
ao . keylen = strlen ( passwd ) ;
2024-02-14 12:48:08 +00:00
memcpy ( ao . key , passwd , ( strlen ( passwd ) > TCP_AO_MAXKEYLEN_ ) ? TCP_AO_MAXKEYLEN_ : strlen ( passwd ) ) ;
2024-02-26 13:58:25 +00:00
2024-02-14 12:48:08 +00:00
if ( setsockopt ( s - > fd , IPPROTO_TCP , TCP_AO_ADD_KEY , & ao , sizeof ( ao ) ) < 0 )
2024-02-28 15:15:28 +00:00
{
2024-03-07 11:25:58 +00:00
if ( errno = = ENOPROTOOPT )
ERR_MSG ( " Kernel does not support extended TCP AO signatures " ) ;
else
ERR ( " TCP_AOSIG_EXT " ) ;
2024-02-28 15:15:28 +00:00
}
s - > use_ao = 1 ;
if ( set_current )
s - > desired_ao_key = passwd_id_rem ;
2024-02-23 08:33:08 +00:00
log_tcp_ao_get_key ( s - > fd ) ;
2024-02-14 12:48:08 +00:00
return 0 ;
}
2020-02-27 15:16:48 +00:00
2024-03-07 11:25:58 +00:00
int
2024-02-23 08:33:08 +00:00
ao_delete_key ( sock * s , ip_addr remote , int pxlen , struct iface * ifa , int passwd_id_loc , int passwd_id_rem )
{
struct tcp_ao_del_ext del ;
memset ( & del , 0 , sizeof ( struct tcp_ao_del_ext ) ) ;
sockaddr_fill ( ( sockaddr * ) & del . addr , s - > af , remote , ifa , 0 ) ;
del . sndid = passwd_id_rem ;
del . rcvid = passwd_id_loc ;
if ( pxlen > = 0 )
del . prefix = pxlen ;
else if ( s - > af = = AF_INET )
del . prefix = 32 ;
else
del . prefix = 128 ;
2024-02-26 13:58:25 +00:00
2024-02-23 08:33:08 +00:00
if ( setsockopt ( s - > fd , IPPROTO_TCP , TCP_AO_DEL_KEY , & del , sizeof ( del ) ) < 0 )
2024-02-28 15:15:28 +00:00
{
2024-03-07 11:25:58 +00:00
log ( L_WARN " TCP AO: deletion of key %i %i on socket fd %i failed with err %i " , passwd_id_loc , passwd_id_rem , s - > fd , errno ) ;
return errno ;
2024-02-28 15:15:28 +00:00
}
2024-03-07 11:25:58 +00:00
log ( L_DEBUG " tcp ao: key %i %i deleted " , passwd_id_loc , passwd_id_rem ) ;
return 0 ;
2024-02-23 08:33:08 +00:00
}
2024-02-19 11:09:36 +00:00
void
2024-02-28 15:15:28 +00:00
ao_try_change_master ( sock * s , int next_master_id_loc , int next_master_id_rem )
2024-02-19 11:09:36 +00:00
{
struct tcp_ao_info_opt_ext tmp ;
memset ( & tmp , 0 , sizeof ( struct tcp_ao_info_opt_ext ) ) ;
tmp . set_rnext = 1 ;
2024-02-28 15:15:28 +00:00
tmp . rnext = next_master_id_loc ;
2024-02-19 11:09:36 +00:00
2024-02-28 15:15:28 +00:00
if ( setsockopt ( s - > fd , IPPROTO_TCP , TCP_AO_INFO , & tmp , sizeof ( tmp ) ) )
2024-02-19 11:09:36 +00:00
{
2024-03-07 11:25:58 +00:00
log ( L_WARN " TCP AO: change master key failed with err code %i " , errno ) ;
2024-02-28 15:15:28 +00:00
log_tcp_ao_get_key ( s - > fd ) ;
2024-02-19 11:09:36 +00:00
return ;
}
else
2024-03-07 11:25:58 +00:00
log ( L_DEBUG " tcp ao: tried to change master to %i %i " , next_master_id_loc , next_master_id_rem ) ;
2024-02-28 15:15:28 +00:00
s - > desired_ao_key = next_master_id_rem ;
2024-02-19 11:09:36 +00:00
2024-02-14 12:48:08 +00:00
}
2024-03-07 11:25:58 +00:00
int check_ao_keys_id ( int sock_fd , struct bgp_ao_key * keys )
2024-02-14 12:48:08 +00:00
{
2024-02-28 15:15:28 +00:00
int errors = 0 ;
int expected_keys [ 256 ] ; //can not have char, because we must support 0 key id
memset ( expected_keys , 0 , sizeof ( int ) * 256 ) ;
2024-03-07 11:25:58 +00:00
for ( struct bgp_ao_key * key = keys ; key ; key = key - > next_key )
expected_keys [ key - > key . local_id ] = key - > key . remote_id + 1 ; // the + 1 because we do not want 0 id be 0
2024-02-28 15:15:28 +00:00
int nkeys = get_num_ao_keys ( sock_fd ) ;
if ( nkeys = = - 1 )
{
2024-03-07 11:25:58 +00:00
log ( L_WARN " TCP AO: unable to get num of keys " ) ;
2024-02-28 15:15:28 +00:00
return 1 ;
}
struct tcp_ao_getsockopt_ext tm_all [ nkeys ] ;
socklen_t len = sizeof ( struct tcp_ao_getsockopt_ext ) ;
memset ( tm_all , 0 , sizeof ( struct tcp_ao_getsockopt_ext ) * nkeys ) ;
tm_all [ 0 ] . nkeys = nkeys ;
tm_all [ 0 ] . get_all = 1 ;
if ( getsockopt ( sock_fd , IPPROTO_TCP , TCP_AO_GET_KEYS , tm_all , & len ) ) // len should be still size of one struct. Because kernel net/ipv4/tcp_ao.c line 2165
{
2024-03-07 11:25:58 +00:00
log ( L_WARN " TCP AO: log tcp ao get keys failed with err code %i " , errno ) ;
return 1 ;
2024-02-28 15:15:28 +00:00
}
for ( int i = 0 ; i < nkeys ; i + + )
{
struct tcp_ao_getsockopt_ext sock_key = tm_all [ i ] ;
if ( expected_keys [ sock_key . rcvid ] - 1 ! = sock_key . sndid )
{
2024-03-07 11:25:58 +00:00
if ( expected_keys [ sock_key . rcvid ] = = 0 )
log ( L_WARN " TCP AO: unexpected ao key %i %i " , sock_key . rcvid , sock_key . sndid ) ;
2024-02-28 15:15:28 +00:00
else
2024-03-07 11:25:58 +00:00
log ( L_WARN " TCP AO: expected key local id %i has different remote id than expected (%i vs %i) " , sock_key . rcvid , expected_keys [ sock_key . rcvid ] - 1 , sock_key . sndid ) ;
2024-02-28 15:15:28 +00:00
errors + + ;
2024-03-07 11:25:58 +00:00
}
expected_keys [ sock_key . rcvid ] = 0 ;
2024-02-28 15:15:28 +00:00
}
for ( int i = 0 ; i < 256 ; i + + )
{
if ( expected_keys [ i ] ! = 0 )
{
2024-03-07 11:25:58 +00:00
log ( L_WARN " TCP AO: key %i %i is not in socket " , i , expected_keys - 1 ) ;
2024-02-28 15:15:28 +00:00
errors + + ;
}
}
return errors ;
2024-02-14 12:48:08 +00:00
}
2014-05-18 09:42:26 +00:00
static inline int
2011-08-16 21:05:35 +00:00
sk_set_min_ttl4 ( sock * s , int ttl )
{
2014-02-06 16:46:01 +00:00
if ( setsockopt ( s - > fd , SOL_IP , IP_MINTTL , & ttl , sizeof ( ttl ) ) < 0 )
2011-08-16 21:05:35 +00:00
{
if ( errno = = ENOPROTOOPT )
2014-05-18 09:42:26 +00:00
ERR_MSG ( " Kernel does not support IPv4 TTL security " ) ;
2011-08-16 21:05:35 +00:00
else
2014-05-18 09:42:26 +00:00
ERR ( " IP_MINTTL " ) ;
2011-08-16 21:05:35 +00:00
}
return 0 ;
}
2014-05-18 09:42:26 +00:00
static inline int
2011-08-16 21:05:35 +00:00
sk_set_min_ttl6 ( sock * s , int ttl )
{
2014-02-06 16:46:01 +00:00
if ( setsockopt ( s - > fd , SOL_IPV6 , IPV6_MINHOPCOUNT , & ttl , sizeof ( ttl ) ) < 0 )
2011-08-16 21:05:35 +00:00
{
if ( errno = = ENOPROTOOPT )
2014-05-18 09:42:26 +00:00
ERR_MSG ( " Kernel does not support IPv6 TTL security " ) ;
2011-08-16 21:05:35 +00:00
else
2014-05-18 09:42:26 +00:00
ERR ( " IPV6_MINHOPCOUNT " ) ;
2011-08-16 21:05:35 +00:00
}
return 0 ;
}
2014-05-18 09:42:26 +00:00
static inline int
sk_disable_mtu_disc4 ( sock * s )
{
int dont = IP_PMTUDISC_DONT ;
2013-06-24 14:37:30 +00:00
2014-05-18 09:42:26 +00:00
if ( setsockopt ( s - > fd , SOL_IP , IP_MTU_DISCOVER , & dont , sizeof ( dont ) ) < 0 )
ERR ( " IP_MTU_DISCOVER " ) ;
2013-06-24 14:37:30 +00:00
2014-05-18 09:42:26 +00:00
return 0 ;
}
static inline int
sk_disable_mtu_disc6 ( sock * s )
{
int dont = IPV6_PMTUDISC_DONT ;
if ( setsockopt ( s - > fd , SOL_IPV6 , IPV6_MTU_DISCOVER , & dont , sizeof ( dont ) ) < 0 )
ERR ( " IPV6_MTU_DISCOVER " ) ;
return 0 ;
}
2013-06-24 14:37:30 +00:00
int sk_priority_control = 7 ;
2014-05-18 09:42:26 +00:00
static inline int
2013-06-24 14:37:30 +00:00
sk_set_priority ( sock * s , int prio )
{
if ( setsockopt ( s - > fd , SOL_SOCKET , SO_PRIORITY , & prio , sizeof ( prio ) ) < 0 )
2014-05-18 09:42:26 +00:00
ERR ( " SO_PRIORITY " ) ;
2013-06-24 14:37:30 +00:00
return 0 ;
}
2014-05-18 09:42:26 +00:00
2022-01-08 17:31:56 +00:00
static inline int
sk_set_freebind ( sock * s )
{
int y = 1 ;
if ( sk_is_ipv4 ( s ) )
if ( setsockopt ( s - > fd , SOL_IP , IP_FREEBIND , & y , sizeof ( y ) ) < 0 )
ERR ( " IP_FREEBIND " ) ;
if ( sk_is_ipv6 ( s ) )
if ( setsockopt ( s - > fd , SOL_IPV6 , IPV6_FREEBIND , & y , sizeof ( y ) ) < 0 )
ERR ( " IPV6_FREEBIND " ) ;
return 0 ;
}