2006-11-27 21:38:13 +11:00
/*
2006-11-28 17:56:10 +11:00
ctdb main protocol code
2006-11-27 21:38:13 +11:00
Copyright ( C ) Andrew Tridgell 2006
2007-05-31 13:50:53 +10: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 15:29:31 +10:00
the Free Software Foundation ; either version 3 of the License , or
2007-05-31 13:50:53 +10:00
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
2006-11-27 21:38:13 +11:00
but WITHOUT ANY WARRANTY ; without even the implied warranty of
2007-05-31 13:50:53 +10: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 15:29:31 +10:00
along with this program ; if not , see < http : //www.gnu.org/licenses/>.
2006-11-27 21:38:13 +11:00
*/
# include "includes.h"
2007-01-23 11:38:45 +11:00
# include "lib/tdb/include/tdb.h"
2010-08-18 09:16:31 +09:30
# include "lib/tevent/tevent.h"
2007-01-23 11:38:45 +11:00
# include "lib/util/dlinklist.h"
2006-11-27 21:38:13 +11:00
# include "system/network.h"
# include "system/filesys.h"
2007-01-23 11:38:45 +11:00
# include "../include/ctdb_private.h"
2006-11-27 21:38:13 +11:00
/*
choose the transport we will use
*/
int ctdb_set_transport ( struct ctdb_context * ctdb , const char * transport )
{
2007-04-20 14:26:19 +02:00
ctdb - > transport = talloc_strdup ( ctdb , transport ) ;
2009-05-21 11:49:16 +10:00
CTDB_NO_MEMORY ( ctdb , ctdb - > transport ) ;
2007-04-20 14:26:19 +02:00
return 0 ;
2006-11-27 21:38:13 +11:00
}
2007-11-26 10:52:55 +11:00
/*
Check whether an ip is a valid node ip
Returns the node id for this ip address or - 1
*/
int ctdb_ip_to_nodeid ( struct ctdb_context * ctdb , const char * nodeip )
{
int nodeid ;
for ( nodeid = 0 ; nodeid < ctdb - > num_nodes ; nodeid + + ) {
2009-06-01 14:18:34 +10:00
if ( ctdb - > nodes [ nodeid ] - > flags & NODE_FLAGS_DELETED ) {
continue ;
}
2007-11-26 10:52:55 +11:00
if ( ! strcmp ( ctdb - > nodes [ nodeid ] - > address . address , nodeip ) ) {
return nodeid ;
}
}
return - 1 ;
}
2007-06-02 11:36:42 +10:00
/*
choose the recovery lock file
*/
int ctdb_set_recovery_lock_file ( struct ctdb_context * ctdb , const char * file )
{
2009-06-25 11:50:45 +10:00
if ( ctdb - > recovery_lock_file ! = NULL ) {
talloc_free ( ctdb - > recovery_lock_file ) ;
ctdb - > recovery_lock_file = NULL ;
}
if ( file = = NULL ) {
DEBUG ( DEBUG_ALERT , ( " Recovery lock file set to \" \" . Disabling recovery lock checking \n " ) ) ;
ctdb - > tunable . verify_recovery_lock = 0 ;
return 0 ;
}
2007-06-02 11:36:42 +10:00
ctdb - > recovery_lock_file = talloc_strdup ( ctdb , file ) ;
2009-05-21 11:49:16 +10:00
CTDB_NO_MEMORY ( ctdb , ctdb - > recovery_lock_file ) ;
2007-06-02 11:36:42 +10:00
return 0 ;
}
2007-04-18 16:36:22 +02:00
/*
set the directory for the local databases
*/
2007-04-19 09:14:25 +10:00
int ctdb_set_tdb_dir ( struct ctdb_context * ctdb , const char * dir )
2007-04-18 16:36:22 +02:00
{
2007-05-01 06:34:55 +10:00
ctdb - > db_directory = talloc_strdup ( ctdb , dir ) ;
2007-04-19 09:14:25 +10:00
if ( ctdb - > db_directory = = NULL ) {
return - 1 ;
}
return 0 ;
2007-04-18 16:36:22 +02:00
}
2007-09-21 12:24:02 +10:00
/*
set the directory for the persistent databases
*/
int ctdb_set_tdb_dir_persistent ( struct ctdb_context * ctdb , const char * dir )
{
ctdb - > db_directory_persistent = talloc_strdup ( ctdb , dir ) ;
if ( ctdb - > db_directory_persistent = = NULL ) {
return - 1 ;
}
return 0 ;
}
2009-11-23 14:38:03 +01:00
/*
set the directory for internal state databases
*/
int ctdb_set_tdb_dir_state ( struct ctdb_context * ctdb , const char * dir )
{
ctdb - > db_directory_state = talloc_strdup ( ctdb , dir ) ;
if ( ctdb - > db_directory_state = = NULL ) {
return - 1 ;
}
return 0 ;
}
2006-11-27 21:38:13 +11:00
/*
2009-06-01 14:18:34 +10:00
add a node to the list of nodes
2006-11-27 21:38:13 +11:00
*/
static int ctdb_add_node ( struct ctdb_context * ctdb , char * nstr )
{
2006-11-28 17:56:10 +11:00
struct ctdb_node * node , * * nodep ;
nodep = talloc_realloc ( ctdb , ctdb - > nodes , struct ctdb_node * , ctdb - > num_nodes + 1 ) ;
CTDB_NO_MEMORY ( ctdb , nodep ) ;
ctdb - > nodes = nodep ;
nodep = & ctdb - > nodes [ ctdb - > num_nodes ] ;
( * nodep ) = talloc_zero ( ctdb - > nodes , struct ctdb_node ) ;
CTDB_NO_MEMORY ( ctdb , * nodep ) ;
node = * nodep ;
2006-11-27 21:38:13 +11:00
if ( ctdb_parse_address ( ctdb , node , nstr , & node - > address ) ! = 0 ) {
return - 1 ;
}
node - > ctdb = ctdb ;
2006-11-28 14:15:46 +11:00
node - > name = talloc_asprintf ( node , " %s:%u " ,
node - > address . address ,
node - > address . port ) ;
2007-05-18 23:23:36 +10:00
/* this assumes that the nodes are kept in sorted order, and no gaps */
2007-09-04 09:50:07 +10:00
node - > pnn = ctdb - > num_nodes ;
2006-11-27 21:38:13 +11:00
2007-09-24 10:00:14 +10:00
/* nodes start out disconnected and unhealthy */
2007-10-18 16:54:00 +10:00
node - > flags = ( NODE_FLAGS_DISCONNECTED | NODE_FLAGS_UNHEALTHY ) ;
2007-06-07 15:18:55 +10:00
2007-05-01 06:34:55 +10:00
if ( ctdb - > address . address & &
ctdb_same_address ( & ctdb - > address , & node - > address ) ) {
2008-02-22 09:42:52 +11:00
/* for automatic binding to interfaces, see tcp_connect.c */
2007-09-04 10:06:36 +10:00
ctdb - > pnn = node - > pnn ;
2007-06-07 15:18:55 +10:00
node - > flags & = ~ NODE_FLAGS_DISCONNECTED ;
2008-02-22 09:42:52 +11:00
/* do we start out in DISABLED mode? */
if ( ctdb - > start_as_disabled ! = 0 ) {
2008-02-22 09:52:57 +11:00
DEBUG ( DEBUG_INFO , ( " This node is configured to start in DISABLED state \n " ) ) ;
2008-02-22 09:42:52 +11:00
node - > flags | = NODE_FLAGS_DISABLED ;
}
2009-07-09 11:57:20 +10:00
/* do we start out in STOPPED mode? */
if ( ctdb - > start_as_stopped ! = 0 ) {
DEBUG ( DEBUG_INFO , ( " This node is configured to start in STOPPED state \n " ) ) ;
node - > flags | = NODE_FLAGS_STOPPED ;
}
2006-11-28 17:56:10 +11:00
}
ctdb - > num_nodes + + ;
2007-05-18 19:19:35 +10:00
node - > dead_count = 0 ;
2006-11-28 17:56:10 +11:00
2006-11-27 21:38:13 +11:00
return 0 ;
}
2009-06-01 14:18:34 +10:00
/*
add an entry for a " deleted " node to the list of nodes .
a " deleted " node is a node that is commented out from the nodes file .
this is used to prevent that subsequent nodes in the nodes list
change their pnn value if a node is " delete " by commenting it out and then
using " ctdb reloadnodes " at runtime .
*/
static int ctdb_add_deleted_node ( struct ctdb_context * ctdb )
{
struct ctdb_node * node , * * nodep ;
nodep = talloc_realloc ( ctdb , ctdb - > nodes , struct ctdb_node * , ctdb - > num_nodes + 1 ) ;
CTDB_NO_MEMORY ( ctdb , nodep ) ;
ctdb - > nodes = nodep ;
nodep = & ctdb - > nodes [ ctdb - > num_nodes ] ;
( * nodep ) = talloc_zero ( ctdb - > nodes , struct ctdb_node ) ;
CTDB_NO_MEMORY ( ctdb , * nodep ) ;
node = * nodep ;
if ( ctdb_parse_address ( ctdb , node , " 0.0.0.0 " , & node - > address ) ! = 0 ) {
DEBUG ( DEBUG_ERR , ( " Failed to setup deleted node %d \n " , ctdb - > num_nodes ) ) ;
return - 1 ;
}
node - > ctdb = ctdb ;
node - > name = talloc_strdup ( node , " 0.0.0.0:0 " ) ;
/* this assumes that the nodes are kept in sorted order, and no gaps */
node - > pnn = ctdb - > num_nodes ;
/* this node is permanently deleted/disconnected */
node - > flags = NODE_FLAGS_DELETED | NODE_FLAGS_DISCONNECTED ;
ctdb - > num_nodes + + ;
node - > dead_count = 0 ;
return 0 ;
}
2006-11-27 21:38:13 +11:00
/*
setup the node list from a file
*/
int ctdb_set_nlist ( struct ctdb_context * ctdb , const char * nlist )
{
char * * lines ;
int nlines ;
2009-06-01 14:44:15 +10:00
int i , j , num_present ;
2006-11-27 21:38:13 +11:00
2008-02-19 14:44:48 +11:00
talloc_free ( ctdb - > nodes ) ;
ctdb - > nodes = NULL ;
ctdb - > num_nodes = 0 ;
2006-11-27 21:38:13 +11:00
lines = file_lines_load ( nlist , & nlines , ctdb ) ;
if ( lines = = NULL ) {
ctdb_set_error ( ctdb , " Failed to load nlist '%s' \n " , nlist ) ;
return - 1 ;
}
2007-05-29 16:23:47 +10:00
while ( nlines > 0 & & strcmp ( lines [ nlines - 1 ] , " " ) = = 0 ) {
nlines - - ;
}
2006-11-27 21:38:13 +11:00
2009-06-01 14:44:15 +10:00
num_present = 0 ;
for ( i = 0 ; i < nlines ; i + + ) {
2008-10-07 19:25:10 +11:00
char * node ;
node = lines [ i ] ;
/* strip leading spaces */
while ( ( * node = = ' ' ) | | ( * node = = ' \t ' ) ) {
node + + ;
}
if ( * node = = ' # ' ) {
2009-06-01 14:18:34 +10:00
if ( ctdb_add_deleted_node ( ctdb ) ! = 0 ) {
talloc_free ( lines ) ;
return - 1 ;
}
2008-10-07 19:25:10 +11:00
continue ;
}
if ( strcmp ( node , " " ) = = 0 ) {
continue ;
}
if ( ctdb_add_node ( ctdb , node ) ! = 0 ) {
2006-11-27 21:38:13 +11:00
talloc_free ( lines ) ;
return - 1 ;
}
2009-06-01 14:44:15 +10:00
num_present + + ;
2006-11-27 21:38:13 +11:00
}
2007-05-10 07:55:46 +10:00
2009-06-01 14:44:15 +10:00
/* initialize the vnn mapping table now that we have the nodes list,
skipping any deleted nodes
*/
2007-05-10 08:13:19 +10:00
ctdb - > vnn_map = talloc ( ctdb , struct ctdb_vnn_map ) ;
CTDB_NO_MEMORY ( ctdb , ctdb - > vnn_map ) ;
2007-08-22 12:38:31 +10:00
ctdb - > vnn_map - > generation = INVALID_GENERATION ;
2009-06-01 14:44:15 +10:00
ctdb - > vnn_map - > size = num_present ;
2007-05-10 08:13:19 +10:00
ctdb - > vnn_map - > map = talloc_array ( ctdb - > vnn_map , uint32_t , ctdb - > vnn_map - > size ) ;
CTDB_NO_MEMORY ( ctdb , ctdb - > vnn_map - > map ) ;
2009-06-01 14:44:15 +10:00
for ( i = 0 , j = 0 ; i < ctdb - > vnn_map - > size ; i + + ) {
2009-06-01 14:18:34 +10:00
if ( ctdb - > nodes [ i ] - > flags & NODE_FLAGS_DELETED ) {
2009-06-01 14:44:15 +10:00
continue ;
2009-06-01 14:18:34 +10:00
}
2009-06-01 14:44:15 +10:00
ctdb - > vnn_map - > map [ j ] = i ;
j + + ;
2007-05-10 07:55:46 +10:00
}
2006-11-27 21:38:13 +11:00
talloc_free ( lines ) ;
return 0 ;
}
2007-05-25 17:04:13 +10:00
2006-11-27 21:38:13 +11:00
/*
setup the local node address
*/
int ctdb_set_address ( struct ctdb_context * ctdb , const char * address )
{
2006-11-28 17:56:10 +11:00
if ( ctdb_parse_address ( ctdb , ctdb , address , & ctdb - > address ) ! = 0 ) {
return - 1 ;
}
ctdb - > name = talloc_asprintf ( ctdb , " %s:%u " ,
ctdb - > address . address ,
ctdb - > address . port ) ;
return 0 ;
2006-11-27 21:38:13 +11:00
}
2007-04-26 14:27:49 +02:00
2007-05-17 23:23:41 +10:00
/*
2007-06-19 03:54:06 +10:00
return the number of active nodes
2007-05-17 23:23:41 +10:00
*/
2007-06-19 03:54:06 +10:00
uint32_t ctdb_get_num_active_nodes ( struct ctdb_context * ctdb )
2007-05-17 23:23:41 +10:00
{
int i ;
uint32_t count = 0 ;
2009-06-01 14:39:34 +10:00
for ( i = 0 ; i < ctdb - > num_nodes ; i + + ) {
if ( ! ( ctdb - > nodes [ i ] - > flags & NODE_FLAGS_INACTIVE ) ) {
2007-05-17 23:23:41 +10:00
count + + ;
}
}
return count ;
}
2007-02-09 12:45:58 +11:00
2006-11-27 21:38:13 +11:00
/*
2007-05-19 13:45:24 +10:00
called when we need to process a packet . This can be a requeued packet
after a lockwait , or a real packet from another node
2006-11-27 21:38:13 +11:00
*/
2007-05-19 13:45:24 +10:00
void ctdb_input_pkt ( struct ctdb_context * ctdb , struct ctdb_req_header * hdr )
2006-11-27 21:38:13 +11:00
{
2007-04-18 11:20:24 +10:00
TALLOC_CTX * tmp_ctx ;
/* place the packet as a child of the tmp_ctx. We then use
talloc_free ( ) below to free it . If any of the calls want
to keep it , then they will steal it somewhere else , and the
talloc_free ( ) will only free the tmp_ctx */
tmp_ctx = talloc_new ( ctdb ) ;
talloc_steal ( tmp_ctx , hdr ) ;
2007-02-07 13:26:07 +11:00
2008-02-04 17:44:24 +11:00
DEBUG ( DEBUG_DEBUG , ( __location__ " ctdb request %u of type %u length %u from "
2007-05-23 20:15:09 +10:00
" node %u to %u \n " , hdr - > reqid , hdr - > operation , hdr - > length ,
2007-04-17 17:59:39 +02:00
hdr - > srcnode , hdr - > destnode ) ) ;
2007-04-17 22:13:06 +10:00
2006-12-01 15:45:24 +11:00
switch ( hdr - > operation ) {
case CTDB_REQ_CALL :
2007-05-12 19:54:40 +10:00
case CTDB_REPLY_CALL :
case CTDB_REQ_DMASTER :
case CTDB_REPLY_DMASTER :
2007-08-22 12:53:24 +10:00
/* we dont allow these calls when banned */
2007-09-04 10:06:36 +10:00
if ( ctdb - > nodes [ ctdb - > pnn ] - > flags & NODE_FLAGS_BANNED ) {
2008-02-04 20:07:15 +11:00
DEBUG ( DEBUG_DEBUG , ( __location__ " ctdb operation %u "
2007-08-22 12:53:24 +10:00
" request %u "
" length %u from node %u to %u while node "
" is banned \n " ,
hdr - > operation , hdr - > reqid ,
hdr - > length ,
hdr - > srcnode , hdr - > destnode ) ) ;
goto done ;
}
2007-05-12 19:54:40 +10:00
/* for ctdb_call inter-node operations verify that the
remote node that sent us the call is running in the
same generation instance as this node
2007-04-28 01:48:31 +10:00
*/
if ( ctdb - > vnn_map - > generation ! = hdr - > generation ) {
2008-02-04 20:07:15 +11:00
DEBUG ( DEBUG_DEBUG , ( __location__ " ctdb operation %u "
2007-07-11 08:41:29 +10:00
" request %u "
2007-05-23 20:15:09 +10:00
" length %u from node %u to %u had an "
" invalid generation id:%u while our "
" generation id is:%u \n " ,
2007-07-11 08:41:29 +10:00
hdr - > operation , hdr - > reqid ,
hdr - > length ,
2007-05-23 20:15:09 +10:00
hdr - > srcnode , hdr - > destnode ,
hdr - > generation , ctdb - > vnn_map - > generation ) ) ;
2007-05-12 19:54:40 +10:00
goto done ;
2007-04-29 22:51:56 +10:00
}
2007-05-12 19:54:40 +10:00
}
2007-04-29 22:51:56 +10:00
2007-05-12 19:54:40 +10:00
switch ( hdr - > operation ) {
case CTDB_REQ_CALL :
2010-09-29 10:38:41 +10:00
CTDB_INCREMENT_STAT ( ctdb , node . req_call ) ;
2006-12-01 15:45:24 +11:00
ctdb_request_call ( ctdb , hdr ) ;
break ;
case CTDB_REPLY_CALL :
2010-09-29 10:38:41 +10:00
CTDB_INCREMENT_STAT ( ctdb , node . reply_call ) ;
2006-12-01 15:45:24 +11:00
ctdb_reply_call ( ctdb , hdr ) ;
break ;
2006-12-18 14:27:20 +11:00
case CTDB_REPLY_ERROR :
2010-09-29 10:38:41 +10:00
CTDB_INCREMENT_STAT ( ctdb , node . reply_error ) ;
2006-12-18 14:27:20 +11:00
ctdb_reply_error ( ctdb , hdr ) ;
break ;
2006-12-18 16:01:11 +11:00
case CTDB_REQ_DMASTER :
2010-09-29 10:38:41 +10:00
CTDB_INCREMENT_STAT ( ctdb , node . req_dmaster ) ;
2006-12-18 16:01:11 +11:00
ctdb_request_dmaster ( ctdb , hdr ) ;
break ;
case CTDB_REPLY_DMASTER :
2010-09-29 10:38:41 +10:00
CTDB_INCREMENT_STAT ( ctdb , node . reply_dmaster ) ;
2006-12-18 16:01:11 +11:00
ctdb_reply_dmaster ( ctdb , hdr ) ;
break ;
2007-02-09 09:42:04 +11:00
case CTDB_REQ_MESSAGE :
2010-09-29 10:38:41 +10:00
CTDB_INCREMENT_STAT ( ctdb , node . req_message ) ;
2007-02-09 09:42:04 +11:00
ctdb_request_message ( ctdb , hdr ) ;
break ;
2007-04-26 14:27:49 +02:00
case CTDB_REQ_CONTROL :
2010-09-29 10:38:41 +10:00
CTDB_INCREMENT_STAT ( ctdb , node . req_control ) ;
2007-04-26 14:27:49 +02:00
ctdb_request_control ( ctdb , hdr ) ;
break ;
case CTDB_REPLY_CONTROL :
2010-09-29 10:38:41 +10:00
CTDB_INCREMENT_STAT ( ctdb , node . reply_control ) ;
2007-04-26 14:27:49 +02:00
ctdb_reply_control ( ctdb , hdr ) ;
break ;
2007-05-18 19:19:35 +10:00
case CTDB_REQ_KEEPALIVE :
2010-09-29 10:38:41 +10:00
CTDB_INCREMENT_STAT ( ctdb , keepalive_packets_recv ) ;
2007-05-18 19:19:35 +10:00
break ;
2006-12-01 15:45:24 +11:00
default :
2008-02-04 20:07:15 +11:00
DEBUG ( DEBUG_CRIT , ( " %s: Packet with unknown operation %u \n " ,
2007-04-17 22:13:06 +10:00
__location__ , hdr - > operation ) ) ;
2006-12-01 15:45:24 +11:00
break ;
}
2007-04-18 11:20:24 +10:00
done :
talloc_free ( tmp_ctx ) ;
2006-11-27 21:38:13 +11:00
}
2007-05-19 13:45:24 +10:00
2006-11-28 11:51:33 +11:00
/*
2006-11-28 17:56:10 +11:00
called by the transport layer when a node is dead
2006-11-28 11:51:33 +11:00
*/
2007-05-19 16:59:10 +10:00
void ctdb_node_dead ( struct ctdb_node * node )
2006-11-28 11:51:33 +11:00
{
2007-06-07 15:18:55 +10:00
if ( node - > flags & NODE_FLAGS_DISCONNECTED ) {
2008-02-04 17:44:24 +11:00
DEBUG ( DEBUG_INFO , ( " %s: node %s is already marked disconnected: %u connected \n " ,
2007-05-30 14:35:22 +10:00
node - > ctdb - > name , node - > name ,
node - > ctdb - > num_connected ) ) ;
return ;
}
2006-12-01 08:58:08 +11:00
node - > ctdb - > num_connected - - ;
2007-10-10 10:45:22 +10:00
node - > flags | = NODE_FLAGS_DISCONNECTED | NODE_FLAGS_UNHEALTHY ;
2007-05-19 14:04:48 +10:00
node - > rx_cnt = 0 ;
2007-05-19 17:21:58 +10:00
node - > dead_count = 0 ;
2007-10-19 08:58:30 +10:00
2008-02-04 20:07:15 +11:00
DEBUG ( DEBUG_NOTICE , ( " %s: node %s is dead: %u connected \n " ,
2007-04-17 22:13:06 +10:00
node - > ctdb - > name , node - > name , node - > ctdb - > num_connected ) ) ;
2007-05-18 23:48:29 +10:00
ctdb_daemon_cancel_controls ( node - > ctdb , node ) ;
2007-10-19 08:58:30 +10:00
2008-05-11 14:28:33 +10:00
if ( node - > ctdb - > methods = = NULL ) {
2008-12-17 12:01:40 +11:00
DEBUG ( DEBUG_ERR , ( __location__ " Can not restart transport while shutting down daemon. \n " ) ) ;
return ;
2008-05-11 14:28:33 +10:00
}
2007-10-19 08:58:30 +10:00
node - > ctdb - > methods - > restart ( node ) ;
2006-11-28 11:51:33 +11:00
}
2006-11-28 14:15:46 +11:00
/*
2007-02-07 13:26:07 +11:00
called by the transport layer when a node is connected
2006-11-28 14:15:46 +11:00
*/
2007-05-19 17:21:58 +10:00
void ctdb_node_connected ( struct ctdb_node * node )
2006-11-28 14:15:46 +11:00
{
2007-06-07 15:18:55 +10:00
if ( ! ( node - > flags & NODE_FLAGS_DISCONNECTED ) ) {
2008-02-04 17:44:24 +11:00
DEBUG ( DEBUG_INFO , ( " %s: node %s is already marked connected: %u connected \n " ,
2007-05-30 14:35:22 +10:00
node - > ctdb - > name , node - > name ,
node - > ctdb - > num_connected ) ) ;
return ;
}
2006-12-01 08:58:08 +11:00
node - > ctdb - > num_connected + + ;
2007-05-19 17:21:58 +10:00
node - > dead_count = 0 ;
2007-06-07 15:18:55 +10:00
node - > flags & = ~ NODE_FLAGS_DISCONNECTED ;
2007-10-10 10:45:22 +10:00
node - > flags | = NODE_FLAGS_UNHEALTHY ;
2008-02-04 17:44:24 +11:00
DEBUG ( DEBUG_INFO , ( " %s: connected to %s - %u connected \n " ,
2007-04-17 22:13:06 +10:00
node - > ctdb - > name , node - > name , node - > ctdb - > num_connected ) ) ;
2006-12-01 08:58:08 +11:00
}
2007-04-20 17:58:37 +10:00
struct queue_next {
struct ctdb_context * ctdb ;
struct ctdb_req_header * hdr ;
} ;
/*
2009-12-09 17:14:16 +01:00
triggered when a deferred packet is due
2007-04-20 17:58:37 +10:00
*/
static void queue_next_trigger ( struct event_context * ev , struct timed_event * te ,
struct timeval t , void * private_data )
{
struct queue_next * q = talloc_get_type ( private_data , struct queue_next ) ;
2007-05-19 13:45:24 +10:00
ctdb_input_pkt ( q - > ctdb , q - > hdr ) ;
2007-05-23 20:06:37 +10:00
talloc_free ( q ) ;
2007-04-20 17:58:37 +10:00
}
/*
defer a packet , so it is processed on the next event loop
this is used for sending packets to ourselves
*/
static void ctdb_defer_packet ( struct ctdb_context * ctdb , struct ctdb_req_header * hdr )
{
struct queue_next * q ;
q = talloc ( ctdb , struct queue_next ) ;
if ( q = = NULL ) {
2008-02-04 20:07:15 +11:00
DEBUG ( DEBUG_ERR , ( __location__ " Failed to allocate deferred packet \n " ) ) ;
2007-04-20 17:58:37 +10:00
return ;
}
q - > ctdb = ctdb ;
q - > hdr = talloc_memdup ( ctdb , hdr , hdr - > length ) ;
if ( q - > hdr = = NULL ) {
2008-02-04 20:07:15 +11:00
DEBUG ( DEBUG_ERR , ( " Error copying deferred packet to self \n " ) ) ;
2007-04-20 17:58:37 +10:00
return ;
}
2007-04-22 14:26:45 +02:00
#if 0
/* use this to put packets directly into our recv function */
2007-05-19 13:45:24 +10:00
ctdb_input_pkt ( q - > ctdb , q - > hdr ) ;
2007-04-22 14:26:45 +02:00
# else
2007-04-20 17:58:37 +10:00
event_add_timed ( ctdb - > ev , q , timeval_zero ( ) , queue_next_trigger , q ) ;
2007-04-22 14:26:45 +02:00
# endif
2007-04-20 17:58:37 +10:00
}
2007-04-30 15:31:40 +02:00
/*
broadcast a packet to all nodes
*/
2007-06-09 21:58:50 +10:00
static void ctdb_broadcast_packet_all ( struct ctdb_context * ctdb ,
struct ctdb_req_header * hdr )
2007-04-30 15:31:40 +02:00
{
int i ;
2009-06-01 14:18:34 +10:00
for ( i = 0 ; i < ctdb - > num_nodes ; i + + ) {
if ( ctdb - > nodes [ i ] - > flags & NODE_FLAGS_DELETED ) {
continue ;
}
2007-09-04 09:50:07 +10:00
hdr - > destnode = ctdb - > nodes [ i ] - > pnn ;
2007-04-30 15:31:40 +02:00
ctdb_queue_packet ( ctdb , hdr ) ;
}
}
2007-05-05 13:17:26 +10:00
/*
broadcast a packet to all nodes in the current vnnmap
*/
2007-06-09 21:58:50 +10:00
static void ctdb_broadcast_packet_vnnmap ( struct ctdb_context * ctdb ,
struct ctdb_req_header * hdr )
2007-05-05 13:17:26 +10:00
{
int i ;
for ( i = 0 ; i < ctdb - > vnn_map - > size ; i + + ) {
hdr - > destnode = ctdb - > vnn_map - > map [ i ] ;
ctdb_queue_packet ( ctdb , hdr ) ;
}
}
2007-06-09 21:58:50 +10:00
/*
broadcast a packet to all connected nodes
*/
static void ctdb_broadcast_packet_connected ( struct ctdb_context * ctdb ,
struct ctdb_req_header * hdr )
{
int i ;
2009-06-01 14:18:34 +10:00
for ( i = 0 ; i < ctdb - > num_nodes ; i + + ) {
if ( ctdb - > nodes [ i ] - > flags & NODE_FLAGS_DELETED ) {
continue ;
}
2007-06-09 21:58:50 +10:00
if ( ! ( ctdb - > nodes [ i ] - > flags & NODE_FLAGS_DISCONNECTED ) ) {
2007-09-04 09:50:07 +10:00
hdr - > destnode = ctdb - > nodes [ i ] - > pnn ;
2007-06-09 21:58:50 +10:00
ctdb_queue_packet ( ctdb , hdr ) ;
}
}
}
2007-02-09 09:42:04 +11:00
/*
queue a packet or die
*/
void ctdb_queue_packet ( struct ctdb_context * ctdb , struct ctdb_req_header * hdr )
{
struct ctdb_node * node ;
2007-04-30 15:31:40 +02:00
2007-05-05 13:17:26 +10:00
switch ( hdr - > destnode ) {
case CTDB_BROADCAST_ALL :
ctdb_broadcast_packet_all ( ctdb , hdr ) ;
return ;
case CTDB_BROADCAST_VNNMAP :
ctdb_broadcast_packet_vnnmap ( ctdb , hdr ) ;
2007-04-30 15:31:40 +02:00
return ;
2007-06-09 21:58:50 +10:00
case CTDB_BROADCAST_CONNECTED :
ctdb_broadcast_packet_connected ( ctdb , hdr ) ;
return ;
2007-04-30 15:31:40 +02:00
}
2010-09-29 10:38:41 +10:00
CTDB_INCREMENT_STAT ( ctdb , node_packets_sent ) ;
2007-04-26 15:38:33 +02:00
2007-09-04 10:09:58 +10:00
if ( ! ctdb_validate_pnn ( ctdb , hdr - > destnode ) ) {
2008-02-04 20:07:15 +11:00
DEBUG ( DEBUG_CRIT , ( __location__ " cant send to node %u that does not exist \n " ,
2007-04-26 15:38:33 +02:00
hdr - > destnode ) ) ;
return ;
}
2007-02-09 09:42:04 +11:00
node = ctdb - > nodes [ hdr - > destnode ] ;
2007-04-26 15:38:33 +02:00
2009-06-01 14:56:19 +10:00
if ( node - > flags & NODE_FLAGS_DELETED ) {
DEBUG ( DEBUG_ERR , ( __location__ " Can not queue packet to DELETED node %d \n " , hdr - > destnode ) ) ;
return ;
}
if ( node - > pnn = = ctdb - > pnn ) {
2007-04-20 17:58:37 +10:00
ctdb_defer_packet ( ctdb , hdr ) ;
2009-12-09 17:20:23 +01:00
return ;
}
2008-05-11 14:28:33 +10:00
2009-12-09 17:20:23 +01:00
if ( ctdb - > methods = = NULL ) {
DEBUG ( DEBUG_ALERT , ( __location__ " Can not queue packet. "
" Transport is DOWN \n " ) ) ;
return ;
}
node - > tx_cnt + + ;
if ( ctdb - > methods - > queue_pkt ( node , ( uint8_t * ) hdr , hdr - > length ) ! = 0 ) {
ctdb_fatal ( ctdb , " Unable to queue packet \n " ) ;
2007-02-09 09:42:04 +11:00
}
}
2008-07-04 18:00:24 +10:00
/*
a valgrind hack to allow us to get opcode specific backtraces
very ugly , and relies on no compiler optimisation !
*/
void ctdb_queue_packet_opcode ( struct ctdb_context * ctdb , struct ctdb_req_header * hdr , unsigned opcode )
{
switch ( opcode ) {
2008-07-07 08:52:04 +10:00
# define DO_OP(x) case x: ctdb_queue_packet(ctdb, hdr); break
2008-07-04 18:00:24 +10:00
DO_OP ( 1 ) ;
DO_OP ( 2 ) ;
DO_OP ( 3 ) ;
DO_OP ( 4 ) ;
DO_OP ( 5 ) ;
DO_OP ( 6 ) ;
DO_OP ( 7 ) ;
DO_OP ( 8 ) ;
DO_OP ( 9 ) ;
DO_OP ( 10 ) ;
DO_OP ( 11 ) ;
DO_OP ( 12 ) ;
DO_OP ( 13 ) ;
DO_OP ( 14 ) ;
DO_OP ( 15 ) ;
DO_OP ( 16 ) ;
DO_OP ( 17 ) ;
DO_OP ( 18 ) ;
DO_OP ( 19 ) ;
DO_OP ( 20 ) ;
DO_OP ( 21 ) ;
DO_OP ( 22 ) ;
DO_OP ( 23 ) ;
DO_OP ( 24 ) ;
DO_OP ( 25 ) ;
DO_OP ( 26 ) ;
DO_OP ( 27 ) ;
DO_OP ( 28 ) ;
DO_OP ( 29 ) ;
DO_OP ( 30 ) ;
DO_OP ( 31 ) ;
DO_OP ( 32 ) ;
DO_OP ( 33 ) ;
DO_OP ( 34 ) ;
DO_OP ( 35 ) ;
DO_OP ( 36 ) ;
DO_OP ( 37 ) ;
DO_OP ( 38 ) ;
DO_OP ( 39 ) ;
DO_OP ( 40 ) ;
DO_OP ( 41 ) ;
DO_OP ( 42 ) ;
DO_OP ( 43 ) ;
DO_OP ( 44 ) ;
DO_OP ( 45 ) ;
DO_OP ( 46 ) ;
DO_OP ( 47 ) ;
DO_OP ( 48 ) ;
DO_OP ( 49 ) ;
DO_OP ( 50 ) ;
DO_OP ( 51 ) ;
DO_OP ( 52 ) ;
DO_OP ( 53 ) ;
DO_OP ( 54 ) ;
DO_OP ( 55 ) ;
DO_OP ( 56 ) ;
DO_OP ( 57 ) ;
DO_OP ( 58 ) ;
DO_OP ( 59 ) ;
DO_OP ( 60 ) ;
DO_OP ( 61 ) ;
DO_OP ( 62 ) ;
DO_OP ( 63 ) ;
DO_OP ( 64 ) ;
DO_OP ( 65 ) ;
DO_OP ( 66 ) ;
DO_OP ( 67 ) ;
DO_OP ( 68 ) ;
DO_OP ( 69 ) ;
DO_OP ( 70 ) ;
DO_OP ( 71 ) ;
DO_OP ( 72 ) ;
DO_OP ( 73 ) ;
DO_OP ( 74 ) ;
DO_OP ( 75 ) ;
DO_OP ( 76 ) ;
DO_OP ( 77 ) ;
DO_OP ( 78 ) ;
DO_OP ( 79 ) ;
DO_OP ( 80 ) ;
DO_OP ( 81 ) ;
DO_OP ( 82 ) ;
DO_OP ( 83 ) ;
DO_OP ( 84 ) ;
DO_OP ( 85 ) ;
DO_OP ( 86 ) ;
DO_OP ( 87 ) ;
DO_OP ( 88 ) ;
DO_OP ( 89 ) ;
DO_OP ( 90 ) ;
DO_OP ( 91 ) ;
DO_OP ( 92 ) ;
DO_OP ( 93 ) ;
DO_OP ( 94 ) ;
DO_OP ( 95 ) ;
DO_OP ( 96 ) ;
DO_OP ( 97 ) ;
DO_OP ( 98 ) ;
DO_OP ( 99 ) ;
DO_OP ( 100 ) ;
default :
ctdb_queue_packet ( ctdb , hdr ) ;
break ;
}
}