2007-01-23 08:46:18 -05:00
/*
* util / netevent . c - event notification
*
* Copyright ( c ) 2007 , NLnet Labs . All rights reserved .
*
2007-01-25 07:39:51 -05:00
* This software is open source .
*
* Redistribution and use in source and binary forms , with or without
* modification , are permitted provided that the following conditions
* are met :
*
* Redistributions of source code must retain the above copyright notice ,
* this list of conditions and the following disclaimer .
*
* Redistributions in binary form must reproduce the above copyright notice ,
* this list of conditions and the following disclaimer in the documentation
* and / or other materials provided with the distribution .
*
* Neither the name of the NLNET LABS nor the names of its contributors may
* be used to endorse or promote products derived from this software without
* specific prior written permission .
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
2014-02-07 08:28:39 -05:00
* " AS IS " AND ANY EXPRESS OR IMPLIED WARRANTIES , INCLUDING , BUT NOT
* LIMITED TO , THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED . IN NO EVENT SHALL THE COPYRIGHT
* HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT , INDIRECT , INCIDENTAL ,
* SPECIAL , EXEMPLARY , OR CONSEQUENTIAL DAMAGES ( INCLUDING , BUT NOT LIMITED
* TO , PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES ; LOSS OF USE , DATA , OR
* PROFITS ; OR BUSINESS INTERRUPTION ) HOWEVER CAUSED AND ON ANY THEORY OF
* LIABILITY , WHETHER IN CONTRACT , STRICT LIABILITY , OR TORT ( INCLUDING
* NEGLIGENCE OR OTHERWISE ) ARISING IN ANY WAY OUT OF THE USE OF THIS
* SOFTWARE , EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE .
2007-01-23 08:46:18 -05:00
*/
/**
* \ file
*
* This file contains event notification functions .
*/
2009-03-24 06:42:57 -04:00
# include "config.h"
2007-01-23 08:46:18 -05:00
# include "util/netevent.h"
2016-03-07 09:10:06 -05:00
# include "util/ub_event.h"
2007-01-23 08:46:18 -05:00
# include "util/log.h"
2008-01-30 09:46:01 -05:00
# include "util/net_help.h"
2018-08-07 07:57:42 -04:00
# include "util/tcp_conn_limit.h"
2007-10-04 11:10:11 -04:00
# include "util/fptr_wlist.h"
2015-03-26 06:21:38 -04:00
# include "sldns/pkthdr.h"
# include "sldns/sbuffer.h"
2018-04-05 04:10:25 -04:00
# include "sldns/str2wire.h"
2014-08-05 03:57:52 -04:00
# include "dnstap/dnstap.h"
2017-03-20 10:55:31 -04:00
# include "dnscrypt/dnscrypt.h"
2019-01-11 09:12:27 -05:00
# include "services/listen_dnsport.h"
2020-12-09 05:00:51 -05:00
# ifdef HAVE_SYS_TYPES_H
# include <sys/types.h>
# endif
# ifdef HAVE_SYS_SOCKET_H
# include <sys/socket.h>
# endif
# ifdef HAVE_NETDB_H
# include <netdb.h>
# endif
2012-06-20 11:11:53 -04:00
# ifdef HAVE_OPENSSL_SSL_H
2011-10-31 10:48:48 -04:00
# include <openssl/ssl.h>
2012-06-20 11:11:53 -04:00
# endif
# ifdef HAVE_OPENSSL_ERR_H
2011-10-31 10:48:48 -04:00
# include <openssl/err.h>
2012-06-20 11:11:53 -04:00
# endif
2007-01-23 08:46:18 -05:00
2007-01-23 11:10:23 -05:00
/* -------- Start of local definitions -------- */
2010-04-16 04:41:08 -04:00
/** if CMSG_ALIGN is not defined on this platform, a workaround */
# ifndef CMSG_ALIGN
2016-01-26 09:11:39 -05:00
# ifdef __CMSG_ALIGN
# define CMSG_ALIGN(n) __CMSG_ALIGN(n)
# elif defined(CMSG_DATA_ALIGN)
2010-04-16 04:41:08 -04:00
# define CMSG_ALIGN _CMSG_DATA_ALIGN
# else
# define CMSG_ALIGN(len) (((len)+sizeof(long)-1) & ~(sizeof(long)-1))
# endif
# endif
/** if CMSG_LEN is not defined on this platform, a workaround */
# ifndef CMSG_LEN
# define CMSG_LEN(len) (CMSG_ALIGN(sizeof(struct cmsghdr))+(len))
# endif
/** if CMSG_SPACE is not defined on this platform, a workaround */
# ifndef CMSG_SPACE
# ifdef _CMSG_HDR_ALIGN
# define CMSG_SPACE(l) (CMSG_ALIGN(l)+_CMSG_HDR_ALIGN(sizeof(struct cmsghdr)))
# else
# define CMSG_SPACE(l) (CMSG_ALIGN(l)+CMSG_ALIGN(sizeof(struct cmsghdr)))
# endif
# endif
2018-07-31 03:15:12 -04:00
/** The TCP writing query timeout in milliseconds */
2016-06-15 10:23:43 -04:00
# define TCP_QUERY_TIMEOUT 120000
2018-07-31 09:36:45 -04:00
/** The minimum actual TCP timeout to use, regardless of what we advertise,
* in msec */
2018-07-31 03:18:34 -04:00
# define TCP_QUERY_TIMEOUT_MINIMUM 200
2007-02-07 09:18:42 -05:00
2008-11-07 09:28:06 -05:00
# ifndef NONBLOCKING_IS_BROKEN
2008-02-19 09:35:40 -05:00
/** number of UDP reads to perform per read indication from select */
# define NUM_UDP_PER_SELECT 100
2008-11-07 09:28:06 -05:00
# else
# define NUM_UDP_PER_SELECT 1
# endif
2008-02-19 09:35:40 -05:00
2007-01-23 08:46:18 -05:00
/**
2016-03-07 09:10:06 -05:00
* The internal event structure for keeping ub_event info for the event .
2007-01-23 08:46:18 -05:00
* Possibly other structures ( list , tree ) this is part of .
*/
struct internal_event {
2008-02-19 08:12:23 -05:00
/** the comm base */
struct comm_base * base ;
2016-03-07 09:10:06 -05:00
/** ub_event event type */
struct ub_event * ev ;
2007-01-23 08:46:18 -05:00
} ;
/**
* Internal base structure , so that every thread has its own events .
*/
struct internal_base {
2016-03-07 09:10:06 -05:00
/** ub_event event_base type. */
struct ub_event_base * base ;
2008-02-19 08:12:23 -05:00
/** seconds time pointer points here */
2013-08-20 08:23:42 -04:00
time_t secs ;
2008-02-19 08:12:23 -05:00
/** timeval with current time */
struct timeval now ;
2012-05-08 08:08:55 -04:00
/** the event used for slow_accept timeouts */
2016-03-07 09:10:06 -05:00
struct ub_event * slow_accept ;
2012-05-08 08:08:55 -04:00
/** true if slow_accept is enabled */
int slow_accept_enabled ;
2007-01-23 08:46:18 -05:00
} ;
2007-02-01 10:06:38 -05:00
/**
* Internal timer structure , to store timer event in .
*/
struct internal_timer {
2016-03-07 09:10:06 -05:00
/** the super struct from which derived */
struct comm_timer super ;
2008-02-19 08:12:23 -05:00
/** the comm base */
struct comm_base * base ;
2016-03-07 09:10:06 -05:00
/** ub_event event type */
struct ub_event * ev ;
2007-02-06 09:00:52 -05:00
/** is timer enabled */
2007-02-01 10:06:38 -05:00
uint8_t enabled ;
} ;
2007-02-05 11:46:40 -05:00
/**
* Internal signal structure , to store signal event in .
*/
struct internal_signal {
2016-03-07 09:10:06 -05:00
/** ub_event event type */
struct ub_event * ev ;
2007-02-05 11:46:40 -05:00
/** next in signal list */
struct internal_signal * next ;
} ;
2007-01-23 11:10:23 -05:00
/** create a tcp handler with a parent */
static struct comm_point * comm_point_create_tcp_handler (
struct comm_base * base , struct comm_point * parent , size_t bufsize ,
2019-01-11 09:12:27 -05:00
struct sldns_buffer * spoolbuf , comm_point_callback_type * callback ,
2020-12-09 05:00:51 -05:00
void * callback_arg , struct unbound_socket * socket ) ;
2007-01-23 11:10:23 -05:00
/* -------- End of local definitions -------- */
struct comm_base *
2009-01-05 10:05:33 -05:00
comm_base_create ( int sigs )
2007-01-23 08:46:18 -05:00
{
struct comm_base * b = ( struct comm_base * ) calloc ( 1 ,
sizeof ( struct comm_base ) ) ;
2016-03-07 09:10:06 -05:00
const char * evnm = " event " , * evsys = " " , * evmethod = " " ;
2007-01-23 08:46:18 -05:00
if ( ! b )
return NULL ;
b - > eb = ( struct internal_base * ) calloc ( 1 , sizeof ( struct internal_base ) ) ;
if ( ! b - > eb ) {
free ( b ) ;
return NULL ;
}
2016-03-07 09:10:06 -05:00
b - > eb - > base = ub_default_event_base ( sigs , & b - > eb - > secs , & b - > eb - > now ) ;
2007-01-23 08:46:18 -05:00
if ( ! b - > eb - > base ) {
free ( b - > eb ) ;
free ( b ) ;
return NULL ;
}
2016-03-07 09:10:06 -05:00
ub_comm_base_now ( b ) ;
ub_get_event_sys ( b - > eb - > base , & evnm , & evsys , & evmethod ) ;
2019-04-04 10:28:39 -04:00
verbose ( VERB_ALGO , " %s %s uses %s method. " , evnm , evsys , evmethod ) ;
2007-01-23 08:46:18 -05:00
return b ;
}
2014-02-18 08:54:19 -05:00
struct comm_base *
2016-03-07 09:10:06 -05:00
comm_base_create_event ( struct ub_event_base * base )
2013-09-26 04:34:14 -04:00
{
struct comm_base * b = ( struct comm_base * ) calloc ( 1 ,
sizeof ( struct comm_base ) ) ;
if ( ! b )
return NULL ;
b - > eb = ( struct internal_base * ) calloc ( 1 , sizeof ( struct internal_base ) ) ;
if ( ! b - > eb ) {
free ( b ) ;
return NULL ;
}
b - > eb - > base = base ;
2016-03-07 11:01:09 -05:00
ub_comm_base_now ( b ) ;
2013-09-26 04:34:14 -04:00
return b ;
}
2007-01-23 11:10:23 -05:00
void
comm_base_delete ( struct comm_base * b )
2007-01-23 08:46:18 -05:00
{
2007-02-27 04:41:34 -05:00
if ( ! b )
return ;
2012-05-08 08:08:55 -04:00
if ( b - > eb - > slow_accept_enabled ) {
2016-03-07 09:10:06 -05:00
if ( ub_event_del ( b - > eb - > slow_accept ) ! = 0 ) {
2012-05-08 08:08:55 -04:00
log_err ( " could not event_del slow_accept " ) ;
}
2016-03-07 09:10:06 -05:00
ub_event_free ( b - > eb - > slow_accept ) ;
2012-05-08 08:08:55 -04:00
}
2016-03-07 09:10:06 -05:00
ub_event_base_free ( b - > eb - > base ) ;
2007-01-23 08:46:18 -05:00
b - > eb - > base = NULL ;
free ( b - > eb ) ;
free ( b ) ;
}
2013-09-26 04:34:14 -04:00
void
comm_base_delete_no_base ( struct comm_base * b )
{
if ( ! b )
return ;
if ( b - > eb - > slow_accept_enabled ) {
2016-03-07 09:10:06 -05:00
if ( ub_event_del ( b - > eb - > slow_accept ) ! = 0 ) {
2013-09-26 04:34:14 -04:00
log_err ( " could not event_del slow_accept " ) ;
}
2016-03-07 09:10:06 -05:00
ub_event_free ( b - > eb - > slow_accept ) ;
2013-09-26 04:34:14 -04:00
}
b - > eb - > base = NULL ;
free ( b - > eb ) ;
free ( b ) ;
}
2008-02-19 08:12:23 -05:00
void
2013-08-20 08:23:42 -04:00
comm_base_timept ( struct comm_base * b , time_t * * tt , struct timeval * * tv )
2008-02-19 08:12:23 -05:00
{
* tt = & b - > eb - > secs ;
* tv = & b - > eb - > now ;
}
2007-01-23 11:10:23 -05:00
void
comm_base_dispatch ( struct comm_base * b )
2007-01-23 08:46:18 -05:00
{
int retval ;
2016-03-07 09:10:06 -05:00
retval = ub_event_base_dispatch ( b - > eb - > base ) ;
2016-03-11 07:57:09 -05:00
if ( retval < 0 ) {
2007-02-05 11:46:40 -05:00
fatal_exit ( " event_dispatch returned error %d, "
" errno is %s " , retval , strerror ( errno ) ) ;
}
}
void comm_base_exit ( struct comm_base * b )
{
2016-03-07 09:10:06 -05:00
if ( ub_event_base_loopexit ( b - > eb - > base ) ! = 0 ) {
2007-02-05 11:46:40 -05:00
log_err ( " Could not loopexit " ) ;
2007-01-23 08:46:18 -05:00
}
}
2012-05-08 08:08:55 -04:00
void comm_base_set_slow_accept_handlers ( struct comm_base * b ,
void ( * stop_acc ) ( void * ) , void ( * start_acc ) ( void * ) , void * arg )
{
b - > stop_accept = stop_acc ;
b - > start_accept = start_acc ;
b - > cb_arg = arg ;
}
2016-03-07 09:10:06 -05:00
struct ub_event_base * comm_base_internal ( struct comm_base * b )
2008-07-25 05:26:15 -04:00
{
return b - > eb - > base ;
}
2010-11-15 09:00:20 -05:00
/** see if errno for udp has to be logged or not uses globals */
static int
udp_send_errno_needs_log ( struct sockaddr * addr , socklen_t addrlen )
{
/* do not log transient errors (unless high verbosity) */
# if defined(ENETUNREACH) || defined(EHOSTDOWN) || defined(EHOSTUNREACH) || defined(ENETDOWN)
switch ( errno ) {
# ifdef ENETUNREACH
case ENETUNREACH :
# endif
# ifdef EHOSTDOWN
case EHOSTDOWN :
# endif
# ifdef EHOSTUNREACH
case EHOSTUNREACH :
# endif
# ifdef ENETDOWN
case ENETDOWN :
# endif
2021-04-12 05:18:23 -04:00
case EPERM :
2021-08-13 03:27:58 -04:00
case EACCES :
2010-11-15 09:00:20 -05:00
if ( verbosity < VERB_ALGO )
return 0 ;
default :
break ;
}
# endif
2013-06-25 03:03:03 -04:00
/* permission denied is gotten for every send if the
* network is disconnected ( on some OS ) , squelch it */
2016-01-19 10:37:54 -05:00
if ( ( ( errno = = EPERM )
# ifdef EADDRNOTAVAIL
/* 'Cannot assign requested address' also when disconnected */
| | ( errno = = EADDRNOTAVAIL )
# endif
2021-02-22 02:24:04 -05:00
) & & verbosity < VERB_ALGO )
2013-06-25 03:03:03 -04:00
return 0 ;
2018-03-05 10:45:34 -05:00
# ifdef EADDRINUSE
/* If SO_REUSEADDR is set, we could try to connect to the same server
* from the same source port twice . */
if ( errno = = EADDRINUSE & & verbosity < VERB_DETAIL )
return 0 ;
# endif
2010-11-15 09:00:20 -05:00
/* squelch errors where people deploy AAAA ::ffff:bla for
* authority servers , which we try for intranets . */
if ( errno = = EINVAL & & addr_is_ip4mapped (
( struct sockaddr_storage * ) addr , addrlen ) & &
verbosity < VERB_DETAIL )
return 0 ;
/* SO_BROADCAST sockopt can give access to 255.255.255.255,
* but a dns cache does not need it . */
if ( errno = = EACCES & & addr_is_broadcast (
( struct sockaddr_storage * ) addr , addrlen ) & &
verbosity < VERB_DETAIL )
return 0 ;
return 1 ;
}
2012-02-23 04:01:46 -05:00
int tcp_connect_errno_needs_log ( struct sockaddr * addr , socklen_t addrlen )
{
return udp_send_errno_needs_log ( addr , addrlen ) ;
}
2008-01-25 05:41:17 -05:00
/* send a UDP reply */
2007-02-01 10:06:38 -05:00
int
2013-12-03 04:11:16 -05:00
comm_point_send_udp_msg ( struct comm_point * c , sldns_buffer * packet ,
2020-12-16 11:11:41 -05:00
struct sockaddr * addr , socklen_t addrlen , int is_connected )
2007-02-01 10:06:38 -05:00
{
2007-01-31 04:32:30 -05:00
ssize_t sent ;
2007-10-16 08:26:09 -04:00
log_assert ( c - > fd ! = - 1 ) ;
2009-01-06 05:00:28 -05:00
# ifdef UNBOUND_DEBUG
2013-12-03 04:11:16 -05:00
if ( sldns_buffer_remaining ( packet ) = = 0 )
2009-01-16 04:48:40 -05:00
log_err ( " error: send empty UDP packet " ) ;
2009-01-06 05:00:28 -05:00
# endif
2020-12-16 11:11:41 -05:00
log_assert ( addr & & addrlen > 0 ) ;
if ( ! is_connected ) {
2020-11-25 03:55:01 -05:00
sent = sendto ( c - > fd , ( void * ) sldns_buffer_begin ( packet ) ,
sldns_buffer_remaining ( packet ) , 0 ,
addr , addrlen ) ;
} else {
sent = send ( c - > fd , ( void * ) sldns_buffer_begin ( packet ) ,
sldns_buffer_remaining ( packet ) , 0 ) ;
}
2016-02-15 04:54:52 -05:00
if ( sent = = - 1 ) {
/* try again and block, waiting for IO to complete,
* we want to send the answer , and we will wait for
* the ethernet interface buffer to have space . */
# ifndef USE_WINSOCK
if ( errno = = EAGAIN | |
# ifdef EWOULDBLOCK
errno = = EWOULDBLOCK | |
# endif
errno = = ENOBUFS ) {
# else
if ( WSAGetLastError ( ) = = WSAEINPROGRESS | |
WSAGetLastError ( ) = = WSAENOBUFS | |
WSAGetLastError ( ) = = WSAEWOULDBLOCK ) {
# endif
int e ;
fd_set_block ( c - > fd ) ;
2020-12-16 11:11:41 -05:00
if ( ! is_connected ) {
sent = sendto ( c - > fd , ( void * ) sldns_buffer_begin ( packet ) ,
sldns_buffer_remaining ( packet ) , 0 ,
addr , addrlen ) ;
} else {
sent = send ( c - > fd , ( void * ) sldns_buffer_begin ( packet ) ,
sldns_buffer_remaining ( packet ) , 0 ) ;
}
2016-02-15 04:54:52 -05:00
e = errno ;
fd_set_nonblock ( c - > fd ) ;
errno = e ;
}
}
2007-01-31 04:32:30 -05:00
if ( sent = = - 1 ) {
2010-11-15 09:00:20 -05:00
if ( ! udp_send_errno_needs_log ( addr , addrlen ) )
2010-04-23 02:48:49 -04:00
return 0 ;
2020-12-16 11:11:41 -05:00
if ( ! is_connected ) {
verbose ( VERB_OPS , " sendto failed: %s " , sock_strerror ( errno ) ) ;
} else {
verbose ( VERB_OPS , " send failed: %s " , sock_strerror ( errno ) ) ;
}
2021-01-06 06:35:22 -05:00
if ( addr )
log_addr ( VERB_OPS , " remote address is " ,
( struct sockaddr_storage * ) addr , addrlen ) ;
2007-02-01 10:06:38 -05:00
return 0 ;
2013-12-03 04:11:16 -05:00
} else if ( ( size_t ) sent ! = sldns_buffer_remaining ( packet ) ) {
2007-01-31 04:32:30 -05:00
log_err ( " sent %d in place of %d bytes " ,
2013-12-03 04:11:16 -05:00
( int ) sent , ( int ) sldns_buffer_remaining ( packet ) ) ;
2007-02-01 10:06:38 -05:00
return 0 ;
2007-01-31 04:32:30 -05:00
}
2007-02-01 10:06:38 -05:00
return 1 ;
2007-01-31 04:32:30 -05:00
}
2010-10-15 05:43:54 -04:00
# if defined(AF_INET6) && defined(IPV6_PKTINFO) && (defined(HAVE_RECVMSG) || defined(HAVE_SENDMSG))
2008-01-17 10:35:34 -05:00
/** print debug ancillary info */
2010-07-07 09:13:36 -04:00
static void p_ancil ( const char * str , struct comm_reply * r )
2008-01-17 10:35:34 -05:00
{
if ( r - > srctype ! = 4 & & r - > srctype ! = 6 ) {
log_info ( " %s: unknown srctype %d " , str , r - > srctype ) ;
return ;
}
2020-12-09 05:00:51 -05:00
2008-01-17 10:35:34 -05:00
if ( r - > srctype = = 6 ) {
2020-12-09 05:00:51 -05:00
# ifdef IPV6_PKTINFO
2008-01-17 10:35:34 -05:00
char buf [ 1024 ] ;
if ( inet_ntop ( AF_INET6 , & r - > pktinfo . v6info . ipi6_addr ,
buf , ( socklen_t ) sizeof ( buf ) ) = = 0 ) {
2014-01-24 08:23:45 -05:00
( void ) strlcpy ( buf , " (inet_ntop error) " , sizeof ( buf ) ) ;
2008-01-17 10:35:34 -05:00
}
buf [ sizeof ( buf ) - 1 ] = 0 ;
log_info ( " %s: %s %d " , str , buf , r - > pktinfo . v6info . ipi6_ifindex ) ;
2020-12-09 05:00:51 -05:00
# endif
2008-01-17 10:35:34 -05:00
} else if ( r - > srctype = = 4 ) {
2010-03-16 11:20:40 -04:00
# ifdef IP_PKTINFO
2008-01-17 10:35:34 -05:00
char buf1 [ 1024 ] , buf2 [ 1024 ] ;
if ( inet_ntop ( AF_INET , & r - > pktinfo . v4info . ipi_addr ,
buf1 , ( socklen_t ) sizeof ( buf1 ) ) = = 0 ) {
2014-01-24 08:23:45 -05:00
( void ) strlcpy ( buf1 , " (inet_ntop error) " , sizeof ( buf1 ) ) ;
2008-01-17 10:35:34 -05:00
}
buf1 [ sizeof ( buf1 ) - 1 ] = 0 ;
2012-02-13 05:42:22 -05:00
# ifdef HAVE_STRUCT_IN_PKTINFO_IPI_SPEC_DST
2008-01-17 10:35:34 -05:00
if ( inet_ntop ( AF_INET , & r - > pktinfo . v4info . ipi_spec_dst ,
buf2 , ( socklen_t ) sizeof ( buf2 ) ) = = 0 ) {
2014-01-24 08:23:45 -05:00
( void ) strlcpy ( buf2 , " (inet_ntop error) " , sizeof ( buf2 ) ) ;
2008-01-17 10:35:34 -05:00
}
buf2 [ sizeof ( buf2 ) - 1 ] = 0 ;
2012-02-13 05:42:22 -05:00
# else
buf2 [ 0 ] = 0 ;
# endif
2008-01-17 10:35:34 -05:00
log_info ( " %s: %d %s %s " , str , r - > pktinfo . v4info . ipi_ifindex ,
buf1 , buf2 ) ;
2010-03-16 11:20:40 -04:00
# elif defined(IP_RECVDSTADDR)
char buf1 [ 1024 ] ;
if ( inet_ntop ( AF_INET , & r - > pktinfo . v4addr ,
buf1 , ( socklen_t ) sizeof ( buf1 ) ) = = 0 ) {
2014-01-24 08:23:45 -05:00
( void ) strlcpy ( buf1 , " (inet_ntop error) " , sizeof ( buf1 ) ) ;
2010-03-16 11:20:40 -04:00
}
buf1 [ sizeof ( buf1 ) - 1 ] = 0 ;
log_info ( " %s: %s " , str , buf1 ) ;
2010-11-02 06:03:18 -04:00
# endif /* IP_PKTINFO or PI_RECVDSTDADDR */
2008-01-17 10:35:34 -05:00
}
}
2010-11-02 06:03:18 -04:00
# endif /* AF_INET6 && IPV6_PKTINFO && HAVE_RECVMSG||HAVE_SENDMSG */
2008-01-17 10:35:34 -05:00
2008-01-15 04:45:30 -05:00
/** send a UDP reply over specified interface*/
2010-07-07 09:13:36 -04:00
static int
2013-12-03 04:11:16 -05:00
comm_point_send_udp_msg_if ( struct comm_point * c , sldns_buffer * packet ,
2008-01-17 10:35:34 -05:00
struct sockaddr * addr , socklen_t addrlen , struct comm_reply * r )
2008-01-15 04:45:30 -05:00
{
2008-06-13 11:32:16 -04:00
# if defined(AF_INET6) && defined(IPV6_PKTINFO) && defined(HAVE_SENDMSG)
2008-01-15 04:45:30 -05:00
ssize_t sent ;
struct msghdr msg ;
struct iovec iov [ 1 ] ;
2020-05-10 10:46:48 -04:00
union {
struct cmsghdr hdr ;
char buf [ 256 ] ;
} control ;
2008-01-15 05:18:36 -05:00
# ifndef S_SPLINT_S
2008-01-15 04:45:30 -05:00
struct cmsghdr * cmsg ;
2008-01-15 05:18:36 -05:00
# endif /* S_SPLINT_S */
2008-01-15 04:45:30 -05:00
log_assert ( c - > fd ! = - 1 ) ;
2011-05-25 07:32:05 -04:00
# ifdef UNBOUND_DEBUG
2013-12-03 04:11:16 -05:00
if ( sldns_buffer_remaining ( packet ) = = 0 )
2011-05-25 07:32:05 -04:00
log_err ( " error: send empty UDP packet " ) ;
# endif
2008-01-15 04:45:30 -05:00
log_assert ( addr & & addrlen > 0 ) ;
msg . msg_name = addr ;
msg . msg_namelen = addrlen ;
2013-12-03 04:11:16 -05:00
iov [ 0 ] . iov_base = sldns_buffer_begin ( packet ) ;
iov [ 0 ] . iov_len = sldns_buffer_remaining ( packet ) ;
2008-01-15 04:45:30 -05:00
msg . msg_iov = iov ;
msg . msg_iovlen = 1 ;
2020-05-10 10:46:48 -04:00
msg . msg_control = control . buf ;
2008-01-15 05:10:22 -05:00
# ifndef S_SPLINT_S
2020-05-10 10:46:48 -04:00
msg . msg_controllen = sizeof ( control . buf ) ;
2008-01-15 05:10:22 -05:00
# endif /* S_SPLINT_S */
2008-01-15 04:45:30 -05:00
msg . msg_flags = 0 ;
2008-01-15 05:10:22 -05:00
# ifndef S_SPLINT_S
2008-01-15 05:18:36 -05:00
cmsg = CMSG_FIRSTHDR ( & msg ) ;
2008-01-17 10:35:34 -05:00
if ( r - > srctype = = 4 ) {
2010-03-16 11:11:58 -04:00
# ifdef IP_PKTINFO
2015-04-02 06:02:01 -04:00
void * cmsg_data ;
2010-03-16 04:37:50 -04:00
msg . msg_controllen = CMSG_SPACE ( sizeof ( struct in_pktinfo ) ) ;
2020-07-17 07:07:03 -04:00
log_assert ( msg . msg_controllen < = sizeof ( control . buf ) ) ;
2008-01-17 10:35:34 -05:00
cmsg - > cmsg_level = IPPROTO_IP ;
cmsg - > cmsg_type = IP_PKTINFO ;
memmove ( CMSG_DATA ( cmsg ) , & r - > pktinfo . v4info ,
sizeof ( struct in_pktinfo ) ) ;
2015-04-02 06:02:01 -04:00
/* unset the ifindex to not bypass the routing tables */
cmsg_data = CMSG_DATA ( cmsg ) ;
( ( struct in_pktinfo * ) cmsg_data ) - > ipi_ifindex = 0 ;
2008-01-17 10:35:34 -05:00
cmsg - > cmsg_len = CMSG_LEN ( sizeof ( struct in_pktinfo ) ) ;
2010-03-16 11:11:58 -04:00
# elif defined(IP_SENDSRCADDR)
msg . msg_controllen = CMSG_SPACE ( sizeof ( struct in_addr ) ) ;
2020-07-17 07:07:03 -04:00
log_assert ( msg . msg_controllen < = sizeof ( control . buf ) ) ;
2010-03-16 11:11:58 -04:00
cmsg - > cmsg_level = IPPROTO_IP ;
cmsg - > cmsg_type = IP_SENDSRCADDR ;
memmove ( CMSG_DATA ( cmsg ) , & r - > pktinfo . v4addr ,
sizeof ( struct in_addr ) ) ;
cmsg - > cmsg_len = CMSG_LEN ( sizeof ( struct in_addr ) ) ;
# else
verbose ( VERB_ALGO , " no IP_PKTINFO or IP_SENDSRCADDR " ) ;
msg . msg_control = NULL ;
2010-11-02 06:03:18 -04:00
# endif /* IP_PKTINFO or IP_SENDSRCADDR */
2008-01-17 10:35:34 -05:00
} else if ( r - > srctype = = 6 ) {
2015-04-02 06:02:01 -04:00
void * cmsg_data ;
2010-03-16 04:37:50 -04:00
msg . msg_controllen = CMSG_SPACE ( sizeof ( struct in6_pktinfo ) ) ;
2020-07-17 07:07:03 -04:00
log_assert ( msg . msg_controllen < = sizeof ( control . buf ) ) ;
2008-01-17 10:35:34 -05:00
cmsg - > cmsg_level = IPPROTO_IPV6 ;
cmsg - > cmsg_type = IPV6_PKTINFO ;
memmove ( CMSG_DATA ( cmsg ) , & r - > pktinfo . v6info ,
sizeof ( struct in6_pktinfo ) ) ;
2015-04-02 06:02:01 -04:00
/* unset the ifindex to not bypass the routing tables */
cmsg_data = CMSG_DATA ( cmsg ) ;
( ( struct in6_pktinfo * ) cmsg_data ) - > ipi6_ifindex = 0 ;
2008-01-17 10:35:34 -05:00
cmsg - > cmsg_len = CMSG_LEN ( sizeof ( struct in6_pktinfo ) ) ;
} else {
/* try to pass all 0 to use default route */
2010-03-16 04:37:50 -04:00
msg . msg_controllen = CMSG_SPACE ( sizeof ( struct in6_pktinfo ) ) ;
2020-07-17 07:07:03 -04:00
log_assert ( msg . msg_controllen < = sizeof ( control . buf ) ) ;
2008-01-17 10:35:34 -05:00
cmsg - > cmsg_level = IPPROTO_IPV6 ;
cmsg - > cmsg_type = IPV6_PKTINFO ;
memset ( CMSG_DATA ( cmsg ) , 0 , sizeof ( struct in6_pktinfo ) ) ;
cmsg - > cmsg_len = CMSG_LEN ( sizeof ( struct in6_pktinfo ) ) ;
}
2008-01-15 05:10:22 -05:00
# endif /* S_SPLINT_S */
2008-01-23 03:49:57 -05:00
if ( verbosity > = VERB_ALGO )
p_ancil ( " send_udp over interface " , r ) ;
2008-01-15 04:45:30 -05:00
sent = sendmsg ( c - > fd , & msg , 0 ) ;
2016-02-15 04:54:52 -05:00
if ( sent = = - 1 ) {
/* try again and block, waiting for IO to complete,
* we want to send the answer , and we will wait for
* the ethernet interface buffer to have space . */
# ifndef USE_WINSOCK
if ( errno = = EAGAIN | |
# ifdef EWOULDBLOCK
errno = = EWOULDBLOCK | |
# endif
errno = = ENOBUFS ) {
# else
if ( WSAGetLastError ( ) = = WSAEINPROGRESS | |
WSAGetLastError ( ) = = WSAENOBUFS | |
WSAGetLastError ( ) = = WSAEWOULDBLOCK ) {
# endif
int e ;
fd_set_block ( c - > fd ) ;
sent = sendmsg ( c - > fd , & msg , 0 ) ;
e = errno ;
fd_set_nonblock ( c - > fd ) ;
errno = e ;
}
}
2008-01-15 04:45:30 -05:00
if ( sent = = - 1 ) {
2010-11-15 09:00:20 -05:00
if ( ! udp_send_errno_needs_log ( addr , addrlen ) )
return 0 ;
2008-01-15 05:39:32 -05:00
verbose ( VERB_OPS , " sendmsg failed: %s " , strerror ( errno ) ) ;
2008-01-30 09:46:01 -05:00
log_addr ( VERB_OPS , " remote address is " ,
( struct sockaddr_storage * ) addr , addrlen ) ;
2016-01-26 11:14:28 -05:00
# ifdef __NetBSD__
/* netbsd 7 has IP_PKTINFO for recv but not send */
if ( errno = = EINVAL & & r - > srctype = = 4 )
2016-01-27 02:44:47 -05:00
log_err ( " sendmsg: No support for sendmsg(IP_PKTINFO). "
2016-01-26 11:14:28 -05:00
" Please disable interface-automatic " ) ;
# endif
2008-01-15 04:45:30 -05:00
return 0 ;
2013-12-03 04:11:16 -05:00
} else if ( ( size_t ) sent ! = sldns_buffer_remaining ( packet ) ) {
2008-01-15 04:45:30 -05:00
log_err ( " sent %d in place of %d bytes " ,
2013-12-03 04:11:16 -05:00
( int ) sent , ( int ) sldns_buffer_remaining ( packet ) ) ;
2008-01-15 04:45:30 -05:00
return 0 ;
}
return 1 ;
# else
2008-04-16 12:09:25 -04:00
( void ) c ;
( void ) packet ;
( void ) addr ;
( void ) addrlen ;
( void ) r ;
2008-01-15 04:45:30 -05:00
log_err ( " sendmsg: IPV6_PKTINFO not supported " ) ;
return 0 ;
2010-11-02 06:03:18 -04:00
# endif /* AF_INET6 && IPV6_PKTINFO && HAVE_SENDMSG */
2008-01-15 04:45:30 -05:00
}
2020-11-26 03:39:54 -05:00
/** return true is UDP receive error needs to be logged */
static int udp_recv_needs_log ( int err )
{
switch ( err ) {
2020-12-11 04:30:54 -05:00
case EACCES : /* some hosts send ICMP 'Permission Denied' */
2020-12-02 05:58:24 -05:00
# ifndef USE_WINSOCK
2020-11-26 03:39:54 -05:00
case ECONNREFUSED :
# ifdef ENETUNREACH
case ENETUNREACH :
# endif
# ifdef EHOSTDOWN
case EHOSTDOWN :
# endif
# ifdef EHOSTUNREACH
case EHOSTUNREACH :
# endif
# ifdef ENETDOWN
case ENETDOWN :
# endif
2020-12-02 05:58:24 -05:00
# else /* USE_WINSOCK */
case WSAECONNREFUSED :
case WSAENETUNREACH :
case WSAEHOSTDOWN :
case WSAEHOSTUNREACH :
case WSAENETDOWN :
# endif
2020-11-26 03:39:54 -05:00
if ( verbosity > = VERB_ALGO )
return 1 ;
return 0 ;
default :
break ;
}
return 1 ;
}
2008-01-15 04:45:30 -05:00
void
comm_point_udp_ancil_callback ( int fd , short event , void * arg )
{
2008-06-13 11:32:16 -04:00
# if defined(AF_INET6) && defined(IPV6_PKTINFO) && defined(HAVE_RECVMSG)
2008-01-15 04:45:30 -05:00
struct comm_reply rep ;
struct msghdr msg ;
struct iovec iov [ 1 ] ;
2010-07-07 09:13:36 -04:00
ssize_t rcv ;
2020-05-10 10:46:48 -04:00
union {
struct cmsghdr hdr ;
char buf [ 256 ] ;
} ancil ;
2008-02-19 09:35:40 -05:00
int i ;
2008-01-15 05:10:22 -05:00
# ifndef S_SPLINT_S
2008-01-15 04:45:30 -05:00
struct cmsghdr * cmsg ;
2008-01-15 05:10:22 -05:00
# endif /* S_SPLINT_S */
2008-01-15 04:45:30 -05:00
rep . c = ( struct comm_point * ) arg ;
log_assert ( rep . c - > type = = comm_udp ) ;
2016-03-07 09:10:06 -05:00
if ( ! ( event & UB_EV_READ ) )
2008-01-15 04:45:30 -05:00
return ;
log_assert ( rep . c & & rep . c - > buffer & & rep . c - > fd = = fd ) ;
2016-03-07 11:01:09 -05:00
ub_comm_base_now ( rep . c - > ev - > base ) ;
2008-02-19 09:35:40 -05:00
for ( i = 0 ; i < NUM_UDP_PER_SELECT ; i + + ) {
2013-12-03 04:11:16 -05:00
sldns_buffer_clear ( rep . c - > buffer ) ;
2008-02-19 09:35:40 -05:00
rep . addrlen = ( socklen_t ) sizeof ( rep . addr ) ;
log_assert ( fd ! = - 1 ) ;
2013-12-03 04:11:16 -05:00
log_assert ( sldns_buffer_remaining ( rep . c - > buffer ) > 0 ) ;
2008-02-19 09:35:40 -05:00
msg . msg_name = & rep . addr ;
msg . msg_namelen = ( socklen_t ) sizeof ( rep . addr ) ;
2013-12-03 04:11:16 -05:00
iov [ 0 ] . iov_base = sldns_buffer_begin ( rep . c - > buffer ) ;
iov [ 0 ] . iov_len = sldns_buffer_remaining ( rep . c - > buffer ) ;
2008-02-19 09:35:40 -05:00
msg . msg_iov = iov ;
msg . msg_iovlen = 1 ;
2020-05-10 10:46:48 -04:00
msg . msg_control = ancil . buf ;
2008-01-15 05:10:22 -05:00
# ifndef S_SPLINT_S
2020-05-10 10:46:48 -04:00
msg . msg_controllen = sizeof ( ancil . buf ) ;
2008-01-15 05:10:22 -05:00
# endif /* S_SPLINT_S */
2008-02-19 09:35:40 -05:00
msg . msg_flags = 0 ;
2010-07-07 09:13:36 -04:00
rcv = recvmsg ( fd , & msg , 0 ) ;
if ( rcv = = - 1 ) {
2020-11-26 03:39:54 -05:00
if ( errno ! = EAGAIN & & errno ! = EINTR
& & udp_recv_needs_log ( errno ) ) {
2008-02-19 09:35:40 -05:00
log_err ( " recvmsg failed: %s " , strerror ( errno ) ) ;
}
return ;
2008-01-15 04:45:30 -05:00
}
2008-02-19 09:35:40 -05:00
rep . addrlen = msg . msg_namelen ;
2013-12-03 04:11:16 -05:00
sldns_buffer_skip ( rep . c - > buffer , rcv ) ;
sldns_buffer_flip ( rep . c - > buffer ) ;
2008-02-19 09:35:40 -05:00
rep . srctype = 0 ;
2008-01-15 05:10:22 -05:00
# ifndef S_SPLINT_S
2008-02-19 09:35:40 -05:00
for ( cmsg = CMSG_FIRSTHDR ( & msg ) ; cmsg ! = NULL ;
cmsg = CMSG_NXTHDR ( & msg , cmsg ) ) {
if ( cmsg - > cmsg_level = = IPPROTO_IPV6 & &
cmsg - > cmsg_type = = IPV6_PKTINFO ) {
rep . srctype = 6 ;
memmove ( & rep . pktinfo . v6info , CMSG_DATA ( cmsg ) ,
sizeof ( struct in6_pktinfo ) ) ;
break ;
2010-03-16 11:11:58 -04:00
# ifdef IP_PKTINFO
2008-02-19 09:35:40 -05:00
} else if ( cmsg - > cmsg_level = = IPPROTO_IP & &
cmsg - > cmsg_type = = IP_PKTINFO ) {
rep . srctype = 4 ;
memmove ( & rep . pktinfo . v4info , CMSG_DATA ( cmsg ) ,
sizeof ( struct in_pktinfo ) ) ;
break ;
2010-03-16 11:11:58 -04:00
# elif defined(IP_RECVDSTADDR)
} else if ( cmsg - > cmsg_level = = IPPROTO_IP & &
cmsg - > cmsg_type = = IP_RECVDSTADDR ) {
rep . srctype = 4 ;
memmove ( & rep . pktinfo . v4addr , CMSG_DATA ( cmsg ) ,
sizeof ( struct in_addr ) ) ;
break ;
2010-11-02 06:03:18 -04:00
# endif /* IP_PKTINFO or IP_RECVDSTADDR */
2008-02-19 09:35:40 -05:00
}
2008-01-15 04:45:30 -05:00
}
2008-02-19 09:35:40 -05:00
if ( verbosity > = VERB_ALGO )
p_ancil ( " receive_udp on interface " , & rep ) ;
2008-01-15 05:10:22 -05:00
# endif /* S_SPLINT_S */
2008-02-20 02:18:42 -05:00
fptr_ok ( fptr_whitelist_comm_point ( rep . c - > callback ) ) ;
2008-02-19 09:35:40 -05:00
if ( ( * rep . c - > callback ) ( rep . c , rep . c - > cb_arg , NETEVENT_NOERROR , & rep ) ) {
/* send back immediate reply */
( void ) comm_point_send_udp_msg_if ( rep . c , rep . c - > buffer ,
( struct sockaddr * ) & rep . addr , rep . addrlen , & rep ) ;
}
2017-10-19 11:11:20 -04:00
if ( ! rep . c | | rep . c - > fd = = - 1 ) /* commpoint closed */
2008-04-11 09:24:49 -04:00
break ;
2008-01-15 04:45:30 -05:00
}
# else
2008-04-16 12:09:25 -04:00
( void ) fd ;
( void ) event ;
( void ) arg ;
2017-01-26 06:07:52 -05:00
fatal_exit ( " recvmsg: No support for IPV6_PKTINFO; IP_PKTINFO or IP_RECVDSTADDR. "
2008-01-15 04:45:30 -05:00
" Please disable interface-automatic " ) ;
2010-11-02 06:03:18 -04:00
# endif /* AF_INET6 && IPV6_PKTINFO && HAVE_RECVMSG */
2008-01-15 04:45:30 -05:00
}
2007-10-05 04:05:06 -04:00
void
2007-01-31 04:32:30 -05:00
comm_point_udp_callback ( int fd , short event , void * arg )
2007-01-23 08:46:18 -05:00
{
2007-01-31 04:32:30 -05:00
struct comm_reply rep ;
2010-07-07 09:13:36 -04:00
ssize_t rcv ;
2008-02-19 09:35:40 -05:00
int i ;
2017-04-18 05:00:52 -04:00
struct sldns_buffer * buffer ;
2007-01-31 04:32:30 -05:00
rep . c = ( struct comm_point * ) arg ;
2007-02-06 09:00:52 -05:00
log_assert ( rep . c - > type = = comm_udp ) ;
2007-01-31 04:32:30 -05:00
2016-03-07 09:10:06 -05:00
if ( ! ( event & UB_EV_READ ) )
2007-01-31 04:32:30 -05:00
return ;
log_assert ( rep . c & & rep . c - > buffer & & rep . c - > fd = = fd ) ;
2016-03-07 11:01:09 -05:00
ub_comm_base_now ( rep . c - > ev - > base ) ;
2008-02-19 09:35:40 -05:00
for ( i = 0 ; i < NUM_UDP_PER_SELECT ; i + + ) {
2013-12-03 04:11:16 -05:00
sldns_buffer_clear ( rep . c - > buffer ) ;
2008-02-19 09:35:40 -05:00
rep . addrlen = ( socklen_t ) sizeof ( rep . addr ) ;
log_assert ( fd ! = - 1 ) ;
2013-12-03 04:11:16 -05:00
log_assert ( sldns_buffer_remaining ( rep . c - > buffer ) > 0 ) ;
rcv = recvfrom ( fd , ( void * ) sldns_buffer_begin ( rep . c - > buffer ) ,
sldns_buffer_remaining ( rep . c - > buffer ) , 0 ,
2008-02-19 09:35:40 -05:00
( struct sockaddr * ) & rep . addr , & rep . addrlen ) ;
2010-07-07 09:13:36 -04:00
if ( rcv = = - 1 ) {
2008-06-18 10:27:30 -04:00
# ifndef USE_WINSOCK
2020-11-25 03:41:06 -05:00
if ( errno ! = EAGAIN & & errno ! = EINTR
2020-11-26 03:39:54 -05:00
& & udp_recv_needs_log ( errno ) )
2008-11-05 09:56:49 -05:00
log_err ( " recvfrom %d failed: %s " ,
fd , strerror ( errno ) ) ;
2008-06-18 10:27:30 -04:00
# else
if ( WSAGetLastError ( ) ! = WSAEINPROGRESS & &
WSAGetLastError ( ) ! = WSAECONNRESET & &
2020-12-02 05:58:24 -05:00
WSAGetLastError ( ) ! = WSAEWOULDBLOCK & &
udp_recv_needs_log ( WSAGetLastError ( ) ) )
2008-06-18 10:27:30 -04:00
log_err ( " recvfrom failed: %s " ,
wsa_strerror ( WSAGetLastError ( ) ) ) ;
# endif
2008-02-19 09:35:40 -05:00
return ;
}
2013-12-03 04:11:16 -05:00
sldns_buffer_skip ( rep . c - > buffer , rcv ) ;
sldns_buffer_flip ( rep . c - > buffer ) ;
2008-02-19 09:35:40 -05:00
rep . srctype = 0 ;
2008-02-20 02:18:42 -05:00
fptr_ok ( fptr_whitelist_comm_point ( rep . c - > callback ) ) ;
2008-02-19 09:35:40 -05:00
if ( ( * rep . c - > callback ) ( rep . c , rep . c - > cb_arg , NETEVENT_NOERROR , & rep ) ) {
/* send back immediate reply */
2017-03-20 10:55:31 -04:00
# ifdef USE_DNSCRYPT
2017-04-18 05:00:52 -04:00
buffer = rep . c - > dnscrypt_buffer ;
2017-03-20 10:55:31 -04:00
# else
2017-04-18 05:00:52 -04:00
buffer = rep . c - > buffer ;
2017-03-20 10:55:31 -04:00
# endif
( void ) comm_point_send_udp_msg ( rep . c , buffer ,
2020-12-16 11:11:41 -05:00
( struct sockaddr * ) & rep . addr , rep . addrlen , 0 ) ;
2007-01-31 04:32:30 -05:00
}
2017-10-19 11:11:20 -04:00
if ( ! rep . c | | rep . c - > fd ! = fd ) /* commpoint closed to -1 or reused for
2008-11-05 09:56:49 -05:00
another UDP port . Note rep . c cannot be reused with TCP fd . */
2008-04-11 09:24:49 -04:00
break ;
2007-01-31 04:32:30 -05:00
}
2007-01-23 08:46:18 -05:00
}
2020-12-02 04:10:27 -05:00
int adjusted_tcp_timeout ( struct comm_point * c )
{
if ( c - > tcp_timeout_msec < TCP_QUERY_TIMEOUT_MINIMUM )
return TCP_QUERY_TIMEOUT_MINIMUM ;
return c - > tcp_timeout_msec ;
}
2007-06-12 10:51:49 -04:00
/** Use a new tcp handler for new query fd, set to read query */
2007-02-07 09:18:42 -05:00
static void
2016-06-15 10:41:23 -04:00
setup_tcp_handler ( struct comm_point * c , int fd , int cur , int max )
2007-02-07 09:18:42 -05:00
{
2018-07-31 03:20:15 -04:00
int handler_usage ;
2020-05-07 10:36:26 -04:00
log_assert ( c - > type = = comm_tcp | | c - > type = = comm_http ) ;
2007-02-07 09:18:42 -05:00
log_assert ( c - > fd = = - 1 ) ;
2013-12-03 04:11:16 -05:00
sldns_buffer_clear ( c - > buffer ) ;
2017-03-20 10:55:31 -04:00
# ifdef USE_DNSCRYPT
2017-04-18 05:00:52 -04:00
if ( c - > dnscrypt )
sldns_buffer_clear ( c - > dnscrypt_buffer ) ;
2017-03-20 10:55:31 -04:00
# endif
2007-02-07 09:18:42 -05:00
c - > tcp_is_reading = 1 ;
c - > tcp_byte_count = 0 ;
2016-06-15 10:41:23 -04:00
/* if more than half the tcp handlers are in use, use a shorter
* timeout for this TCP connection , we need to make space for
* other connections to be able to get attention */
2018-07-31 04:00:57 -04:00
/* If > 50% TCP handler structures in use, set timeout to 1/100th
* configured value .
* If > 65 % TCP handler structures in use , set to 1 / 500 th configured
* value .
* If > 80 % TCP handler structures in use , set to 0.
*
* If the timeout to use falls below 200 milliseconds , an actual
* timeout of 200 ms is used .
*/
2018-07-31 03:20:15 -04:00
handler_usage = ( cur * 100 ) / max ;
if ( handler_usage > 50 & & handler_usage < = 65 )
c - > tcp_timeout_msec / = 100 ;
else if ( handler_usage > 65 & & handler_usage < = 80 )
c - > tcp_timeout_msec / = 500 ;
else if ( handler_usage > 80 )
2018-07-31 03:18:34 -04:00
c - > tcp_timeout_msec = 0 ;
2020-12-02 04:10:27 -05:00
comm_point_start_listening ( c , fd , adjusted_tcp_timeout ( c ) ) ;
2007-02-07 09:18:42 -05:00
}
2012-05-08 09:03:25 -04:00
void comm_base_handle_slow_accept ( int ATTR_UNUSED ( fd ) ,
short ATTR_UNUSED ( event ) , void * arg )
2012-05-08 08:08:55 -04:00
{
struct comm_base * b = ( struct comm_base * ) arg ;
/* timeout for the slow accept, re-enable accepts again */
if ( b - > start_accept ) {
verbose ( VERB_ALGO , " wait is over, slow accept disabled " ) ;
fptr_ok ( fptr_whitelist_start_accept ( b - > start_accept ) ) ;
( * b - > start_accept ) ( b - > cb_arg ) ;
b - > eb - > slow_accept_enabled = 0 ;
}
}
2008-09-10 11:23:01 -04:00
int comm_point_perform_accept ( struct comm_point * c ,
struct sockaddr_storage * addr , socklen_t * addrlen )
2007-01-23 08:46:18 -05:00
{
2007-02-07 09:18:42 -05:00
int new_fd ;
2008-09-10 11:23:01 -04:00
* addrlen = ( socklen_t ) sizeof ( * addr ) ;
2018-05-23 09:55:09 -04:00
# ifndef HAVE_ACCEPT4
2008-09-10 11:23:01 -04:00
new_fd = accept ( c - > fd , ( struct sockaddr * ) addr , addrlen ) ;
2018-05-23 09:55:09 -04:00
# else
/* SOCK_NONBLOCK saves extra calls to fcntl for the same result */
new_fd = accept4 ( c - > fd , ( struct sockaddr * ) addr , addrlen , SOCK_NONBLOCK ) ;
# endif
2007-02-07 09:18:42 -05:00
if ( new_fd = = - 1 ) {
2008-06-18 10:27:30 -04:00
# ifndef USE_WINSOCK
2007-02-07 09:18:42 -05:00
/* EINTR is signal interrupt. others are closed connection. */
2008-11-21 04:15:44 -05:00
if ( errno = = EINTR | | errno = = EAGAIN
2008-06-13 11:32:16 -04:00
# ifdef EWOULDBLOCK
2008-11-21 04:15:44 -05:00
| | errno = = EWOULDBLOCK
2008-06-13 11:32:16 -04:00
# endif
# ifdef ECONNABORTED
2008-11-21 04:15:44 -05:00
| | errno = = ECONNABORTED
2008-06-13 11:32:16 -04:00
# endif
2007-02-15 05:48:43 -05:00
# ifdef EPROTO
2008-11-21 04:15:44 -05:00
| | errno = = EPROTO
2007-02-15 05:48:43 -05:00
# endif /* EPROTO */
)
2008-09-10 11:23:01 -04:00
return - 1 ;
2012-05-08 08:08:55 -04:00
# if defined(ENFILE) && defined(EMFILE)
if ( errno = = ENFILE | | errno = = EMFILE ) {
/* out of file descriptors, likely outside of our
* control . stop accept ( ) calls for some time */
if ( c - > ev - > base - > stop_accept ) {
struct comm_base * b = c - > ev - > base ;
struct timeval tv ;
verbose ( VERB_ALGO , " out of file descriptors: "
" slow accept " ) ;
b - > eb - > slow_accept_enabled = 1 ;
fptr_ok ( fptr_whitelist_stop_accept (
b - > stop_accept ) ) ;
( * b - > stop_accept ) ( b - > cb_arg ) ;
/* set timeout, no mallocs */
tv . tv_sec = NETEVENT_SLOW_ACCEPT_TIME / 1000 ;
2016-06-16 03:21:07 -04:00
tv . tv_usec = ( NETEVENT_SLOW_ACCEPT_TIME % 1000 ) * 1000 ;
2016-03-07 09:10:06 -05:00
b - > eb - > slow_accept = ub_event_new ( b - > eb - > base ,
- 1 , UB_EV_TIMEOUT ,
2012-05-08 08:08:55 -04:00
comm_base_handle_slow_accept , b ) ;
2016-03-07 09:10:06 -05:00
if ( b - > eb - > slow_accept = = NULL ) {
2012-05-08 08:08:55 -04:00
/* we do not want to log here, because
* that would spam the logfiles .
* error : " event_base_set failed. " */
}
2016-03-07 09:10:06 -05:00
else if ( ub_event_add ( b - > eb - > slow_accept , & tv )
! = 0 ) {
2012-05-08 08:08:55 -04:00
/* we do not want to log here,
* error : " event_add failed. " */
}
}
return - 1 ;
}
# endif
2008-06-18 10:27:30 -04:00
# else /* USE_WINSOCK */
if ( WSAGetLastError ( ) = = WSAEINPROGRESS | |
WSAGetLastError ( ) = = WSAECONNRESET )
2008-09-10 11:23:01 -04:00
return - 1 ;
2008-06-23 09:52:03 -04:00
if ( WSAGetLastError ( ) = = WSAEWOULDBLOCK ) {
2016-03-07 09:10:06 -05:00
ub_winsock_tcp_wouldblock ( c - > ev - > ev , UB_EV_READ ) ;
2008-09-10 11:23:01 -04:00
return - 1 ;
2008-06-23 09:52:03 -04:00
}
2008-06-18 10:27:30 -04:00
# endif
2020-08-31 03:12:01 -04:00
log_err_addr ( " accept failed " , sock_strerror ( errno ) , addr ,
* addrlen ) ;
2008-09-10 11:23:01 -04:00
return - 1 ;
}
2018-08-07 07:57:42 -04:00
if ( c - > tcp_conn_limit & & c - > type = = comm_tcp_accept ) {
c - > tcl_addr = tcl_addr_lookup ( c - > tcp_conn_limit , addr , * addrlen ) ;
if ( ! tcl_new_connection ( c - > tcl_addr ) ) {
2018-08-07 08:48:49 -04:00
if ( verbosity > = 3 )
log_err_addr ( " accept rejected " ,
2018-08-07 07:57:42 -04:00
" connection limit exceeded " , addr , * addrlen ) ;
close ( new_fd ) ;
return - 1 ;
}
}
2018-05-23 09:55:09 -04:00
# ifndef HAVE_ACCEPT4
2008-09-11 10:14:12 -04:00
fd_set_nonblock ( new_fd ) ;
2018-05-23 09:55:09 -04:00
# endif
2008-09-10 11:23:01 -04:00
return new_fd ;
}
2011-10-31 10:48:48 -04:00
# ifdef USE_WINSOCK
static long win_bio_cb ( BIO * b , int oper , const char * ATTR_UNUSED ( argp ) ,
2021-09-21 07:51:34 -04:00
# ifdef HAVE_BIO_SET_CALLBACK_EX
size_t ATTR_UNUSED ( len ) ,
# endif
int ATTR_UNUSED ( argi ) , long argl ,
# ifndef HAVE_BIO_SET_CALLBACK_EX
long retvalue
# else
int retvalue , size_t * ATTR_UNUSED ( processed )
# endif
)
2011-10-31 10:48:48 -04:00
{
2018-05-28 08:30:34 -04:00
int wsa_err = WSAGetLastError ( ) ; /* store errcode before it is gone */
2011-10-31 10:48:48 -04:00
verbose ( VERB_ALGO , " bio_cb %d, %s %s %s " , oper ,
( oper & BIO_CB_RETURN ) ? " return " : " before " ,
( oper & BIO_CB_READ ) ? " read " : ( ( oper & BIO_CB_WRITE ) ? " write " : " other " ) ,
2018-05-28 08:30:34 -04:00
wsa_err = = WSAEWOULDBLOCK ? " wsawb " : " " ) ;
2011-10-31 10:48:48 -04:00
/* on windows, check if previous operation caused EWOULDBLOCK */
if ( ( oper = = ( BIO_CB_READ | BIO_CB_RETURN ) & & argl = = 0 ) | |
( oper = = ( BIO_CB_GETS | BIO_CB_RETURN ) & & argl = = 0 ) ) {
2018-05-28 08:30:34 -04:00
if ( wsa_err = = WSAEWOULDBLOCK )
2016-03-07 09:10:06 -05:00
ub_winsock_tcp_wouldblock ( ( struct ub_event * )
BIO_get_callback_arg ( b ) , UB_EV_READ ) ;
2011-10-31 10:48:48 -04:00
}
if ( ( oper = = ( BIO_CB_WRITE | BIO_CB_RETURN ) & & argl = = 0 ) | |
( oper = = ( BIO_CB_PUTS | BIO_CB_RETURN ) & & argl = = 0 ) ) {
2018-05-28 08:30:34 -04:00
if ( wsa_err = = WSAEWOULDBLOCK )
2016-03-07 09:10:06 -05:00
ub_winsock_tcp_wouldblock ( ( struct ub_event * )
BIO_get_callback_arg ( b ) , UB_EV_WRITE ) ;
2011-10-31 10:48:48 -04:00
}
/* return original return value */
return retvalue ;
}
2011-11-01 05:26:58 -04:00
/** set win bio callbacks for nonblocking operations */
void
comm_point_tcp_win_bio_cb ( struct comm_point * c , void * thessl )
{
SSL * ssl = ( SSL * ) thessl ;
/* set them both just in case, but usually they are the same BIO */
2021-09-21 07:51:34 -04:00
# ifdef HAVE_BIO_SET_CALLBACK_EX
BIO_set_callback_ex ( SSL_get_rbio ( ssl ) , & win_bio_cb ) ;
# else
2011-11-01 05:26:58 -04:00
BIO_set_callback ( SSL_get_rbio ( ssl ) , & win_bio_cb ) ;
2021-09-21 07:51:34 -04:00
# endif
2016-03-07 09:10:06 -05:00
BIO_set_callback_arg ( SSL_get_rbio ( ssl ) , ( char * ) c - > ev - > ev ) ;
2021-09-21 07:51:34 -04:00
# ifdef HAVE_BIO_SET_CALLBACK_EX
BIO_set_callback_ex ( SSL_get_wbio ( ssl ) , & win_bio_cb ) ;
# else
2011-11-01 05:26:58 -04:00
BIO_set_callback ( SSL_get_wbio ( ssl ) , & win_bio_cb ) ;
2021-09-21 07:51:34 -04:00
# endif
2016-03-07 09:10:06 -05:00
BIO_set_callback_arg ( SSL_get_wbio ( ssl ) , ( char * ) c - > ev - > ev ) ;
2011-11-01 05:26:58 -04:00
}
2011-10-31 10:48:48 -04:00
# endif
2020-05-07 10:36:26 -04:00
# ifdef HAVE_NGHTTP2
/** Create http2 session server. Per connection, after TCP accepted.*/
static int http2_session_server_create ( struct http2_session * h2_session )
{
log_assert ( h2_session - > callbacks ) ;
h2_session - > is_drop = 0 ;
if ( nghttp2_session_server_new ( & h2_session - > session ,
h2_session - > callbacks ,
h2_session ) = = NGHTTP2_ERR_NOMEM ) {
log_err ( " failed to create nghttp2 session server " ) ;
return 0 ;
}
return 1 ;
}
/** Submit http2 setting to session. Once per session. */
static int http2_submit_settings ( struct http2_session * h2_session )
{
int ret ;
nghttp2_settings_entry settings [ 1 ] = {
2020-05-12 12:12:19 -04:00
{ NGHTTP2_SETTINGS_MAX_CONCURRENT_STREAMS ,
h2_session - > c - > http2_max_streams } } ;
2020-05-07 10:36:26 -04:00
ret = nghttp2_submit_settings ( h2_session - > session , NGHTTP2_FLAG_NONE ,
settings , 1 ) ;
if ( ret ) {
verbose ( VERB_QUERY , " http2: submit_settings failed, "
" error: %s " , nghttp2_strerror ( ret ) ) ;
return 0 ;
}
return 1 ;
}
# endif /* HAVE_NGHTTP2 */
2008-09-10 11:23:01 -04:00
void
comm_point_tcp_accept_callback ( int fd , short event , void * arg )
{
struct comm_point * c = ( struct comm_point * ) arg , * c_hdl ;
int new_fd ;
log_assert ( c - > type = = comm_tcp_accept ) ;
2016-03-07 09:10:06 -05:00
if ( ! ( event & UB_EV_READ ) ) {
2008-09-10 11:23:01 -04:00
log_info ( " ignoring tcp accept event %d " , ( int ) event ) ;
2007-02-07 09:18:42 -05:00
return ;
}
2016-03-07 11:01:09 -05:00
ub_comm_base_now ( c - > ev - > base ) ;
2008-09-10 11:23:01 -04:00
/* find free tcp handler. */
if ( ! c - > tcp_free ) {
log_warn ( " accepted too many tcp, connections full " ) ;
return ;
}
/* accept incoming connection. */
c_hdl = c - > tcp_free ;
2019-04-05 10:11:28 -04:00
/* clear leftover flags from previous use, and then set the
* correct event base for the event structure for libevent */
ub_event_free ( c_hdl - > ev - > ev ) ;
2021-11-05 06:15:19 -04:00
c_hdl - > ev - > ev = NULL ;
2020-10-19 07:36:53 -04:00
if ( ( c_hdl - > type = = comm_tcp & & c_hdl - > tcp_req_info ) | |
c_hdl - > type = = comm_local | | c_hdl - > type = = comm_raw )
c_hdl - > tcp_do_toggle_rw = 0 ;
else c_hdl - > tcp_do_toggle_rw = 1 ;
2020-05-07 10:36:26 -04:00
2020-05-12 12:12:19 -04:00
if ( c_hdl - > type = = comm_http ) {
2020-05-07 10:36:26 -04:00
# ifdef HAVE_NGHTTP2
2020-05-12 12:12:19 -04:00
if ( ! c_hdl - > h2_session | |
! http2_session_server_create ( c_hdl - > h2_session ) ) {
2020-05-07 10:36:26 -04:00
log_warn ( " failed to create nghttp2 " ) ;
return ;
}
2020-05-12 12:12:19 -04:00
if ( ! c_hdl - > h2_session | |
! http2_submit_settings ( c_hdl - > h2_session ) ) {
2020-05-07 10:36:26 -04:00
log_warn ( " failed to submit http2 settings " ) ;
return ;
}
2020-10-19 04:24:03 -04:00
if ( ! c - > ssl ) {
c_hdl - > tcp_do_toggle_rw = 0 ;
c_hdl - > use_h2 = 1 ;
}
2020-05-12 12:12:19 -04:00
# endif
2020-05-07 10:36:26 -04:00
c_hdl - > ev - > ev = ub_event_new ( c_hdl - > ev - > base - > eb - > base , - 1 ,
UB_EV_PERSIST | UB_EV_READ | UB_EV_TIMEOUT ,
comm_point_http_handle_callback , c_hdl ) ;
} else {
c_hdl - > ev - > ev = ub_event_new ( c_hdl - > ev - > base - > eb - > base , - 1 ,
UB_EV_PERSIST | UB_EV_READ | UB_EV_TIMEOUT ,
comm_point_tcp_handle_callback , c_hdl ) ;
}
2019-04-05 10:11:28 -04:00
if ( ! c_hdl - > ev - > ev ) {
log_warn ( " could not ub_event_new, dropped tcp " ) ;
return ;
}
2008-09-10 11:23:01 -04:00
log_assert ( fd ! = - 1 ) ;
2016-07-22 03:05:52 -04:00
( void ) fd ;
2008-09-10 11:23:01 -04:00
new_fd = comm_point_perform_accept ( c , & c_hdl - > repinfo . addr ,
& c_hdl - > repinfo . addrlen ) ;
2008-11-20 03:06:05 -05:00
if ( new_fd = = - 1 )
return ;
2011-10-31 10:48:48 -04:00
if ( c - > ssl ) {
c_hdl - > ssl = incoming_ssl_fd ( c - > ssl , new_fd ) ;
2011-11-01 05:26:58 -04:00
if ( ! c_hdl - > ssl ) {
c_hdl - > fd = new_fd ;
comm_point_close ( c_hdl ) ;
2011-10-31 10:48:48 -04:00
return ;
2011-11-01 05:26:58 -04:00
}
2011-10-31 10:48:48 -04:00
c_hdl - > ssl_shake_state = comm_ssl_shake_read ;
# ifdef USE_WINSOCK
2011-11-01 05:26:58 -04:00
comm_point_tcp_win_bio_cb ( c_hdl , c_hdl - > ssl ) ;
2011-10-31 10:48:48 -04:00
# endif
}
2008-09-10 11:23:01 -04:00
2007-11-19 11:31:22 -05:00
/* grab the tcp handler buffers */
2015-03-05 10:23:14 -05:00
c - > cur_tcp_count + + ;
2007-02-07 09:18:42 -05:00
c - > tcp_free = c_hdl - > tcp_free ;
2021-11-05 06:15:19 -04:00
c_hdl - > tcp_free = NULL ;
2007-02-07 09:18:42 -05:00
if ( ! c - > tcp_free ) {
/* stop accepting incoming queries for now. */
comm_point_stop_listening ( c ) ;
}
2016-06-15 10:41:23 -04:00
setup_tcp_handler ( c_hdl , new_fd , c - > cur_tcp_count , c - > max_tcp_count ) ;
2007-02-07 09:18:42 -05:00
}
2007-06-12 10:51:49 -04:00
/** Make tcp handler free for next assignment */
2007-02-07 09:18:42 -05:00
static void
reclaim_tcp_handler ( struct comm_point * c )
{
log_assert ( c - > type = = comm_tcp ) ;
2011-10-31 10:48:48 -04:00
if ( c - > ssl ) {
2012-06-22 05:11:41 -04:00
# ifdef HAVE_SSL
2011-10-31 10:48:48 -04:00
SSL_shutdown ( c - > ssl ) ;
SSL_free ( c - > ssl ) ;
c - > ssl = NULL ;
2012-06-22 05:11:41 -04:00
# endif
2011-10-31 10:48:48 -04:00
}
2007-02-07 09:18:42 -05:00
comm_point_close ( c ) ;
2007-05-08 09:25:21 -04:00
if ( c - > tcp_parent ) {
2021-11-05 06:15:19 -04:00
if ( c ! = c - > tcp_parent - > tcp_free ) {
c - > tcp_parent - > cur_tcp_count - - ;
c - > tcp_free = c - > tcp_parent - > tcp_free ;
c - > tcp_parent - > tcp_free = c ;
}
2007-05-08 09:25:21 -04:00
if ( ! c - > tcp_free ) {
/* re-enable listening on accept socket */
comm_point_start_listening ( c - > tcp_parent , - 1 , - 1 ) ;
}
2007-02-07 09:18:42 -05:00
}
2020-11-25 07:46:28 -05:00
c - > tcp_more_read_again = NULL ;
c - > tcp_more_write_again = NULL ;
2007-02-07 09:18:42 -05:00
}
2007-02-07 10:44:19 -05:00
/** do the callback when writing is done */
static void
tcp_callback_writer ( struct comm_point * c )
{
log_assert ( c - > type = = comm_tcp ) ;
2020-06-26 10:05:15 -04:00
if ( ! c - > tcp_write_and_read ) {
sldns_buffer_clear ( c - > buffer ) ;
c - > tcp_byte_count = 0 ;
}
2007-02-26 09:49:11 -05:00
if ( c - > tcp_do_toggle_rw )
c - > tcp_is_reading = 1 ;
2008-02-08 05:59:18 -05:00
/* switch from listening(write) to listening(read) */
2019-01-11 09:12:27 -05:00
if ( c - > tcp_req_info ) {
tcp_req_info_handle_writedone ( c - > tcp_req_info ) ;
} else {
2019-02-25 10:48:27 -05:00
comm_point_stop_listening ( c ) ;
2020-06-26 04:54:13 -04:00
if ( c - > tcp_write_and_read ) {
fptr_ok ( fptr_whitelist_comm_point ( c - > callback ) ) ;
if ( ( * c - > callback ) ( c , c - > cb_arg , NETEVENT_PKT_WRITTEN ,
& c - > repinfo ) ) {
comm_point_start_listening ( c , - 1 ,
2020-12-02 04:10:27 -05:00
adjusted_tcp_timeout ( c ) ) ;
2020-06-26 04:54:13 -04:00
}
} else {
2020-12-02 04:10:27 -05:00
comm_point_start_listening ( c , - 1 ,
adjusted_tcp_timeout ( c ) ) ;
2020-06-26 04:54:13 -04:00
}
2019-01-11 09:12:27 -05:00
}
2007-02-07 10:44:19 -05:00
}
2007-02-07 09:18:42 -05:00
/** do the callback when reading is done */
static void
tcp_callback_reader ( struct comm_point * c )
{
2007-02-26 09:49:11 -05:00
log_assert ( c - > type = = comm_tcp | | c - > type = = comm_local ) ;
2013-12-03 04:11:16 -05:00
sldns_buffer_flip ( c - > buffer ) ;
2007-02-26 09:49:11 -05:00
if ( c - > tcp_do_toggle_rw )
c - > tcp_is_reading = 0 ;
2007-02-07 09:18:42 -05:00
c - > tcp_byte_count = 0 ;
2019-01-11 09:12:27 -05:00
if ( c - > tcp_req_info ) {
tcp_req_info_handle_readdone ( c - > tcp_req_info ) ;
} else {
2019-02-25 10:48:27 -05:00
if ( c - > type = = comm_tcp )
comm_point_stop_listening ( c ) ;
2019-01-11 09:12:27 -05:00
fptr_ok ( fptr_whitelist_comm_point ( c - > callback ) ) ;
if ( ( * c - > callback ) ( c , c - > cb_arg , NETEVENT_NOERROR , & c - > repinfo ) ) {
2020-12-02 04:10:27 -05:00
comm_point_start_listening ( c , - 1 ,
adjusted_tcp_timeout ( c ) ) ;
2019-01-11 09:12:27 -05:00
}
2007-02-07 09:18:42 -05:00
}
2007-01-23 08:46:18 -05:00
}
2019-09-03 03:47:27 -04:00
# ifdef HAVE_SSL
/** true if the ssl handshake error has to be squelched from the logs */
2020-01-31 08:03:28 -05:00
int
2019-09-03 03:47:27 -04:00
squelch_err_ssl_handshake ( unsigned long err )
{
if ( verbosity > = VERB_QUERY )
return 0 ; /* only squelch on low verbosity */
/* this is very specific, we could filter on ERR_GET_REASON()
* ( the third element in ERR_PACK ) */
if ( err = = ERR_PACK ( ERR_LIB_SSL , SSL_F_SSL3_GET_RECORD , SSL_R_HTTPS_PROXY_REQUEST ) | |
err = = ERR_PACK ( ERR_LIB_SSL , SSL_F_SSL3_GET_RECORD , SSL_R_HTTP_REQUEST ) | |
err = = ERR_PACK ( ERR_LIB_SSL , SSL_F_SSL3_GET_RECORD , SSL_R_WRONG_VERSION_NUMBER ) | |
2019-09-04 02:44:19 -04:00
err = = ERR_PACK ( ERR_LIB_SSL , SSL_F_SSL3_READ_BYTES , SSL_R_SSLV3_ALERT_BAD_CERTIFICATE )
# ifdef SSL_F_TLS_POST_PROCESS_CLIENT_HELLO
| | err = = ERR_PACK ( ERR_LIB_SSL , SSL_F_TLS_POST_PROCESS_CLIENT_HELLO , SSL_R_NO_SHARED_CIPHER )
# endif
# ifdef SSL_F_TLS_EARLY_POST_PROCESS_CLIENT_HELLO
| | err = = ERR_PACK ( ERR_LIB_SSL , SSL_F_TLS_EARLY_POST_PROCESS_CLIENT_HELLO , SSL_R_UNKNOWN_PROTOCOL )
| | err = = ERR_PACK ( ERR_LIB_SSL , SSL_F_TLS_EARLY_POST_PROCESS_CLIENT_HELLO , SSL_R_UNSUPPORTED_PROTOCOL )
# ifdef SSL_R_VERSION_TOO_LOW
| | err = = ERR_PACK ( ERR_LIB_SSL , SSL_F_TLS_EARLY_POST_PROCESS_CLIENT_HELLO , SSL_R_VERSION_TOO_LOW )
# endif
# endif
)
2019-09-03 03:47:27 -04:00
return 1 ;
return 0 ;
}
# endif /* HAVE_SSL */
2011-10-31 10:48:48 -04:00
/** continue ssl handshake */
2012-07-04 08:33:32 -04:00
# ifdef HAVE_SSL
2011-10-31 10:48:48 -04:00
static int
ssl_handshake ( struct comm_point * c )
{
int r ;
if ( c - > ssl_shake_state = = comm_ssl_shake_hs_read ) {
/* read condition satisfied back to writing */
2021-08-03 06:18:58 -04:00
comm_point_listen_for_rw ( c , 0 , 1 ) ;
2011-10-31 10:48:48 -04:00
c - > ssl_shake_state = comm_ssl_shake_none ;
return 1 ;
}
if ( c - > ssl_shake_state = = comm_ssl_shake_hs_write ) {
/* write condition satisfied, back to reading */
comm_point_listen_for_rw ( c , 1 , 0 ) ;
c - > ssl_shake_state = comm_ssl_shake_none ;
return 1 ;
}
ERR_clear_error ( ) ;
r = SSL_do_handshake ( c - > ssl ) ;
if ( r ! = 1 ) {
int want = SSL_get_error ( c - > ssl , r ) ;
if ( want = = SSL_ERROR_WANT_READ ) {
if ( c - > ssl_shake_state = = comm_ssl_shake_read )
return 1 ;
c - > ssl_shake_state = comm_ssl_shake_read ;
comm_point_listen_for_rw ( c , 1 , 0 ) ;
return 1 ;
} else if ( want = = SSL_ERROR_WANT_WRITE ) {
if ( c - > ssl_shake_state = = comm_ssl_shake_write )
return 1 ;
c - > ssl_shake_state = comm_ssl_shake_write ;
comm_point_listen_for_rw ( c , 0 , 1 ) ;
return 1 ;
} else if ( r = = 0 ) {
return 0 ; /* closed */
} else if ( want = = SSL_ERROR_SYSCALL ) {
/* SYSCALL and errno==0 means closed uncleanly */
2020-01-28 08:32:06 -05:00
# ifdef EPIPE
if ( errno = = EPIPE & & verbosity < 2 )
return 0 ; /* silence 'broken pipe' */
# endif
# ifdef ECONNRESET
if ( errno = = ECONNRESET & & verbosity < 2 )
return 0 ; /* silence reset by peer */
# endif
2011-10-31 10:48:48 -04:00
if ( errno ! = 0 )
log_err ( " SSL_handshake syscall: %s " ,
strerror ( errno ) ) ;
return 0 ;
} else {
2019-09-03 03:47:27 -04:00
unsigned long err = ERR_get_error ( ) ;
if ( ! squelch_err_ssl_handshake ( err ) ) {
log_crypto_err_code ( " ssl handshake failed " , err ) ;
2019-11-20 08:22:06 -05:00
log_addr ( VERB_OPS , " ssl handshake failed " , & c - > repinfo . addr ,
2019-09-03 03:47:27 -04:00
c - > repinfo . addrlen ) ;
}
2011-10-31 10:48:48 -04:00
return 0 ;
}
}
/* this is where peer verification could take place */
2018-02-13 05:35:09 -05:00
if ( ( SSL_get_verify_mode ( c - > ssl ) & SSL_VERIFY_PEER ) ) {
/* verification */
if ( SSL_get_verify_result ( c - > ssl ) = = X509_V_OK ) {
2021-07-30 07:54:43 -04:00
# ifdef HAVE_SSL_GET1_PEER_CERTIFICATE
X509 * x = SSL_get1_peer_certificate ( c - > ssl ) ;
# else
2018-02-13 05:35:09 -05:00
X509 * x = SSL_get_peer_certificate ( c - > ssl ) ;
2021-07-30 07:54:43 -04:00
# endif
2018-02-13 05:35:09 -05:00
if ( ! x ) {
log_addr ( VERB_ALGO , " SSL connection failed: "
" no certificate " ,
& c - > repinfo . addr , c - > repinfo . addrlen ) ;
return 0 ;
}
log_cert ( VERB_ALGO , " peer certificate " , x ) ;
# ifdef HAVE_SSL_GET0_PEERNAME
if ( SSL_get0_peername ( c - > ssl ) ) {
char buf [ 255 ] ;
snprintf ( buf , sizeof ( buf ) , " SSL connection "
" to %s authenticated " ,
SSL_get0_peername ( c - > ssl ) ) ;
log_addr ( VERB_ALGO , buf , & c - > repinfo . addr ,
c - > repinfo . addrlen ) ;
} else {
# endif
log_addr ( VERB_ALGO , " SSL connection "
" authenticated " , & c - > repinfo . addr ,
c - > repinfo . addrlen ) ;
# ifdef HAVE_SSL_GET0_PEERNAME
}
# endif
X509_free ( x ) ;
} else {
2021-07-30 07:54:43 -04:00
# ifdef HAVE_SSL_GET1_PEER_CERTIFICATE
X509 * x = SSL_get1_peer_certificate ( c - > ssl ) ;
# else
2018-02-13 05:35:09 -05:00
X509 * x = SSL_get_peer_certificate ( c - > ssl ) ;
2021-07-30 07:54:43 -04:00
# endif
2018-02-13 05:35:09 -05:00
if ( x ) {
log_cert ( VERB_ALGO , " peer certificate " , x ) ;
X509_free ( x ) ;
}
log_addr ( VERB_ALGO , " SSL connection failed: "
" failed to authenticate " ,
& c - > repinfo . addr , c - > repinfo . addrlen ) ;
return 0 ;
}
} else {
/* unauthenticated, the verify peer flag was not set
* in c - > ssl when the ssl object was created from ssl_ctx */
log_addr ( VERB_ALGO , " SSL connection " , & c - > repinfo . addr ,
c - > repinfo . addrlen ) ;
}
2011-10-31 10:48:48 -04:00
2021-04-19 13:05:50 -04:00
# ifdef HAVE_SSL_GET0_ALPN_SELECTED
2020-05-07 10:36:26 -04:00
/* check if http2 use is negotiated */
if ( c - > type = = comm_http & & c - > h2_session ) {
const unsigned char * alpn ;
unsigned int alpnlen = 0 ;
SSL_get0_alpn_selected ( c - > ssl , & alpn , & alpnlen ) ;
if ( alpnlen = = 2 & & memcmp ( " h2 " , alpn , 2 ) = = 0 ) {
/* connection upgraded to HTTP2 */
c - > tcp_do_toggle_rw = 0 ;
2020-09-16 12:25:02 -04:00
c - > use_h2 = 1 ;
2020-05-07 10:36:26 -04:00
}
}
2021-04-19 13:05:50 -04:00
# endif
2020-05-07 10:36:26 -04:00
2011-10-31 10:48:48 -04:00
/* setup listen rw correctly */
if ( c - > tcp_is_reading ) {
if ( c - > ssl_shake_state ! = comm_ssl_shake_read )
comm_point_listen_for_rw ( c , 1 , 0 ) ;
} else {
2021-08-03 06:18:58 -04:00
comm_point_listen_for_rw ( c , 0 , 1 ) ;
2011-10-31 10:48:48 -04:00
}
c - > ssl_shake_state = comm_ssl_shake_none ;
return 1 ;
}
2012-07-04 08:33:32 -04:00
# endif /* HAVE_SSL */
2011-10-31 10:48:48 -04:00
/** ssl read callback on TCP */
static int
ssl_handle_read ( struct comm_point * c )
{
2012-06-20 11:11:53 -04:00
# ifdef HAVE_SSL
2011-10-31 10:48:48 -04:00
int r ;
if ( c - > ssl_shake_state ! = comm_ssl_shake_none ) {
if ( ! ssl_handshake ( c ) )
return 0 ;
if ( c - > ssl_shake_state ! = comm_ssl_shake_none )
return 1 ;
}
if ( c - > tcp_byte_count < sizeof ( uint16_t ) ) {
/* read length bytes */
ERR_clear_error ( ) ;
2013-12-03 04:11:16 -05:00
if ( ( r = SSL_read ( c - > ssl , ( void * ) sldns_buffer_at ( c - > buffer ,
2011-10-31 10:48:48 -04:00
c - > tcp_byte_count ) , ( int ) ( sizeof ( uint16_t ) -
c - > tcp_byte_count ) ) ) < = 0 ) {
int want = SSL_get_error ( c - > ssl , r ) ;
if ( want = = SSL_ERROR_ZERO_RETURN ) {
2019-01-11 09:12:27 -05:00
if ( c - > tcp_req_info )
return tcp_req_info_handle_read_close ( c - > tcp_req_info ) ;
2011-10-31 10:48:48 -04:00
return 0 ; /* shutdown, closed */
} else if ( want = = SSL_ERROR_WANT_READ ) {
2021-07-16 03:12:06 -04:00
# ifdef USE_WINSOCK
2018-05-07 04:31:17 -04:00
ub_winsock_tcp_wouldblock ( c - > ev - > ev , UB_EV_READ ) ;
2021-07-16 03:12:06 -04:00
# endif
2011-10-31 10:48:48 -04:00
return 1 ; /* read more later */
} else if ( want = = SSL_ERROR_WANT_WRITE ) {
c - > ssl_shake_state = comm_ssl_shake_hs_write ;
comm_point_listen_for_rw ( c , 0 , 1 ) ;
return 1 ;
} else if ( want = = SSL_ERROR_SYSCALL ) {
2019-04-11 11:04:32 -04:00
# ifdef ECONNRESET
if ( errno = = ECONNRESET & & verbosity < 2 )
return 0 ; /* silence reset by peer */
# endif
2011-10-31 10:48:48 -04:00
if ( errno ! = 0 )
log_err ( " SSL_read syscall: %s " ,
strerror ( errno ) ) ;
return 0 ;
}
log_crypto_err ( " could not SSL_read " ) ;
return 0 ;
}
c - > tcp_byte_count + = r ;
2018-05-07 04:31:17 -04:00
if ( c - > tcp_byte_count < sizeof ( uint16_t ) )
2011-10-31 10:48:48 -04:00
return 1 ;
2013-12-03 04:11:16 -05:00
if ( sldns_buffer_read_u16_at ( c - > buffer , 0 ) >
sldns_buffer_capacity ( c - > buffer ) ) {
2011-10-31 10:48:48 -04:00
verbose ( VERB_QUERY , " ssl: dropped larger than buffer " ) ;
return 0 ;
}
2013-12-03 04:11:16 -05:00
sldns_buffer_set_limit ( c - > buffer ,
sldns_buffer_read_u16_at ( c - > buffer , 0 ) ) ;
if ( sldns_buffer_limit ( c - > buffer ) < LDNS_HEADER_SIZE ) {
2011-10-31 10:48:48 -04:00
verbose ( VERB_QUERY , " ssl: dropped bogus too short. " ) ;
return 0 ;
}
2018-05-07 04:31:17 -04:00
sldns_buffer_skip ( c - > buffer , ( ssize_t ) ( c - > tcp_byte_count - sizeof ( uint16_t ) ) ) ;
2011-10-31 10:48:48 -04:00
verbose ( VERB_ALGO , " Reading ssl tcp query of length %d " ,
2013-12-03 04:11:16 -05:00
( int ) sldns_buffer_limit ( c - > buffer ) ) ;
2011-10-31 10:48:48 -04:00
}
2018-05-07 04:31:17 -04:00
if ( sldns_buffer_remaining ( c - > buffer ) > 0 ) {
ERR_clear_error ( ) ;
r = SSL_read ( c - > ssl , ( void * ) sldns_buffer_current ( c - > buffer ) ,
( int ) sldns_buffer_remaining ( c - > buffer ) ) ;
if ( r < = 0 ) {
int want = SSL_get_error ( c - > ssl , r ) ;
if ( want = = SSL_ERROR_ZERO_RETURN ) {
2019-01-11 09:12:27 -05:00
if ( c - > tcp_req_info )
return tcp_req_info_handle_read_close ( c - > tcp_req_info ) ;
2018-05-07 04:31:17 -04:00
return 0 ; /* shutdown, closed */
} else if ( want = = SSL_ERROR_WANT_READ ) {
2021-07-16 03:12:06 -04:00
# ifdef USE_WINSOCK
2018-05-07 04:31:17 -04:00
ub_winsock_tcp_wouldblock ( c - > ev - > ev , UB_EV_READ ) ;
2021-07-16 03:12:06 -04:00
# endif
2018-05-07 04:31:17 -04:00
return 1 ; /* read more later */
} else if ( want = = SSL_ERROR_WANT_WRITE ) {
c - > ssl_shake_state = comm_ssl_shake_hs_write ;
comm_point_listen_for_rw ( c , 0 , 1 ) ;
return 1 ;
} else if ( want = = SSL_ERROR_SYSCALL ) {
2019-04-11 11:04:32 -04:00
# ifdef ECONNRESET
if ( errno = = ECONNRESET & & verbosity < 2 )
return 0 ; /* silence reset by peer */
# endif
2018-05-07 04:31:17 -04:00
if ( errno ! = 0 )
log_err ( " SSL_read syscall: %s " ,
strerror ( errno ) ) ;
return 0 ;
}
log_crypto_err ( " could not SSL_read " ) ;
2011-10-31 10:48:48 -04:00
return 0 ;
}
2018-05-07 04:31:17 -04:00
sldns_buffer_skip ( c - > buffer , ( ssize_t ) r ) ;
2011-10-31 10:48:48 -04:00
}
2013-12-03 04:11:16 -05:00
if ( sldns_buffer_remaining ( c - > buffer ) < = 0 ) {
2011-10-31 10:48:48 -04:00
tcp_callback_reader ( c ) ;
}
return 1 ;
2012-07-04 08:33:32 -04:00
# else
( void ) c ;
return 0 ;
2012-06-20 11:11:53 -04:00
# endif /* HAVE_SSL */
2011-10-31 10:48:48 -04:00
}
/** ssl write callback on TCP */
static int
ssl_handle_write ( struct comm_point * c )
{
2012-06-20 11:11:53 -04:00
# ifdef HAVE_SSL
2011-10-31 10:48:48 -04:00
int r ;
if ( c - > ssl_shake_state ! = comm_ssl_shake_none ) {
if ( ! ssl_handshake ( c ) )
return 0 ;
if ( c - > ssl_shake_state ! = comm_ssl_shake_none )
return 1 ;
}
/* ignore return, if fails we may simply block */
2019-11-13 09:16:27 -05:00
( void ) SSL_set_mode ( c - > ssl , ( long ) SSL_MODE_ENABLE_PARTIAL_WRITE ) ;
2020-06-26 10:05:15 -04:00
if ( ( c - > tcp_write_and_read ? c - > tcp_write_byte_count : c - > tcp_byte_count ) < sizeof ( uint16_t ) ) {
uint16_t len = htons ( c - > tcp_write_and_read ? c - > tcp_write_pkt_len : sldns_buffer_limit ( c - > buffer ) ) ;
2011-10-31 10:48:48 -04:00
ERR_clear_error ( ) ;
2020-06-26 10:05:15 -04:00
if ( c - > tcp_write_and_read ) {
if ( c - > tcp_write_pkt_len + 2 < LDNS_RR_BUF_SIZE ) {
/* combine the tcp length and the query for
* write , this emulates writev */
uint8_t buf [ LDNS_RR_BUF_SIZE ] ;
memmove ( buf , & len , sizeof ( uint16_t ) ) ;
memmove ( buf + sizeof ( uint16_t ) ,
c - > tcp_write_pkt ,
c - > tcp_write_pkt_len ) ;
r = SSL_write ( c - > ssl ,
( void * ) ( buf + c - > tcp_write_byte_count ) ,
c - > tcp_write_pkt_len + 2 -
c - > tcp_write_byte_count ) ;
} else {
r = SSL_write ( c - > ssl ,
( void * ) ( ( ( uint8_t * ) & len ) + c - > tcp_write_byte_count ) ,
( int ) ( sizeof ( uint16_t ) - c - > tcp_write_byte_count ) ) ;
}
} else if ( sizeof ( uint16_t ) + sldns_buffer_remaining ( c - > buffer ) <
2018-04-05 04:10:25 -04:00
LDNS_RR_BUF_SIZE ) {
/* combine the tcp length and the query for write,
* this emulates writev */
uint8_t buf [ LDNS_RR_BUF_SIZE ] ;
memmove ( buf , & len , sizeof ( uint16_t ) ) ;
memmove ( buf + sizeof ( uint16_t ) ,
sldns_buffer_current ( c - > buffer ) ,
sldns_buffer_remaining ( c - > buffer ) ) ;
r = SSL_write ( c - > ssl , ( void * ) ( buf + c - > tcp_byte_count ) ,
( int ) ( sizeof ( uint16_t ) +
sldns_buffer_remaining ( c - > buffer )
- c - > tcp_byte_count ) ) ;
} else {
r = SSL_write ( c - > ssl ,
( void * ) ( ( ( uint8_t * ) & len ) + c - > tcp_byte_count ) ,
( int ) ( sizeof ( uint16_t ) - c - > tcp_byte_count ) ) ;
}
2011-10-31 10:48:48 -04:00
if ( r < = 0 ) {
int want = SSL_get_error ( c - > ssl , r ) ;
if ( want = = SSL_ERROR_ZERO_RETURN ) {
return 0 ; /* closed */
} else if ( want = = SSL_ERROR_WANT_READ ) {
2019-04-15 07:52:21 -04:00
c - > ssl_shake_state = comm_ssl_shake_hs_read ;
2011-10-31 10:48:48 -04:00
comm_point_listen_for_rw ( c , 1 , 0 ) ;
return 1 ; /* wait for read condition */
} else if ( want = = SSL_ERROR_WANT_WRITE ) {
2021-07-16 03:12:06 -04:00
# ifdef USE_WINSOCK
2018-05-07 04:31:17 -04:00
ub_winsock_tcp_wouldblock ( c - > ev - > ev , UB_EV_WRITE ) ;
2021-07-16 03:12:06 -04:00
# endif
2011-10-31 10:48:48 -04:00
return 1 ; /* write more later */
} else if ( want = = SSL_ERROR_SYSCALL ) {
2019-04-11 11:04:32 -04:00
# ifdef EPIPE
if ( errno = = EPIPE & & verbosity < 2 )
return 0 ; /* silence 'broken pipe' */
# endif
2011-10-31 10:48:48 -04:00
if ( errno ! = 0 )
log_err ( " SSL_write syscall: %s " ,
strerror ( errno ) ) ;
return 0 ;
}
log_crypto_err ( " could not SSL_write " ) ;
return 0 ;
}
2020-06-26 10:05:15 -04:00
if ( c - > tcp_write_and_read ) {
c - > tcp_write_byte_count + = r ;
if ( c - > tcp_write_byte_count < sizeof ( uint16_t ) )
return 1 ;
} else {
c - > tcp_byte_count + = r ;
if ( c - > tcp_byte_count < sizeof ( uint16_t ) )
return 1 ;
sldns_buffer_set_position ( c - > buffer , c - > tcp_byte_count -
sizeof ( uint16_t ) ) ;
}
if ( ( ! c - > tcp_write_and_read & & sldns_buffer_remaining ( c - > buffer ) = = 0 ) | | ( c - > tcp_write_and_read & & c - > tcp_write_byte_count = = c - > tcp_write_pkt_len + 2 ) ) {
2011-10-31 10:48:48 -04:00
tcp_callback_writer ( c ) ;
return 1 ;
}
}
2020-06-26 10:05:15 -04:00
log_assert ( c - > tcp_write_and_read | | sldns_buffer_remaining ( c - > buffer ) > 0 ) ;
log_assert ( ! c - > tcp_write_and_read | | c - > tcp_write_byte_count < c - > tcp_write_pkt_len + 2 ) ;
2011-10-31 10:48:48 -04:00
ERR_clear_error ( ) ;
2020-06-26 10:05:15 -04:00
if ( c - > tcp_write_and_read ) {
r = SSL_write ( c - > ssl , ( void * ) ( c - > tcp_write_pkt + c - > tcp_write_byte_count - 2 ) ,
( int ) ( c - > tcp_write_pkt_len + 2 - c - > tcp_write_byte_count ) ) ;
} else {
r = SSL_write ( c - > ssl , ( void * ) sldns_buffer_current ( c - > buffer ) ,
( int ) sldns_buffer_remaining ( c - > buffer ) ) ;
}
2011-10-31 10:48:48 -04:00
if ( r < = 0 ) {
int want = SSL_get_error ( c - > ssl , r ) ;
if ( want = = SSL_ERROR_ZERO_RETURN ) {
return 0 ; /* closed */
} else if ( want = = SSL_ERROR_WANT_READ ) {
2019-04-15 07:52:21 -04:00
c - > ssl_shake_state = comm_ssl_shake_hs_read ;
2011-10-31 10:48:48 -04:00
comm_point_listen_for_rw ( c , 1 , 0 ) ;
return 1 ; /* wait for read condition */
} else if ( want = = SSL_ERROR_WANT_WRITE ) {
2021-07-16 03:12:06 -04:00
# ifdef USE_WINSOCK
2018-05-07 04:31:17 -04:00
ub_winsock_tcp_wouldblock ( c - > ev - > ev , UB_EV_WRITE ) ;
2021-07-16 03:12:06 -04:00
# endif
2011-10-31 10:48:48 -04:00
return 1 ; /* write more later */
} else if ( want = = SSL_ERROR_SYSCALL ) {
2019-04-11 11:04:32 -04:00
# ifdef EPIPE
if ( errno = = EPIPE & & verbosity < 2 )
return 0 ; /* silence 'broken pipe' */
# endif
2011-10-31 10:48:48 -04:00
if ( errno ! = 0 )
log_err ( " SSL_write syscall: %s " ,
strerror ( errno ) ) ;
return 0 ;
}
log_crypto_err ( " could not SSL_write " ) ;
return 0 ;
}
2020-06-26 10:05:15 -04:00
if ( c - > tcp_write_and_read ) {
c - > tcp_write_byte_count + = r ;
} else {
sldns_buffer_skip ( c - > buffer , ( ssize_t ) r ) ;
}
2011-10-31 10:48:48 -04:00
2020-06-26 10:05:15 -04:00
if ( ( ! c - > tcp_write_and_read & & sldns_buffer_remaining ( c - > buffer ) = = 0 ) | | ( c - > tcp_write_and_read & & c - > tcp_write_byte_count = = c - > tcp_write_pkt_len + 2 ) ) {
2011-10-31 10:48:48 -04:00
tcp_callback_writer ( c ) ;
}
return 1 ;
2012-07-04 08:33:32 -04:00
# else
( void ) c ;
return 0 ;
2012-06-20 11:11:53 -04:00
# endif /* HAVE_SSL */
2011-10-31 10:48:48 -04:00
}
2011-11-01 05:26:58 -04:00
/** handle ssl tcp connection with dns contents */
static int
2020-07-10 09:37:30 -04:00
ssl_handle_it ( struct comm_point * c , int is_write )
2011-11-01 05:26:58 -04:00
{
2020-07-10 09:37:30 -04:00
/* handle case where renegotiation wants read during write call
* or write during read calls */
if ( is_write & & c - > ssl_shake_state = = comm_ssl_shake_hs_write )
return ssl_handle_read ( c ) ;
else if ( ! is_write & & c - > ssl_shake_state = = comm_ssl_shake_hs_read )
return ssl_handle_write ( c ) ;
/* handle read events for read operation and write events for a
* write operation */
else if ( ! is_write )
2011-11-01 05:26:58 -04:00
return ssl_handle_read ( c ) ;
return ssl_handle_write ( c ) ;
}
2007-02-07 10:44:19 -05:00
/** Handle tcp reading callback.
* @ param fd : file descriptor of socket .
* @ param c : comm point to read from into buffer .
2007-02-26 09:49:11 -05:00
* @ param short_ok : if true , very short packets are OK ( for comm_local ) .
2007-02-07 10:44:19 -05:00
* @ return : 0 on error
*/
static int
2007-02-26 09:49:11 -05:00
comm_point_tcp_handle_read ( int fd , struct comm_point * c , int short_ok )
2007-01-23 11:10:23 -05:00
{
2007-02-07 09:18:42 -05:00
ssize_t r ;
2007-02-26 09:49:11 -05:00
log_assert ( c - > type = = comm_tcp | | c - > type = = comm_local ) ;
2011-11-01 05:26:58 -04:00
if ( c - > ssl )
2020-07-10 09:37:30 -04:00
return ssl_handle_it ( c , 0 ) ;
if ( ! c - > tcp_is_reading & & ! c - > tcp_write_and_read )
2007-02-07 10:44:19 -05:00
return 0 ;
2007-02-07 09:18:42 -05:00
2007-10-16 08:26:09 -04:00
log_assert ( fd ! = - 1 ) ;
2007-02-07 09:18:42 -05:00
if ( c - > tcp_byte_count < sizeof ( uint16_t ) ) {
/* read length bytes */
2013-12-03 04:11:16 -05:00
r = recv ( fd , ( void * ) sldns_buffer_at ( c - > buffer , c - > tcp_byte_count ) ,
2008-06-18 10:27:30 -04:00
sizeof ( uint16_t ) - c - > tcp_byte_count , 0 ) ;
2019-01-11 09:12:27 -05:00
if ( r = = 0 ) {
if ( c - > tcp_req_info )
return tcp_req_info_handle_read_close ( c - > tcp_req_info ) ;
2007-02-07 10:44:19 -05:00
return 0 ;
2019-01-11 09:12:27 -05:00
} else if ( r = = - 1 ) {
2008-06-18 10:27:30 -04:00
# ifndef USE_WINSOCK
2007-02-07 10:44:19 -05:00
if ( errno = = EINTR | | errno = = EAGAIN )
return 1 ;
2008-06-13 11:32:16 -04:00
# ifdef ECONNRESET
2007-10-30 10:59:01 -04:00
if ( errno = = ECONNRESET & & verbosity < 2 )
return 0 ; /* silence reset by peer */
2008-06-13 11:32:16 -04:00
# endif
2021-03-19 12:43:36 -04:00
# ifdef ECONNREFUSED
if ( errno = = ECONNREFUSED & & verbosity < 2 )
return 0 ; /* silence reset by peer */
# endif
2020-12-11 04:30:54 -05:00
# ifdef ENETUNREACH
if ( errno = = ENETUNREACH & & verbosity < 2 )
return 0 ; /* silence it */
# endif
# ifdef EHOSTDOWN
if ( errno = = EHOSTDOWN & & verbosity < 2 )
return 0 ; /* silence it */
# endif
# ifdef EHOSTUNREACH
if ( errno = = EHOSTUNREACH & & verbosity < 2 )
return 0 ; /* silence it */
# endif
# ifdef ENETDOWN
if ( errno = = ENETDOWN & & verbosity < 2 )
return 0 ; /* silence it */
# endif
# ifdef EACCES
if ( errno = = EACCES & & verbosity < 2 )
return 0 ; /* silence it */
# endif
2020-12-02 03:51:26 -05:00
# ifdef ENOTCONN
if ( errno = = ENOTCONN ) {
log_err_addr ( " read (in tcp s) failed and this could be because TCP Fast Open is enabled [--disable-tfo-client --disable-tfo-server] but does not work " , sock_strerror ( errno ) ,
& c - > repinfo . addr , c - > repinfo . addrlen ) ;
return 0 ;
}
# endif
2008-06-18 10:27:30 -04:00
# else /* USE_WINSOCK */
2021-03-19 12:43:36 -04:00
if ( WSAGetLastError ( ) = = WSAECONNREFUSED & & verbosity < 2 )
return 0 ;
if ( WSAGetLastError ( ) = = WSAEHOSTDOWN & & verbosity < 2 )
return 0 ;
if ( WSAGetLastError ( ) = = WSAEHOSTUNREACH & & verbosity < 2 )
return 0 ;
if ( WSAGetLastError ( ) = = WSAENETDOWN & & verbosity < 2 )
return 0 ;
if ( WSAGetLastError ( ) = = WSAENETUNREACH & & verbosity < 2 )
return 0 ;
2008-06-23 09:52:03 -04:00
if ( WSAGetLastError ( ) = = WSAECONNRESET )
return 0 ;
if ( WSAGetLastError ( ) = = WSAEINPROGRESS )
return 1 ;
if ( WSAGetLastError ( ) = = WSAEWOULDBLOCK ) {
2016-03-07 09:10:06 -05:00
ub_winsock_tcp_wouldblock ( c - > ev - > ev ,
UB_EV_READ ) ;
2008-06-18 10:27:30 -04:00
return 1 ;
2008-06-23 09:52:03 -04:00
}
2008-06-18 10:27:30 -04:00
# endif
2020-08-31 03:12:01 -04:00
log_err_addr ( " read (in tcp s) " , sock_strerror ( errno ) ,
& c - > repinfo . addr , c - > repinfo . addrlen ) ;
2007-02-07 10:44:19 -05:00
return 0 ;
2007-02-07 09:18:42 -05:00
}
c - > tcp_byte_count + = r ;
if ( c - > tcp_byte_count ! = sizeof ( uint16_t ) )
2007-02-07 10:44:19 -05:00
return 1 ;
2013-12-03 04:11:16 -05:00
if ( sldns_buffer_read_u16_at ( c - > buffer , 0 ) >
sldns_buffer_capacity ( c - > buffer ) ) {
2008-02-07 04:46:49 -05:00
verbose ( VERB_QUERY , " tcp: dropped larger than buffer " ) ;
2007-02-07 10:44:19 -05:00
return 0 ;
}
2013-12-03 04:11:16 -05:00
sldns_buffer_set_limit ( c - > buffer ,
sldns_buffer_read_u16_at ( c - > buffer , 0 ) ) ;
2007-02-26 09:49:11 -05:00
if ( ! short_ok & &
2013-12-03 04:11:16 -05:00
sldns_buffer_limit ( c - > buffer ) < LDNS_HEADER_SIZE ) {
2008-02-07 04:46:49 -05:00
verbose ( VERB_QUERY , " tcp: dropped bogus too short. " ) ;
2007-02-07 10:44:19 -05:00
return 0 ;
2007-02-07 09:18:42 -05:00
}
2007-02-15 10:50:22 -05:00
verbose ( VERB_ALGO , " Reading tcp query of length %d " ,
2013-12-03 04:11:16 -05:00
( int ) sldns_buffer_limit ( c - > buffer ) ) ;
2007-02-07 09:18:42 -05:00
}
2021-08-03 08:08:30 -04:00
if ( sldns_buffer_remaining ( c - > buffer ) = = 0 )
log_err ( " in comm_point_tcp_handle_read buffer_remaining is not > 0 as expected, continuing with (harmless) 0 length recv " ) ;
2013-12-03 04:11:16 -05:00
r = recv ( fd , ( void * ) sldns_buffer_current ( c - > buffer ) ,
sldns_buffer_remaining ( c - > buffer ) , 0 ) ;
2007-02-07 09:18:42 -05:00
if ( r = = 0 ) {
2019-01-11 09:12:27 -05:00
if ( c - > tcp_req_info )
return tcp_req_info_handle_read_close ( c - > tcp_req_info ) ;
2007-02-07 10:44:19 -05:00
return 0 ;
2007-02-07 09:18:42 -05:00
} else if ( r = = - 1 ) {
2008-06-18 10:27:30 -04:00
# ifndef USE_WINSOCK
2007-02-07 10:44:19 -05:00
if ( errno = = EINTR | | errno = = EAGAIN )
return 1 ;
2008-06-18 10:27:30 -04:00
# else /* USE_WINSOCK */
2008-06-23 09:52:03 -04:00
if ( WSAGetLastError ( ) = = WSAECONNRESET )
return 0 ;
if ( WSAGetLastError ( ) = = WSAEINPROGRESS )
2008-06-18 10:27:30 -04:00
return 1 ;
2008-06-23 09:52:03 -04:00
if ( WSAGetLastError ( ) = = WSAEWOULDBLOCK ) {
2016-03-07 09:10:06 -05:00
ub_winsock_tcp_wouldblock ( c - > ev - > ev , UB_EV_READ ) ;
2008-06-23 09:52:03 -04:00
return 1 ;
}
2008-06-18 10:27:30 -04:00
# endif
2020-08-31 03:12:01 -04:00
log_err_addr ( " read (in tcp r) " , sock_strerror ( errno ) ,
& c - > repinfo . addr , c - > repinfo . addrlen ) ;
2007-02-07 10:44:19 -05:00
return 0 ;
2007-02-07 09:18:42 -05:00
}
2013-12-03 04:11:16 -05:00
sldns_buffer_skip ( c - > buffer , r ) ;
if ( sldns_buffer_remaining ( c - > buffer ) < = 0 ) {
2007-02-07 09:18:42 -05:00
tcp_callback_reader ( c ) ;
}
2007-02-07 10:44:19 -05:00
return 1 ;
}
2007-05-08 09:25:21 -04:00
/**
* Handle tcp writing callback .
2007-02-07 10:44:19 -05:00
* @ param fd : file descriptor of socket .
* @ param c : comm point to write buffer out of .
* @ return : 0 on error
*/
static int
comm_point_tcp_handle_write ( int fd , struct comm_point * c )
{
ssize_t r ;
2017-03-20 11:09:06 -04:00
struct sldns_buffer * buffer ;
2007-02-07 10:44:19 -05:00
log_assert ( c - > type = = comm_tcp ) ;
2017-03-20 10:55:31 -04:00
# ifdef USE_DNSCRYPT
2017-03-20 11:09:06 -04:00
buffer = c - > dnscrypt_buffer ;
# else
buffer = c - > buffer ;
2017-03-20 10:55:31 -04:00
# endif
2020-07-10 09:37:30 -04:00
if ( c - > tcp_is_reading & & ! c - > ssl & & ! c - > tcp_write_and_read )
2007-02-07 10:44:19 -05:00
return 0 ;
2007-10-16 08:26:09 -04:00
log_assert ( fd ! = - 1 ) ;
2020-06-26 10:05:15 -04:00
if ( ( ( ! c - > tcp_write_and_read & & c - > tcp_byte_count = = 0 ) | | ( c - > tcp_write_and_read & & c - > tcp_write_byte_count = = 0 ) ) & & c - > tcp_check_nb_connect ) {
2007-05-08 09:25:21 -04:00
/* check for pending error from nonblocking connect */
/* from Stevens, unix network programming, vol1, 3rd ed, p450*/
int error = 0 ;
socklen_t len = ( socklen_t ) sizeof ( error ) ;
2008-06-13 11:32:16 -04:00
if ( getsockopt ( fd , SOL_SOCKET , SO_ERROR , ( void * ) & error ,
& len ) < 0 ) {
2008-06-25 10:36:10 -04:00
# ifndef USE_WINSOCK
2007-05-08 09:25:21 -04:00
error = errno ; /* on solaris errno is error */
2008-06-25 10:36:10 -04:00
# else /* USE_WINSOCK */
error = WSAGetLastError ( ) ;
# endif
2007-05-08 09:25:21 -04:00
}
2008-06-25 10:36:10 -04:00
# ifndef USE_WINSOCK
2008-06-13 11:32:16 -04:00
# if defined(EINPROGRESS) && defined(EWOULDBLOCK)
2007-05-08 09:25:21 -04:00
if ( error = = EINPROGRESS | | error = = EWOULDBLOCK )
return 1 ; /* try again later */
2012-02-28 10:06:59 -05:00
else
2008-06-13 11:32:16 -04:00
# endif
2012-02-28 10:06:59 -05:00
if ( error ! = 0 & & verbosity < 2 )
2010-12-17 04:10:46 -05:00
return 0 ; /* silence lots of chatter in the logs */
2008-01-30 09:46:01 -05:00
else if ( error ! = 0 ) {
2014-04-10 10:40:20 -04:00
log_err_addr ( " tcp connect " , strerror ( error ) ,
& c - > repinfo . addr , c - > repinfo . addrlen ) ;
2008-06-25 10:36:10 -04:00
# else /* USE_WINSOCK */
/* examine error */
if ( error = = WSAEINPROGRESS )
return 1 ;
else if ( error = = WSAEWOULDBLOCK ) {
2016-03-07 09:10:06 -05:00
ub_winsock_tcp_wouldblock ( c - > ev - > ev , UB_EV_WRITE ) ;
2008-06-25 10:36:10 -04:00
return 1 ;
2010-12-17 04:10:46 -05:00
} else if ( error ! = 0 & & verbosity < 2 )
2008-06-25 10:36:10 -04:00
return 0 ;
else if ( error ! = 0 ) {
2014-04-10 10:40:20 -04:00
log_err_addr ( " tcp connect " , wsa_strerror ( error ) ,
& c - > repinfo . addr , c - > repinfo . addrlen ) ;
2008-06-25 10:36:10 -04:00
# endif /* USE_WINSOCK */
2007-05-08 09:25:21 -04:00
return 0 ;
}
}
2011-10-31 10:48:48 -04:00
if ( c - > ssl )
2020-07-10 09:37:30 -04:00
return ssl_handle_it ( c , 1 ) ;
2007-02-07 10:44:19 -05:00
2016-07-14 03:06:34 -04:00
# ifdef USE_MSG_FASTOPEN
/* Only try this on first use of a connection that uses tfo,
otherwise fall through to normal write */
2016-08-03 02:20:26 -04:00
/* Also, TFO support on WINDOWS not implemented at the moment */
2016-07-14 03:06:34 -04:00
if ( c - > tcp_do_fastopen = = 1 ) {
2016-08-03 02:20:26 -04:00
/* this form of sendmsg() does both a connect() and send() so need to
2016-07-14 03:06:34 -04:00
look for various flavours of error */
2020-06-26 10:05:15 -04:00
uint16_t len = htons ( c - > tcp_write_and_read ? c - > tcp_write_pkt_len : sldns_buffer_limit ( buffer ) ) ;
2016-08-03 02:20:26 -04:00
struct msghdr msg ;
struct iovec iov [ 2 ] ;
c - > tcp_do_fastopen = 0 ;
memset ( & msg , 0 , sizeof ( msg ) ) ;
2020-06-26 10:05:15 -04:00
if ( c - > tcp_write_and_read ) {
iov [ 0 ] . iov_base = ( uint8_t * ) & len + c - > tcp_write_byte_count ;
iov [ 0 ] . iov_len = sizeof ( uint16_t ) - c - > tcp_write_byte_count ;
iov [ 1 ] . iov_base = c - > tcp_write_pkt ;
iov [ 1 ] . iov_len = c - > tcp_write_pkt_len ;
} else {
iov [ 0 ] . iov_base = ( uint8_t * ) & len + c - > tcp_byte_count ;
iov [ 0 ] . iov_len = sizeof ( uint16_t ) - c - > tcp_byte_count ;
iov [ 1 ] . iov_base = sldns_buffer_begin ( buffer ) ;
iov [ 1 ] . iov_len = sldns_buffer_limit ( buffer ) ;
}
2016-08-03 02:20:26 -04:00
log_assert ( iov [ 0 ] . iov_len > 0 ) ;
msg . msg_name = & c - > repinfo . addr ;
msg . msg_namelen = c - > repinfo . addrlen ;
msg . msg_iov = iov ;
msg . msg_iovlen = 2 ;
r = sendmsg ( fd , & msg , MSG_FASTOPEN ) ;
2016-07-14 03:06:34 -04:00
if ( r = = - 1 ) {
# if defined(EINPROGRESS) && defined(EWOULDBLOCK)
/* Handshake is underway, maybe because no TFO cookie available.
2017-09-15 10:29:28 -04:00
Come back to write the message */
2016-07-14 03:06:34 -04:00
if ( errno = = EINPROGRESS | | errno = = EWOULDBLOCK )
return 1 ;
# endif
if ( errno = = EINTR | | errno = = EAGAIN )
return 1 ;
/* Not handling EISCONN here as shouldn't ever hit that case.*/
2021-09-01 10:21:10 -04:00
if ( errno ! = EPIPE
# ifdef EOPNOTSUPP
/* if /proc/sys/net/ipv4/tcp_fastopen is
* disabled on Linux , sendmsg may return
* ' Operation not supported ' , if so
* fallthrough to ordinary connect . */
& & errno ! = EOPNOTSUPP
# endif
& & errno ! = 0 ) {
if ( verbosity < 2 )
return 0 ; /* silence lots of chatter in the logs */
2016-08-03 02:20:26 -04:00
log_err_addr ( " tcp sendmsg " , strerror ( errno ) ,
2016-07-14 03:06:34 -04:00
& c - > repinfo . addr , c - > repinfo . addrlen ) ;
2017-05-31 07:09:15 -04:00
return 0 ;
}
2021-09-01 10:21:10 -04:00
verbose ( VERB_ALGO , " tcp sendmsg for fastopen failed (with %s), try normal connect " , strerror ( errno ) ) ;
2017-05-31 07:09:15 -04:00
/* fallthrough to nonFASTOPEN
2017-06-01 05:06:14 -04:00
* ( MSG_FASTOPEN on Linux 3 produces EPIPE )
* we need to perform connect ( ) */
2017-06-06 03:59:32 -04:00
if ( connect ( fd , ( struct sockaddr * ) & c - > repinfo . addr , c - > repinfo . addrlen ) = = - 1 ) {
2017-06-01 05:06:14 -04:00
# ifdef EINPROGRESS
if ( errno = = EINPROGRESS )
return 1 ; /* wait until connect done*/
# endif
# ifdef USE_WINSOCK
if ( WSAGetLastError ( ) = = WSAEINPROGRESS | |
WSAGetLastError ( ) = = WSAEWOULDBLOCK )
return 1 ; /* wait until connect done*/
# endif
if ( tcp_connect_errno_needs_log (
2017-06-06 03:59:32 -04:00
( struct sockaddr * ) & c - > repinfo . addr , c - > repinfo . addrlen ) ) {
2017-06-01 05:06:14 -04:00
log_err_addr ( " outgoing tcp: connect after EPIPE for fastopen " ,
strerror ( errno ) , & c - > repinfo . addr , c - > repinfo . addrlen ) ;
}
return 0 ;
}
2016-07-14 03:06:34 -04:00
} else {
2020-06-26 10:05:15 -04:00
if ( c - > tcp_write_and_read ) {
c - > tcp_write_byte_count + = r ;
if ( c - > tcp_write_byte_count < sizeof ( uint16_t ) )
return 1 ;
} else {
c - > tcp_byte_count + = r ;
if ( c - > tcp_byte_count < sizeof ( uint16_t ) )
return 1 ;
sldns_buffer_set_position ( buffer , c - > tcp_byte_count -
sizeof ( uint16_t ) ) ;
}
if ( ( ! c - > tcp_write_and_read & & sldns_buffer_remaining ( buffer ) = = 0 ) | | ( c - > tcp_write_and_read & & c - > tcp_write_byte_count = = c - > tcp_write_pkt_len + 2 ) ) {
2016-07-14 03:06:34 -04:00
tcp_callback_writer ( c ) ;
return 1 ;
}
}
}
# endif /* USE_MSG_FASTOPEN */
2020-06-26 10:05:15 -04:00
if ( ( c - > tcp_write_and_read ? c - > tcp_write_byte_count : c - > tcp_byte_count ) < sizeof ( uint16_t ) ) {
uint16_t len = htons ( c - > tcp_write_and_read ? c - > tcp_write_pkt_len : sldns_buffer_limit ( buffer ) ) ;
2008-06-13 11:32:16 -04:00
# ifdef HAVE_WRITEV
2007-05-04 08:45:59 -04:00
struct iovec iov [ 2 ] ;
2020-06-26 10:05:15 -04:00
if ( c - > tcp_write_and_read ) {
iov [ 0 ] . iov_base = ( uint8_t * ) & len + c - > tcp_write_byte_count ;
iov [ 0 ] . iov_len = sizeof ( uint16_t ) - c - > tcp_write_byte_count ;
iov [ 1 ] . iov_base = c - > tcp_write_pkt ;
iov [ 1 ] . iov_len = c - > tcp_write_pkt_len ;
} else {
iov [ 0 ] . iov_base = ( uint8_t * ) & len + c - > tcp_byte_count ;
iov [ 0 ] . iov_len = sizeof ( uint16_t ) - c - > tcp_byte_count ;
iov [ 1 ] . iov_base = sldns_buffer_begin ( buffer ) ;
iov [ 1 ] . iov_len = sldns_buffer_limit ( buffer ) ;
}
2007-10-16 08:26:09 -04:00
log_assert ( iov [ 0 ] . iov_len > 0 ) ;
2007-05-04 09:01:20 -04:00
r = writev ( fd , iov , 2 ) ;
2008-06-13 11:32:16 -04:00
# else /* HAVE_WRITEV */
2020-06-26 10:05:15 -04:00
if ( c - > tcp_write_and_read ) {
r = send ( fd , ( void * ) ( ( ( uint8_t * ) & len ) + c - > tcp_write_byte_count ) ,
sizeof ( uint16_t ) - c - > tcp_write_byte_count , 0 ) ;
} else {
r = send ( fd , ( void * ) ( ( ( uint8_t * ) & len ) + c - > tcp_byte_count ) ,
sizeof ( uint16_t ) - c - > tcp_byte_count , 0 ) ;
}
2008-06-13 11:32:16 -04:00
# endif /* HAVE_WRITEV */
2007-02-07 10:44:19 -05:00
if ( r = = - 1 ) {
2008-06-18 10:27:30 -04:00
# ifndef USE_WINSOCK
2014-04-10 10:40:20 -04:00
# ifdef EPIPE
2010-11-15 08:15:23 -05:00
if ( errno = = EPIPE & & verbosity < 2 )
return 0 ; /* silence 'broken pipe' */
2014-04-10 10:40:20 -04:00
# endif
2007-02-07 10:44:19 -05:00
if ( errno = = EINTR | | errno = = EAGAIN )
return 1 ;
2019-05-13 04:39:39 -04:00
# ifdef ECONNRESET
if ( errno = = ECONNRESET & & verbosity < 2 )
return 0 ; /* silence reset by peer */
# endif
2014-04-10 10:40:20 -04:00
# ifdef HAVE_WRITEV
log_err_addr ( " tcp writev " , strerror ( errno ) ,
& c - > repinfo . addr , c - > repinfo . addrlen ) ;
# else /* HAVE_WRITEV */
log_err_addr ( " tcp send s " , strerror ( errno ) ,
& c - > repinfo . addr , c - > repinfo . addrlen ) ;
# endif /* HAVE_WRITEV */
2008-06-18 10:27:30 -04:00
# else
2009-04-14 09:35:10 -04:00
if ( WSAGetLastError ( ) = = WSAENOTCONN )
return 1 ;
2008-06-23 09:52:03 -04:00
if ( WSAGetLastError ( ) = = WSAEINPROGRESS )
2008-06-18 10:27:30 -04:00
return 1 ;
2008-06-23 09:52:03 -04:00
if ( WSAGetLastError ( ) = = WSAEWOULDBLOCK ) {
2016-03-07 09:10:06 -05:00
ub_winsock_tcp_wouldblock ( c - > ev - > ev ,
UB_EV_WRITE ) ;
2008-06-23 09:52:03 -04:00
return 1 ;
}
2019-05-13 04:39:39 -04:00
if ( WSAGetLastError ( ) = = WSAECONNRESET & & verbosity < 2 )
return 0 ; /* silence reset by peer */
2014-04-10 10:40:20 -04:00
log_err_addr ( " tcp send s " ,
wsa_strerror ( WSAGetLastError ( ) ) ,
& c - > repinfo . addr , c - > repinfo . addrlen ) ;
2008-06-18 10:27:30 -04:00
# endif
2007-02-07 10:44:19 -05:00
return 0 ;
}
2020-06-26 10:05:15 -04:00
if ( c - > tcp_write_and_read ) {
c - > tcp_write_byte_count + = r ;
if ( c - > tcp_write_byte_count < sizeof ( uint16_t ) )
return 1 ;
} else {
c - > tcp_byte_count + = r ;
if ( c - > tcp_byte_count < sizeof ( uint16_t ) )
return 1 ;
sldns_buffer_set_position ( buffer , c - > tcp_byte_count -
sizeof ( uint16_t ) ) ;
}
if ( ( ! c - > tcp_write_and_read & & sldns_buffer_remaining ( buffer ) = = 0 ) | | ( c - > tcp_write_and_read & & c - > tcp_write_byte_count = = c - > tcp_write_pkt_len + 2 ) ) {
2007-05-04 09:01:20 -04:00
tcp_callback_writer ( c ) ;
2008-06-18 10:27:30 -04:00
return 1 ;
2007-05-04 09:01:20 -04:00
}
2007-02-07 10:44:19 -05:00
}
2020-06-26 10:05:15 -04:00
log_assert ( c - > tcp_write_and_read | | sldns_buffer_remaining ( buffer ) > 0 ) ;
log_assert ( ! c - > tcp_write_and_read | | c - > tcp_write_byte_count < c - > tcp_write_pkt_len + 2 ) ;
if ( c - > tcp_write_and_read ) {
2020-12-11 08:00:20 -05:00
r = send ( fd , ( void * ) ( c - > tcp_write_pkt + c - > tcp_write_byte_count - 2 ) ,
2020-06-26 10:05:15 -04:00
c - > tcp_write_pkt_len + 2 - c - > tcp_write_byte_count , 0 ) ;
} else {
r = send ( fd , ( void * ) sldns_buffer_current ( buffer ) ,
sldns_buffer_remaining ( buffer ) , 0 ) ;
}
2007-02-07 10:44:19 -05:00
if ( r = = - 1 ) {
2008-06-18 10:27:30 -04:00
# ifndef USE_WINSOCK
2007-02-07 10:44:19 -05:00
if ( errno = = EINTR | | errno = = EAGAIN )
return 1 ;
2019-05-13 04:39:39 -04:00
# ifdef ECONNRESET
if ( errno = = ECONNRESET & & verbosity < 2 )
return 0 ; /* silence reset by peer */
# endif
2008-06-18 10:27:30 -04:00
# else
2008-06-23 09:52:03 -04:00
if ( WSAGetLastError ( ) = = WSAEINPROGRESS )
2008-06-18 10:27:30 -04:00
return 1 ;
2008-06-23 09:52:03 -04:00
if ( WSAGetLastError ( ) = = WSAEWOULDBLOCK ) {
2016-03-07 09:10:06 -05:00
ub_winsock_tcp_wouldblock ( c - > ev - > ev , UB_EV_WRITE ) ;
2008-06-23 09:52:03 -04:00
return 1 ;
}
2019-05-13 04:39:39 -04:00
if ( WSAGetLastError ( ) = = WSAECONNRESET & & verbosity < 2 )
return 0 ; /* silence reset by peer */
2008-06-18 10:27:30 -04:00
# endif
2020-08-31 03:12:01 -04:00
log_err_addr ( " tcp send r " , sock_strerror ( errno ) ,
& c - > repinfo . addr , c - > repinfo . addrlen ) ;
2007-02-07 10:44:19 -05:00
return 0 ;
}
2020-06-26 10:05:15 -04:00
if ( c - > tcp_write_and_read ) {
c - > tcp_write_byte_count + = r ;
} else {
sldns_buffer_skip ( buffer , r ) ;
}
2007-02-07 10:44:19 -05:00
2020-06-26 10:05:15 -04:00
if ( ( ! c - > tcp_write_and_read & & sldns_buffer_remaining ( buffer ) = = 0 ) | | ( c - > tcp_write_and_read & & c - > tcp_write_byte_count = = c - > tcp_write_pkt_len + 2 ) ) {
2007-02-07 10:44:19 -05:00
tcp_callback_writer ( c ) ;
}
return 1 ;
}
2019-01-21 09:14:12 -05:00
/** read again to drain buffers when there could be more to read */
static void
tcp_req_info_read_again ( int fd , struct comm_point * c )
{
while ( c - > tcp_req_info - > read_again ) {
int r ;
c - > tcp_req_info - > read_again = 0 ;
if ( c - > tcp_is_reading )
r = comm_point_tcp_handle_read ( fd , c , 0 ) ;
else r = comm_point_tcp_handle_write ( fd , c ) ;
if ( ! r ) {
reclaim_tcp_handler ( c ) ;
if ( ! c - > tcp_do_close ) {
fptr_ok ( fptr_whitelist_comm_point (
c - > callback ) ) ;
( void ) ( * c - > callback ) ( c , c - > cb_arg ,
NETEVENT_CLOSED , NULL ) ;
}
return ;
}
}
}
2020-07-13 09:16:59 -04:00
/** read again to drain buffers when there could be more to read */
static void
tcp_more_read_again ( int fd , struct comm_point * c )
{
/* if the packet is done, but another one could be waiting on
* the connection , the callback signals this , and we try again */
/* this continues until the read routines get EAGAIN or so,
* and thus does not call the callback , and the bool is 0 */
2020-11-25 07:46:28 -05:00
int * moreread = c - > tcp_more_read_again ;
while ( moreread & & * moreread ) {
* moreread = 0 ;
2020-07-13 09:16:59 -04:00
if ( ! comm_point_tcp_handle_read ( fd , c , 0 ) ) {
reclaim_tcp_handler ( c ) ;
if ( ! c - > tcp_do_close ) {
fptr_ok ( fptr_whitelist_comm_point (
c - > callback ) ) ;
2020-07-13 09:45:16 -04:00
( void ) ( * c - > callback ) ( c , c - > cb_arg ,
2020-07-13 09:16:59 -04:00
NETEVENT_CLOSED , NULL ) ;
}
return ;
}
}
}
/** write again to fill up when there could be more to write */
static void
tcp_more_write_again ( int fd , struct comm_point * c )
{
/* if the packet is done, but another is waiting to be written,
* the callback signals it and we try again . */
/* this continues until the write routines get EAGAIN or so,
* and thus does not call the callback , and the bool is 0 */
2020-11-25 07:46:28 -05:00
int * morewrite = c - > tcp_more_write_again ;
while ( morewrite & & * morewrite ) {
* morewrite = 0 ;
2020-07-13 09:16:59 -04:00
if ( ! comm_point_tcp_handle_write ( fd , c ) ) {
reclaim_tcp_handler ( c ) ;
if ( ! c - > tcp_do_close ) {
fptr_ok ( fptr_whitelist_comm_point (
c - > callback ) ) ;
2020-07-13 09:45:16 -04:00
( void ) ( * c - > callback ) ( c , c - > cb_arg ,
2020-07-13 09:16:59 -04:00
NETEVENT_CLOSED , NULL ) ;
}
return ;
}
}
}
2007-10-05 04:05:06 -04:00
void
2007-02-07 10:44:19 -05:00
comm_point_tcp_handle_callback ( int fd , short event , void * arg )
{
struct comm_point * c = ( struct comm_point * ) arg ;
log_assert ( c - > type = = comm_tcp ) ;
2016-03-07 11:01:09 -05:00
ub_comm_base_now ( c - > ev - > base ) ;
2007-02-07 10:44:19 -05:00
2017-03-20 10:55:31 -04:00
# ifdef USE_DNSCRYPT
/* Initialize if this is a dnscrypt socket */
if ( c - > tcp_parent ) {
c - > dnscrypt = c - > tcp_parent - > dnscrypt ;
}
2017-04-18 05:00:52 -04:00
if ( c - > dnscrypt & & c - > dnscrypt_buffer = = c - > buffer ) {
c - > dnscrypt_buffer = sldns_buffer_new ( sldns_buffer_capacity ( c - > buffer ) ) ;
if ( ! c - > dnscrypt_buffer ) {
log_err ( " Could not allocate dnscrypt buffer " ) ;
2017-08-22 09:40:44 -04:00
reclaim_tcp_handler ( c ) ;
if ( ! c - > tcp_do_close ) {
fptr_ok ( fptr_whitelist_comm_point (
c - > callback ) ) ;
2020-07-13 09:45:16 -04:00
( void ) ( * c - > callback ) ( c , c - > cb_arg ,
2017-08-22 09:40:44 -04:00
NETEVENT_CLOSED , NULL ) ;
}
2017-04-18 05:00:52 -04:00
return ;
}
}
2017-03-20 10:55:31 -04:00
# endif
2019-04-11 09:41:53 -04:00
if ( event & UB_EV_TIMEOUT ) {
verbose ( VERB_QUERY , " tcp took too long, dropped " ) ;
reclaim_tcp_handler ( c ) ;
if ( ! c - > tcp_do_close ) {
fptr_ok ( fptr_whitelist_comm_point ( c - > callback ) ) ;
( void ) ( * c - > callback ) ( c , c - > cb_arg ,
NETEVENT_TIMEOUT , NULL ) ;
}
return ;
}
2020-12-02 09:42:24 -05:00
if ( event & UB_EV_READ
# ifdef USE_MSG_FASTOPEN
2020-12-02 10:17:26 -05:00
& & ! ( c - > tcp_do_fastopen & & ( event & UB_EV_WRITE ) )
2020-12-02 09:42:24 -05:00
# endif
) {
2019-01-25 07:46:15 -05:00
int has_tcpq = ( c - > tcp_req_info ! = NULL ) ;
2020-11-25 07:46:28 -05:00
int * moreread = c - > tcp_more_read_again ;
2007-02-26 09:49:11 -05:00
if ( ! comm_point_tcp_handle_read ( fd , c , 0 ) ) {
2007-02-07 10:44:19 -05:00
reclaim_tcp_handler ( c ) ;
2007-10-04 11:10:11 -04:00
if ( ! c - > tcp_do_close ) {
2008-02-20 02:18:42 -05:00
fptr_ok ( fptr_whitelist_comm_point (
2007-10-04 11:10:11 -04:00
c - > callback ) ) ;
2020-07-13 09:45:16 -04:00
( void ) ( * c - > callback ) ( c , c - > cb_arg ,
2007-02-07 10:44:19 -05:00
NETEVENT_CLOSED , NULL ) ;
2007-10-04 11:10:11 -04:00
}
2020-11-26 06:12:52 -05:00
return ;
2007-02-07 10:44:19 -05:00
}
2019-01-25 07:46:15 -05:00
if ( has_tcpq & & c - > tcp_req_info & & c - > tcp_req_info - > read_again )
2019-01-21 09:14:12 -05:00
tcp_req_info_read_again ( fd , c ) ;
2020-11-25 07:46:28 -05:00
if ( moreread & & * moreread )
2020-07-13 09:16:59 -04:00
tcp_more_read_again ( fd , c ) ;
2007-02-07 10:44:19 -05:00
return ;
}
2016-03-07 09:10:06 -05:00
if ( event & UB_EV_WRITE ) {
2019-01-25 07:46:15 -05:00
int has_tcpq = ( c - > tcp_req_info ! = NULL ) ;
2020-11-25 07:46:28 -05:00
int * morewrite = c - > tcp_more_write_again ;
2007-02-07 10:44:19 -05:00
if ( ! comm_point_tcp_handle_write ( fd , c ) ) {
reclaim_tcp_handler ( c ) ;
2007-10-04 11:10:11 -04:00
if ( ! c - > tcp_do_close ) {
2008-02-20 02:18:42 -05:00
fptr_ok ( fptr_whitelist_comm_point (
2007-10-04 11:10:11 -04:00
c - > callback ) ) ;
2020-07-13 09:45:16 -04:00
( void ) ( * c - > callback ) ( c , c - > cb_arg ,
2007-02-07 10:44:19 -05:00
NETEVENT_CLOSED , NULL ) ;
2007-10-04 11:10:11 -04:00
}
2020-11-26 06:12:52 -05:00
return ;
2007-02-07 10:44:19 -05:00
}
2019-01-25 07:46:15 -05:00
if ( has_tcpq & & c - > tcp_req_info & & c - > tcp_req_info - > read_again )
2019-01-21 09:14:12 -05:00
tcp_req_info_read_again ( fd , c ) ;
2020-11-25 07:46:28 -05:00
if ( morewrite & & * morewrite )
2020-07-13 09:16:59 -04:00
tcp_more_write_again ( fd , c ) ;
2007-02-07 10:44:19 -05:00
return ;
}
log_err ( " Ignored event %d for tcphdl. " , event ) ;
2007-01-23 11:10:23 -05:00
}
2018-02-07 11:10:31 -05:00
/** Make http handler free for next assignment */
static void
reclaim_http_handler ( struct comm_point * c )
{
log_assert ( c - > type = = comm_http ) ;
if ( c - > ssl ) {
# ifdef HAVE_SSL
SSL_shutdown ( c - > ssl ) ;
SSL_free ( c - > ssl ) ;
c - > ssl = NULL ;
# endif
}
comm_point_close ( c ) ;
if ( c - > tcp_parent ) {
2021-11-05 06:15:19 -04:00
if ( c ! = c - > tcp_parent - > tcp_free ) {
c - > tcp_parent - > cur_tcp_count - - ;
c - > tcp_free = c - > tcp_parent - > tcp_free ;
c - > tcp_parent - > tcp_free = c ;
}
2018-02-07 11:10:31 -05:00
if ( ! c - > tcp_free ) {
/* re-enable listening on accept socket */
comm_point_start_listening ( c - > tcp_parent , - 1 , - 1 ) ;
}
}
}
/** read more data for http (with ssl) */
static int
ssl_http_read_more ( struct comm_point * c )
{
# ifdef HAVE_SSL
int r ;
log_assert ( sldns_buffer_remaining ( c - > buffer ) > 0 ) ;
ERR_clear_error ( ) ;
r = SSL_read ( c - > ssl , ( void * ) sldns_buffer_current ( c - > buffer ) ,
( int ) sldns_buffer_remaining ( c - > buffer ) ) ;
if ( r < = 0 ) {
int want = SSL_get_error ( c - > ssl , r ) ;
if ( want = = SSL_ERROR_ZERO_RETURN ) {
return 0 ; /* shutdown, closed */
} else if ( want = = SSL_ERROR_WANT_READ ) {
return 1 ; /* read more later */
} else if ( want = = SSL_ERROR_WANT_WRITE ) {
c - > ssl_shake_state = comm_ssl_shake_hs_write ;
comm_point_listen_for_rw ( c , 0 , 1 ) ;
return 1 ;
} else if ( want = = SSL_ERROR_SYSCALL ) {
2019-04-11 11:04:32 -04:00
# ifdef ECONNRESET
if ( errno = = ECONNRESET & & verbosity < 2 )
return 0 ; /* silence reset by peer */
# endif
2018-02-07 11:10:31 -05:00
if ( errno ! = 0 )
log_err ( " SSL_read syscall: %s " ,
strerror ( errno ) ) ;
return 0 ;
}
log_crypto_err ( " could not SSL_read " ) ;
return 0 ;
}
2021-02-19 08:42:02 -05:00
verbose ( VERB_ALGO , " ssl http read more skip to %d + %d " ,
( int ) sldns_buffer_position ( c - > buffer ) , ( int ) r ) ;
2018-02-07 11:10:31 -05:00
sldns_buffer_skip ( c - > buffer , ( ssize_t ) r ) ;
return 1 ;
# else
( void ) c ;
return 0 ;
# endif /* HAVE_SSL */
}
/** read more data for http */
static int
http_read_more ( int fd , struct comm_point * c )
{
ssize_t r ;
log_assert ( sldns_buffer_remaining ( c - > buffer ) > 0 ) ;
r = recv ( fd , ( void * ) sldns_buffer_current ( c - > buffer ) ,
sldns_buffer_remaining ( c - > buffer ) , 0 ) ;
if ( r = = 0 ) {
return 0 ;
} else if ( r = = - 1 ) {
# ifndef USE_WINSOCK
if ( errno = = EINTR | | errno = = EAGAIN )
return 1 ;
# else /* USE_WINSOCK */
if ( WSAGetLastError ( ) = = WSAECONNRESET )
return 0 ;
if ( WSAGetLastError ( ) = = WSAEINPROGRESS )
return 1 ;
if ( WSAGetLastError ( ) = = WSAEWOULDBLOCK ) {
ub_winsock_tcp_wouldblock ( c - > ev - > ev , UB_EV_READ ) ;
return 1 ;
}
# endif
2020-08-31 03:12:01 -04:00
log_err_addr ( " read (in http r) " , sock_strerror ( errno ) ,
& c - > repinfo . addr , c - > repinfo . addrlen ) ;
2018-02-07 11:10:31 -05:00
return 0 ;
}
2021-02-19 08:42:02 -05:00
verbose ( VERB_ALGO , " http read more skip to %d + %d " ,
( int ) sldns_buffer_position ( c - > buffer ) , ( int ) r ) ;
2018-02-07 11:10:31 -05:00
sldns_buffer_skip ( c - > buffer , r ) ;
return 1 ;
}
/** return true if http header has been read (one line complete) */
static int
http_header_done ( sldns_buffer * buf )
{
size_t i ;
for ( i = sldns_buffer_position ( buf ) ; i < sldns_buffer_limit ( buf ) ; i + + ) {
/* there was a \r before the \n, but we ignore that */
if ( ( char ) sldns_buffer_read_u8_at ( buf , i ) = = ' \n ' )
return 1 ;
}
return 0 ;
}
/** return character string into buffer for header line, moves buffer
2018-02-08 05:00:10 -05:00
* past that line and puts zero terminator into linefeed - newline */
2018-02-07 11:10:31 -05:00
static char *
http_header_line ( sldns_buffer * buf )
{
char * result = ( char * ) sldns_buffer_current ( buf ) ;
size_t i ;
for ( i = sldns_buffer_position ( buf ) ; i < sldns_buffer_limit ( buf ) ; i + + ) {
/* terminate the string on the \r */
if ( ( char ) sldns_buffer_read_u8_at ( buf , i ) = = ' \r ' )
sldns_buffer_write_u8_at ( buf , i , 0 ) ;
/* terminate on the \n and skip past the it and done */
if ( ( char ) sldns_buffer_read_u8_at ( buf , i ) = = ' \n ' ) {
sldns_buffer_write_u8_at ( buf , i , 0 ) ;
2018-02-08 06:59:30 -05:00
sldns_buffer_set_position ( buf , i + 1 ) ;
2018-02-07 11:10:31 -05:00
return result ;
}
}
return NULL ;
}
/** move unread buffer to start and clear rest for putting the rest into it */
static void
http_moveover_buffer ( sldns_buffer * buf )
{
size_t pos = sldns_buffer_position ( buf ) ;
size_t len = sldns_buffer_remaining ( buf ) ;
sldns_buffer_clear ( buf ) ;
memmove ( sldns_buffer_begin ( buf ) , sldns_buffer_at ( buf , pos ) , len ) ;
sldns_buffer_set_position ( buf , len ) ;
}
/** a http header is complete, process it */
static int
http_process_initial_header ( struct comm_point * c )
{
char * line = http_header_line ( c - > buffer ) ;
if ( ! line ) return 1 ;
verbose ( VERB_ALGO , " http header: %s " , line ) ;
if ( strncasecmp ( line , " HTTP/1.1 " , 9 ) = = 0 ) {
/* check returncode */
if ( line [ 9 ] ! = ' 2 ' ) {
verbose ( VERB_ALGO , " http bad status %s " , line + 9 ) ;
return 0 ;
}
} else if ( strncasecmp ( line , " Content-Length: " , 16 ) = = 0 ) {
if ( ! c - > http_is_chunked )
c - > tcp_byte_count = ( size_t ) atoi ( line + 16 ) ;
} else if ( strncasecmp ( line , " Transfer-Encoding: chunked " , 19 + 7 ) = = 0 ) {
c - > tcp_byte_count = 0 ;
c - > http_is_chunked = 1 ;
} else if ( line [ 0 ] = = 0 ) {
/* end of initial headers */
c - > http_in_headers = 0 ;
if ( c - > http_is_chunked )
c - > http_in_chunk_headers = 1 ;
2018-02-08 05:43:58 -05:00
/* remove header text from front of buffer
* the buffer is going to be used to return the data segment
* itself and we don ' t want the header to get returned
* prepended with it */
2018-02-07 11:10:31 -05:00
http_moveover_buffer ( c - > buffer ) ;
sldns_buffer_flip ( c - > buffer ) ;
return 1 ;
}
/* ignore other headers */
return 1 ;
}
/** a chunk header is complete, process it, return 0=fail, 1=continue next
* header line , 2 = done with chunked transfer */
static int
http_process_chunk_header ( struct comm_point * c )
{
char * line = http_header_line ( c - > buffer ) ;
if ( ! line ) return 1 ;
if ( c - > http_in_chunk_headers = = 3 ) {
verbose ( VERB_ALGO , " http chunk trailer: %s " , line ) ;
/* are we done ? */
if ( line [ 0 ] = = 0 & & c - > tcp_byte_count = = 0 ) {
/* callback of http reader when NETEVENT_DONE,
* end of data , with no data in buffer */
sldns_buffer_set_position ( c - > buffer , 0 ) ;
sldns_buffer_set_limit ( c - > buffer , 0 ) ;
fptr_ok ( fptr_whitelist_comm_point ( c - > callback ) ) ;
( void ) ( * c - > callback ) ( c , c - > cb_arg , NETEVENT_DONE , NULL ) ;
2018-02-08 05:43:58 -05:00
/* return that we are done */
2018-02-07 11:10:31 -05:00
return 2 ;
}
if ( line [ 0 ] = = 0 ) {
/* continue with header of the next chunk */
c - > http_in_chunk_headers = 1 ;
/* remove header text from front of buffer */
http_moveover_buffer ( c - > buffer ) ;
sldns_buffer_flip ( c - > buffer ) ;
return 1 ;
}
/* ignore further trail headers */
return 1 ;
}
verbose ( VERB_ALGO , " http chunk header: %s " , line ) ;
if ( c - > http_in_chunk_headers = = 1 ) {
/* read chunked start line */
char * end = NULL ;
c - > tcp_byte_count = ( size_t ) strtol ( line , & end , 16 ) ;
if ( end = = line )
return 0 ;
c - > http_in_chunk_headers = 0 ;
/* remove header text from front of buffer */
http_moveover_buffer ( c - > buffer ) ;
sldns_buffer_flip ( c - > buffer ) ;
if ( c - > tcp_byte_count = = 0 ) {
/* done with chunks, process chunk_trailer lines */
c - > http_in_chunk_headers = 3 ;
}
2018-02-08 06:59:30 -05:00
return 1 ;
2018-02-07 11:10:31 -05:00
}
/* ignore other headers */
return 1 ;
}
2021-03-25 07:18:49 -04:00
/** handle nonchunked data segment, 0=fail, 1=wait */
2018-02-07 11:10:31 -05:00
static int
http_nonchunk_segment ( struct comm_point * c )
{
/* c->buffer at position..limit has new data we read in.
* the buffer itself is full of nonchunked data .
* we are looking to read tcp_byte_count more data
* and then the transfer is done . */
size_t remainbufferlen ;
2021-03-25 07:18:49 -04:00
size_t got_now = sldns_buffer_limit ( c - > buffer ) ;
2018-02-07 11:10:31 -05:00
if ( c - > tcp_byte_count < = got_now ) {
/* done, this is the last data fragment */
2018-02-08 11:11:27 -05:00
c - > http_stored = 0 ;
2018-02-07 11:10:31 -05:00
sldns_buffer_set_position ( c - > buffer , 0 ) ;
fptr_ok ( fptr_whitelist_comm_point ( c - > callback ) ) ;
( void ) ( * c - > callback ) ( c , c - > cb_arg , NETEVENT_DONE , NULL ) ;
return 1 ;
}
2018-02-08 05:43:58 -05:00
/* if we have the buffer space,
2018-02-07 11:10:31 -05:00
* read more data collected into the buffer */
remainbufferlen = sldns_buffer_capacity ( c - > buffer ) -
sldns_buffer_limit ( c - > buffer ) ;
2021-02-19 08:42:02 -05:00
if ( remainbufferlen + got_now > = c - > tcp_byte_count | |
remainbufferlen > = ( c - > ssl ? 16384 : 2048 ) ) {
2018-02-07 11:10:31 -05:00
size_t total = sldns_buffer_limit ( c - > buffer ) ;
sldns_buffer_clear ( c - > buffer ) ;
sldns_buffer_set_position ( c - > buffer , total ) ;
2018-02-08 11:11:27 -05:00
c - > http_stored = total ;
2018-02-07 11:10:31 -05:00
/* return and wait to read more */
return 1 ;
}
/* call callback with this data amount, then
* wait for more */
2021-03-25 07:18:49 -04:00
c - > tcp_byte_count - = got_now ;
2018-02-08 11:11:27 -05:00
c - > http_stored = 0 ;
2018-02-07 11:10:31 -05:00
sldns_buffer_set_position ( c - > buffer , 0 ) ;
fptr_ok ( fptr_whitelist_comm_point ( c - > callback ) ) ;
( void ) ( * c - > callback ) ( c , c - > cb_arg , NETEVENT_NOERROR , NULL ) ;
/* c->callback has to buffer_clear(c->buffer). */
/* return and wait to read more */
return 1 ;
}
2021-01-06 04:36:23 -05:00
/** handle chunked data segment, return 0=fail, 1=wait, 2=process more */
2018-02-07 11:10:31 -05:00
static int
http_chunked_segment ( struct comm_point * c )
{
/* the c->buffer has from position..limit new data we read. */
/* the current chunk has length tcp_byte_count.
* once we read that read more chunk headers .
*/
size_t remainbufferlen ;
2018-02-08 11:11:27 -05:00
size_t got_now = sldns_buffer_limit ( c - > buffer ) - c - > http_stored ;
2021-01-06 04:36:23 -05:00
verbose ( VERB_ALGO , " http_chunked_segment: got now %d, tcpbytcount %d, http_stored %d, buffer pos %d, buffer limit %d " , ( int ) got_now , ( int ) c - > tcp_byte_count , ( int ) c - > http_stored , ( int ) sldns_buffer_position ( c - > buffer ) , ( int ) sldns_buffer_limit ( c - > buffer ) ) ;
2018-02-07 11:10:31 -05:00
if ( c - > tcp_byte_count < = got_now ) {
/* the chunk has completed (with perhaps some extra data
* from next chunk header and next chunk ) */
/* save too much info into temp buffer */
size_t fraglen ;
struct comm_reply repinfo ;
2018-02-08 11:11:27 -05:00
c - > http_stored = 0 ;
2018-02-08 05:00:10 -05:00
sldns_buffer_skip ( c - > buffer , ( ssize_t ) c - > tcp_byte_count ) ;
2018-02-07 11:10:31 -05:00
sldns_buffer_clear ( c - > http_temp ) ;
sldns_buffer_write ( c - > http_temp ,
sldns_buffer_current ( c - > buffer ) ,
sldns_buffer_remaining ( c - > buffer ) ) ;
sldns_buffer_flip ( c - > http_temp ) ;
/* callback with this fragment */
fraglen = sldns_buffer_position ( c - > buffer ) ;
sldns_buffer_set_position ( c - > buffer , 0 ) ;
sldns_buffer_set_limit ( c - > buffer , fraglen ) ;
repinfo = c - > repinfo ;
fptr_ok ( fptr_whitelist_comm_point ( c - > callback ) ) ;
( void ) ( * c - > callback ) ( c , c - > cb_arg , NETEVENT_NOERROR , & repinfo ) ;
/* c->callback has to buffer_clear(). */
/* is commpoint deleted? */
if ( ! repinfo . c ) {
return 1 ;
}
/* copy waiting info */
sldns_buffer_clear ( c - > buffer ) ;
sldns_buffer_write ( c - > buffer ,
sldns_buffer_begin ( c - > http_temp ) ,
sldns_buffer_remaining ( c - > http_temp ) ) ;
2018-02-08 06:59:30 -05:00
sldns_buffer_flip ( c - > buffer ) ;
2018-02-07 11:10:31 -05:00
/* process end of chunk trailer header lines, until
* an empty line */
c - > http_in_chunk_headers = 3 ;
/* process more data in buffer (if any) */
return 2 ;
}
c - > tcp_byte_count - = got_now ;
2018-02-08 05:43:58 -05:00
/* if we have the buffer space,
2018-02-07 11:10:31 -05:00
* read more data collected into the buffer */
remainbufferlen = sldns_buffer_capacity ( c - > buffer ) -
sldns_buffer_limit ( c - > buffer ) ;
if ( remainbufferlen > = c - > tcp_byte_count | |
2018-02-08 11:11:27 -05:00
remainbufferlen > = 2048 ) {
2018-02-07 11:10:31 -05:00
size_t total = sldns_buffer_limit ( c - > buffer ) ;
sldns_buffer_clear ( c - > buffer ) ;
sldns_buffer_set_position ( c - > buffer , total ) ;
2018-02-08 11:11:27 -05:00
c - > http_stored = total ;
2018-02-07 11:10:31 -05:00
/* return and wait to read more */
return 1 ;
}
/* callback of http reader for a new part of the data */
2018-02-08 11:11:27 -05:00
c - > http_stored = 0 ;
2018-02-07 11:10:31 -05:00
sldns_buffer_set_position ( c - > buffer , 0 ) ;
fptr_ok ( fptr_whitelist_comm_point ( c - > callback ) ) ;
( void ) ( * c - > callback ) ( c , c - > cb_arg , NETEVENT_NOERROR , NULL ) ;
/* c->callback has to buffer_clear(c->buffer). */
/* return and wait to read more */
return 1 ;
}
2020-05-07 10:36:26 -04:00
# ifdef HAVE_NGHTTP2
/** Create new http2 session. Called when creating handling comm point. */
2020-10-28 09:15:23 -04:00
static struct http2_session * http2_session_create ( struct comm_point * c )
2020-05-07 10:36:26 -04:00
{
struct http2_session * session = calloc ( 1 , sizeof ( * session ) ) ;
if ( ! session ) {
log_err ( " malloc failure while creating http2 session " ) ;
return NULL ;
}
session - > c = c ;
return session ;
}
# endif
/** Delete http2 session. After closing connection or on error */
2020-10-28 09:15:23 -04:00
static void http2_session_delete ( struct http2_session * h2_session )
2020-05-07 10:36:26 -04:00
{
# ifdef HAVE_NGHTTP2
if ( h2_session - > callbacks )
nghttp2_session_callbacks_del ( h2_session - > callbacks ) ;
free ( h2_session ) ;
# else
( void ) h2_session ;
# endif
}
# ifdef HAVE_NGHTTP2
struct http2_stream * http2_stream_create ( int32_t stream_id )
{
struct http2_stream * h2_stream = calloc ( 1 , sizeof ( * h2_stream ) ) ;
if ( ! h2_stream ) {
log_err ( " malloc failure while creating http2 stream " ) ;
return NULL ;
}
h2_stream - > stream_id = stream_id ;
return h2_stream ;
}
/** Delete http2 stream. After session delete or stream close callback */
2020-06-24 08:04:34 -04:00
static void http2_stream_delete ( struct http2_session * h2_session ,
2020-05-07 10:36:26 -04:00
struct http2_stream * h2_stream )
{
if ( h2_stream - > mesh_state ) {
mesh_state_remove_reply ( h2_stream - > mesh , h2_stream - > mesh_state ,
h2_session - > c ) ;
2020-06-24 08:04:34 -04:00
h2_stream - > mesh_state = NULL ;
2020-05-07 10:36:26 -04:00
}
2020-05-12 12:12:19 -04:00
http2_req_stream_clear ( h2_stream ) ;
2020-05-07 10:36:26 -04:00
free ( h2_stream ) ;
}
# endif
void http2_stream_add_meshstate ( struct http2_stream * h2_stream ,
struct mesh_area * mesh , struct mesh_state * m )
{
h2_stream - > mesh = mesh ;
h2_stream - > mesh_state = m ;
}
/** delete http2 session server. After closing connection. */
static void http2_session_server_delete ( struct http2_session * h2_session )
{
# ifdef HAVE_NGHTTP2
struct http2_stream * h2_stream , * next ;
nghttp2_session_del ( h2_session - > session ) ; /* NULL input is fine */
h2_session - > session = NULL ;
for ( h2_stream = h2_session - > first_stream ; h2_stream ; ) {
next = h2_stream - > next ;
http2_stream_delete ( h2_session , h2_stream ) ;
h2_stream = next ;
}
h2_session - > first_stream = NULL ;
h2_session - > is_drop = 0 ;
h2_session - > postpone_drop = 0 ;
h2_session - > c - > h2_stream = NULL ;
# endif
( void ) h2_session ;
}
# ifdef HAVE_NGHTTP2
void http2_session_add_stream ( struct http2_session * h2_session ,
struct http2_stream * h2_stream )
{
if ( h2_session - > first_stream )
h2_session - > first_stream - > prev = h2_stream ;
h2_stream - > next = h2_session - > first_stream ;
h2_session - > first_stream = h2_stream ;
}
/** remove stream from session linked list. After stream close callback or
* closing connection */
2020-10-28 09:15:23 -04:00
static void http2_session_remove_stream ( struct http2_session * h2_session ,
2020-05-07 10:36:26 -04:00
struct http2_stream * h2_stream )
{
if ( h2_stream - > prev )
h2_stream - > prev - > next = h2_stream - > next ;
else
h2_session - > first_stream = h2_stream - > next ;
if ( h2_stream - > next )
h2_stream - > next - > prev = h2_stream - > prev ;
}
int http2_stream_close_cb ( nghttp2_session * ATTR_UNUSED ( session ) ,
int32_t stream_id , uint32_t ATTR_UNUSED ( error_code ) , void * cb_arg )
{
struct http2_stream * h2_stream ;
struct http2_session * h2_session = ( struct http2_session * ) cb_arg ;
if ( ! ( h2_stream = nghttp2_session_get_stream_user_data (
h2_session - > session , stream_id ) ) ) {
return 0 ;
}
http2_session_remove_stream ( h2_session , h2_stream ) ;
http2_stream_delete ( h2_session , h2_stream ) ;
return 0 ;
}
ssize_t http2_recv_cb ( nghttp2_session * ATTR_UNUSED ( session ) , uint8_t * buf ,
size_t len , int ATTR_UNUSED ( flags ) , void * cb_arg )
{
struct http2_session * h2_session = ( struct http2_session * ) cb_arg ;
2020-10-19 04:24:03 -04:00
ssize_t ret ;
2020-05-07 10:36:26 -04:00
log_assert ( h2_session - > c - > type = = comm_http ) ;
log_assert ( h2_session - > c - > h2_session ) ;
2020-10-19 04:24:03 -04:00
# ifdef HAVE_SSL
if ( h2_session - > c - > ssl ) {
int r ;
ERR_clear_error ( ) ;
r = SSL_read ( h2_session - > c - > ssl , buf , len ) ;
if ( r < = 0 ) {
int want = SSL_get_error ( h2_session - > c - > ssl , r ) ;
if ( want = = SSL_ERROR_ZERO_RETURN ) {
return NGHTTP2_ERR_EOF ;
} else if ( want = = SSL_ERROR_WANT_READ ) {
return NGHTTP2_ERR_WOULDBLOCK ;
} else if ( want = = SSL_ERROR_WANT_WRITE ) {
h2_session - > c - > ssl_shake_state = comm_ssl_shake_hs_write ;
comm_point_listen_for_rw ( h2_session - > c , 0 , 1 ) ;
return NGHTTP2_ERR_WOULDBLOCK ;
} else if ( want = = SSL_ERROR_SYSCALL ) {
# ifdef ECONNRESET
if ( errno = = ECONNRESET & & verbosity < 2 )
return NGHTTP2_ERR_CALLBACK_FAILURE ;
# endif
if ( errno ! = 0 )
log_err ( " SSL_read syscall: %s " ,
strerror ( errno ) ) ;
return NGHTTP2_ERR_CALLBACK_FAILURE ;
}
log_crypto_err ( " could not SSL_read " ) ;
return NGHTTP2_ERR_CALLBACK_FAILURE ;
}
return r ;
}
# endif /* HAVE_SSL */
2020-05-07 10:36:26 -04:00
2020-10-19 04:24:03 -04:00
ret = recv ( h2_session - > c - > fd , buf , len , 0 ) ;
if ( ret = = 0 ) {
return NGHTTP2_ERR_EOF ;
} else if ( ret < 0 ) {
# ifndef USE_WINSOCK
if ( errno = = EINTR | | errno = = EAGAIN )
2020-05-07 10:36:26 -04:00
return NGHTTP2_ERR_WOULDBLOCK ;
# ifdef ECONNRESET
2020-10-19 04:24:03 -04:00
if ( errno = = ECONNRESET & & verbosity < 2 )
return NGHTTP2_ERR_CALLBACK_FAILURE ;
2020-05-07 10:36:26 -04:00
# endif
2020-10-19 04:24:03 -04:00
log_err_addr ( " could not http2 recv: %s " , strerror ( errno ) ,
& h2_session - > c - > repinfo . addr ,
h2_session - > c - > repinfo . addrlen ) ;
# else /* USE_WINSOCK */
if ( WSAGetLastError ( ) = = WSAECONNRESET )
2020-05-07 10:36:26 -04:00
return NGHTTP2_ERR_CALLBACK_FAILURE ;
2020-10-19 04:24:03 -04:00
if ( WSAGetLastError ( ) = = WSAEINPROGRESS )
return NGHTTP2_ERR_WOULDBLOCK ;
if ( WSAGetLastError ( ) = = WSAEWOULDBLOCK ) {
ub_winsock_tcp_wouldblock ( h2_session - > c - > ev - > ev ,
UB_EV_READ ) ;
return NGHTTP2_ERR_WOULDBLOCK ;
2020-05-07 10:36:26 -04:00
}
2020-10-19 04:24:03 -04:00
log_err_addr ( " could not http2 recv: %s " ,
wsa_strerror ( WSAGetLastError ( ) ) ,
& h2_session - > c - > repinfo . addr ,
h2_session - > c - > repinfo . addrlen ) ;
# endif
2020-05-07 10:36:26 -04:00
return NGHTTP2_ERR_CALLBACK_FAILURE ;
}
2020-10-19 04:24:03 -04:00
return ret ;
2020-05-07 10:36:26 -04:00
}
# endif /* HAVE_NGHTTP2 */
/** Handle http2 read */
static int
comm_point_http2_handle_read ( int ATTR_UNUSED ( fd ) , struct comm_point * c )
{
# ifdef HAVE_NGHTTP2
int ret ;
log_assert ( c - > h2_session ) ;
/* reading until recv cb returns NGHTTP2_ERR_WOULDBLOCK */
ret = nghttp2_session_recv ( c - > h2_session - > session ) ;
if ( ret ) {
2020-06-24 08:04:34 -04:00
if ( ret ! = NGHTTP2_ERR_EOF & &
ret ! = NGHTTP2_ERR_CALLBACK_FAILURE ) {
2020-10-19 05:06:55 -04:00
char a [ 256 ] ;
addr_to_str ( & c - > repinfo . addr , c - > repinfo . addrlen ,
a , sizeof ( a ) ) ;
verbose ( VERB_QUERY , " http2: session_recv from %s failed, "
" error: %s " , a , nghttp2_strerror ( ret ) ) ;
2020-05-07 10:36:26 -04:00
}
return 0 ;
}
if ( nghttp2_session_want_write ( c - > h2_session - > session ) ) {
c - > tcp_is_reading = 0 ;
comm_point_stop_listening ( c ) ;
2020-12-02 04:10:27 -05:00
comm_point_start_listening ( c , - 1 , adjusted_tcp_timeout ( c ) ) ;
2020-05-07 10:36:26 -04:00
} else if ( ! nghttp2_session_want_read ( c - > h2_session - > session ) )
return 0 ; /* connection can be closed */
return 1 ;
# else
( void ) c ;
return 0 ;
# endif
}
2018-02-07 11:10:31 -05:00
/**
2020-05-07 10:36:26 -04:00
* Handle http reading callback .
2018-02-07 11:10:31 -05:00
* @ param fd : file descriptor of socket .
* @ param c : comm point to read from into buffer .
2020-05-07 10:36:26 -04:00
* @ return : 0 on error
2018-02-07 11:10:31 -05:00
*/
static int
comm_point_http_handle_read ( int fd , struct comm_point * c )
{
log_assert ( c - > type = = comm_http ) ;
log_assert ( fd ! = - 1 ) ;
/* if we are in ssl handshake, handle SSL handshake */
2018-02-27 08:03:54 -05:00
# ifdef HAVE_SSL
2018-02-07 11:10:31 -05:00
if ( c - > ssl & & c - > ssl_shake_state ! = comm_ssl_shake_none ) {
if ( ! ssl_handshake ( c ) )
return 0 ;
if ( c - > ssl_shake_state ! = comm_ssl_shake_none )
return 1 ;
}
2018-02-27 08:03:54 -05:00
# endif /* HAVE_SSL */
2018-02-07 11:10:31 -05:00
2018-02-08 06:59:30 -05:00
if ( ! c - > tcp_is_reading )
return 1 ;
2020-05-07 10:36:26 -04:00
2020-09-16 12:25:02 -04:00
if ( c - > use_h2 ) {
2020-05-07 10:36:26 -04:00
return comm_point_http2_handle_read ( fd , c ) ;
}
/* http version is <= http/1.1 */
if ( c - > http_min_version > = http_version_2 ) {
/* HTTP/2 failed, not allowed to use lower version. */
return 0 ;
}
2018-02-07 11:10:31 -05:00
/* read more data */
if ( c - > ssl ) {
if ( ! ssl_http_read_more ( c ) )
return 0 ;
} else {
if ( ! http_read_more ( fd , c ) )
return 0 ;
}
2021-02-19 08:42:02 -05:00
if ( c - > http_stored > = sldns_buffer_position ( c - > buffer ) ) {
/* read did not work but we wanted more data, there is
* no bytes to process now . */
return 1 ;
}
2018-02-07 11:10:31 -05:00
sldns_buffer_flip ( c - > buffer ) ;
2021-01-06 04:36:23 -05:00
/* if we are partway in a segment of data, position us at the point
* where we left off previously */
if ( c - > http_stored < sldns_buffer_limit ( c - > buffer ) )
sldns_buffer_set_position ( c - > buffer , c - > http_stored ) ;
else sldns_buffer_set_position ( c - > buffer , sldns_buffer_limit ( c - > buffer ) ) ;
2020-05-07 10:36:26 -04:00
2018-02-07 11:10:31 -05:00
while ( sldns_buffer_remaining ( c - > buffer ) > 0 ) {
2020-05-07 10:36:26 -04:00
/* Handle HTTP/1.x data */
2018-02-07 11:10:31 -05:00
/* if we are reading headers, read more headers */
if ( c - > http_in_headers | | c - > http_in_chunk_headers ) {
/* if header is done, process the header */
if ( ! http_header_done ( c - > buffer ) ) {
/* copy remaining data to front of buffer
* and set rest for writing into it */
http_moveover_buffer ( c - > buffer ) ;
/* return and wait to read more */
return 1 ;
}
if ( ! c - > http_in_chunk_headers ) {
/* process initial headers */
if ( ! http_process_initial_header ( c ) )
return 0 ;
} else {
/* process chunk headers */
int r = http_process_chunk_header ( c ) ;
if ( r = = 0 ) return 0 ;
if ( r = = 2 ) return 1 ; /* done */
/* r == 1, continue */
}
/* see if we have more to process */
continue ;
}
if ( ! c - > http_is_chunked ) {
/* if we are reading nonchunks, process that*/
return http_nonchunk_segment ( c ) ;
} else {
/* if we are reading chunks, read the chunk */
int r = http_chunked_segment ( c ) ;
if ( r = = 0 ) return 0 ;
if ( r = = 1 ) return 1 ;
continue ;
}
}
/* broke out of the loop; could not process header instead need
* to read more */
/* moveover any remaining data and read more data */
http_moveover_buffer ( c - > buffer ) ;
/* return and wait to read more */
return 1 ;
}
/** check pending connect for http */
static int
http_check_connect ( int fd , struct comm_point * c )
{
/* check for pending error from nonblocking connect */
/* from Stevens, unix network programming, vol1, 3rd ed, p450*/
int error = 0 ;
socklen_t len = ( socklen_t ) sizeof ( error ) ;
if ( getsockopt ( fd , SOL_SOCKET , SO_ERROR , ( void * ) & error ,
& len ) < 0 ) {
# ifndef USE_WINSOCK
error = errno ; /* on solaris errno is error */
# else /* USE_WINSOCK */
error = WSAGetLastError ( ) ;
# endif
}
# ifndef USE_WINSOCK
# if defined(EINPROGRESS) && defined(EWOULDBLOCK)
if ( error = = EINPROGRESS | | error = = EWOULDBLOCK )
return 1 ; /* try again later */
else
# endif
if ( error ! = 0 & & verbosity < 2 )
return 0 ; /* silence lots of chatter in the logs */
else if ( error ! = 0 ) {
log_err_addr ( " http connect " , strerror ( error ) ,
& c - > repinfo . addr , c - > repinfo . addrlen ) ;
# else /* USE_WINSOCK */
/* examine error */
if ( error = = WSAEINPROGRESS )
return 1 ;
else if ( error = = WSAEWOULDBLOCK ) {
ub_winsock_tcp_wouldblock ( c - > ev - > ev , UB_EV_WRITE ) ;
return 1 ;
} else if ( error ! = 0 & & verbosity < 2 )
return 0 ;
else if ( error ! = 0 ) {
log_err_addr ( " http connect " , wsa_strerror ( error ) ,
& c - > repinfo . addr , c - > repinfo . addrlen ) ;
# endif /* USE_WINSOCK */
return 0 ;
}
/* keep on processing this socket */
return 2 ;
}
/** write more data for http (with ssl) */
static int
ssl_http_write_more ( struct comm_point * c )
{
# ifdef HAVE_SSL
int r ;
log_assert ( sldns_buffer_remaining ( c - > buffer ) > 0 ) ;
ERR_clear_error ( ) ;
r = SSL_write ( c - > ssl , ( void * ) sldns_buffer_current ( c - > buffer ) ,
( int ) sldns_buffer_remaining ( c - > buffer ) ) ;
if ( r < = 0 ) {
int want = SSL_get_error ( c - > ssl , r ) ;
if ( want = = SSL_ERROR_ZERO_RETURN ) {
return 0 ; /* closed */
} else if ( want = = SSL_ERROR_WANT_READ ) {
2019-04-15 07:52:21 -04:00
c - > ssl_shake_state = comm_ssl_shake_hs_read ;
2018-02-07 11:10:31 -05:00
comm_point_listen_for_rw ( c , 1 , 0 ) ;
return 1 ; /* wait for read condition */
} else if ( want = = SSL_ERROR_WANT_WRITE ) {
return 1 ; /* write more later */
} else if ( want = = SSL_ERROR_SYSCALL ) {
2019-04-11 11:04:32 -04:00
# ifdef EPIPE
if ( errno = = EPIPE & & verbosity < 2 )
return 0 ; /* silence 'broken pipe' */
# endif
2018-02-07 11:10:31 -05:00
if ( errno ! = 0 )
log_err ( " SSL_write syscall: %s " ,
strerror ( errno ) ) ;
return 0 ;
}
log_crypto_err ( " could not SSL_write " ) ;
return 0 ;
}
sldns_buffer_skip ( c - > buffer , ( ssize_t ) r ) ;
return 1 ;
# else
( void ) c ;
return 0 ;
# endif /* HAVE_SSL */
}
/** write more data for http */
static int
http_write_more ( int fd , struct comm_point * c )
{
ssize_t r ;
log_assert ( sldns_buffer_remaining ( c - > buffer ) > 0 ) ;
r = send ( fd , ( void * ) sldns_buffer_current ( c - > buffer ) ,
sldns_buffer_remaining ( c - > buffer ) , 0 ) ;
if ( r = = - 1 ) {
# ifndef USE_WINSOCK
if ( errno = = EINTR | | errno = = EAGAIN )
return 1 ;
# else
if ( WSAGetLastError ( ) = = WSAEINPROGRESS )
return 1 ;
if ( WSAGetLastError ( ) = = WSAEWOULDBLOCK ) {
ub_winsock_tcp_wouldblock ( c - > ev - > ev , UB_EV_WRITE ) ;
return 1 ;
}
# endif
2020-08-31 03:12:01 -04:00
log_err_addr ( " http send r " , sock_strerror ( errno ) ,
& c - > repinfo . addr , c - > repinfo . addrlen ) ;
2018-02-07 11:10:31 -05:00
return 0 ;
}
sldns_buffer_skip ( c - > buffer , r ) ;
return 1 ;
}
2020-05-07 10:36:26 -04:00
# ifdef HAVE_NGHTTP2
ssize_t http2_send_cb ( nghttp2_session * ATTR_UNUSED ( session ) , const uint8_t * buf ,
size_t len , int ATTR_UNUSED ( flags ) , void * cb_arg )
{
2020-10-19 04:24:03 -04:00
ssize_t ret ;
2020-05-07 10:36:26 -04:00
struct http2_session * h2_session = ( struct http2_session * ) cb_arg ;
log_assert ( h2_session - > c - > type = = comm_http ) ;
log_assert ( h2_session - > c - > h2_session ) ;
2020-10-19 04:24:03 -04:00
# ifdef HAVE_SSL
if ( h2_session - > c - > ssl ) {
int r ;
ERR_clear_error ( ) ;
r = SSL_write ( h2_session - > c - > ssl , buf , len ) ;
if ( r < = 0 ) {
int want = SSL_get_error ( h2_session - > c - > ssl , r ) ;
if ( want = = SSL_ERROR_ZERO_RETURN ) {
return NGHTTP2_ERR_CALLBACK_FAILURE ;
} else if ( want = = SSL_ERROR_WANT_READ ) {
h2_session - > c - > ssl_shake_state = comm_ssl_shake_hs_read ;
comm_point_listen_for_rw ( h2_session - > c , 1 , 0 ) ;
return NGHTTP2_ERR_WOULDBLOCK ;
} else if ( want = = SSL_ERROR_WANT_WRITE ) {
return NGHTTP2_ERR_WOULDBLOCK ;
} else if ( want = = SSL_ERROR_SYSCALL ) {
# ifdef EPIPE
if ( errno = = EPIPE & & verbosity < 2 )
return NGHTTP2_ERR_CALLBACK_FAILURE ;
# endif
if ( errno ! = 0 )
log_err ( " SSL_write syscall: %s " ,
strerror ( errno ) ) ;
return NGHTTP2_ERR_CALLBACK_FAILURE ;
}
log_crypto_err ( " could not SSL_write " ) ;
2020-05-07 10:36:26 -04:00
return NGHTTP2_ERR_CALLBACK_FAILURE ;
2020-10-19 04:24:03 -04:00
}
return r ;
}
# endif /* HAVE_SSL */
ret = send ( h2_session - > c - > fd , buf , len , 0 ) ;
if ( ret = = 0 ) {
return NGHTTP2_ERR_CALLBACK_FAILURE ;
} else if ( ret < 0 ) {
# ifndef USE_WINSOCK
if ( errno = = EINTR | | errno = = EAGAIN )
2020-05-07 10:36:26 -04:00
return NGHTTP2_ERR_WOULDBLOCK ;
# ifdef EPIPE
2020-10-19 04:24:03 -04:00
if ( errno = = EPIPE & & verbosity < 2 )
return NGHTTP2_ERR_CALLBACK_FAILURE ;
2020-05-07 10:36:26 -04:00
# endif
2020-10-19 04:24:03 -04:00
# ifdef ECONNRESET
if ( errno = = ECONNRESET & & verbosity < 2 )
2020-05-07 10:36:26 -04:00
return NGHTTP2_ERR_CALLBACK_FAILURE ;
2020-10-19 04:24:03 -04:00
# endif
log_err_addr ( " could not http2 write: %s " , strerror ( errno ) ,
& h2_session - > c - > repinfo . addr ,
h2_session - > c - > repinfo . addrlen ) ;
# else /* USE_WINSOCK */
if ( WSAGetLastError ( ) = = WSAENOTCONN )
return NGHTTP2_ERR_WOULDBLOCK ;
if ( WSAGetLastError ( ) = = WSAEINPROGRESS )
return NGHTTP2_ERR_WOULDBLOCK ;
if ( WSAGetLastError ( ) = = WSAEWOULDBLOCK ) {
ub_winsock_tcp_wouldblock ( h2_session - > c - > ev - > ev ,
UB_EV_WRITE ) ;
return NGHTTP2_ERR_WOULDBLOCK ;
2020-05-07 10:36:26 -04:00
}
2020-10-19 04:24:03 -04:00
if ( WSAGetLastError ( ) = = WSAECONNRESET & & verbosity < 2 )
return NGHTTP2_ERR_CALLBACK_FAILURE ;
log_err_addr ( " could not http2 write: %s " ,
wsa_strerror ( WSAGetLastError ( ) ) ,
& h2_session - > c - > repinfo . addr ,
h2_session - > c - > repinfo . addrlen ) ;
# endif
2020-05-07 10:36:26 -04:00
return NGHTTP2_ERR_CALLBACK_FAILURE ;
}
2020-10-19 04:24:03 -04:00
return ret ;
2020-05-07 10:36:26 -04:00
}
# endif /* HAVE_NGHTTP2 */
/** Handle http2 writing */
static int
comm_point_http2_handle_write ( int ATTR_UNUSED ( fd ) , struct comm_point * c )
{
# ifdef HAVE_NGHTTP2
int ret ;
log_assert ( c - > h2_session ) ;
ret = nghttp2_session_send ( c - > h2_session - > session ) ;
if ( ret ) {
verbose ( VERB_QUERY , " http2: session_send failed, "
" error: %s " , nghttp2_strerror ( ret ) ) ;
return 0 ;
}
if ( nghttp2_session_want_read ( c - > h2_session - > session ) ) {
c - > tcp_is_reading = 1 ;
comm_point_stop_listening ( c ) ;
2020-12-02 04:10:27 -05:00
comm_point_start_listening ( c , - 1 , adjusted_tcp_timeout ( c ) ) ;
2020-05-07 10:36:26 -04:00
} else if ( ! nghttp2_session_want_write ( c - > h2_session - > session ) )
return 0 ; /* connection can be closed */
return 1 ;
# else
( void ) c ;
return 0 ;
# endif
}
2018-02-07 11:10:31 -05:00
/**
* Handle http writing callback .
* @ param fd : file descriptor of socket .
* @ param c : comm point to write buffer out of .
* @ return : 0 on error
*/
static int
comm_point_http_handle_write ( int fd , struct comm_point * c )
{
log_assert ( c - > type = = comm_http ) ;
log_assert ( fd ! = - 1 ) ;
/* check pending connect errors, if that fails, we wait for more,
* or we can continue to write contents */
if ( c - > tcp_check_nb_connect ) {
int r = http_check_connect ( fd , c ) ;
if ( r = = 0 ) return 0 ;
if ( r = = 1 ) return 1 ;
c - > tcp_check_nb_connect = 0 ;
}
/* if we are in ssl handshake, handle SSL handshake */
2018-02-27 08:03:54 -05:00
# ifdef HAVE_SSL
2018-02-07 11:10:31 -05:00
if ( c - > ssl & & c - > ssl_shake_state ! = comm_ssl_shake_none ) {
if ( ! ssl_handshake ( c ) )
return 0 ;
if ( c - > ssl_shake_state ! = comm_ssl_shake_none )
return 1 ;
}
2018-02-27 08:03:54 -05:00
# endif /* HAVE_SSL */
2018-02-08 06:59:30 -05:00
if ( c - > tcp_is_reading )
return 1 ;
2020-05-07 10:36:26 -04:00
2020-09-16 12:25:02 -04:00
if ( c - > use_h2 ) {
2020-05-07 10:36:26 -04:00
return comm_point_http2_handle_write ( fd , c ) ;
}
/* http version is <= http/1.1 */
if ( c - > http_min_version > = http_version_2 ) {
/* HTTP/2 failed, not allowed to use lower version. */
return 0 ;
}
2018-02-07 11:10:31 -05:00
/* if we are writing, write more */
if ( c - > ssl ) {
if ( ! ssl_http_write_more ( c ) )
return 0 ;
} else {
if ( ! http_write_more ( fd , c ) )
return 0 ;
}
/* we write a single buffer contents, that can contain
* the http request , and then flip to read the results */
/* see if write is done */
if ( sldns_buffer_remaining ( c - > buffer ) = = 0 ) {
sldns_buffer_clear ( c - > buffer ) ;
if ( c - > tcp_do_toggle_rw )
c - > tcp_is_reading = 1 ;
c - > tcp_byte_count = 0 ;
/* switch from listening(write) to listening(read) */
comm_point_stop_listening ( c ) ;
comm_point_start_listening ( c , - 1 , - 1 ) ;
}
return 1 ;
}
void
comm_point_http_handle_callback ( int fd , short event , void * arg )
{
struct comm_point * c = ( struct comm_point * ) arg ;
log_assert ( c - > type = = comm_http ) ;
ub_comm_base_now ( c - > ev - > base ) ;
2019-04-11 09:41:53 -04:00
if ( event & UB_EV_TIMEOUT ) {
verbose ( VERB_QUERY , " http took too long, dropped " ) ;
reclaim_http_handler ( c ) ;
if ( ! c - > tcp_do_close ) {
fptr_ok ( fptr_whitelist_comm_point ( c - > callback ) ) ;
( void ) ( * c - > callback ) ( c , c - > cb_arg ,
NETEVENT_TIMEOUT , NULL ) ;
}
return ;
}
2018-02-07 11:10:31 -05:00
if ( event & UB_EV_READ ) {
if ( ! comm_point_http_handle_read ( fd , c ) ) {
reclaim_http_handler ( c ) ;
if ( ! c - > tcp_do_close ) {
fptr_ok ( fptr_whitelist_comm_point (
c - > callback ) ) ;
2020-07-13 09:45:16 -04:00
( void ) ( * c - > callback ) ( c , c - > cb_arg ,
2018-02-07 11:10:31 -05:00
NETEVENT_CLOSED , NULL ) ;
}
}
return ;
}
if ( event & UB_EV_WRITE ) {
if ( ! comm_point_http_handle_write ( fd , c ) ) {
reclaim_http_handler ( c ) ;
if ( ! c - > tcp_do_close ) {
fptr_ok ( fptr_whitelist_comm_point (
c - > callback ) ) ;
2020-07-13 09:45:16 -04:00
( void ) ( * c - > callback ) ( c , c - > cb_arg ,
2018-02-07 11:10:31 -05:00
NETEVENT_CLOSED , NULL ) ;
}
}
return ;
}
log_err ( " Ignored event %d for httphdl. " , event ) ;
}
2007-10-05 04:05:06 -04:00
void comm_point_local_handle_callback ( int fd , short event , void * arg )
2007-02-26 09:49:11 -05:00
{
struct comm_point * c = ( struct comm_point * ) arg ;
log_assert ( c - > type = = comm_local ) ;
2016-03-07 11:01:09 -05:00
ub_comm_base_now ( c - > ev - > base ) ;
2007-02-26 09:49:11 -05:00
2016-03-07 09:10:06 -05:00
if ( event & UB_EV_READ ) {
2007-02-26 09:49:11 -05:00
if ( ! comm_point_tcp_handle_read ( fd , c , 1 ) ) {
2008-02-20 02:18:42 -05:00
fptr_ok ( fptr_whitelist_comm_point ( c - > callback ) ) ;
2007-02-27 05:33:04 -05:00
( void ) ( * c - > callback ) ( c , c - > cb_arg , NETEVENT_CLOSED ,
NULL ) ;
2007-02-26 09:49:11 -05:00
}
return ;
}
log_err ( " Ignored event %d for localhdl. " , event ) ;
}
2008-01-21 11:03:59 -05:00
void comm_point_raw_handle_callback ( int ATTR_UNUSED ( fd ) ,
2008-09-11 10:14:12 -04:00
short event , void * arg )
2008-01-21 11:03:59 -05:00
{
struct comm_point * c = ( struct comm_point * ) arg ;
2008-09-11 10:14:12 -04:00
int err = NETEVENT_NOERROR ;
2008-01-21 11:03:59 -05:00
log_assert ( c - > type = = comm_raw ) ;
2016-03-07 11:01:09 -05:00
ub_comm_base_now ( c - > ev - > base ) ;
2008-09-11 10:14:12 -04:00
2016-03-07 09:10:06 -05:00
if ( event & UB_EV_TIMEOUT )
2008-09-11 10:14:12 -04:00
err = NETEVENT_TIMEOUT ;
2008-07-23 05:23:03 -04:00
fptr_ok ( fptr_whitelist_comm_point_raw ( c - > callback ) ) ;
2008-09-11 10:14:12 -04:00
( void ) ( * c - > callback ) ( c , c - > cb_arg , err , NULL ) ;
2008-01-21 11:03:59 -05:00
}
2007-01-23 11:10:23 -05:00
struct comm_point *
2013-12-03 04:11:16 -05:00
comm_point_create_udp ( struct comm_base * base , int fd , sldns_buffer * buffer ,
2020-12-09 05:00:51 -05:00
comm_point_callback_type * callback , void * callback_arg , struct unbound_socket * socket )
2007-01-23 08:46:18 -05:00
{
struct comm_point * c = ( struct comm_point * ) calloc ( 1 ,
sizeof ( struct comm_point ) ) ;
short evbits ;
if ( ! c )
return NULL ;
c - > ev = ( struct internal_event * ) calloc ( 1 ,
sizeof ( struct internal_event ) ) ;
if ( ! c - > ev ) {
free ( c ) ;
return NULL ;
}
2008-02-19 08:12:23 -05:00
c - > ev - > base = base ;
2007-01-23 08:46:18 -05:00
c - > fd = fd ;
c - > buffer = buffer ;
c - > timeout = NULL ;
c - > tcp_is_reading = 0 ;
c - > tcp_byte_count = 0 ;
c - > tcp_parent = NULL ;
c - > max_tcp_count = 0 ;
2015-03-05 10:23:14 -05:00
c - > cur_tcp_count = 0 ;
2007-01-23 08:46:18 -05:00
c - > tcp_handlers = NULL ;
c - > tcp_free = NULL ;
c - > type = comm_udp ;
c - > tcp_do_close = 0 ;
2007-02-23 05:04:50 -05:00
c - > do_not_close = 0 ;
2007-01-23 08:46:18 -05:00
c - > tcp_do_toggle_rw = 0 ;
2007-05-08 09:25:21 -04:00
c - > tcp_check_nb_connect = 0 ;
2016-07-14 03:06:34 -04:00
# ifdef USE_MSG_FASTOPEN
c - > tcp_do_fastopen = 0 ;
2017-03-20 10:55:31 -04:00
# endif
# ifdef USE_DNSCRYPT
c - > dnscrypt = 0 ;
c - > dnscrypt_buffer = buffer ;
2016-07-14 03:06:34 -04:00
# endif
2008-04-09 07:58:53 -04:00
c - > inuse = 0 ;
2007-01-23 08:46:18 -05:00
c - > callback = callback ;
c - > cb_arg = callback_arg ;
2020-12-09 05:00:51 -05:00
c - > socket = socket ;
2016-03-07 09:10:06 -05:00
evbits = UB_EV_READ | UB_EV_PERSIST ;
/* ub_event stuff */
c - > ev - > ev = ub_event_new ( base - > eb - > base , c - > fd , evbits ,
comm_point_udp_callback , c ) ;
if ( c - > ev - > ev = = NULL ) {
2008-04-11 09:24:49 -04:00
log_err ( " could not baseset udp event " ) ;
comm_point_delete ( c ) ;
return NULL ;
}
2016-03-07 09:10:06 -05:00
if ( fd ! = - 1 & & ub_event_add ( c - > ev - > ev , c - > timeout ) ! = 0 ) {
2007-01-23 08:46:18 -05:00
log_err ( " could not add udp event " ) ;
comm_point_delete ( c ) ;
return NULL ;
}
2021-01-04 08:05:50 -05:00
c - > event_added = 1 ;
2007-01-23 08:46:18 -05:00
return c ;
}
2008-01-15 04:45:30 -05:00
struct comm_point *
comm_point_create_udp_ancil ( struct comm_base * base , int fd ,
2013-12-03 04:11:16 -05:00
sldns_buffer * buffer ,
2020-12-09 05:00:51 -05:00
comm_point_callback_type * callback , void * callback_arg , struct unbound_socket * socket )
2008-01-15 04:45:30 -05:00
{
struct comm_point * c = ( struct comm_point * ) calloc ( 1 ,
sizeof ( struct comm_point ) ) ;
short evbits ;
if ( ! c )
return NULL ;
c - > ev = ( struct internal_event * ) calloc ( 1 ,
sizeof ( struct internal_event ) ) ;
if ( ! c - > ev ) {
free ( c ) ;
return NULL ;
}
2008-02-19 08:12:23 -05:00
c - > ev - > base = base ;
2008-01-15 04:45:30 -05:00
c - > fd = fd ;
c - > buffer = buffer ;
c - > timeout = NULL ;
c - > tcp_is_reading = 0 ;
c - > tcp_byte_count = 0 ;
c - > tcp_parent = NULL ;
c - > max_tcp_count = 0 ;
2015-03-05 10:23:14 -05:00
c - > cur_tcp_count = 0 ;
2008-01-15 04:45:30 -05:00
c - > tcp_handlers = NULL ;
c - > tcp_free = NULL ;
c - > type = comm_udp ;
c - > tcp_do_close = 0 ;
c - > do_not_close = 0 ;
2017-03-20 10:55:31 -04:00
# ifdef USE_DNSCRYPT
2017-04-18 05:00:52 -04:00
c - > dnscrypt = 0 ;
c - > dnscrypt_buffer = buffer ;
2017-03-20 10:55:31 -04:00
# endif
2008-04-09 07:58:53 -04:00
c - > inuse = 0 ;
2008-01-15 04:45:30 -05:00
c - > tcp_do_toggle_rw = 0 ;
c - > tcp_check_nb_connect = 0 ;
2016-07-14 03:06:34 -04:00
# ifdef USE_MSG_FASTOPEN
c - > tcp_do_fastopen = 0 ;
# endif
2008-01-15 04:45:30 -05:00
c - > callback = callback ;
c - > cb_arg = callback_arg ;
2020-12-09 05:00:51 -05:00
c - > socket = socket ;
2016-03-07 09:10:06 -05:00
evbits = UB_EV_READ | UB_EV_PERSIST ;
/* ub_event stuff */
c - > ev - > ev = ub_event_new ( base - > eb - > base , c - > fd , evbits ,
comm_point_udp_ancil_callback , c ) ;
if ( c - > ev - > ev = = NULL ) {
2008-04-11 09:24:49 -04:00
log_err ( " could not baseset udp event " ) ;
comm_point_delete ( c ) ;
return NULL ;
}
2016-03-07 09:10:06 -05:00
if ( fd ! = - 1 & & ub_event_add ( c - > ev - > ev , c - > timeout ) ! = 0 ) {
2008-01-15 04:45:30 -05:00
log_err ( " could not add udp event " ) ;
comm_point_delete ( c ) ;
return NULL ;
}
2021-01-04 08:05:50 -05:00
c - > event_added = 1 ;
2008-01-15 04:45:30 -05:00
return c ;
}
2007-01-23 11:10:23 -05:00
static struct comm_point *
2007-01-23 08:46:18 -05:00
comm_point_create_tcp_handler ( struct comm_base * base ,
struct comm_point * parent , size_t bufsize ,
2019-01-11 09:12:27 -05:00
struct sldns_buffer * spoolbuf , comm_point_callback_type * callback ,
2020-12-09 05:00:51 -05:00
void * callback_arg , struct unbound_socket * socket )
2007-01-23 08:46:18 -05:00
{
2007-01-23 11:10:23 -05:00
struct comm_point * c = ( struct comm_point * ) calloc ( 1 ,
sizeof ( struct comm_point ) ) ;
short evbits ;
if ( ! c )
return NULL ;
c - > ev = ( struct internal_event * ) calloc ( 1 ,
sizeof ( struct internal_event ) ) ;
if ( ! c - > ev ) {
free ( c ) ;
return NULL ;
}
2008-02-19 08:12:23 -05:00
c - > ev - > base = base ;
2007-01-23 11:10:23 -05:00
c - > fd = - 1 ;
2013-12-03 04:11:16 -05:00
c - > buffer = sldns_buffer_new ( bufsize ) ;
2007-02-07 09:18:42 -05:00
if ( ! c - > buffer ) {
free ( c - > ev ) ;
free ( c ) ;
return NULL ;
}
c - > timeout = ( struct timeval * ) malloc ( sizeof ( struct timeval ) ) ;
if ( ! c - > timeout ) {
2013-12-03 04:11:16 -05:00
sldns_buffer_free ( c - > buffer ) ;
2007-02-07 09:18:42 -05:00
free ( c - > ev ) ;
free ( c ) ;
return NULL ;
}
2007-01-23 11:10:23 -05:00
c - > tcp_is_reading = 0 ;
c - > tcp_byte_count = 0 ;
c - > tcp_parent = parent ;
2018-07-31 03:15:12 -04:00
c - > tcp_timeout_msec = parent - > tcp_timeout_msec ;
2018-08-07 07:57:42 -04:00
c - > tcp_conn_limit = parent - > tcp_conn_limit ;
c - > tcl_addr = NULL ;
2018-07-31 03:18:34 -04:00
c - > tcp_keepalive = 0 ;
2007-01-23 11:10:23 -05:00
c - > max_tcp_count = 0 ;
2015-03-05 10:23:14 -05:00
c - > cur_tcp_count = 0 ;
2007-01-23 11:10:23 -05:00
c - > tcp_handlers = NULL ;
c - > tcp_free = NULL ;
c - > type = comm_tcp ;
c - > tcp_do_close = 0 ;
2007-02-23 05:04:50 -05:00
c - > do_not_close = 0 ;
2007-02-26 09:49:11 -05:00
c - > tcp_do_toggle_rw = 1 ;
2007-05-08 09:25:21 -04:00
c - > tcp_check_nb_connect = 0 ;
2016-07-14 03:06:34 -04:00
# ifdef USE_MSG_FASTOPEN
c - > tcp_do_fastopen = 0 ;
2017-03-20 10:55:31 -04:00
# endif
# ifdef USE_DNSCRYPT
2017-04-18 05:00:52 -04:00
c - > dnscrypt = 0 ;
/* We don't know just yet if this is a dnscrypt channel. Allocation
* will be done when handling the callback . */
c - > dnscrypt_buffer = c - > buffer ;
2016-07-14 03:06:34 -04:00
# endif
2007-11-19 11:31:22 -05:00
c - > repinfo . c = c ;
2007-01-23 11:10:23 -05:00
c - > callback = callback ;
c - > cb_arg = callback_arg ;
2020-12-09 05:00:51 -05:00
c - > socket = socket ;
2019-01-11 09:12:27 -05:00
if ( spoolbuf ) {
c - > tcp_req_info = tcp_req_info_create ( spoolbuf ) ;
if ( ! c - > tcp_req_info ) {
log_err ( " could not create tcp commpoint " ) ;
sldns_buffer_free ( c - > buffer ) ;
free ( c - > timeout ) ;
free ( c - > ev ) ;
free ( c ) ;
return NULL ;
}
c - > tcp_req_info - > cp = c ;
c - > tcp_do_close = 1 ;
c - > tcp_do_toggle_rw = 0 ;
}
2007-01-23 11:10:23 -05:00
/* add to parent free list */
c - > tcp_free = parent - > tcp_free ;
parent - > tcp_free = c ;
2016-03-07 09:10:06 -05:00
/* ub_event stuff */
evbits = UB_EV_PERSIST | UB_EV_READ | UB_EV_TIMEOUT ;
c - > ev - > ev = ub_event_new ( base - > eb - > base , c - > fd , evbits ,
comm_point_tcp_handle_callback , c ) ;
if ( c - > ev - > ev = = NULL )
2007-01-23 11:10:23 -05:00
{
log_err ( " could not basetset tcphdl event " ) ;
parent - > tcp_free = c - > tcp_free ;
2019-01-11 09:12:27 -05:00
tcp_req_info_delete ( c - > tcp_req_info ) ;
sldns_buffer_free ( c - > buffer ) ;
free ( c - > timeout ) ;
2007-01-23 11:10:23 -05:00
free ( c - > ev ) ;
free ( c ) ;
return NULL ;
}
return c ;
2007-01-23 08:46:18 -05:00
}
2020-05-07 10:36:26 -04:00
static struct comm_point *
comm_point_create_http_handler ( struct comm_base * base ,
struct comm_point * parent , size_t bufsize , int harden_large_queries ,
2020-05-12 12:12:19 -04:00
uint32_t http_max_streams , char * http_endpoint ,
2020-12-09 05:29:57 -05:00
comm_point_callback_type * callback , void * callback_arg ,
struct unbound_socket * socket )
2020-05-07 10:36:26 -04:00
{
struct comm_point * c = ( struct comm_point * ) calloc ( 1 ,
sizeof ( struct comm_point ) ) ;
short evbits ;
if ( ! c )
return NULL ;
c - > ev = ( struct internal_event * ) calloc ( 1 ,
sizeof ( struct internal_event ) ) ;
if ( ! c - > ev ) {
free ( c ) ;
return NULL ;
}
c - > ev - > base = base ;
c - > fd = - 1 ;
c - > buffer = sldns_buffer_new ( bufsize ) ;
if ( ! c - > buffer ) {
free ( c - > ev ) ;
free ( c ) ;
return NULL ;
}
c - > timeout = ( struct timeval * ) malloc ( sizeof ( struct timeval ) ) ;
if ( ! c - > timeout ) {
sldns_buffer_free ( c - > buffer ) ;
free ( c - > ev ) ;
free ( c ) ;
return NULL ;
}
c - > tcp_is_reading = 0 ;
c - > tcp_byte_count = 0 ;
c - > tcp_parent = parent ;
c - > tcp_timeout_msec = parent - > tcp_timeout_msec ;
c - > tcp_conn_limit = parent - > tcp_conn_limit ;
c - > tcl_addr = NULL ;
c - > tcp_keepalive = 0 ;
c - > max_tcp_count = 0 ;
c - > cur_tcp_count = 0 ;
c - > tcp_handlers = NULL ;
c - > tcp_free = NULL ;
c - > type = comm_http ;
c - > tcp_do_close = 1 ;
c - > do_not_close = 0 ;
c - > tcp_do_toggle_rw = 1 ; /* will be set to 0 after http2 upgrade */
c - > tcp_check_nb_connect = 0 ;
# ifdef USE_MSG_FASTOPEN
c - > tcp_do_fastopen = 0 ;
# endif
# ifdef USE_DNSCRYPT
c - > dnscrypt = 0 ;
c - > dnscrypt_buffer = NULL ;
# endif
c - > repinfo . c = c ;
c - > callback = callback ;
c - > cb_arg = callback_arg ;
2020-12-09 05:29:57 -05:00
c - > socket = socket ;
2020-05-07 10:36:26 -04:00
c - > http_min_version = http_version_2 ;
2020-05-12 12:12:19 -04:00
c - > http2_stream_max_qbuffer_size = bufsize ;
2020-05-07 10:36:26 -04:00
if ( harden_large_queries & & bufsize > 512 )
2020-05-12 12:12:19 -04:00
c - > http2_stream_max_qbuffer_size = 512 ;
c - > http2_max_streams = http_max_streams ;
2020-09-10 07:05:55 -04:00
if ( ! ( c - > http_endpoint = strdup ( http_endpoint ) ) ) {
log_err ( " could not strdup http_endpoint " ) ;
sldns_buffer_free ( c - > buffer ) ;
free ( c - > timeout ) ;
free ( c - > ev ) ;
free ( c ) ;
return NULL ;
}
2020-09-16 12:25:02 -04:00
c - > use_h2 = 0 ;
2020-05-07 10:36:26 -04:00
# ifdef HAVE_NGHTTP2
if ( ! ( c - > h2_session = http2_session_create ( c ) ) ) {
log_err ( " could not create http2 session " ) ;
2020-09-10 07:05:55 -04:00
free ( c - > http_endpoint ) ;
2020-05-07 10:36:26 -04:00
sldns_buffer_free ( c - > buffer ) ;
free ( c - > timeout ) ;
free ( c - > ev ) ;
free ( c ) ;
return NULL ;
}
if ( ! ( c - > h2_session - > callbacks = http2_req_callbacks_create ( ) ) ) {
log_err ( " could not create http2 callbacks " ) ;
http2_session_delete ( c - > h2_session ) ;
2020-09-10 07:05:55 -04:00
free ( c - > http_endpoint ) ;
2020-05-07 10:36:26 -04:00
sldns_buffer_free ( c - > buffer ) ;
free ( c - > timeout ) ;
free ( c - > ev ) ;
free ( c ) ;
return NULL ;
}
# endif
/* add to parent free list */
c - > tcp_free = parent - > tcp_free ;
parent - > tcp_free = c ;
/* ub_event stuff */
evbits = UB_EV_PERSIST | UB_EV_READ | UB_EV_TIMEOUT ;
c - > ev - > ev = ub_event_new ( base - > eb - > base , c - > fd , evbits ,
comm_point_http_handle_callback , c ) ;
if ( c - > ev - > ev = = NULL )
{
log_err ( " could not set http handler event " ) ;
parent - > tcp_free = c - > tcp_free ;
http2_session_delete ( c - > h2_session ) ;
sldns_buffer_free ( c - > buffer ) ;
free ( c - > timeout ) ;
free ( c - > ev ) ;
free ( c ) ;
return NULL ;
}
return c ;
}
2007-01-23 08:46:18 -05:00
struct comm_point *
2018-07-31 03:15:12 -04:00
comm_point_create_tcp ( struct comm_base * base , int fd , int num ,
2020-05-07 10:36:26 -04:00
int idle_timeout , int harden_large_queries ,
2020-05-12 12:12:19 -04:00
uint32_t http_max_streams , char * http_endpoint ,
2020-05-07 10:36:26 -04:00
struct tcl_list * tcp_conn_limit , size_t bufsize ,
struct sldns_buffer * spoolbuf , enum listen_type port_type ,
2020-12-09 05:00:51 -05:00
comm_point_callback_type * callback , void * callback_arg , struct unbound_socket * socket )
2007-01-23 08:46:18 -05:00
{
struct comm_point * c = ( struct comm_point * ) calloc ( 1 ,
sizeof ( struct comm_point ) ) ;
short evbits ;
int i ;
/* first allocate the TCP accept listener */
if ( ! c )
return NULL ;
c - > ev = ( struct internal_event * ) calloc ( 1 ,
sizeof ( struct internal_event ) ) ;
if ( ! c - > ev ) {
free ( c ) ;
return NULL ;
}
2008-02-19 08:12:23 -05:00
c - > ev - > base = base ;
2007-01-23 08:46:18 -05:00
c - > fd = fd ;
c - > buffer = NULL ;
c - > timeout = NULL ;
c - > tcp_is_reading = 0 ;
c - > tcp_byte_count = 0 ;
2018-07-31 03:15:12 -04:00
c - > tcp_timeout_msec = idle_timeout ;
2018-08-07 07:57:42 -04:00
c - > tcp_conn_limit = tcp_conn_limit ;
c - > tcl_addr = NULL ;
2018-07-31 03:18:34 -04:00
c - > tcp_keepalive = 0 ;
2007-01-23 08:46:18 -05:00
c - > tcp_parent = NULL ;
c - > max_tcp_count = num ;
2015-03-05 10:23:14 -05:00
c - > cur_tcp_count = 0 ;
2007-01-25 04:48:37 -05:00
c - > tcp_handlers = ( struct comm_point * * ) calloc ( ( size_t ) num ,
2007-01-23 08:46:18 -05:00
sizeof ( struct comm_point * ) ) ;
2007-02-06 09:00:52 -05:00
if ( ! c - > tcp_handlers ) {
free ( c - > ev ) ;
free ( c ) ;
return NULL ;
}
2007-01-23 08:46:18 -05:00
c - > tcp_free = NULL ;
c - > type = comm_tcp_accept ;
c - > tcp_do_close = 0 ;
2007-02-23 05:04:50 -05:00
c - > do_not_close = 0 ;
2007-01-23 08:46:18 -05:00
c - > tcp_do_toggle_rw = 0 ;
2007-05-08 09:25:21 -04:00
c - > tcp_check_nb_connect = 0 ;
2016-07-14 03:06:34 -04:00
# ifdef USE_MSG_FASTOPEN
c - > tcp_do_fastopen = 0 ;
2017-03-20 10:55:31 -04:00
# endif
# ifdef USE_DNSCRYPT
c - > dnscrypt = 0 ;
c - > dnscrypt_buffer = NULL ;
2016-07-14 03:06:34 -04:00
# endif
2007-01-23 08:46:18 -05:00
c - > callback = NULL ;
c - > cb_arg = NULL ;
2020-12-09 05:00:51 -05:00
c - > socket = socket ;
2016-03-07 09:10:06 -05:00
evbits = UB_EV_READ | UB_EV_PERSIST ;
/* ub_event stuff */
c - > ev - > ev = ub_event_new ( base - > eb - > base , c - > fd , evbits ,
comm_point_tcp_accept_callback , c ) ;
if ( c - > ev - > ev = = NULL ) {
log_err ( " could not baseset tcpacc event " ) ;
comm_point_delete ( c ) ;
return NULL ;
}
if ( ub_event_add ( c - > ev - > ev , c - > timeout ) ! = 0 ) {
2007-01-23 08:46:18 -05:00
log_err ( " could not add tcpacc event " ) ;
2007-02-06 09:00:52 -05:00
comm_point_delete ( c ) ;
2007-01-23 08:46:18 -05:00
return NULL ;
}
2021-01-04 08:05:50 -05:00
c - > event_added = 1 ;
2020-05-07 10:36:26 -04:00
/* now prealloc the handlers */
2007-01-23 08:46:18 -05:00
for ( i = 0 ; i < num ; i + + ) {
2020-05-07 10:36:26 -04:00
if ( port_type = = listen_type_tcp | |
port_type = = listen_type_ssl | |
port_type = = listen_type_tcp_dnscrypt ) {
c - > tcp_handlers [ i ] = comm_point_create_tcp_handler ( base ,
2020-12-09 05:00:51 -05:00
c , bufsize , spoolbuf , callback , callback_arg , socket ) ;
2020-05-07 10:36:26 -04:00
} else if ( port_type = = listen_type_http ) {
c - > tcp_handlers [ i ] = comm_point_create_http_handler (
base , c , bufsize , harden_large_queries ,
2020-05-12 12:12:19 -04:00
http_max_streams , http_endpoint ,
2020-12-09 05:29:57 -05:00
callback , callback_arg , socket ) ;
2020-05-07 10:36:26 -04:00
}
else {
log_err ( " could not create tcp handler, unknown listen "
" type " ) ;
return NULL ;
}
2007-01-23 11:10:23 -05:00
if ( ! c - > tcp_handlers [ i ] ) {
comm_point_delete ( c ) ;
return NULL ;
}
2007-01-23 08:46:18 -05:00
}
return c ;
}
2007-05-08 09:25:21 -04:00
struct comm_point *
2007-05-09 03:00:10 -04:00
comm_point_create_tcp_out ( struct comm_base * base , size_t bufsize ,
2017-01-19 05:25:41 -05:00
comm_point_callback_type * callback , void * callback_arg )
2007-05-08 09:25:21 -04:00
{
struct comm_point * c = ( struct comm_point * ) calloc ( 1 ,
sizeof ( struct comm_point ) ) ;
2007-05-09 03:00:10 -04:00
short evbits ;
2007-05-08 09:25:21 -04:00
if ( ! c )
return NULL ;
c - > ev = ( struct internal_event * ) calloc ( 1 ,
sizeof ( struct internal_event ) ) ;
if ( ! c - > ev ) {
free ( c ) ;
return NULL ;
}
2008-02-19 08:12:23 -05:00
c - > ev - > base = base ;
2007-05-08 09:25:21 -04:00
c - > fd = - 1 ;
2013-12-03 04:11:16 -05:00
c - > buffer = sldns_buffer_new ( bufsize ) ;
2007-05-08 09:25:21 -04:00
if ( ! c - > buffer ) {
free ( c - > ev ) ;
free ( c ) ;
return NULL ;
}
c - > timeout = NULL ;
c - > tcp_is_reading = 0 ;
c - > tcp_byte_count = 0 ;
2018-07-31 03:15:12 -04:00
c - > tcp_timeout_msec = TCP_QUERY_TIMEOUT ;
2018-08-07 07:57:42 -04:00
c - > tcp_conn_limit = NULL ;
c - > tcl_addr = NULL ;
2018-07-31 03:18:34 -04:00
c - > tcp_keepalive = 0 ;
2007-05-08 09:25:21 -04:00
c - > tcp_parent = NULL ;
c - > max_tcp_count = 0 ;
2015-03-05 10:23:14 -05:00
c - > cur_tcp_count = 0 ;
2007-05-08 09:25:21 -04:00
c - > tcp_handlers = NULL ;
c - > tcp_free = NULL ;
c - > type = comm_tcp ;
c - > tcp_do_close = 0 ;
c - > do_not_close = 0 ;
c - > tcp_do_toggle_rw = 1 ;
c - > tcp_check_nb_connect = 1 ;
2016-07-14 03:06:34 -04:00
# ifdef USE_MSG_FASTOPEN
c - > tcp_do_fastopen = 1 ;
2017-03-20 10:55:31 -04:00
# endif
# ifdef USE_DNSCRYPT
c - > dnscrypt = 0 ;
c - > dnscrypt_buffer = c - > buffer ;
2016-07-14 03:06:34 -04:00
# endif
2007-11-19 11:31:22 -05:00
c - > repinfo . c = c ;
2007-05-08 09:25:21 -04:00
c - > callback = callback ;
c - > cb_arg = callback_arg ;
2016-03-07 09:10:06 -05:00
evbits = UB_EV_PERSIST | UB_EV_WRITE ;
c - > ev - > ev = ub_event_new ( base - > eb - > base , c - > fd , evbits ,
comm_point_tcp_handle_callback , c ) ;
if ( c - > ev - > ev = = NULL )
2007-05-09 03:00:10 -04:00
{
2016-03-07 09:10:06 -05:00
log_err ( " could not baseset tcpout event " ) ;
2013-12-03 04:11:16 -05:00
sldns_buffer_free ( c - > buffer ) ;
2007-05-09 03:00:10 -04:00
free ( c - > ev ) ;
free ( c ) ;
return NULL ;
2018-02-07 11:10:31 -05:00
}
return c ;
}
struct comm_point *
comm_point_create_http_out ( struct comm_base * base , size_t bufsize ,
comm_point_callback_type * callback , void * callback_arg ,
sldns_buffer * temp )
{
struct comm_point * c = ( struct comm_point * ) calloc ( 1 ,
sizeof ( struct comm_point ) ) ;
short evbits ;
if ( ! c )
return NULL ;
c - > ev = ( struct internal_event * ) calloc ( 1 ,
sizeof ( struct internal_event ) ) ;
if ( ! c - > ev ) {
free ( c ) ;
return NULL ;
}
c - > ev - > base = base ;
c - > fd = - 1 ;
c - > buffer = sldns_buffer_new ( bufsize ) ;
if ( ! c - > buffer ) {
free ( c - > ev ) ;
free ( c ) ;
return NULL ;
}
c - > timeout = NULL ;
c - > tcp_is_reading = 0 ;
c - > tcp_byte_count = 0 ;
c - > tcp_parent = NULL ;
c - > max_tcp_count = 0 ;
c - > cur_tcp_count = 0 ;
c - > tcp_handlers = NULL ;
c - > tcp_free = NULL ;
c - > type = comm_http ;
c - > tcp_do_close = 0 ;
c - > do_not_close = 0 ;
c - > tcp_do_toggle_rw = 1 ;
c - > tcp_check_nb_connect = 1 ;
c - > http_in_headers = 1 ;
c - > http_in_chunk_headers = 0 ;
c - > http_is_chunked = 0 ;
c - > http_temp = temp ;
# ifdef USE_MSG_FASTOPEN
c - > tcp_do_fastopen = 1 ;
# endif
# ifdef USE_DNSCRYPT
c - > dnscrypt = 0 ;
c - > dnscrypt_buffer = c - > buffer ;
# endif
c - > repinfo . c = c ;
c - > callback = callback ;
c - > cb_arg = callback_arg ;
evbits = UB_EV_PERSIST | UB_EV_WRITE ;
c - > ev - > ev = ub_event_new ( base - > eb - > base , c - > fd , evbits ,
comm_point_http_handle_callback , c ) ;
if ( c - > ev - > ev = = NULL )
{
log_err ( " could not baseset tcpout event " ) ;
2018-02-27 08:03:54 -05:00
# ifdef HAVE_SSL
2018-02-07 11:10:31 -05:00
SSL_free ( c - > ssl ) ;
2018-02-27 08:03:54 -05:00
# endif
2018-02-07 11:10:31 -05:00
sldns_buffer_free ( c - > buffer ) ;
free ( c - > ev ) ;
free ( c ) ;
return NULL ;
2007-05-09 03:00:10 -04:00
}
2007-05-08 09:25:21 -04:00
return c ;
}
2007-02-26 09:49:11 -05:00
struct comm_point *
comm_point_create_local ( struct comm_base * base , int fd , size_t bufsize ,
2017-01-19 05:25:41 -05:00
comm_point_callback_type * callback , void * callback_arg )
2007-02-26 09:49:11 -05:00
{
struct comm_point * c = ( struct comm_point * ) calloc ( 1 ,
sizeof ( struct comm_point ) ) ;
short evbits ;
if ( ! c )
return NULL ;
c - > ev = ( struct internal_event * ) calloc ( 1 ,
sizeof ( struct internal_event ) ) ;
if ( ! c - > ev ) {
free ( c ) ;
return NULL ;
}
2008-02-19 08:12:23 -05:00
c - > ev - > base = base ;
2007-02-26 09:49:11 -05:00
c - > fd = fd ;
2013-12-03 04:11:16 -05:00
c - > buffer = sldns_buffer_new ( bufsize ) ;
2007-02-26 09:49:11 -05:00
if ( ! c - > buffer ) {
free ( c - > ev ) ;
free ( c ) ;
return NULL ;
}
c - > timeout = NULL ;
c - > tcp_is_reading = 1 ;
c - > tcp_byte_count = 0 ;
c - > tcp_parent = NULL ;
c - > max_tcp_count = 0 ;
2015-03-05 10:23:14 -05:00
c - > cur_tcp_count = 0 ;
2007-02-26 09:49:11 -05:00
c - > tcp_handlers = NULL ;
c - > tcp_free = NULL ;
c - > type = comm_local ;
c - > tcp_do_close = 0 ;
c - > do_not_close = 1 ;
c - > tcp_do_toggle_rw = 0 ;
2007-05-08 09:25:21 -04:00
c - > tcp_check_nb_connect = 0 ;
2016-07-14 03:06:34 -04:00
# ifdef USE_MSG_FASTOPEN
c - > tcp_do_fastopen = 0 ;
2017-03-20 10:55:31 -04:00
# endif
# ifdef USE_DNSCRYPT
c - > dnscrypt = 0 ;
c - > dnscrypt_buffer = c - > buffer ;
2016-07-14 03:06:34 -04:00
# endif
2007-02-26 09:49:11 -05:00
c - > callback = callback ;
c - > cb_arg = callback_arg ;
2016-03-07 09:10:06 -05:00
/* ub_event stuff */
evbits = UB_EV_PERSIST | UB_EV_READ ;
c - > ev - > ev = ub_event_new ( base - > eb - > base , c - > fd , evbits ,
comm_point_local_handle_callback , c ) ;
if ( c - > ev - > ev = = NULL ) {
log_err ( " could not baseset localhdl event " ) ;
free ( c - > ev ) ;
free ( c ) ;
return NULL ;
}
if ( ub_event_add ( c - > ev - > ev , c - > timeout ) ! = 0 ) {
2008-01-21 11:03:59 -05:00
log_err ( " could not add localhdl event " ) ;
2016-03-07 09:10:06 -05:00
ub_event_free ( c - > ev - > ev ) ;
2008-01-21 11:03:59 -05:00
free ( c - > ev ) ;
free ( c ) ;
return NULL ;
}
2021-01-04 08:05:50 -05:00
c - > event_added = 1 ;
2008-01-21 11:03:59 -05:00
return c ;
}
struct comm_point *
comm_point_create_raw ( struct comm_base * base , int fd , int writing ,
2017-01-19 05:25:41 -05:00
comm_point_callback_type * callback , void * callback_arg )
2008-01-21 11:03:59 -05:00
{
struct comm_point * c = ( struct comm_point * ) calloc ( 1 ,
sizeof ( struct comm_point ) ) ;
short evbits ;
if ( ! c )
return NULL ;
c - > ev = ( struct internal_event * ) calloc ( 1 ,
sizeof ( struct internal_event ) ) ;
if ( ! c - > ev ) {
free ( c ) ;
return NULL ;
}
2008-02-19 08:12:23 -05:00
c - > ev - > base = base ;
2008-01-21 11:03:59 -05:00
c - > fd = fd ;
c - > buffer = NULL ;
c - > timeout = NULL ;
c - > tcp_is_reading = 0 ;
c - > tcp_byte_count = 0 ;
c - > tcp_parent = NULL ;
c - > max_tcp_count = 0 ;
2015-03-05 10:23:14 -05:00
c - > cur_tcp_count = 0 ;
2008-01-21 11:03:59 -05:00
c - > tcp_handlers = NULL ;
c - > tcp_free = NULL ;
c - > type = comm_raw ;
c - > tcp_do_close = 0 ;
c - > do_not_close = 1 ;
c - > tcp_do_toggle_rw = 0 ;
c - > tcp_check_nb_connect = 0 ;
2016-07-14 03:06:34 -04:00
# ifdef USE_MSG_FASTOPEN
c - > tcp_do_fastopen = 0 ;
2017-03-20 10:55:31 -04:00
# endif
# ifdef USE_DNSCRYPT
c - > dnscrypt = 0 ;
c - > dnscrypt_buffer = c - > buffer ;
2016-07-14 03:06:34 -04:00
# endif
2008-01-21 11:03:59 -05:00
c - > callback = callback ;
c - > cb_arg = callback_arg ;
2016-03-07 09:10:06 -05:00
/* ub_event stuff */
2008-01-21 11:03:59 -05:00
if ( writing )
2016-03-07 09:10:06 -05:00
evbits = UB_EV_PERSIST | UB_EV_WRITE ;
else evbits = UB_EV_PERSIST | UB_EV_READ ;
c - > ev - > ev = ub_event_new ( base - > eb - > base , c - > fd , evbits ,
comm_point_raw_handle_callback , c ) ;
if ( c - > ev - > ev = = NULL ) {
log_err ( " could not baseset rawhdl event " ) ;
free ( c - > ev ) ;
free ( c ) ;
return NULL ;
}
if ( ub_event_add ( c - > ev - > ev , c - > timeout ) ! = 0 ) {
2008-01-21 11:03:59 -05:00
log_err ( " could not add rawhdl event " ) ;
2016-03-07 09:10:06 -05:00
ub_event_free ( c - > ev - > ev ) ;
2007-02-26 09:49:11 -05:00
free ( c - > ev ) ;
free ( c ) ;
return NULL ;
}
2021-01-04 08:05:50 -05:00
c - > event_added = 1 ;
2007-02-26 09:49:11 -05:00
return c ;
}
2007-01-23 11:10:23 -05:00
void
comm_point_close ( struct comm_point * c )
2007-01-23 08:46:18 -05:00
{
2007-02-06 09:00:52 -05:00
if ( ! c )
return ;
2018-05-25 06:02:06 -04:00
if ( c - > fd ! = - 1 ) {
2020-06-03 11:24:26 -04:00
verbose ( 5 , " comm_point_close of %d: event_del " , c - > fd ) ;
2021-01-04 08:05:50 -05:00
if ( c - > event_added ) {
if ( ub_event_del ( c - > ev - > ev ) ! = 0 ) {
log_err ( " could not event_del on close " ) ;
}
c - > event_added = 0 ;
2007-05-08 09:25:21 -04:00
}
2018-05-25 06:02:06 -04:00
}
2018-08-07 07:57:42 -04:00
tcl_close_connection ( c - > tcl_addr ) ;
2019-01-11 09:12:27 -05:00
if ( c - > tcp_req_info )
tcp_req_info_clear ( c - > tcp_req_info ) ;
2020-05-07 10:36:26 -04:00
if ( c - > h2_session )
http2_session_server_delete ( c - > h2_session ) ;
2007-02-06 09:00:52 -05:00
/* close fd after removing from event lists, or epoll.. is messed up */
2008-04-11 09:24:49 -04:00
if ( c - > fd ! = - 1 & & ! c - > do_not_close ) {
2021-07-16 03:12:06 -04:00
# ifdef USE_WINSOCK
2018-05-25 06:41:24 -04:00
if ( c - > type = = comm_tcp | | c - > type = = comm_http ) {
/* delete sticky events for the fd, it gets closed */
ub_winsock_tcp_wouldblock ( c - > ev - > ev , UB_EV_READ ) ;
ub_winsock_tcp_wouldblock ( c - > ev - > ev , UB_EV_WRITE ) ;
}
2021-07-16 03:12:06 -04:00
# endif
2008-04-11 09:24:49 -04:00
verbose ( VERB_ALGO , " close fd %d " , c - > fd ) ;
2020-08-31 02:41:34 -04:00
sock_close ( c - > fd ) ;
2008-04-11 09:24:49 -04:00
}
2007-01-25 11:08:52 -05:00
c - > fd = - 1 ;
2007-01-23 08:46:18 -05:00
}
2007-01-23 11:10:23 -05:00
void
comm_point_delete ( struct comm_point * c )
2007-01-23 08:46:18 -05:00
{
2007-01-23 11:10:23 -05:00
if ( ! c )
return ;
2018-02-08 06:59:30 -05:00
if ( ( c - > type = = comm_tcp | | c - > type = = comm_http ) & & c - > ssl ) {
2012-06-22 05:11:41 -04:00
# ifdef HAVE_SSL
2011-10-31 10:48:48 -04:00
SSL_shutdown ( c - > ssl ) ;
SSL_free ( c - > ssl ) ;
2012-06-22 05:11:41 -04:00
# endif
2011-10-31 10:48:48 -04:00
}
2020-05-12 12:12:19 -04:00
if ( c - > type = = comm_http & & c - > http_endpoint ) {
free ( c - > http_endpoint ) ;
c - > http_endpoint = NULL ;
}
2007-01-23 08:46:18 -05:00
comm_point_close ( c ) ;
if ( c - > tcp_handlers ) {
int i ;
for ( i = 0 ; i < c - > max_tcp_count ; i + + )
comm_point_delete ( c - > tcp_handlers [ i ] ) ;
free ( c - > tcp_handlers ) ;
}
2007-02-22 11:22:54 -05:00
free ( c - > timeout ) ;
2018-02-08 06:59:30 -05:00
if ( c - > type = = comm_tcp | | c - > type = = comm_local | | c - > type = = comm_http ) {
2013-12-03 04:11:16 -05:00
sldns_buffer_free ( c - > buffer ) ;
2017-03-20 10:55:31 -04:00
# ifdef USE_DNSCRYPT
2017-04-18 05:00:52 -04:00
if ( c - > dnscrypt & & c - > dnscrypt_buffer ! = c - > buffer ) {
sldns_buffer_free ( c - > dnscrypt_buffer ) ;
}
2017-03-20 10:55:31 -04:00
# endif
2019-01-11 09:12:27 -05:00
if ( c - > tcp_req_info ) {
tcp_req_info_delete ( c - > tcp_req_info ) ;
}
2020-05-07 10:36:26 -04:00
if ( c - > h2_session ) {
http2_session_delete ( c - > h2_session ) ;
}
2017-04-18 05:00:52 -04:00
}
2016-03-07 09:10:06 -05:00
ub_event_free ( c - > ev - > ev ) ;
2007-01-23 08:46:18 -05:00
free ( c - > ev ) ;
free ( c ) ;
}
2007-02-06 09:00:52 -05:00
void
comm_point_send_reply ( struct comm_reply * repinfo )
2007-01-31 04:32:30 -05:00
{
2017-03-20 11:09:06 -04:00
struct sldns_buffer * buffer ;
2007-01-31 04:32:30 -05:00
log_assert ( repinfo & & repinfo - > c ) ;
2017-03-20 10:55:31 -04:00
# ifdef USE_DNSCRYPT
2017-03-20 11:09:06 -04:00
buffer = repinfo - > c - > dnscrypt_buffer ;
2017-04-18 05:00:52 -04:00
if ( ! dnsc_handle_uncurved_request ( repinfo ) ) {
2017-03-20 11:09:06 -04:00
return ;
}
# else
buffer = repinfo - > c - > buffer ;
2017-03-20 10:55:31 -04:00
# endif
2007-01-31 04:32:30 -05:00
if ( repinfo - > c - > type = = comm_udp ) {
2008-01-17 10:35:34 -05:00
if ( repinfo - > srctype )
2008-01-15 04:45:30 -05:00
comm_point_send_udp_msg_if ( repinfo - > c ,
2017-03-20 10:55:31 -04:00
buffer , ( struct sockaddr * ) & repinfo - > addr ,
2008-01-17 10:35:34 -05:00
repinfo - > addrlen , repinfo ) ;
2008-01-15 04:45:30 -05:00
else
2017-03-20 10:55:31 -04:00
comm_point_send_udp_msg ( repinfo - > c , buffer ,
2020-12-16 11:11:41 -05:00
( struct sockaddr * ) & repinfo - > addr , repinfo - > addrlen , 0 ) ;
2014-08-05 03:57:52 -04:00
# ifdef USE_DNSTAP
2020-12-09 05:00:51 -05:00
/*
* sending src ( client ) / dst ( local service ) addresses over DNSTAP from udp callback
*/
if ( repinfo - > c - > dtenv ! = NULL & & repinfo - > c - > dtenv - > log_client_response_messages ) {
2020-12-09 05:56:35 -05:00
log_addr ( VERB_ALGO , " from local addr " , ( void * ) repinfo - > c - > socket - > addr - > ai_addr , repinfo - > c - > socket - > addr - > ai_addrlen ) ;
2020-12-09 05:00:51 -05:00
log_addr ( VERB_ALGO , " response to client " , & repinfo - > addr , repinfo - > addrlen ) ;
2020-12-09 05:56:35 -05:00
dt_msg_send_client_response ( repinfo - > c - > dtenv , & repinfo - > addr , ( void * ) repinfo - > c - > socket - > addr - > ai_addr , repinfo - > c - > type , repinfo - > c - > buffer ) ;
2020-12-09 05:00:51 -05:00
}
2014-08-05 03:57:52 -04:00
# endif
2007-01-31 04:32:30 -05:00
} else {
2020-03-30 06:19:17 -04:00
# ifdef USE_DNSTAP
2020-12-09 05:00:51 -05:00
/*
* sending src ( client ) / dst ( local service ) addresses over DNSTAP from TCP callback
*/
if ( repinfo - > c - > tcp_parent - > dtenv ! = NULL & & repinfo - > c - > tcp_parent - > dtenv - > log_client_response_messages ) {
2020-12-09 05:56:35 -05:00
log_addr ( VERB_ALGO , " from local addr " , ( void * ) repinfo - > c - > socket - > addr - > ai_addr , repinfo - > c - > socket - > addr - > ai_addrlen ) ;
2020-12-09 05:00:51 -05:00
log_addr ( VERB_ALGO , " response to client " , & repinfo - > addr , repinfo - > addrlen ) ;
2020-12-09 05:56:35 -05:00
dt_msg_send_client_response ( repinfo - > c - > tcp_parent - > dtenv , & repinfo - > addr , ( void * ) repinfo - > c - > socket - > addr - > ai_addr , repinfo - > c - > type ,
2020-12-09 05:00:51 -05:00
( repinfo - > c - > tcp_req_info ? repinfo - > c - > tcp_req_info - > spool_buffer : repinfo - > c - > buffer ) ) ;
}
2020-03-30 06:19:17 -04:00
# endif
2019-01-11 09:12:27 -05:00
if ( repinfo - > c - > tcp_req_info ) {
tcp_req_info_send_reply ( repinfo - > c - > tcp_req_info ) ;
2020-09-16 12:25:02 -04:00
} else if ( repinfo - > c - > use_h2 ) {
2020-05-07 10:36:26 -04:00
if ( ! http2_submit_dns_response ( repinfo - > c - > h2_session ) ) {
comm_point_drop_reply ( repinfo ) ;
return ;
}
repinfo - > c - > h2_stream = NULL ;
repinfo - > c - > tcp_is_reading = 0 ;
comm_point_stop_listening ( repinfo - > c ) ;
comm_point_start_listening ( repinfo - > c , - 1 ,
2020-12-02 04:10:27 -05:00
adjusted_tcp_timeout ( repinfo - > c ) ) ;
2020-05-07 10:36:26 -04:00
return ;
2019-01-11 09:12:27 -05:00
} else {
comm_point_start_listening ( repinfo - > c , - 1 ,
2020-12-02 04:10:27 -05:00
adjusted_tcp_timeout ( repinfo - > c ) ) ;
2019-01-11 09:12:27 -05:00
}
2007-01-31 04:32:30 -05:00
}
}
2007-03-28 11:40:12 -04:00
void
2007-02-07 10:53:51 -05:00
comm_point_drop_reply ( struct comm_reply * repinfo )
{
if ( ! repinfo )
return ;
2019-11-20 08:05:54 -05:00
log_assert ( repinfo - > c ) ;
2007-02-07 10:53:51 -05:00
log_assert ( repinfo - > c - > type ! = comm_tcp_accept ) ;
if ( repinfo - > c - > type = = comm_udp )
return ;
2019-01-11 09:12:27 -05:00
if ( repinfo - > c - > tcp_req_info )
repinfo - > c - > tcp_req_info - > is_drop = 1 ;
2020-05-07 10:36:26 -04:00
if ( repinfo - > c - > type = = comm_http ) {
if ( repinfo - > c - > h2_session ) {
repinfo - > c - > h2_session - > is_drop = 1 ;
if ( ! repinfo - > c - > h2_session - > postpone_drop )
reclaim_http_handler ( repinfo - > c ) ;
return ;
}
reclaim_http_handler ( repinfo - > c ) ;
return ;
}
2007-02-07 10:53:51 -05:00
reclaim_tcp_handler ( repinfo - > c ) ;
}
void
comm_point_stop_listening ( struct comm_point * c )
2007-02-07 09:18:42 -05:00
{
2007-02-15 10:50:22 -05:00
verbose ( VERB_ALGO , " comm point stop listening %d " , c - > fd ) ;
2021-01-04 08:05:50 -05:00
if ( c - > event_added ) {
if ( ub_event_del ( c - > ev - > ev ) ! = 0 ) {
log_err ( " event_del error to stoplisten " ) ;
}
c - > event_added = 0 ;
2007-02-07 09:18:42 -05:00
}
}
2007-02-07 10:53:51 -05:00
void
2016-06-15 10:23:43 -04:00
comm_point_start_listening ( struct comm_point * c , int newfd , int msec )
2007-02-07 09:18:42 -05:00
{
2019-04-11 09:41:53 -04:00
verbose ( VERB_ALGO , " comm point start listening %d (%d msec) " ,
c - > fd = = - 1 ? newfd : c - > fd , msec ) ;
2007-02-07 09:18:42 -05:00
if ( c - > type = = comm_tcp_accept & & ! c - > tcp_free ) {
/* no use to start listening no free slots. */
return ;
}
2021-01-04 08:05:50 -05:00
if ( c - > event_added ) {
if ( ub_event_del ( c - > ev - > ev ) ! = 0 ) {
log_err ( " event_del error to startlisten " ) ;
}
c - > event_added = 0 ;
}
2016-06-15 10:23:43 -04:00
if ( msec ! = - 1 & & msec ! = 0 ) {
2007-02-07 09:18:42 -05:00
if ( ! c - > timeout ) {
c - > timeout = ( struct timeval * ) malloc ( sizeof (
struct timeval ) ) ;
if ( ! c - > timeout ) {
log_err ( " cpsl: malloc failed. No net read. " ) ;
return ;
}
}
2016-03-07 09:10:06 -05:00
ub_event_add_bits ( c - > ev - > ev , UB_EV_TIMEOUT ) ;
2007-02-07 10:44:19 -05:00
# ifndef S_SPLINT_S /* splint fails on struct timeval. */
2016-06-15 10:23:43 -04:00
c - > timeout - > tv_sec = msec / 1000 ;
2016-06-16 03:21:07 -04:00
c - > timeout - > tv_usec = ( msec % 1000 ) * 1000 ;
2007-02-07 10:44:19 -05:00
# endif /* S_SPLINT_S */
2021-11-05 06:21:30 -04:00
} else {
if ( msec = = 0 | | ! c - > timeout ) {
ub_event_del_bits ( c - > ev - > ev , UB_EV_TIMEOUT ) ;
}
2007-02-07 10:44:19 -05:00
}
2018-02-08 10:14:51 -05:00
if ( c - > type = = comm_tcp | | c - > type = = comm_http ) {
2016-03-07 09:10:06 -05:00
ub_event_del_bits ( c - > ev - > ev , UB_EV_READ | UB_EV_WRITE ) ;
2020-07-09 07:41:03 -04:00
if ( c - > tcp_write_and_read ) {
2020-07-09 07:41:50 -04:00
verbose ( 5 , " startlistening %d mode rw " , ( newfd = = - 1 ? c - > fd : newfd ) ) ;
2020-06-25 08:26:29 -04:00
ub_event_add_bits ( c - > ev - > ev , UB_EV_READ | UB_EV_WRITE ) ;
2020-07-09 07:41:03 -04:00
} else if ( c - > tcp_is_reading ) {
2020-07-09 07:41:50 -04:00
verbose ( 5 , " startlistening %d mode r " , ( newfd = = - 1 ? c - > fd : newfd ) ) ;
2016-03-07 09:10:06 -05:00
ub_event_add_bits ( c - > ev - > ev , UB_EV_READ ) ;
2020-07-09 07:41:03 -04:00
} else {
2020-07-09 07:41:50 -04:00
verbose ( 5 , " startlistening %d mode w " , ( newfd = = - 1 ? c - > fd : newfd ) ) ;
2020-07-09 07:41:03 -04:00
ub_event_add_bits ( c - > ev - > ev , UB_EV_WRITE ) ;
}
2007-02-07 10:44:19 -05:00
}
if ( newfd ! = - 1 ) {
2020-06-03 11:24:26 -04:00
if ( c - > fd ! = - 1 & & c - > fd ! = newfd ) {
verbose ( 5 , " cpsl close of fd %d for %d " , c - > fd , newfd ) ;
2020-08-31 02:41:34 -04:00
sock_close ( c - > fd ) ;
2009-04-29 11:23:08 -04:00
}
2007-02-07 10:44:19 -05:00
c - > fd = newfd ;
2016-03-07 09:10:06 -05:00
ub_event_set_fd ( c - > ev - > ev , c - > fd ) ;
2007-02-07 09:18:42 -05:00
}
2016-06-15 10:23:43 -04:00
if ( ub_event_add ( c - > ev - > ev , msec = = 0 ? NULL : c - > timeout ) ! = 0 ) {
2007-02-07 09:18:42 -05:00
log_err ( " event_add failed. in cpsl. " ) ;
2021-11-05 06:21:30 -04:00
return ;
2007-02-07 09:18:42 -05:00
}
2021-01-04 08:05:50 -05:00
c - > event_added = 1 ;
2007-02-07 09:18:42 -05:00
}
2008-09-11 10:14:12 -04:00
void comm_point_listen_for_rw ( struct comm_point * c , int rd , int wr )
{
verbose ( VERB_ALGO , " comm point listen_for_rw %d %d " , c - > fd , wr ) ;
2021-01-04 08:05:50 -05:00
if ( c - > event_added ) {
if ( ub_event_del ( c - > ev - > ev ) ! = 0 ) {
log_err ( " event_del error to cplf " ) ;
}
c - > event_added = 0 ;
2008-09-11 10:14:12 -04:00
}
2021-11-05 06:21:30 -04:00
if ( ! c - > timeout ) {
ub_event_del_bits ( c - > ev - > ev , UB_EV_TIMEOUT ) ;
}
2016-03-07 09:10:06 -05:00
ub_event_del_bits ( c - > ev - > ev , UB_EV_READ | UB_EV_WRITE ) ;
if ( rd ) ub_event_add_bits ( c - > ev - > ev , UB_EV_READ ) ;
if ( wr ) ub_event_add_bits ( c - > ev - > ev , UB_EV_WRITE ) ;
if ( ub_event_add ( c - > ev - > ev , c - > timeout ) ! = 0 ) {
2008-09-11 10:14:12 -04:00
log_err ( " event_add failed. in cplf. " ) ;
2021-11-05 06:21:30 -04:00
return ;
2008-09-11 10:14:12 -04:00
}
2021-01-04 08:05:50 -05:00
c - > event_added = 1 ;
2008-09-11 10:14:12 -04:00
}
2007-07-20 11:51:06 -04:00
size_t comm_point_get_mem ( struct comm_point * c )
{
size_t s ;
if ( ! c )
return 0 ;
s = sizeof ( * c ) + sizeof ( * c - > ev ) ;
if ( c - > timeout )
s + = sizeof ( * c - > timeout ) ;
2017-03-20 10:55:31 -04:00
if ( c - > type = = comm_tcp | | c - > type = = comm_local ) {
2013-12-03 04:11:16 -05:00
s + = sizeof ( * c - > buffer ) + sldns_buffer_capacity ( c - > buffer ) ;
2017-03-20 10:55:31 -04:00
# ifdef USE_DNSCRYPT
2017-04-18 05:00:52 -04:00
s + = sizeof ( * c - > dnscrypt_buffer ) ;
if ( c - > buffer ! = c - > dnscrypt_buffer ) {
s + = sldns_buffer_capacity ( c - > dnscrypt_buffer ) ;
}
2017-03-20 10:55:31 -04:00
# endif
2017-04-18 05:00:52 -04:00
}
2007-07-20 11:51:06 -04:00
if ( c - > type = = comm_tcp_accept ) {
int i ;
for ( i = 0 ; i < c - > max_tcp_count ; i + + )
s + = comm_point_get_mem ( c - > tcp_handlers [ i ] ) ;
}
return s ;
}
2007-02-06 09:00:52 -05:00
struct comm_timer *
comm_timer_create ( struct comm_base * base , void ( * cb ) ( void * ) , void * cb_arg )
2007-02-01 10:06:38 -05:00
{
2016-03-07 09:10:06 -05:00
struct internal_timer * tm = ( struct internal_timer * ) calloc ( 1 ,
2007-02-01 10:06:38 -05:00
sizeof ( struct internal_timer ) ) ;
2016-03-07 09:10:06 -05:00
if ( ! tm ) {
2007-02-01 10:06:38 -05:00
log_err ( " malloc failed " ) ;
return NULL ;
}
2016-03-07 09:10:06 -05:00
tm - > super . ev_timer = tm ;
tm - > base = base ;
tm - > super . callback = cb ;
tm - > super . cb_arg = cb_arg ;
tm - > ev = ub_event_new ( base - > eb - > base , - 1 , UB_EV_TIMEOUT ,
comm_timer_callback , & tm - > super ) ;
if ( tm - > ev = = NULL ) {
2007-02-01 10:06:38 -05:00
log_err ( " timer_create: event_base_set failed. " ) ;
free ( tm ) ;
return NULL ;
}
2016-03-07 09:10:06 -05:00
return & tm - > super ;
2007-02-01 10:06:38 -05:00
}
2007-02-06 09:00:52 -05:00
void
comm_timer_disable ( struct comm_timer * timer )
2007-02-01 10:06:38 -05:00
{
if ( ! timer )
return ;
2016-03-07 09:10:06 -05:00
ub_timer_del ( timer - > ev_timer - > ev ) ;
2007-02-01 10:06:38 -05:00
timer - > ev_timer - > enabled = 0 ;
}
2007-02-06 09:00:52 -05:00
void
comm_timer_set ( struct comm_timer * timer , struct timeval * tv )
2007-02-01 10:06:38 -05:00
{
2007-10-16 08:26:09 -04:00
log_assert ( tv ) ;
2007-02-01 10:06:38 -05:00
if ( timer - > ev_timer - > enabled )
comm_timer_disable ( timer ) ;
2016-03-07 09:10:06 -05:00
if ( ub_timer_add ( timer - > ev_timer - > ev , timer - > ev_timer - > base - > eb - > base ,
comm_timer_callback , timer , tv ) ! = 0 )
2007-12-04 12:54:14 -05:00
log_err ( " comm_timer_set: evtimer_add failed. " ) ;
2007-02-01 10:06:38 -05:00
timer - > ev_timer - > enabled = 1 ;
}
2007-02-06 09:00:52 -05:00
void
comm_timer_delete ( struct comm_timer * timer )
2007-02-01 10:06:38 -05:00
{
if ( ! timer )
return ;
comm_timer_disable ( timer ) ;
2016-03-07 09:10:06 -05:00
/* Free the sub struct timer->ev_timer derived from the super struct timer.
* i . e . assert ( timer = = timer - > ev_timer )
*/
ub_event_free ( timer - > ev_timer - > ev ) ;
2007-02-01 10:06:38 -05:00
free ( timer - > ev_timer ) ;
}
2007-10-05 04:05:06 -04:00
void
2007-02-01 10:06:38 -05:00
comm_timer_callback ( int ATTR_UNUSED ( fd ) , short event , void * arg )
{
struct comm_timer * tm = ( struct comm_timer * ) arg ;
2016-03-07 09:10:06 -05:00
if ( ! ( event & UB_EV_TIMEOUT ) )
2007-02-01 10:06:38 -05:00
return ;
2016-03-07 09:10:06 -05:00
ub_comm_base_now ( tm - > ev_timer - > base ) ;
2007-02-01 10:06:38 -05:00
tm - > ev_timer - > enabled = 0 ;
2008-02-20 02:18:42 -05:00
fptr_ok ( fptr_whitelist_comm_timer ( tm - > callback ) ) ;
2007-02-01 10:06:38 -05:00
( * tm - > callback ) ( tm - > cb_arg ) ;
}
int
comm_timer_is_set ( struct comm_timer * timer )
{
return ( int ) timer - > ev_timer - > enabled ;
}
2007-02-05 11:46:40 -05:00
2007-07-20 11:51:06 -04:00
size_t
2016-03-10 09:39:48 -05:00
comm_timer_get_mem ( struct comm_timer * ATTR_UNUSED ( timer ) )
2007-07-20 11:51:06 -04:00
{
2016-03-07 09:10:06 -05:00
return sizeof ( struct internal_timer ) ;
2007-07-20 11:51:06 -04:00
}
2007-02-06 09:00:52 -05:00
struct comm_signal *
comm_signal_create ( struct comm_base * base ,
2007-02-05 11:46:40 -05:00
void ( * callback ) ( int , void * ) , void * cb_arg )
{
struct comm_signal * com = ( struct comm_signal * ) malloc (
sizeof ( struct comm_signal ) ) ;
if ( ! com ) {
log_err ( " malloc failed " ) ;
return NULL ;
}
com - > base = base ;
com - > callback = callback ;
com - > cb_arg = cb_arg ;
com - > ev_signal = NULL ;
return com ;
}
2007-10-05 04:05:06 -04:00
void
2007-02-06 09:00:52 -05:00
comm_signal_callback ( int sig , short event , void * arg )
2007-02-05 11:46:40 -05:00
{
2007-02-06 09:00:52 -05:00
struct comm_signal * comsig = ( struct comm_signal * ) arg ;
2016-03-07 09:10:06 -05:00
if ( ! ( event & UB_EV_SIGNAL ) )
2007-02-05 11:46:40 -05:00
return ;
2016-03-07 11:01:09 -05:00
ub_comm_base_now ( comsig - > base ) ;
2008-02-20 02:18:42 -05:00
fptr_ok ( fptr_whitelist_comm_signal ( comsig - > callback ) ) ;
2007-02-06 09:00:52 -05:00
( * comsig - > callback ) ( sig , comsig - > cb_arg ) ;
2007-02-05 11:46:40 -05:00
}
2007-02-06 09:00:52 -05:00
int
comm_signal_bind ( struct comm_signal * comsig , int sig )
2007-02-05 11:46:40 -05:00
{
struct internal_signal * entry = ( struct internal_signal * ) calloc ( 1 ,
sizeof ( struct internal_signal ) ) ;
if ( ! entry ) {
log_err ( " malloc failed " ) ;
return 0 ;
}
log_assert ( comsig ) ;
/* add signal event */
2016-03-07 09:10:06 -05:00
entry - > ev = ub_signal_new ( comsig - > base - > eb - > base , sig ,
comm_signal_callback , comsig ) ;
if ( entry - > ev = = NULL ) {
log_err ( " Could not create signal event " ) ;
2007-02-05 11:46:40 -05:00
free ( entry ) ;
return 0 ;
}
2016-03-07 09:10:06 -05:00
if ( ub_signal_add ( entry - > ev , NULL ) ! = 0 ) {
2007-02-05 11:46:40 -05:00
log_err ( " Could not add signal handler " ) ;
2016-03-07 09:10:06 -05:00
ub_event_free ( entry - > ev ) ;
2007-02-05 11:46:40 -05:00
free ( entry ) ;
return 0 ;
}
/* link into list */
entry - > next = comsig - > ev_signal ;
comsig - > ev_signal = entry ;
return 1 ;
}
2007-02-06 09:00:52 -05:00
void
comm_signal_delete ( struct comm_signal * comsig )
2007-02-05 11:46:40 -05:00
{
struct internal_signal * p , * np ;
if ( ! comsig )
return ;
p = comsig - > ev_signal ;
while ( p ) {
np = p - > next ;
2016-03-07 09:10:06 -05:00
ub_signal_del ( p - > ev ) ;
ub_event_free ( p - > ev ) ;
2007-02-05 11:46:40 -05:00
free ( p ) ;
p = np ;
}
free ( comsig ) ;
}