2006-11-18 02:41:20 +03:00
/*
ctdb over TCP
Copyright ( C ) Andrew Tridgell 2006
2008-12-05 02:33:38 +03:00
Copyright ( C ) Ronnie Sahlberg 2008
2006-11-18 02:41:20 +03:00
2007-05-31 07:50:53 +04:00
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
2007-07-10 09:29:31 +04:00
the Free Software Foundation ; either version 3 of the License , or
2007-05-31 07:50:53 +04:00
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
2006-11-18 02:41:20 +03:00
but WITHOUT ANY WARRANTY ; without even the implied warranty of
2007-05-31 07:50:53 +04:00
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
2007-07-10 09:29:31 +04:00
along with this program ; if not , see < http : //www.gnu.org/licenses/>.
2006-11-18 02:41:20 +03:00
*/
2015-10-26 08:50:46 +03:00
# include "replace.h"
2006-11-18 02:41:20 +03:00
# include "system/network.h"
# include "system/filesys.h"
2015-10-26 08:50:46 +03:00
# include <talloc.h>
# include <tevent.h>
# include "lib/util/debug.h"
# include "lib/util/time.h"
2016-05-27 06:50:06 +03:00
# include "lib/util/blocking.h"
2015-10-26 08:50:46 +03:00
# include "ctdb_private.h"
2015-10-23 06:11:53 +03:00
# include "common/system.h"
2015-10-23 06:17:34 +03:00
# include "common/common.h"
2015-11-11 07:22:52 +03:00
# include "common/logging.h"
2015-10-26 08:50:46 +03:00
2006-11-27 13:38:13 +03:00
# include "ctdb_tcp.h"
2006-11-18 02:41:20 +03:00
2007-10-22 10:41:11 +04:00
/*
stop any connecting ( established or pending ) to a node
*/
void ctdb_tcp_stop_connection ( struct ctdb_node * node )
{
struct ctdb_tcp_node * tnode = talloc_get_type (
node - > private_data , struct ctdb_tcp_node ) ;
ctdb_queue_set_fd ( tnode - > out_queue , - 1 ) ;
talloc_free ( tnode - > connect_te ) ;
talloc_free ( tnode - > connect_fde ) ;
tnode - > connect_fde = NULL ;
tnode - > connect_te = NULL ;
if ( tnode - > fd ! = - 1 ) {
close ( tnode - > fd ) ;
tnode - > fd = - 1 ;
}
}
2007-04-10 13:33:21 +04:00
/*
called when a complete packet has come in - should not happen on this socket
2008-12-02 05:26:30 +03:00
unless the other side closes the connection with RST or FIN
2007-04-10 13:33:21 +04:00
*/
2007-04-13 14:38:24 +04:00
void ctdb_tcp_tnode_cb ( uint8_t * data , size_t cnt , void * private_data )
2007-04-10 13:33:21 +04:00
{
2007-04-13 14:38:24 +04:00
struct ctdb_node * node = talloc_get_type ( private_data , struct ctdb_node ) ;
2007-04-11 22:12:15 +04:00
struct ctdb_tcp_node * tnode = talloc_get_type (
node - > private_data , struct ctdb_tcp_node ) ;
2007-04-10 13:33:21 +04:00
2007-04-17 13:41:29 +04:00
if ( data = = NULL ) {
node - > ctdb - > upcalls - > node_dead ( node ) ;
}
2007-10-22 10:41:11 +04:00
ctdb_tcp_stop_connection ( node ) ;
2015-10-26 08:50:09 +03:00
tnode - > connect_te = tevent_add_timer ( node - > ctdb - > ev , tnode ,
timeval_current_ofs ( 3 , 0 ) ,
ctdb_tcp_node_connect , node ) ;
2007-04-10 13:33:21 +04:00
}
2006-11-18 02:41:20 +03:00
/*
called when socket becomes writeable on connect
*/
2015-10-26 08:50:09 +03:00
static void ctdb_node_connect_write ( struct tevent_context * ev ,
struct tevent_fd * fde ,
2007-04-11 22:12:15 +04:00
uint16_t flags , void * private_data )
2006-11-18 02:41:20 +03:00
{
2007-04-11 22:12:15 +04:00
struct ctdb_node * node = talloc_get_type ( private_data ,
struct ctdb_node ) ;
struct ctdb_tcp_node * tnode = talloc_get_type ( node - > private_data ,
2006-11-27 13:38:13 +03:00
struct ctdb_tcp_node ) ;
2006-11-18 02:41:20 +03:00
struct ctdb_context * ctdb = node - > ctdb ;
2006-11-28 09:56:10 +03:00
int error = 0 ;
2006-12-01 07:45:24 +03:00
socklen_t len = sizeof ( error ) ;
2007-02-20 06:57:13 +03:00
int one = 1 ;
2006-11-18 02:41:20 +03:00
2007-05-25 16:07:45 +04:00
talloc_free ( tnode - > connect_te ) ;
tnode - > connect_te = NULL ;
2006-11-27 13:38:13 +03:00
if ( getsockopt ( tnode - > fd , SOL_SOCKET , SO_ERROR , & error , & len ) ! = 0 | |
2006-11-18 07:27:36 +03:00
error ! = 0 ) {
2007-10-22 10:41:11 +04:00
ctdb_tcp_stop_connection ( node ) ;
2015-10-26 08:50:09 +03:00
tnode - > connect_te = tevent_add_timer ( ctdb - > ev , tnode ,
2007-10-22 10:41:11 +04:00
timeval_current_ofs ( 1 , 0 ) ,
ctdb_tcp_node_connect , node ) ;
2006-11-18 02:41:20 +03:00
return ;
}
2007-10-22 08:07:35 +04:00
talloc_free ( tnode - > connect_fde ) ;
tnode - > connect_fde = NULL ;
2007-10-22 10:41:11 +04:00
2013-11-11 05:41:00 +04:00
if ( setsockopt ( tnode - > fd , IPPROTO_TCP , TCP_NODELAY , ( char * ) & one , sizeof ( one ) ) = = - 1 ) {
DEBUG ( DEBUG_WARNING , ( " Failed to set TCP_NODELAY on fd - %s \n " ,
strerror ( errno ) ) ) ;
}
if ( setsockopt ( tnode - > fd , SOL_SOCKET , SO_KEEPALIVE , ( char * ) & one , sizeof ( one ) ) = = - 1 ) {
DEBUG ( DEBUG_WARNING , ( " Failed to set KEEPALIVE on fd - %s \n " ,
strerror ( errno ) ) ) ;
}
2007-04-10 13:33:21 +04:00
2007-07-02 08:26:50 +04:00
ctdb_queue_set_fd ( tnode - > out_queue , tnode - > fd ) ;
2006-11-28 09:56:10 +03:00
2007-10-22 10:41:11 +04:00
/* the queue subsystem now owns this fd */
tnode - > fd = - 1 ;
2006-11-18 02:41:20 +03:00
}
2007-02-20 05:22:18 +03:00
2006-11-18 02:41:20 +03:00
/*
called when we should try and establish a tcp connection to a node
*/
2015-10-26 08:50:09 +03:00
void ctdb_tcp_node_connect ( struct tevent_context * ev , struct tevent_timer * te ,
2007-04-11 22:12:15 +04:00
struct timeval t , void * private_data )
2006-11-18 02:41:20 +03:00
{
2007-04-11 22:12:15 +04:00
struct ctdb_node * node = talloc_get_type ( private_data ,
struct ctdb_node ) ;
struct ctdb_tcp_node * tnode = talloc_get_type ( node - > private_data ,
2006-11-27 13:38:13 +03:00
struct ctdb_tcp_node ) ;
2006-11-18 02:41:20 +03:00
struct ctdb_context * ctdb = node - > ctdb ;
2008-08-19 08:58:29 +04:00
ctdb_sock_addr sock_in ;
2008-12-05 02:33:38 +03:00
int sockin_size ;
2009-03-16 01:21:24 +03:00
int sockout_size ;
2008-08-19 08:58:29 +04:00
ctdb_sock_addr sock_out ;
2016-08-10 10:12:21 +03:00
int ret ;
2006-11-18 02:41:20 +03:00
2007-10-22 10:41:11 +04:00
ctdb_tcp_stop_connection ( node ) ;
2007-05-25 16:07:45 +04:00
2015-02-20 03:47:23 +03:00
sock_out = node - > address ;
2007-04-06 03:08:41 +04:00
2008-08-19 08:58:29 +04:00
tnode - > fd = socket ( sock_out . sa . sa_family , SOCK_STREAM , IPPROTO_TCP ) ;
2011-08-10 19:53:56 +04:00
if ( tnode - > fd = = - 1 ) {
2016-05-26 14:40:38 +03:00
DEBUG ( DEBUG_ERR , ( __location__ " Failed to create socket \n " ) ) ;
2011-08-10 19:53:56 +04:00
return ;
}
2016-08-10 10:12:21 +03:00
ret = set_blocking ( tnode - > fd , false ) ;
if ( ret ! = 0 ) {
DEBUG ( DEBUG_ERR ,
( __location__
" failed to set socket non-blocking (%s) \n " ,
strerror ( errno ) ) ) ;
close ( tnode - > fd ) ;
tnode - > fd = - 1 ;
return ;
}
2008-08-19 08:58:29 +04:00
set_close_on_exec ( tnode - > fd ) ;
2007-04-06 03:08:41 +04:00
2009-10-21 08:26:24 +04:00
DEBUG ( DEBUG_DEBUG , ( __location__ " Created TCP SOCKET FD:%d \n " , tnode - > fd ) ) ;
2009-10-15 04:24:54 +04:00
2007-04-06 03:08:41 +04:00
/* Bind our side of the socketpair to the same address we use to listen
* on incoming CTDB traffic .
* We must specify this address to make sure that the address we expose to
* the remote side is actually routable in case CTDB traffic will run on
* a dedicated non - routeable network .
*/
2015-02-20 03:47:23 +03:00
sock_in = * ctdb - > address ;
2009-03-16 01:21:24 +03:00
/* AIX libs check to see if the socket address and length
arguments are consistent with each other on calls like
connect ( ) . Can not get by with just sizeof ( sock_in ) ,
need sizeof ( sock_in . ip ) .
*/
2008-12-05 02:33:38 +03:00
switch ( sock_in . sa . sa_family ) {
case AF_INET :
2015-02-20 03:47:23 +03:00
sock_in . ip . sin_port = 0 /* Any port */ ;
2008-12-05 02:33:38 +03:00
sockin_size = sizeof ( sock_in . ip ) ;
2009-03-16 01:21:24 +03:00
sockout_size = sizeof ( sock_out . ip ) ;
2008-12-05 02:33:38 +03:00
break ;
case AF_INET6 :
2015-02-20 03:47:23 +03:00
sock_in . ip6 . sin6_port = 0 /* Any port */ ;
2008-12-05 02:33:38 +03:00
sockin_size = sizeof ( sock_in . ip6 ) ;
2009-03-16 01:21:24 +03:00
sockout_size = sizeof ( sock_out . ip6 ) ;
2008-12-05 02:33:38 +03:00
break ;
default :
DEBUG ( DEBUG_ERR , ( __location__ " unknown family %u \n " ,
sock_in . sa . sa_family ) ) ;
2009-10-15 04:24:54 +04:00
close ( tnode - > fd ) ;
2016-08-15 02:43:46 +03:00
tnode - > fd = - 1 ;
2008-12-05 02:33:38 +03:00
return ;
}
2015-02-20 03:47:23 +03:00
2011-08-10 19:53:56 +04:00
if ( bind ( tnode - > fd , ( struct sockaddr * ) & sock_in , sockin_size ) = = - 1 ) {
2016-05-26 14:40:38 +03:00
DEBUG ( DEBUG_ERR , ( __location__ " Failed to bind socket %s(%d) \n " ,
2011-08-10 19:53:56 +04:00
strerror ( errno ) , errno ) ) ;
close ( tnode - > fd ) ;
2016-08-15 02:43:46 +03:00
tnode - > fd = - 1 ;
2011-08-10 19:53:56 +04:00
return ;
}
2007-04-06 03:08:41 +04:00
2009-03-16 01:21:24 +03:00
if ( connect ( tnode - > fd , ( struct sockaddr * ) & sock_out , sockout_size ) ! = 0 & &
2006-11-18 02:41:20 +03:00
errno ! = EINPROGRESS ) {
2007-10-22 10:41:11 +04:00
ctdb_tcp_stop_connection ( node ) ;
2015-10-26 08:50:09 +03:00
tnode - > connect_te = tevent_add_timer ( ctdb - > ev , tnode ,
timeval_current_ofs ( 1 , 0 ) ,
ctdb_tcp_node_connect , node ) ;
2006-11-18 02:41:20 +03:00
return ;
}
/* non-blocking connect - wait for write event */
2015-10-26 08:50:09 +03:00
tnode - > connect_fde = tevent_add_fd ( node - > ctdb - > ev , tnode , tnode - > fd ,
TEVENT_FD_WRITE | TEVENT_FD_READ ,
ctdb_node_connect_write , node ) ;
2007-05-25 16:07:45 +04:00
/* don't give it long to connect - retry in one second. This ensures
that we find a node is up quickly ( tcp normally backs off a syn reply
delay by quite a lot ) */
2015-10-26 08:50:09 +03:00
tnode - > connect_te = tevent_add_timer ( ctdb - > ev , tnode ,
timeval_current_ofs ( 1 , 0 ) ,
ctdb_tcp_node_connect , node ) ;
2006-11-18 02:41:20 +03:00
}
2006-11-18 05:45:04 +03:00
/*
called when we get contacted by another node
currently makes no attempt to check if the connection is really from a ctdb
node in our cluster
*/
2015-10-26 08:50:09 +03:00
static void ctdb_listen_event ( struct tevent_context * ev , struct tevent_fd * fde ,
2007-04-13 14:38:24 +04:00
uint16_t flags , void * private_data )
2006-11-18 05:45:04 +03:00
{
2007-06-02 02:41:19 +04:00
struct ctdb_context * ctdb = talloc_get_type ( private_data , struct ctdb_context ) ;
struct ctdb_tcp * ctcp = talloc_get_type ( ctdb - > private_data , struct ctdb_tcp ) ;
2008-08-19 08:58:29 +04:00
ctdb_sock_addr addr ;
2006-11-18 05:45:04 +03:00
socklen_t len ;
2007-07-02 08:10:20 +04:00
int fd , nodeid ;
2006-11-18 05:45:04 +03:00
struct ctdb_incoming * in ;
2007-05-15 12:40:56 +04:00
int one = 1 ;
2016-08-10 10:31:52 +03:00
int ret ;
2006-11-18 05:45:04 +03:00
2006-11-18 07:27:36 +03:00
memset ( & addr , 0 , sizeof ( addr ) ) ;
len = sizeof ( addr ) ;
2006-11-27 13:38:13 +03:00
fd = accept ( ctcp - > listen_fd , ( struct sockaddr * ) & addr , & len ) ;
2006-11-18 05:45:04 +03:00
if ( fd = = - 1 ) return ;
2015-02-20 03:47:23 +03:00
nodeid = ctdb_ip_to_nodeid ( ctdb , & addr ) ;
2007-11-26 02:52:55 +03:00
if ( nodeid = = - 1 ) {
2015-02-20 03:47:23 +03:00
DEBUG ( DEBUG_ERR , ( " Refused connection from unknown node %s \n " , ctdb_addr_to_str ( & addr ) ) ) ;
2007-07-02 08:10:20 +04:00
close ( fd ) ;
return ;
}
2007-06-02 02:41:19 +04:00
in = talloc_zero ( ctcp , struct ctdb_incoming ) ;
2006-11-18 05:45:04 +03:00
in - > fd = fd ;
in - > ctdb = ctdb ;
2016-08-10 10:31:52 +03:00
ret = set_blocking ( in - > fd , false ) ;
if ( ret ! = 0 ) {
DEBUG ( DEBUG_ERR ,
( __location__
" failed to set socket non-blocking (%s) \n " ,
strerror ( errno ) ) ) ;
close ( in - > fd ) ;
in - > fd = - 1 ;
return ;
}
2007-05-30 09:43:25 +04:00
set_close_on_exec ( in - > fd ) ;
2006-11-28 06:15:46 +03:00
2010-02-03 22:37:41 +03:00
DEBUG ( DEBUG_DEBUG , ( __location__ " Created SOCKET FD:%d to incoming ctdb connection \n " , fd ) ) ;
2009-10-15 04:24:54 +04:00
2013-11-11 05:41:00 +04:00
if ( setsockopt ( in - > fd , SOL_SOCKET , SO_KEEPALIVE , ( char * ) & one , sizeof ( one ) ) = = - 1 ) {
DEBUG ( DEBUG_WARNING , ( " Failed to set KEEPALIVE on fd - %s \n " ,
strerror ( errno ) ) ) ;
}
2007-05-15 12:40:56 +04:00
2015-02-20 03:47:23 +03:00
in - > queue = ctdb_queue_setup ( ctdb , in , in - > fd , CTDB_TCP_ALIGNMENT ,
ctdb_tcp_read_cb , in , " ctdbd-%s " , ctdb_addr_to_str ( & addr ) ) ;
2006-11-18 05:45:04 +03:00
}
/*
2007-05-01 00:34:55 +04:00
automatically find which address to listen on
2006-11-18 05:45:04 +03:00
*/
2007-05-01 00:34:55 +04:00
static int ctdb_tcp_listen_automatic ( struct ctdb_context * ctdb )
2006-11-18 05:45:04 +03:00
{
2007-04-11 22:12:15 +04:00
struct ctdb_tcp * ctcp = talloc_get_type ( ctdb - > private_data ,
struct ctdb_tcp ) ;
2008-08-19 08:58:29 +04:00
ctdb_sock_addr sock ;
2007-05-01 00:34:55 +04:00
int lock_fd , i ;
2014-06-23 09:20:44 +04:00
const char * lock_path = CTDB_RUNDIR " /.socket_lock " ;
2007-05-01 00:34:55 +04:00
struct flock lock ;
2008-08-19 08:58:29 +04:00
int one = 1 ;
2008-12-05 02:33:38 +03:00
int sock_size ;
2010-08-18 03:46:31 +04:00
struct tevent_fd * fde ;
2007-05-01 00:34:55 +04:00
2012-07-04 01:21:01 +04:00
/* If there are no nodes, then it won't be possible to find
* the first one . Log a failure and short circuit the whole
* process .
*/
if ( ctdb - > num_nodes = = 0 ) {
DEBUG ( DEBUG_CRIT , ( " No nodes available to attempt bind to - is the nodes file empty? \n " ) ) ;
return - 1 ;
}
2012-12-03 08:44:12 +04:00
/* in order to ensure that we don't get two nodes with the
same adddress , we must make the bind ( ) and listen ( ) calls
atomic . The SO_REUSEADDR setsockopt only prevents double
binds if the first socket is in LISTEN state */
lock_fd = open ( lock_path , O_RDWR | O_CREAT , 0666 ) ;
if ( lock_fd = = - 1 ) {
DEBUG ( DEBUG_CRIT , ( " Unable to open %s \n " , lock_path ) ) ;
return - 1 ;
}
2007-05-01 00:34:55 +04:00
2012-12-03 08:44:12 +04:00
lock . l_type = F_WRLCK ;
lock . l_whence = SEEK_SET ;
lock . l_start = 0 ;
lock . l_len = 1 ;
lock . l_pid = 0 ;
2006-11-18 05:45:04 +03:00
2012-12-03 08:44:12 +04:00
if ( fcntl ( lock_fd , F_SETLKW , & lock ) ! = 0 ) {
DEBUG ( DEBUG_CRIT , ( " Unable to lock %s \n " , lock_path ) ) ;
close ( lock_fd ) ;
return - 1 ;
2007-02-20 05:22:18 +03:00
}
2006-11-18 05:45:04 +03:00
2009-06-01 08:18:34 +04:00
for ( i = 0 ; i < ctdb - > num_nodes ; i + + ) {
if ( ctdb - > nodes [ i ] - > flags & NODE_FLAGS_DELETED ) {
continue ;
}
2015-02-20 03:47:23 +03:00
sock = ctdb - > nodes [ i ] - > address ;
2008-08-19 08:58:29 +04:00
switch ( sock . sa . sa_family ) {
case AF_INET :
2008-12-05 02:33:38 +03:00
sock_size = sizeof ( sock . ip ) ;
2008-08-19 08:58:29 +04:00
break ;
case AF_INET6 :
2008-12-05 02:33:38 +03:00
sock_size = sizeof ( sock . ip6 ) ;
2008-08-19 08:58:29 +04:00
break ;
default :
DEBUG ( DEBUG_ERR , ( __location__ " unknown family %u \n " ,
sock . sa . sa_family ) ) ;
continue ;
}
ctcp - > listen_fd = socket ( sock . sa . sa_family , SOCK_STREAM , IPPROTO_TCP ) ;
if ( ctcp - > listen_fd = = - 1 ) {
ctdb_set_error ( ctdb , " socket failed \n " ) ;
continue ;
}
set_close_on_exec ( ctcp - > listen_fd ) ;
2013-11-11 05:41:00 +04:00
if ( setsockopt ( ctcp - > listen_fd , SOL_SOCKET , SO_REUSEADDR ,
( char * ) & one , sizeof ( one ) ) = = - 1 ) {
DEBUG ( DEBUG_WARNING , ( " Failed to set REUSEADDR on fd - %s \n " ,
strerror ( errno ) ) ) ;
}
2008-08-19 08:58:29 +04:00
2008-12-05 02:33:38 +03:00
if ( bind ( ctcp - > listen_fd , ( struct sockaddr * ) & sock , sock_size ) = = 0 ) {
2007-05-01 00:34:55 +04:00
break ;
}
2009-10-21 19:06:48 +04:00
if ( errno = = EADDRNOTAVAIL ) {
DEBUG ( DEBUG_DEBUG , ( __location__ " Failed to bind() to socket. %s(%d) \n " ,
strerror ( errno ) , errno ) ) ;
} else {
DEBUG ( DEBUG_ERR , ( __location__ " Failed to bind() to socket. %s(%d) \n " ,
strerror ( errno ) , errno ) ) ;
}
2015-09-08 11:01:31 +03:00
close ( ctcp - > listen_fd ) ;
2015-09-15 11:19:03 +03:00
ctcp - > listen_fd = - 1 ;
2007-05-01 00:34:55 +04:00
}
2015-02-17 07:21:15 +03:00
2007-05-01 00:34:55 +04:00
if ( i = = ctdb - > num_nodes ) {
2008-02-04 12:07:15 +03:00
DEBUG ( DEBUG_CRIT , ( " Unable to bind to any of the node addresses - giving up \n " ) ) ;
2007-05-01 00:34:55 +04:00
goto failed ;
}
2015-02-20 03:47:23 +03:00
ctdb - > address = talloc_memdup ( ctdb ,
& ctdb - > nodes [ i ] - > address ,
sizeof ( ctdb_sock_addr ) ) ;
2015-03-31 16:13:36 +03:00
if ( ctdb - > address = = NULL ) {
ctdb_set_error ( ctdb , " Out of memory at %s:%d " ,
__FILE__ , __LINE__ ) ;
goto failed ;
}
2015-02-17 07:21:15 +03:00
ctdb - > name = talloc_asprintf ( ctdb , " %s:%u " ,
2015-02-20 03:47:23 +03:00
ctdb_addr_to_str ( ctdb - > address ) ,
ctdb_addr_to_port ( ctdb - > address ) ) ;
2015-03-31 19:06:43 +03:00
if ( ctdb - > name = = NULL ) {
ctdb_set_error ( ctdb , " Out of memory at %s:%d " ,
__FILE__ , __LINE__ ) ;
goto failed ;
}
2015-02-20 03:47:23 +03:00
DEBUG ( DEBUG_INFO , ( " ctdb chose network address %s \n " , ctdb - > name ) ) ;
2015-02-17 07:21:15 +03:00
2007-05-01 00:34:55 +04:00
if ( listen ( ctcp - > listen_fd , 10 ) = = - 1 ) {
goto failed ;
}
2015-10-26 08:50:09 +03:00
fde = tevent_add_fd ( ctdb - > ev , ctcp , ctcp - > listen_fd , TEVENT_FD_READ ,
ctdb_listen_event , ctdb ) ;
2010-08-18 03:46:31 +04:00
tevent_fd_set_auto_close ( fde ) ;
2007-05-01 00:34:55 +04:00
2012-12-03 08:44:12 +04:00
close ( lock_fd ) ;
2007-05-01 00:34:55 +04:00
return 0 ;
2015-02-20 03:47:23 +03:00
2007-05-01 00:34:55 +04:00
failed :
2012-12-03 08:44:12 +04:00
close ( lock_fd ) ;
2013-11-11 05:41:00 +04:00
if ( ctcp - > listen_fd ! = - 1 ) {
close ( ctcp - > listen_fd ) ;
ctcp - > listen_fd = - 1 ;
}
2007-05-01 00:34:55 +04:00
return - 1 ;
}
/*
listen on our own address
*/
int ctdb_tcp_listen ( struct ctdb_context * ctdb )
{
struct ctdb_tcp * ctcp = talloc_get_type ( ctdb - > private_data ,
struct ctdb_tcp ) ;
2008-08-19 08:58:29 +04:00
ctdb_sock_addr sock ;
2008-12-05 02:33:38 +03:00
int sock_size ;
2007-05-01 00:34:55 +04:00
int one = 1 ;
2010-08-18 03:46:31 +04:00
struct tevent_fd * fde ;
2007-05-01 00:34:55 +04:00
/* we can either auto-bind to the first available address, or we can
use a specified address */
2015-02-20 03:47:23 +03:00
if ( ! ctdb - > address ) {
2007-05-01 00:34:55 +04:00
return ctdb_tcp_listen_automatic ( ctdb ) ;
}
2015-02-20 03:47:23 +03:00
sock = * ctdb - > address ;
2008-08-19 08:58:29 +04:00
switch ( sock . sa . sa_family ) {
case AF_INET :
2008-12-05 02:33:38 +03:00
sock_size = sizeof ( sock . ip ) ;
2008-08-19 08:58:29 +04:00
break ;
case AF_INET6 :
2008-12-05 02:33:38 +03:00
sock_size = sizeof ( sock . ip6 ) ;
2008-08-19 08:58:29 +04:00
break ;
default :
DEBUG ( DEBUG_ERR , ( __location__ " unknown family %u \n " ,
sock . sa . sa_family ) ) ;
goto failed ;
}
ctcp - > listen_fd = socket ( sock . sa . sa_family , SOCK_STREAM , IPPROTO_TCP ) ;
if ( ctcp - > listen_fd = = - 1 ) {
ctdb_set_error ( ctdb , " socket failed \n " ) ;
return - 1 ;
}
set_close_on_exec ( ctcp - > listen_fd ) ;
2013-11-11 05:41:00 +04:00
if ( setsockopt ( ctcp - > listen_fd , SOL_SOCKET , SO_REUSEADDR , ( char * ) & one , sizeof ( one ) ) = = - 1 ) {
DEBUG ( DEBUG_WARNING , ( " Failed to set REUSEADDR on fd - %s \n " ,
strerror ( errno ) ) ) ;
}
2008-08-19 08:58:29 +04:00
2008-12-05 02:33:38 +03:00
if ( bind ( ctcp - > listen_fd , ( struct sockaddr * ) & sock , sock_size ) ! = 0 ) {
DEBUG ( DEBUG_ERR , ( __location__ " Failed to bind() to socket. %s(%d) \n " , strerror ( errno ) , errno ) ) ;
2007-05-01 00:34:55 +04:00
goto failed ;
}
2006-11-18 05:45:04 +03:00
2006-11-27 13:38:13 +03:00
if ( listen ( ctcp - > listen_fd , 10 ) = = - 1 ) {
2007-05-01 00:34:55 +04:00
goto failed ;
2006-11-18 05:45:04 +03:00
}
2015-10-26 08:50:09 +03:00
fde = tevent_add_fd ( ctdb - > ev , ctcp , ctcp - > listen_fd , TEVENT_FD_READ ,
ctdb_listen_event , ctdb ) ;
2010-08-18 03:46:31 +04:00
tevent_fd_set_auto_close ( fde ) ;
2006-11-18 05:45:04 +03:00
return 0 ;
2007-05-01 00:34:55 +04:00
failed :
2008-08-19 08:58:29 +04:00
if ( ctcp - > listen_fd ! = - 1 ) {
close ( ctcp - > listen_fd ) ;
}
2007-05-01 00:34:55 +04:00
ctcp - > listen_fd = - 1 ;
return - 1 ;
2006-11-18 05:45:04 +03:00
}