2007-04-26 16:27:49 +04:00
/*
ctdb_control protocol code
Copyright ( C ) Andrew Tridgell 2007
This library is free software ; you can redistribute it and / or
modify it under the terms of the GNU Lesser General Public
License as published by the Free Software Foundation ; either
version 2 of the License , or ( at your option ) any later version .
This library 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
Lesser General Public License for more details .
You should have received a copy of the GNU Lesser General Public
License along with this library ; if not , write to the Free Software
Foundation , Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*/
# include "includes.h"
# include "lib/events/events.h"
# include "lib/tdb/include/tdb.h"
# include "system/network.h"
# include "system/filesys.h"
# include "system/wait.h"
# include "../include/ctdb_private.h"
# include "lib/util/dlinklist.h"
struct ctdb_control_state {
struct ctdb_context * ctdb ;
uint32_t reqid ;
ctdb_control_callback_fn_t callback ;
void * private_data ;
} ;
2007-04-27 16:49:44 +04:00
# define CHECK_CONTROL_DATA_SIZE(size) do { \
if ( indata . dsize ! = sizeof ( pid_t ) ) { \
DEBUG ( 0 , ( __location__ " Invalid data in opcode %u \n " , opcode ) ) ; \
return - 1 ; \
} \
} while ( 0 )
2007-04-26 16:27:49 +04:00
/*
process a control request
*/
static int32_t ctdb_control_dispatch ( struct ctdb_context * ctdb ,
uint32_t opcode , TDB_DATA indata ,
TDB_DATA * outdata )
{
switch ( opcode ) {
case CTDB_CONTROL_PROCESS_EXISTS : {
pid_t pid ;
2007-04-27 16:49:44 +04:00
CHECK_CONTROL_DATA_SIZE ( sizeof ( pid ) ) ;
2007-04-26 16:27:49 +04:00
pid = * ( pid_t * ) indata . dptr ;
return kill ( pid , 0 ) ;
}
2007-04-26 21:27:07 +04:00
2007-04-26 16:51:41 +04:00
case CTDB_CONTROL_STATUS : {
2007-04-27 16:49:44 +04:00
CHECK_CONTROL_DATA_SIZE ( 0 ) ;
2007-04-26 16:51:41 +04:00
outdata - > dptr = ( uint8_t * ) & ctdb - > status ;
outdata - > dsize = sizeof ( ctdb - > status ) ;
return 0 ;
}
2007-04-26 21:27:07 +04:00
2007-04-27 14:56:10 +04:00
case CTDB_CONTROL_GETVNNMAP : {
uint32_t i , len ;
2007-04-27 16:49:44 +04:00
CHECK_CONTROL_DATA_SIZE ( 0 ) ;
2007-04-27 14:56:10 +04:00
len = 2 + ctdb - > vnn_map - > size ;
outdata - > dsize = 4 * len ;
outdata - > dptr = ( unsigned char * ) talloc_array ( outdata , uint32_t , len ) ;
( ( uint32_t * ) outdata - > dptr ) [ 0 ] = ctdb - > vnn_map - > generation ;
( ( uint32_t * ) outdata - > dptr ) [ 1 ] = ctdb - > vnn_map - > size ;
for ( i = 0 ; i < ctdb - > vnn_map - > size ; i + + ) {
( ( uint32_t * ) outdata - > dptr ) [ i + 2 ] = ctdb - > vnn_map - > map [ i ] ;
}
return 0 ;
}
2007-04-27 16:08:12 +04:00
case CTDB_CONTROL_SETVNNMAP : {
uint32_t * ptr , i ;
ptr = ( uint32_t * ) ( & indata . dptr [ 0 ] ) ;
ctdb - > vnn_map - > generation = ptr [ 0 ] ;
ctdb - > vnn_map - > size = ptr [ 1 ] ;
if ( ctdb - > vnn_map - > map ) {
talloc_free ( ctdb - > vnn_map - > map ) ;
ctdb - > vnn_map - > map = NULL ;
}
ctdb - > vnn_map - > map = talloc_array ( ctdb - > vnn_map , uint32_t , ctdb - > vnn_map - > size ) ;
if ( ctdb - > vnn_map - > map = = NULL ) {
DEBUG ( 0 , ( __location__ " Unable to allocate vnn_map->map structure \n " ) ) ;
exit ( 1 ) ;
}
for ( i = 0 ; i < ctdb - > vnn_map - > size ; i + + ) {
ctdb - > vnn_map - > map [ i ] = ptr [ i + 2 ] ;
}
return 0 ;
}
2007-04-26 21:27:07 +04:00
case CTDB_CONTROL_CONFIG : {
2007-04-27 16:49:44 +04:00
CHECK_CONTROL_DATA_SIZE ( 0 ) ;
2007-04-26 21:27:07 +04:00
outdata - > dptr = ( uint8_t * ) ctdb ;
outdata - > dsize = sizeof ( * ctdb ) ;
return 0 ;
}
case CTDB_CONTROL_PING :
2007-04-27 16:49:44 +04:00
CHECK_CONTROL_DATA_SIZE ( 0 ) ;
2007-04-26 21:27:07 +04:00
return 0 ;
2007-04-27 01:10:35 +04:00
case CTDB_CONTROL_GETDBPATH : {
uint32_t db_id ;
struct ctdb_db_context * ctdb_db ;
2007-04-27 16:49:44 +04:00
CHECK_CONTROL_DATA_SIZE ( db_id ) ;
2007-04-27 01:10:35 +04:00
db_id = * ( uint32_t * ) indata . dptr ;
ctdb_db = find_ctdb_db ( ctdb , db_id ) ;
if ( ctdb_db = = NULL ) return - 1 ;
outdata - > dptr = discard_const ( ctdb_db - > db_path ) ;
2007-04-27 01:29:26 +04:00
outdata - > dsize = strlen ( ctdb_db - > db_path ) + 1 ;
2007-04-27 01:10:35 +04:00
return 0 ;
}
2007-04-26 16:27:49 +04:00
default :
DEBUG ( 0 , ( __location__ " Unknown CTDB control opcode %u \n " , opcode ) ) ;
return - 1 ;
}
}
/*
called when a CTDB_REQ_CONTROL packet comes in
*/
void ctdb_request_control ( struct ctdb_context * ctdb , struct ctdb_req_header * hdr )
{
struct ctdb_req_control * c = ( struct ctdb_req_control * ) hdr ;
2007-04-27 14:56:10 +04:00
TDB_DATA data , * outdata ;
2007-04-26 16:27:49 +04:00
struct ctdb_reply_control * r ;
int32_t status ;
size_t len ;
data . dptr = & c - > data [ 0 ] ;
data . dsize = c - > datalen ;
2007-04-27 14:56:10 +04:00
outdata = talloc_zero ( c , TDB_DATA ) ;
status = ctdb_control_dispatch ( ctdb , c - > opcode , data , outdata ) ;
2007-04-26 16:27:49 +04:00
2007-04-27 14:56:10 +04:00
len = offsetof ( struct ctdb_reply_control , data ) + outdata - > dsize ;
2007-04-26 16:27:49 +04:00
r = ctdb - > methods - > allocate_pkt ( ctdb , len ) ;
CTDB_NO_MEMORY_VOID ( ctdb , r ) ;
talloc_set_name_const ( r , " ctdb_reply_control packet " ) ;
r - > hdr . length = len ;
r - > hdr . ctdb_magic = CTDB_MAGIC ;
r - > hdr . ctdb_version = CTDB_VERSION ;
r - > hdr . operation = CTDB_REPLY_CONTROL ;
r - > hdr . destnode = hdr - > srcnode ;
r - > hdr . srcnode = ctdb - > vnn ;
r - > hdr . reqid = hdr - > reqid ;
r - > status = status ;
2007-04-27 14:56:10 +04:00
r - > datalen = outdata - > dsize ;
if ( outdata - > dsize ) {
memcpy ( & r - > data [ 0 ] , outdata - > dptr , outdata - > dsize ) ;
2007-04-26 16:27:49 +04:00
}
ctdb_queue_packet ( ctdb , & r - > hdr ) ;
talloc_free ( r ) ;
}
/*
called when a CTDB_REPLY_CONTROL packet comes in
*/
void ctdb_reply_control ( struct ctdb_context * ctdb , struct ctdb_req_header * hdr )
{
struct ctdb_reply_control * c = ( struct ctdb_reply_control * ) hdr ;
TDB_DATA data ;
struct ctdb_control_state * state ;
state = ctdb_reqid_find ( ctdb , hdr - > reqid , struct ctdb_control_state ) ;
if ( state = = NULL ) {
return ;
}
if ( hdr - > reqid ! = state - > reqid ) {
/* we found a record but it was the wrong one */
DEBUG ( 0 , ( " Dropped orphaned control reply with reqid:%d \n " , hdr - > reqid ) ) ;
return ;
}
data . dptr = & c - > data [ 0 ] ;
data . dsize = c - > datalen ;
state - > callback ( ctdb , c - > status , data , state - > private_data ) ;
talloc_free ( state ) ;
}
static int ctdb_control_destructor ( struct ctdb_control_state * state )
{
ctdb_reqid_remove ( state - > ctdb , state - > reqid ) ;
return 0 ;
}
/*
send a control message to a node
*/
int ctdb_daemon_send_control ( struct ctdb_context * ctdb , uint32_t destnode ,
uint32_t srvid , uint32_t opcode , TDB_DATA data ,
ctdb_control_callback_fn_t callback ,
void * private_data )
{
struct ctdb_req_control * c ;
struct ctdb_control_state * state ;
size_t len ;
state = talloc ( ctdb , struct ctdb_control_state ) ;
CTDB_NO_MEMORY ( ctdb , state ) ;
state - > reqid = ctdb_reqid_new ( ctdb , state ) ;
state - > callback = callback ;
state - > private_data = private_data ;
state - > ctdb = ctdb ;
talloc_set_destructor ( state , ctdb_control_destructor ) ;
len = offsetof ( struct ctdb_req_control , data ) + data . dsize ;
c = ctdb - > methods - > allocate_pkt ( state , len ) ;
CTDB_NO_MEMORY ( ctdb , c ) ;
talloc_set_name_const ( c , " ctdb_req_control packet " ) ;
c - > hdr . length = len ;
c - > hdr . ctdb_magic = CTDB_MAGIC ;
c - > hdr . ctdb_version = CTDB_VERSION ;
c - > hdr . operation = CTDB_REQ_CONTROL ;
c - > hdr . destnode = destnode ;
c - > hdr . srcnode = ctdb - > vnn ;
c - > hdr . reqid = state - > reqid ;
c - > opcode = opcode ;
c - > srvid = srvid ;
c - > datalen = data . dsize ;
if ( data . dsize ) {
memcpy ( & c - > data [ 0 ] , data . dptr , data . dsize ) ;
}
ctdb_queue_packet ( ctdb , & c - > hdr ) ;
# if CTDB_REQ_TIMEOUT
event_add_timed ( ctdb - > ev , state , timeval_current_ofs ( CTDB_REQ_TIMEOUT , 0 ) ,
ctdb_control_timeout , state ) ;
# endif
talloc_free ( c ) ;
return 0 ;
}