2007-01-02 18:16:39 +01:00
/*
* Unix SMB / CIFS implementation .
* Join infiniband wrapper and ctdb .
*
* Copyright ( C ) Sven Oehme < oehmes @ de . ibm . com > 2006
*
* Major code contributions by Peter Somogyi < psomogyi @ gamax . hu >
*
* 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 15:29:31 +10:00
* the Free Software Foundation ; either version 3 of the License , or
2007-01-02 18:16:39 +01:00
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* 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 15:46:05 +10:00
* along with this program ; if not , see < http : //www.gnu.org/licenses/>.
2007-01-02 18:16:39 +01:00
*/
2015-10-26 16:50:46 +11:00
# include "replace.h"
# include "system/network.h"
2007-01-02 18:16:39 +01:00
# include <assert.h>
2015-10-26 16:50:46 +11:00
# include <talloc.h>
# include <tevent.h>
# include "lib/util/time.h"
# include "lib/util/debug.h"
2007-01-02 18:16:39 +01:00
# include "ctdb_private.h"
2015-10-26 16:50:46 +11:00
2015-10-23 14:17:34 +11:00
# include "common/common.h"
2015-11-11 15:22:52 +11:00
# include "common/logging.h"
2015-10-23 14:17:34 +11:00
2007-01-02 18:16:39 +01:00
# include "ibwrapper.h"
# include "ibw_ctdb.h"
2007-04-26 16:54:24 +02:00
int ctdb_ibw_get_address ( struct ctdb_context * ctdb ,
const char * address , struct in_addr * addr )
{
if ( inet_pton ( AF_INET , address , addr ) < = 0 ) {
struct hostent * he = gethostbyname ( address ) ;
if ( he = = NULL | | he - > h_length > sizeof ( * addr ) ) {
ctdb_set_error ( ctdb , " invalid nework address '%s' \n " ,
address ) ;
return - 1 ;
}
memcpy ( addr , he - > h_addr , he - > h_length ) ;
}
return 0 ;
}
2007-02-26 11:59:20 +01:00
int ctdb_ibw_node_connect ( struct ctdb_node * node )
2007-02-15 17:02:38 +01:00
{
2007-04-13 20:38:24 +10:00
struct ctdb_ibw_node * cn = talloc_get_type ( node - > private_data , struct ctdb_ibw_node ) ;
2007-02-26 11:59:20 +01:00
int rc ;
assert ( cn ! = NULL ) ;
assert ( cn - > conn ! = NULL ) ;
2007-02-15 17:02:38 +01:00
2015-10-13 16:19:21 +11:00
rc = ibw_connect ( cn - > conn , & node - > address . ip , node ) ;
2007-02-26 11:59:20 +01:00
if ( rc ) {
2008-02-04 20:07:15 +11:00
DEBUG ( DEBUG_ERR , ( " ctdb_ibw_node_connect/ibw_connect failed - retrying... \n " ) ) ;
2007-02-15 17:02:38 +01:00
/* try again once a second */
2015-10-26 16:50:09 +11:00
tevent_add_timer ( node - > ctdb - > ev , node ,
timeval_current_ofs ( 1 , 0 ) ,
ctdb_ibw_node_connect_event , node ) ;
2007-02-15 17:02:38 +01:00
}
/* continues at ibw_ctdb.c/IBWC_CONNECTED in good case */
return 0 ;
}
2015-10-26 16:50:09 +11:00
void ctdb_ibw_node_connect_event ( struct tevent_context * ev ,
struct tevent_timer * te ,
struct timeval t , void * private_data )
2007-02-15 17:02:38 +01:00
{
2007-04-13 20:38:24 +10:00
struct ctdb_node * node = talloc_get_type ( private_data , struct ctdb_node ) ;
2007-02-15 17:02:38 +01:00
2007-02-26 11:59:20 +01:00
ctdb_ibw_node_connect ( node ) ;
2007-02-15 17:02:38 +01:00
}
2007-01-02 18:16:39 +01:00
int ctdb_ibw_connstate_handler ( struct ibw_ctx * ctx , struct ibw_conn * conn )
{
if ( ctx ! = NULL ) {
/* ctx->state changed */
switch ( ctx - > state ) {
case IBWS_INIT : /* ctx start - after ibw_init */
break ;
case IBWS_READY : /* after ibw_bind & ibw_listen */
break ;
case IBWS_CONNECT_REQUEST : /* after [IBWS_READY + incoming request] */
/* => [(ibw_accept)IBWS_READY | (ibw_disconnect)STOPPED | ERROR] */
if ( ibw_accept ( ctx , conn , NULL ) ) {
2008-02-04 20:07:15 +11:00
DEBUG ( DEBUG_ERR , ( " connstate_handler/ibw_accept failed \n " ) ) ;
2007-01-02 18:16:39 +01:00
return - 1 ;
} /* else continue in IBWC_CONNECTED */
break ;
case IBWS_STOPPED : /* normal stop <= ibw_disconnect+(IBWS_READY | IBWS_CONNECT_REQUEST) */
/* TODO: have a CTDB upcall for which CTDB should wait in a (final) loop */
break ;
case IBWS_ERROR : /* abnormal state; ibw_stop must be called after this */
break ;
default :
assert ( 0 ) ;
break ;
}
}
if ( conn ! = NULL ) {
/* conn->state changed */
switch ( conn - > state ) {
case IBWC_INIT : /* conn start - internal state */
break ;
case IBWC_CONNECTED : { /* after ibw_accept or ibw_connect */
struct ctdb_node * node = talloc_get_type ( conn - > conn_userdata , struct ctdb_node ) ;
2007-02-14 18:58:20 +01:00
if ( node ! = NULL ) { /* after ibw_connect */
2007-04-13 20:38:24 +10:00
struct ctdb_ibw_node * cn = talloc_get_type ( node - > private_data , struct ctdb_ibw_node ) ;
2007-02-26 11:59:20 +01:00
2007-01-02 18:16:39 +01:00
node - > ctdb - > upcalls - > node_connected ( node ) ;
2007-02-26 11:59:20 +01:00
ctdb_flush_cn_queue ( cn ) ;
2007-02-14 18:58:20 +01:00
} else { /* after ibw_accept */
2007-01-02 18:16:39 +01:00
/* NOP in CTDB case */
}
} break ;
2007-02-14 18:58:20 +01:00
case IBWC_DISCONNECTED : { /* after ibw_disconnect */
2007-01-02 18:16:39 +01:00
struct ctdb_node * node = talloc_get_type ( conn - > conn_userdata , struct ctdb_node ) ;
if ( node ! = NULL )
2007-02-14 18:58:20 +01:00
node - > ctdb - > upcalls - > node_dead ( node ) ;
talloc_free ( conn ) ;
2007-02-15 17:02:38 +01:00
/* normal + intended disconnect => not reconnecting in this layer */
2007-02-14 18:58:20 +01:00
} break ;
case IBWC_ERROR : {
2007-02-15 17:02:38 +01:00
struct ctdb_node * node = talloc_get_type ( conn - > conn_userdata , struct ctdb_node ) ;
2007-02-26 11:59:20 +01:00
if ( node ! = NULL ) {
2007-04-13 20:38:24 +10:00
struct ctdb_ibw_node * cn = talloc_get_type ( node - > private_data , struct ctdb_ibw_node ) ;
2007-02-26 11:59:20 +01:00
struct ibw_ctx * ictx = cn - > conn - > ctx ;
2008-02-04 17:44:24 +11:00
DEBUG ( DEBUG_DEBUG , ( " IBWC_ERROR, reconnecting... \n " ) ) ;
2007-02-26 11:59:20 +01:00
talloc_free ( cn - > conn ) ; /* internal queue content is destroyed */
cn - > conn = ( void * ) ibw_conn_new ( ictx , node ) ;
2015-10-26 16:50:09 +11:00
tevent_add_timer ( node - > ctdb - > ev , node ,
timeval_current_ofs ( 1 , 0 ) ,
ctdb_ibw_node_connect_event , node ) ;
2007-02-26 11:59:20 +01:00
}
2007-01-02 18:16:39 +01:00
} break ;
default :
assert ( 0 ) ;
break ;
}
}
return 0 ;
}
int ctdb_ibw_receive_handler ( struct ibw_conn * conn , void * buf , int n )
{
struct ctdb_context * ctdb = talloc_get_type ( conn - > ctx - > ctx_userdata , struct ctdb_context ) ;
2007-02-15 17:02:38 +01:00
void * buf2 ; /* future TODO: a solution for removal of this */
2007-01-02 18:16:39 +01:00
assert ( ctdb ! = NULL ) ;
2007-02-15 17:02:38 +01:00
assert ( buf ! = NULL ) ;
assert ( conn ! = NULL ) ;
2007-01-02 18:16:39 +01:00
assert ( conn - > state = = IBWC_CONNECTED ) ;
2007-02-15 17:02:38 +01:00
/* so far "buf" is an ib-registered memory area
* and being reused for next receive
* noticed that HL requires talloc - ed memory to be stolen */
buf2 = talloc_zero_size ( conn , n ) ;
2009-05-21 11:49:16 +10:00
CTDB_NO_MEMORY ( ctdb , buf2 ) ;
2007-02-15 17:02:38 +01:00
memcpy ( buf2 , buf , n ) ;
2007-01-02 18:16:39 +01:00
2007-02-15 17:02:38 +01:00
ctdb - > upcalls - > recv_pkt ( ctdb , ( uint8_t * ) buf2 , ( uint32_t ) n ) ;
2007-01-02 18:16:39 +01:00
return 0 ;
}