2007-06-10 17:02:09 +00:00
/*
Unix SMB / CIFS implementation .
Samba internal messaging functions
Copyright ( C ) 2007 by Volker Lendecke
Copyright ( C ) 2007 by Andrew Tridgell
2009-08-07 12:09:21 +02:00
2007-06-10 17:02:09 +00: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-09 19:25:36 +00:00
the Free Software Foundation ; either version 3 of the License , or
2007-06-10 17:02:09 +00:00
( at your option ) any later version .
2009-08-07 12:09:21 +02:00
2007-06-10 17:02:09 +00:00
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 .
2009-08-07 12:09:21 +02:00
2007-06-10 17:02:09 +00:00
You should have received a copy of the GNU General Public License
2007-07-10 00:52:41 +00:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
2007-06-10 17:02:09 +00:00
*/
2016-04-08 16:14:33 +02:00
# include "replace.h"
2017-01-09 08:17:02 +01:00
# include <tevent.h>
2011-05-05 11:25:29 +02:00
# include "util_tdb.h"
2012-02-15 11:22:45 +01:00
# include "serverid.h"
2012-04-04 11:56:06 +02:00
# include "ctdbd_conn.h"
2014-05-06 12:21:42 +02:00
# include "system/select.h"
2020-06-25 15:11:44 +02:00
# include "lib/util/util_net.h"
2015-10-12 15:57:34 +02:00
# include "lib/util/sys_rw_data.h"
2015-05-25 08:50:35 +02:00
# include "lib/util/iov_buf.h"
2016-04-19 16:14:30 +02:00
# include "lib/util/select.h"
2016-04-08 16:14:33 +02:00
# include "lib/util/debug.h"
# include "lib/util/talloc_stack.h"
# include "lib/util/genrand.h"
# include "lib/util/fault.h"
2017-01-09 08:17:02 +01:00
# include "lib/util/dlinklist.h"
2018-06-29 08:00:10 +02:00
# include "lib/util/tevent_unix.h"
2017-01-09 08:17:02 +01:00
# include "lib/util/sys_rw.h"
# include "lib/util/blocking.h"
2017-06-09 08:48:21 +02:00
# include "ctdb/include/ctdb_protocol.h"
2020-03-20 13:58:21 +01:00
# include "lib/async_req/async_sock.h"
2007-06-10 17:02:09 +00:00
/* paths to these include files come from --with-ctdb= in configure */
2011-08-30 17:02:54 +02:00
2015-05-19 07:01:55 +02:00
struct ctdbd_srvid_cb {
uint64_t srvid ;
2017-06-01 17:45:47 +02:00
int ( * cb ) ( struct tevent_context * ev ,
uint32_t src_vnn , uint32_t dst_vnn ,
2015-06-23 16:55:09 +02:00
uint64_t dst_srvid ,
const uint8_t * msg , size_t msglen ,
void * private_data ) ;
2015-05-19 07:01:55 +02:00
void * private_data ;
} ;
2007-06-10 17:02:09 +00:00
struct ctdbd_connection {
2012-07-30 09:31:59 -04:00
uint32_t reqid ;
uint32_t our_vnn ;
uint64_t rand_srvid ;
2015-05-19 07:01:55 +02:00
struct ctdbd_srvid_cb * callbacks ;
2014-05-06 12:21:42 +02:00
int fd ;
2015-09-25 14:36:35 -07:00
int timeout ;
2017-01-09 08:17:02 +01:00
2020-03-12 16:05:58 +01:00
/*
* Outgoing queue for writev_send of asynchronous ctdb requests
*/
struct tevent_queue * outgoing ;
2020-03-20 13:58:21 +01:00
struct tevent_req * * pending ;
struct tevent_req * read_req ;
2007-06-10 17:02:09 +00:00
} ;
2017-01-09 08:17:02 +01:00
static bool ctdbd_conn_has_async_reqs ( struct ctdbd_connection * conn )
{
2020-03-20 13:58:21 +01:00
size_t len = talloc_array_length ( conn - > pending ) ;
2020-03-24 14:33:28 +01:00
return ( len ! = 0 ) ;
2017-01-09 08:17:02 +01:00
}
2011-10-26 10:56:32 +02:00
static uint32_t ctdbd_next_reqid ( struct ctdbd_connection * conn )
{
conn - > reqid + = 1 ;
if ( conn - > reqid = = 0 ) {
conn - > reqid + = 1 ;
}
return conn - > reqid ;
}
2015-10-02 20:31:52 -07:00
static int ctdbd_control ( struct ctdbd_connection * conn ,
uint32_t vnn , uint32_t opcode ,
uint64_t srvid , uint32_t flags ,
TDB_DATA data ,
TALLOC_CTX * mem_ctx , TDB_DATA * outdata ,
2016-04-21 13:31:39 +02:00
int32_t * cstatus ) ;
2007-06-10 17:02:09 +00:00
/*
* exit on fatal communications errors with the ctdbd daemon
*/
static void cluster_fatal ( const char * why )
{
DEBUG ( 0 , ( " cluster fatal event: %s - exiting immediately \n " , why ) ) ;
/* we don't use smb_panic() as we don't want to delay to write
a core file . We need to release this process id immediately
so that someone else can take over without getting sharing
violations */
2009-10-22 13:03:20 +02:00
_exit ( 1 ) ;
2007-06-10 17:02:09 +00:00
}
2007-08-29 11:46:44 +00:00
/*
*
*/
static void ctdb_packet_dump ( struct ctdb_req_header * hdr )
{
2012-02-07 16:41:25 +01:00
if ( DEBUGLEVEL < 11 ) {
2007-08-29 11:46:44 +00:00
return ;
}
2020-03-20 13:27:43 +01:00
DEBUGADD ( 11 , ( " len=% " PRIu32 " , magic=% " PRIu32 " , vers=% " PRIu32 " , "
" gen=% " PRIu32 " , op=% " PRIu32 " , reqid=% " PRIu32 " \n " ,
hdr - > length ,
hdr - > ctdb_magic ,
hdr - > ctdb_version ,
hdr - > generation ,
hdr - > operation ,
hdr - > reqid ) ) ;
2007-08-29 11:46:44 +00:00
}
2007-06-10 17:02:09 +00:00
/*
* Register a srvid with ctdbd
*/
2015-10-02 20:42:05 -07:00
int register_with_ctdbd ( struct ctdbd_connection * conn , uint64_t srvid ,
2017-06-01 17:45:47 +02:00
int ( * cb ) ( struct tevent_context * ev ,
uint32_t src_vnn , uint32_t dst_vnn ,
2015-10-02 20:42:05 -07:00
uint64_t dst_srvid ,
const uint8_t * msg , size_t msglen ,
void * private_data ) ,
void * private_data )
2007-06-10 17:02:09 +00:00
{
2016-04-21 13:31:39 +02:00
int ret ;
int32_t cstatus ;
2015-05-19 07:01:55 +02:00
size_t num_callbacks ;
struct ctdbd_srvid_cb * tmp ;
2014-11-21 16:20:10 +01:00
2016-04-13 18:47:46 +02:00
ret = ctdbd_control_local ( conn , CTDB_CONTROL_REGISTER_SRVID , srvid , 0 ,
tdb_null , NULL , NULL , & cstatus ) ;
2015-10-02 20:00:32 -07:00
if ( ret ! = 0 ) {
2015-10-02 20:42:05 -07:00
return ret ;
2014-11-21 16:20:10 +01:00
}
2015-05-19 07:01:55 +02:00
num_callbacks = talloc_array_length ( conn - > callbacks ) ;
2014-11-21 16:20:10 +01:00
2015-05-19 07:01:55 +02:00
tmp = talloc_realloc ( conn , conn - > callbacks , struct ctdbd_srvid_cb ,
num_callbacks + 1 ) ;
2014-11-21 16:20:10 +01:00
if ( tmp = = NULL ) {
2015-10-02 20:42:05 -07:00
return ENOMEM ;
2014-11-21 16:20:10 +01:00
}
2015-05-19 07:01:55 +02:00
conn - > callbacks = tmp ;
conn - > callbacks [ num_callbacks ] = ( struct ctdbd_srvid_cb ) {
2015-05-19 07:05:24 +02:00
. srvid = srvid , . cb = cb , . private_data = private_data
2015-05-19 07:01:55 +02:00
} ;
2014-11-21 16:20:10 +01:00
2015-10-02 20:42:05 -07:00
return 0 ;
2014-11-21 16:20:10 +01:00
}
2017-06-01 17:49:56 +02:00
static int ctdbd_msg_call_back ( struct tevent_context * ev ,
struct ctdbd_connection * conn ,
2015-10-29 16:36:30 +11:00
struct ctdb_req_message_old * msg )
2015-05-19 15:07:33 +02:00
{
2016-04-19 16:02:49 +02:00
uint32_t msg_len ;
2015-05-19 15:07:33 +02:00
size_t i , num_callbacks ;
2015-06-02 20:30:06 +00:00
msg_len = msg - > hdr . length ;
2015-10-29 16:36:30 +11:00
if ( msg_len < offsetof ( struct ctdb_req_message_old , data ) ) {
2016-04-19 16:02:49 +02:00
DBG_DEBUG ( " len % " PRIu32 " too small \n " , msg_len ) ;
2015-06-23 16:59:00 +02:00
return 0 ;
2015-06-02 20:30:06 +00:00
}
2015-10-29 16:36:30 +11:00
msg_len - = offsetof ( struct ctdb_req_message_old , data ) ;
2015-06-02 20:30:06 +00:00
if ( msg_len < msg - > datalen ) {
2016-04-19 16:02:49 +02:00
DBG_DEBUG ( " msg_len=% " PRIu32 " < msg->datalen=% " PRIu32 " \n " ,
msg_len , msg - > datalen ) ;
2015-06-23 16:59:00 +02:00
return 0 ;
2015-06-02 20:30:06 +00:00
}
2015-05-19 15:07:33 +02:00
num_callbacks = talloc_array_length ( conn - > callbacks ) ;
for ( i = 0 ; i < num_callbacks ; i + + ) {
struct ctdbd_srvid_cb * cb = & conn - > callbacks [ i ] ;
if ( ( cb - > srvid = = msg - > srvid ) & & ( cb - > cb ! = NULL ) ) {
2015-06-23 16:59:00 +02:00
int ret ;
2017-06-01 17:49:56 +02:00
ret = cb - > cb ( ev ,
2017-06-01 17:45:47 +02:00
msg - > hdr . srcnode , msg - > hdr . destnode ,
2015-06-23 16:59:00 +02:00
msg - > srvid , msg - > data , msg - > datalen ,
cb - > private_data ) ;
if ( ret ! = 0 ) {
return ret ;
}
2015-05-19 15:07:33 +02:00
}
}
2015-06-23 16:59:00 +02:00
return 0 ;
2015-05-19 15:07:33 +02:00
}
2007-06-10 17:02:09 +00:00
/*
* get our vnn from the cluster
*/
2015-10-02 20:42:05 -07:00
static int get_cluster_vnn ( struct ctdbd_connection * conn , uint32_t * vnn )
2007-06-10 17:02:09 +00:00
{
int32_t cstatus = - 1 ;
2015-10-02 20:00:32 -07:00
int ret ;
2016-04-13 18:47:46 +02:00
ret = ctdbd_control_local ( conn , CTDB_CONTROL_GET_PNN , 0 , 0 ,
tdb_null , NULL , NULL , & cstatus ) ;
2015-10-02 20:00:32 -07:00
if ( ret ! = 0 ) {
DEBUG ( 1 , ( " ctdbd_control failed: %s \n " , strerror ( ret ) ) ) ;
2015-10-02 20:42:05 -07:00
return ret ;
2007-06-10 17:02:09 +00:00
}
* vnn = ( uint32_t ) cstatus ;
2015-10-02 20:42:05 -07:00
return ret ;
2007-06-10 17:02:09 +00:00
}
2009-11-16 12:03:24 +01:00
/*
* Are we active ( i . e . not banned or stopped ? )
*/
static bool ctdbd_working ( struct ctdbd_connection * conn , uint32_t vnn )
{
int32_t cstatus = - 1 ;
2020-03-30 12:22:46 +02:00
TDB_DATA outdata = { 0 } ;
2015-10-29 17:22:48 +11:00
struct ctdb_node_map_old * m ;
2015-10-02 20:05:15 -07:00
bool ok = false ;
2016-04-08 15:59:08 +02:00
uint32_t i ;
int ret ;
2009-11-16 12:03:24 +01:00
2016-04-13 18:47:46 +02:00
ret = ctdbd_control_local ( conn , CTDB_CONTROL_GET_NODEMAP , 0 , 0 ,
tdb_null , talloc_tos ( ) , & outdata , & cstatus ) ;
2015-10-02 20:06:59 -07:00
if ( ret ! = 0 ) {
DEBUG ( 1 , ( " ctdbd_control failed: %s \n " , strerror ( ret ) ) ) ;
2013-01-31 11:02:52 +01:00
return false ;
2009-11-16 12:03:24 +01:00
}
if ( ( cstatus ! = 0 ) | | ( outdata . dptr = = NULL ) ) {
DEBUG ( 2 , ( " Received invalid ctdb data \n " ) ) ;
2020-03-30 12:22:46 +02:00
goto fail ;
2009-11-16 12:03:24 +01:00
}
2015-10-29 17:22:48 +11:00
m = ( struct ctdb_node_map_old * ) outdata . dptr ;
2009-11-16 12:03:24 +01:00
for ( i = 0 ; i < m - > num ; i + + ) {
if ( vnn = = m - > nodes [ i ] . pnn ) {
break ;
}
}
if ( i = = m - > num ) {
DEBUG ( 2 , ( " Did not find ourselves (node %d) in nodemap \n " ,
( int ) vnn ) ) ;
goto fail ;
}
2016-06-23 16:17:51 +10:00
if ( ( m - > nodes [ i ] . flags & NODE_FLAGS_INACTIVE ) ! = 0 ) {
2009-11-16 12:03:24 +01:00
DEBUG ( 2 , ( " Node has status %x, not active \n " ,
( int ) m - > nodes [ i ] . flags ) ) ;
goto fail ;
}
2015-10-02 20:05:15 -07:00
ok = true ;
2009-11-16 12:03:24 +01:00
fail :
TALLOC_FREE ( outdata . dptr ) ;
2015-10-02 20:05:15 -07:00
return ok ;
2009-11-16 12:03:24 +01:00
}
2012-07-30 09:31:59 -04:00
uint32_t ctdbd_vnn ( const struct ctdbd_connection * conn )
2007-06-10 17:02:09 +00:00
{
return conn - > our_vnn ;
}
/*
* Get us a ctdb connection
*/
2015-09-25 14:32:09 -07:00
static int ctdbd_connect ( const char * sockname , int * pfd )
2007-06-10 17:02:09 +00:00
{
2020-06-25 15:11:44 +02:00
struct samba_sockaddr addr = {
. sa_socklen = sizeof ( struct sockaddr_un ) ,
. u = {
. un = {
. sun_family = AF_UNIX ,
} ,
} ,
} ;
2007-06-10 17:02:09 +00:00
int fd ;
2014-08-19 09:20:49 +00:00
size_t namelen ;
2020-06-25 15:11:44 +02:00
int ret ;
2007-06-10 17:02:09 +00:00
fd = socket ( AF_UNIX , SOCK_STREAM , 0 ) ;
if ( fd = = - 1 ) {
2014-05-06 12:21:42 +02:00
int err = errno ;
DEBUG ( 3 , ( " Could not create socket: %s \n " , strerror ( err ) ) ) ;
return err ;
2007-06-10 17:02:09 +00:00
}
2020-06-25 15:11:44 +02:00
namelen = strlcpy ( addr . u . un . sun_path ,
sockname ,
sizeof ( addr . u . un . sun_path ) ) ;
if ( namelen > = sizeof ( addr . u . un . sun_path ) ) {
2014-08-19 09:20:49 +00:00
DEBUG ( 3 , ( " %s: Socket name too long: %s \n " , __func__ ,
sockname ) ) ;
close ( fd ) ;
return ENAMETOOLONG ;
}
2007-06-10 17:02:09 +00:00
2020-06-25 15:11:44 +02:00
ret = connect ( fd , & addr . u . sa , addr . sa_socklen ) ;
if ( ret = = - 1 ) {
2014-05-06 12:21:42 +02:00
int err = errno ;
2008-06-24 15:09:37 +02:00
DEBUG ( 1 , ( " connect(%s) failed: %s \n " , sockname ,
2014-05-06 12:21:42 +02:00
strerror ( err ) ) ) ;
2007-06-10 17:02:09 +00:00
close ( fd ) ;
2014-05-06 12:21:42 +02:00
return err ;
2007-06-10 17:02:09 +00:00
}
2014-05-06 12:21:42 +02:00
* pfd = fd ;
return 0 ;
2007-06-10 17:02:09 +00:00
}
2015-09-25 14:36:35 -07:00
static int ctdb_read_packet ( int fd , int timeout , TALLOC_CTX * mem_ctx ,
2015-07-11 12:23:22 +02:00
struct ctdb_req_header * * result )
2009-11-03 05:41:02 +01:00
{
2014-05-06 12:21:42 +02:00
struct ctdb_req_header * req ;
uint32_t msglen ;
2015-05-17 20:23:35 +02:00
ssize_t nread ;
2009-11-03 05:41:02 +01:00
2014-07-21 12:35:39 +00:00
if ( timeout ! = - 1 ) {
2016-04-19 16:14:30 +02:00
struct pollfd pfd = { . fd = fd , . events = POLLIN } ;
int ret ;
ret = sys_poll_intr ( & pfd , 1 , timeout ) ;
2014-07-21 12:35:39 +00:00
if ( ret = = - 1 ) {
2015-07-11 12:23:22 +02:00
return errno ;
2014-07-21 12:35:39 +00:00
}
if ( ret = = 0 ) {
2015-07-11 12:23:22 +02:00
return ETIMEDOUT ;
2014-07-21 12:35:39 +00:00
}
if ( ret ! = 1 ) {
2015-07-11 12:23:22 +02:00
return EIO ;
2014-07-21 12:35:39 +00:00
}
2014-05-06 12:21:42 +02:00
}
2015-05-17 20:23:35 +02:00
nread = read_data ( fd , & msglen , sizeof ( msglen ) ) ;
if ( nread = = - 1 ) {
2015-07-11 12:23:22 +02:00
return errno ;
2015-05-17 20:23:35 +02:00
}
if ( nread = = 0 ) {
2015-07-11 12:23:22 +02:00
return EIO ;
2014-05-06 12:21:42 +02:00
}
if ( msglen < sizeof ( struct ctdb_req_header ) ) {
2015-07-11 12:23:22 +02:00
return EIO ;
2014-05-06 12:21:42 +02:00
}
req = talloc_size ( mem_ctx , msglen ) ;
if ( req = = NULL ) {
2015-07-11 12:23:22 +02:00
return ENOMEM ;
2014-05-06 12:21:42 +02:00
}
talloc_set_name_const ( req , " struct ctdb_req_header " ) ;
req - > length = msglen ;
2015-05-17 20:23:35 +02:00
nread = read_data ( fd , ( ( char * ) req ) + sizeof ( msglen ) ,
msglen - sizeof ( msglen ) ) ;
if ( nread = = - 1 ) {
2015-10-02 19:51:01 -07:00
TALLOC_FREE ( req ) ;
2015-07-11 12:23:22 +02:00
return errno ;
2015-05-17 20:23:35 +02:00
}
if ( nread = = 0 ) {
2015-10-02 19:51:01 -07:00
TALLOC_FREE ( req ) ;
2015-07-11 12:23:22 +02:00
return EIO ;
2014-05-06 12:21:42 +02:00
}
* result = req ;
2015-07-11 12:23:22 +02:00
return 0 ;
2009-11-03 05:41:02 +01:00
}
2007-06-10 17:02:09 +00:00
/*
* Read a full ctdbd request . If we have a messaging context , defer incoming
* messages that might come in between .
*/
2015-06-19 11:47:01 +02:00
static int ctdb_read_req ( struct ctdbd_connection * conn , uint32_t reqid ,
TALLOC_CTX * mem_ctx , struct ctdb_req_header * * result )
2007-06-10 17:02:09 +00:00
{
2019-07-09 16:02:47 +00:00
struct ctdb_req_header * hdr = NULL ;
2015-07-11 12:23:22 +02:00
int ret ;
2007-06-10 17:02:09 +00:00
2007-08-29 11:46:44 +00:00
next_pkt :
2007-06-10 17:02:09 +00:00
2015-09-25 14:36:35 -07:00
ret = ctdb_read_packet ( conn - > fd , conn - > timeout , mem_ctx , & hdr ) ;
2019-09-03 15:26:19 +02:00
if ( ret ! = 0 ) {
2018-10-05 14:04:18 +10:00
DBG_ERR ( " ctdb_read_packet failed: %s \n " , strerror ( ret ) ) ;
cluster_fatal ( " failed to read data from ctdbd \n " ) ;
2019-07-09 16:02:47 +00:00
return - 1 ;
2007-06-10 17:02:09 +00:00
}
2019-09-03 15:26:19 +02:00
SMB_ASSERT ( hdr ! = NULL ) ;
2007-06-10 17:02:09 +00:00
2012-02-07 16:41:25 +01:00
DEBUG ( 11 , ( " Received ctdb packet \n " ) ) ;
2007-08-29 11:46:44 +00:00
ctdb_packet_dump ( hdr ) ;
2007-06-10 17:02:09 +00:00
if ( hdr - > operation = = CTDB_REQ_MESSAGE ) {
2015-10-29 16:36:30 +11:00
struct ctdb_req_message_old * msg = ( struct ctdb_req_message_old * ) hdr ;
2007-06-10 17:02:09 +00:00
2017-06-01 17:49:56 +02:00
ret = ctdbd_msg_call_back ( NULL , conn , msg ) ;
2015-06-23 17:04:59 +02:00
if ( ret ! = 0 ) {
TALLOC_FREE ( hdr ) ;
return ret ;
}
2007-06-10 17:02:09 +00:00
TALLOC_FREE ( hdr ) ;
2007-08-29 11:46:44 +00:00
goto next_pkt ;
2007-06-10 17:02:09 +00:00
}
2011-10-26 10:58:25 +02:00
if ( ( reqid ! = 0 ) & & ( hdr - > reqid ! = reqid ) ) {
2007-06-10 17:02:09 +00:00
/* we got the wrong reply */
DEBUG ( 0 , ( " Discarding mismatched ctdb reqid %u should have "
" been %u \n " , hdr - > reqid , reqid ) ) ;
TALLOC_FREE ( hdr ) ;
2011-10-27 15:21:29 +02:00
goto next_pkt ;
2007-06-10 17:02:09 +00:00
}
2014-05-06 12:21:42 +02:00
* result = talloc_move ( mem_ctx , & hdr ) ;
2007-06-10 17:02:09 +00:00
2015-06-19 11:47:01 +02:00
return 0 ;
2007-06-10 17:02:09 +00:00
}
2017-01-09 08:17:02 +01:00
static int ctdbd_connection_destructor ( struct ctdbd_connection * c ) ;
2007-06-10 17:02:09 +00:00
/*
* Get us a ctdbd connection
*/
2016-07-09 08:48:49 +02:00
static int ctdbd_init_connection_internal ( TALLOC_CTX * mem_ctx ,
const char * sockname , int timeout ,
struct ctdbd_connection * conn )
2007-06-10 17:02:09 +00:00
{
2014-05-06 12:21:42 +02:00
int ret ;
2007-06-10 17:02:09 +00:00
2015-09-25 16:09:23 -07:00
conn - > timeout = timeout ;
2015-09-25 14:36:35 -07:00
if ( conn - > timeout = = 0 ) {
conn - > timeout = - 1 ;
}
2016-04-19 21:40:40 +02:00
ret = ctdbd_connect ( sockname , & conn - > fd ) ;
2014-05-06 12:21:42 +02:00
if ( ret ! = 0 ) {
2015-06-26 13:17:01 +02:00
DEBUG ( 1 , ( " ctdbd_connect failed: %s \n " , strerror ( ret ) ) ) ;
2016-07-09 08:48:49 +02:00
return ret ;
2007-06-10 17:02:09 +00:00
}
2014-05-06 12:21:42 +02:00
talloc_set_destructor ( conn , ctdbd_connection_destructor ) ;
2007-06-10 17:02:09 +00:00
2015-10-02 20:42:05 -07:00
ret = get_cluster_vnn ( conn , & conn - > our_vnn ) ;
if ( ret ! = 0 ) {
DEBUG ( 10 , ( " get_cluster_vnn failed: %s \n " , strerror ( ret ) ) ) ;
2016-07-09 08:48:49 +02:00
return ret ;
2007-06-10 17:02:09 +00:00
}
2009-11-16 12:03:24 +01:00
if ( ! ctdbd_working ( conn , conn - > our_vnn ) ) {
DEBUG ( 2 , ( " Node is not working, can not connect \n " ) ) ;
2016-07-09 08:48:49 +02:00
return EIO ;
2009-11-16 12:03:24 +01:00
}
2007-06-10 17:02:09 +00:00
generate_random_buffer ( ( unsigned char * ) & conn - > rand_srvid ,
sizeof ( conn - > rand_srvid ) ) ;
2015-10-02 20:42:05 -07:00
ret = register_with_ctdbd ( conn , conn - > rand_srvid , NULL , NULL ) ;
if ( ret ! = 0 ) {
2007-06-10 17:02:09 +00:00
DEBUG ( 5 , ( " Could not register random srvid: %s \n " ,
2015-10-02 20:42:05 -07:00
strerror ( ret ) ) ) ;
2016-07-09 08:48:49 +02:00
return ret ;
}
return 0 ;
}
int ctdbd_init_connection ( TALLOC_CTX * mem_ctx ,
const char * sockname , int timeout ,
struct ctdbd_connection * * pconn )
{
struct ctdbd_connection * conn ;
int ret ;
if ( ! ( conn = talloc_zero ( mem_ctx , struct ctdbd_connection ) ) ) {
DEBUG ( 0 , ( " talloc failed \n " ) ) ;
return ENOMEM ;
}
ret = ctdbd_init_connection_internal ( mem_ctx ,
sockname ,
timeout ,
conn ) ;
if ( ret ! = 0 ) {
DBG_ERR ( " ctdbd_init_connection_internal failed (%s) \n " ,
strerror ( ret ) ) ;
2007-06-10 17:02:09 +00:00
goto fail ;
}
* pconn = conn ;
2015-10-02 20:42:05 -07:00
return 0 ;
2007-06-10 17:02:09 +00:00
fail :
TALLOC_FREE ( conn ) ;
2015-10-02 20:42:05 -07:00
return ret ;
2007-06-10 17:02:09 +00:00
}
2016-07-09 08:59:09 +02:00
int ctdbd_reinit_connection ( TALLOC_CTX * mem_ctx ,
const char * sockname , int timeout ,
struct ctdbd_connection * conn )
{
int ret ;
ret = ctdbd_connection_destructor ( conn ) ;
if ( ret ! = 0 ) {
DBG_ERR ( " ctdbd_connection_destructor failed \n " ) ;
return ret ;
}
ret = ctdbd_init_connection_internal ( mem_ctx ,
sockname ,
timeout ,
conn ) ;
if ( ret ! = 0 ) {
DBG_ERR ( " ctdbd_init_connection_internal failed (%s) \n " ,
strerror ( ret ) ) ;
return ret ;
}
return 0 ;
}
2020-03-12 16:05:58 +01:00
int ctdbd_init_async_connection (
TALLOC_CTX * mem_ctx ,
const char * sockname ,
int timeout ,
struct ctdbd_connection * * pconn )
{
struct ctdbd_connection * conn = NULL ;
int ret ;
2022-06-13 11:32:30 +02:00
* pconn = NULL ;
2020-03-12 16:05:58 +01:00
ret = ctdbd_init_connection ( mem_ctx , sockname , timeout , & conn ) ;
if ( ret ! = 0 ) {
return ret ;
}
ret = set_blocking ( conn - > fd , false ) ;
if ( ret = = - 1 ) {
int err = errno ;
2022-06-13 11:32:30 +02:00
SMB_ASSERT ( err ! = 0 ) ;
2020-03-12 16:05:58 +01:00
TALLOC_FREE ( conn ) ;
return err ;
}
conn - > outgoing = tevent_queue_create ( conn , " ctdb async outgoing " ) ;
if ( conn - > outgoing = = NULL ) {
TALLOC_FREE ( conn ) ;
return ENOMEM ;
}
* pconn = conn ;
return 0 ;
}
2010-01-23 00:05:15 +01:00
int ctdbd_conn_get_fd ( struct ctdbd_connection * conn )
{
2014-05-06 12:21:42 +02:00
return conn - > fd ;
2010-01-23 00:05:15 +01:00
}
2007-06-10 17:02:09 +00:00
/*
* Packet handler to receive and handle a ctdb message
*/
2017-06-01 17:55:06 +02:00
static int ctdb_handle_message ( struct tevent_context * ev ,
struct ctdbd_connection * conn ,
2015-06-19 11:47:01 +02:00
struct ctdb_req_header * hdr )
2007-06-10 17:02:09 +00:00
{
2015-10-29 16:36:30 +11:00
struct ctdb_req_message_old * msg ;
2007-06-10 17:02:09 +00:00
2014-05-06 12:21:42 +02:00
if ( hdr - > operation ! = CTDB_REQ_MESSAGE ) {
2007-06-10 17:02:09 +00:00
DEBUG ( 0 , ( " Received async msg of type %u, discarding \n " ,
2014-05-06 12:21:42 +02:00
hdr - > operation ) ) ;
2015-06-19 11:47:01 +02:00
return EINVAL ;
2007-06-10 17:02:09 +00:00
}
2015-10-29 16:36:30 +11:00
msg = ( struct ctdb_req_message_old * ) hdr ;
2014-05-06 12:21:42 +02:00
2017-06-01 17:55:06 +02:00
ctdbd_msg_call_back ( ev , conn , msg ) ;
2015-05-19 15:07:33 +02:00
2015-06-19 11:47:01 +02:00
return 0 ;
2007-06-10 17:02:09 +00:00
}
2017-06-01 18:00:45 +02:00
void ctdbd_socket_readable ( struct tevent_context * ev ,
struct ctdbd_connection * conn )
2007-06-10 17:02:09 +00:00
{
2015-03-03 08:48:00 +01:00
struct ctdb_req_header * hdr = NULL ;
2015-07-11 12:23:22 +02:00
int ret ;
2007-06-10 17:02:09 +00:00
2015-09-25 14:36:35 -07:00
ret = ctdb_read_packet ( conn - > fd , conn - > timeout , talloc_tos ( ) , & hdr ) ;
2015-07-11 12:23:22 +02:00
if ( ret ! = 0 ) {
2018-10-05 14:04:18 +10:00
DBG_ERR ( " ctdb_read_packet failed: %s \n " , strerror ( ret ) ) ;
cluster_fatal ( " failed to read data from ctdbd \n " ) ;
2007-06-10 17:02:09 +00:00
}
2019-09-03 15:26:19 +02:00
SMB_ASSERT ( hdr ! = NULL ) ;
2007-06-10 17:02:09 +00:00
2017-06-01 18:00:45 +02:00
ret = ctdb_handle_message ( ev , conn , hdr ) ;
2015-06-15 14:47:50 +00:00
TALLOC_FREE ( hdr ) ;
2015-06-19 11:47:01 +02:00
if ( ret ! = 0 ) {
2014-05-06 12:21:42 +02:00
DEBUG ( 10 , ( " could not handle incoming message: %s \n " ,
2015-06-19 11:47:01 +02:00
strerror ( ret ) ) ) ;
2007-06-10 17:02:09 +00:00
}
}
2015-10-02 20:42:05 -07:00
int ctdbd_messaging_send_iov ( struct ctdbd_connection * conn ,
uint32_t dst_vnn , uint64_t dst_srvid ,
const struct iovec * iov , int iovlen )
2012-03-30 11:10:47 +02:00
{
2015-10-29 16:36:30 +11:00
struct ctdb_req_message_old r ;
2015-05-25 08:50:35 +02:00
struct iovec iov2 [ iovlen + 1 ] ;
size_t buflen = iov_buflen ( iov , iovlen ) ;
2014-05-06 12:21:42 +02:00
ssize_t nwritten ;
2012-03-30 11:10:47 +02:00
2015-10-29 16:36:30 +11:00
r . hdr . length = offsetof ( struct ctdb_req_message_old , data ) + buflen ;
2007-06-10 17:02:09 +00:00
r . hdr . ctdb_magic = CTDB_MAGIC ;
2014-10-21 11:53:29 +11:00
r . hdr . ctdb_version = CTDB_PROTOCOL ;
2007-06-10 17:02:09 +00:00
r . hdr . generation = 1 ;
r . hdr . operation = CTDB_REQ_MESSAGE ;
r . hdr . destnode = dst_vnn ;
r . hdr . srcnode = conn - > our_vnn ;
r . hdr . reqid = 0 ;
r . srvid = dst_srvid ;
2012-03-30 11:10:47 +02:00
r . datalen = buflen ;
2007-06-10 17:02:09 +00:00
2007-08-29 11:46:44 +00:00
DEBUG ( 10 , ( " ctdbd_messaging_send: Sending ctdb packet \n " ) ) ;
ctdb_packet_dump ( & r . hdr ) ;
2015-05-25 08:50:35 +02:00
iov2 [ 0 ] . iov_base = & r ;
2015-10-29 16:36:30 +11:00
iov2 [ 0 ] . iov_len = offsetof ( struct ctdb_req_message_old , data ) ;
2015-05-25 08:50:35 +02:00
memcpy ( & iov2 [ 1 ] , iov , iovlen * sizeof ( struct iovec ) ) ;
2007-06-10 17:02:09 +00:00
2015-05-25 08:50:35 +02:00
nwritten = write_data_iov ( conn - > fd , iov2 , iovlen + 1 ) ;
2014-05-06 12:21:42 +02:00
if ( nwritten = = - 1 ) {
DEBUG ( 3 , ( " write_data_iov failed: %s \n " , strerror ( errno ) ) ) ;
2007-06-10 17:02:09 +00:00
cluster_fatal ( " cluster dispatch daemon msg write error \n " ) ;
}
2014-05-06 12:21:42 +02:00
2015-10-02 20:42:05 -07:00
return 0 ;
2007-06-10 17:02:09 +00:00
}
/*
* send / recv a generic ctdb control message
*/
2015-10-02 20:31:52 -07:00
static int ctdbd_control ( struct ctdbd_connection * conn ,
uint32_t vnn , uint32_t opcode ,
uint64_t srvid , uint32_t flags ,
TDB_DATA data ,
TALLOC_CTX * mem_ctx , TDB_DATA * outdata ,
2016-04-21 13:31:39 +02:00
int32_t * cstatus )
2007-06-10 17:02:09 +00:00
{
2015-10-29 16:42:05 +11:00
struct ctdb_req_control_old req ;
2014-05-06 12:21:42 +02:00
struct ctdb_req_header * hdr ;
2015-10-29 16:44:08 +11:00
struct ctdb_reply_control_old * reply = NULL ;
2014-05-06 12:21:42 +02:00
struct iovec iov [ 2 ] ;
ssize_t nwritten ;
2015-06-19 11:47:01 +02:00
int ret ;
2007-06-10 17:02:09 +00:00
2017-01-09 08:17:02 +01:00
if ( ctdbd_conn_has_async_reqs ( conn ) ) {
/*
* Can ' t use sync call while an async call is in flight . Adding
* this check as a safety net . We ' ll be using different
* connections for sync and async requests , so this shouldn ' t
* happen , but who knows . . .
*/
DBG_ERR ( " Async ctdb req on sync connection \n " ) ;
return EINVAL ;
}
2007-06-10 17:02:09 +00:00
ZERO_STRUCT ( req ) ;
2015-10-29 16:42:05 +11:00
req . hdr . length = offsetof ( struct ctdb_req_control_old , data ) + data . dsize ;
2007-06-10 17:02:09 +00:00
req . hdr . ctdb_magic = CTDB_MAGIC ;
2014-10-21 11:53:29 +11:00
req . hdr . ctdb_version = CTDB_PROTOCOL ;
2007-06-10 17:02:09 +00:00
req . hdr . operation = CTDB_REQ_CONTROL ;
2011-10-26 10:56:32 +02:00
req . hdr . reqid = ctdbd_next_reqid ( conn ) ;
2007-06-10 17:02:09 +00:00
req . hdr . destnode = vnn ;
req . opcode = opcode ;
req . srvid = srvid ;
req . datalen = data . dsize ;
2011-03-08 16:26:34 +01:00
req . flags = flags ;
2007-06-10 17:02:09 +00:00
2016-04-21 11:21:26 +02:00
DBG_DEBUG ( " Sending ctdb packet reqid=% " PRIu32 " , vnn=% " PRIu32 " , "
" opcode=% " PRIu32 " , srvid=% " PRIu64 " \n " , req . hdr . reqid ,
req . hdr . destnode , req . opcode , req . srvid ) ;
2007-08-29 11:46:44 +00:00
ctdb_packet_dump ( & req . hdr ) ;
2014-05-06 12:21:42 +02:00
iov [ 0 ] . iov_base = & req ;
2015-10-29 16:42:05 +11:00
iov [ 0 ] . iov_len = offsetof ( struct ctdb_req_control_old , data ) ;
2014-05-06 12:21:42 +02:00
iov [ 1 ] . iov_base = data . dptr ;
iov [ 1 ] . iov_len = data . dsize ;
2007-06-10 17:02:09 +00:00
2014-05-06 12:21:42 +02:00
nwritten = write_data_iov ( conn - > fd , iov , ARRAY_SIZE ( iov ) ) ;
if ( nwritten = = - 1 ) {
DEBUG ( 3 , ( " write_data_iov failed: %s \n " , strerror ( errno ) ) ) ;
cluster_fatal ( " cluster dispatch daemon msg write error \n " ) ;
2007-06-10 17:02:09 +00:00
}
2008-08-07 16:20:05 +10:00
if ( flags & CTDB_CTRL_FLAG_NOREPLY ) {
2010-12-23 16:43:55 +01:00
if ( cstatus ) {
* cstatus = 0 ;
}
2015-10-02 19:54:31 -07:00
return 0 ;
2008-08-07 16:20:05 +10:00
}
2015-06-19 11:47:01 +02:00
ret = ctdb_read_req ( conn , req . hdr . reqid , NULL , & hdr ) ;
if ( ret ! = 0 ) {
DEBUG ( 10 , ( " ctdb_read_req failed: %s \n " , strerror ( ret ) ) ) ;
2015-10-02 19:54:31 -07:00
return ret ;
2007-06-10 17:02:09 +00:00
}
2014-05-06 12:21:42 +02:00
if ( hdr - > operation ! = CTDB_REPLY_CONTROL ) {
2007-06-10 17:02:09 +00:00
DEBUG ( 0 , ( " received invalid reply \n " ) ) ;
2015-10-02 19:54:31 -07:00
TALLOC_FREE ( hdr ) ;
return EIO ;
2007-06-10 17:02:09 +00:00
}
2015-10-29 16:44:08 +11:00
reply = ( struct ctdb_reply_control_old * ) hdr ;
2007-06-10 17:02:09 +00:00
if ( outdata ) {
2015-05-09 16:33:10 -07:00
if ( ! ( outdata - > dptr = ( uint8_t * ) talloc_memdup (
2007-06-10 17:02:09 +00:00
mem_ctx , reply - > data , reply - > datalen ) ) ) {
TALLOC_FREE ( reply ) ;
2015-10-02 19:54:31 -07:00
return ENOMEM ;
2007-06-10 17:02:09 +00:00
}
outdata - > dsize = reply - > datalen ;
}
if ( cstatus ) {
( * cstatus ) = reply - > status ;
}
TALLOC_FREE ( reply ) ;
2015-10-02 19:54:31 -07:00
return ret ;
}
2007-06-10 17:02:09 +00:00
/*
* see if a remote process exists
*/
2017-08-29 13:26:20 +02:00
bool ctdbd_process_exists ( struct ctdbd_connection * conn , uint32_t vnn ,
pid_t pid , uint64_t unique_id )
2007-06-10 17:02:09 +00:00
{
2017-08-29 13:31:07 +02:00
uint8_t buf [ sizeof ( pid ) + sizeof ( unique_id ) ] ;
2016-04-11 17:15:29 +02:00
int32_t cstatus = 0 ;
int ret ;
2007-06-10 17:02:09 +00:00
2017-08-29 13:31:07 +02:00
if ( unique_id = = SERVERID_UNIQUE_ID_NOT_TO_VERIFY ) {
ret = ctdbd_control ( conn , vnn , CTDB_CONTROL_PROCESS_EXISTS ,
0 , 0 ,
( TDB_DATA ) { . dptr = ( uint8_t * ) & pid ,
. dsize = sizeof ( pid ) } ,
NULL , NULL , & cstatus ) ;
if ( ret ! = 0 ) {
return false ;
}
return ( cstatus = = 0 ) ;
}
memcpy ( buf , & pid , sizeof ( pid ) ) ;
memcpy ( buf + sizeof ( pid ) , & unique_id , sizeof ( unique_id ) ) ;
ret = ctdbd_control ( conn , vnn , CTDB_CONTROL_CHECK_PID_SRVID , 0 , 0 ,
( TDB_DATA ) { . dptr = buf , . dsize = sizeof ( buf ) } ,
2016-04-11 17:15:29 +02:00
NULL , NULL , & cstatus ) ;
if ( ret ! = 0 ) {
2011-10-26 11:36:21 +02:00
return false ;
2007-06-10 17:02:09 +00:00
}
2016-04-11 17:15:29 +02:00
return ( cstatus = = 0 ) ;
2011-10-23 21:38:54 +02:00
}
2007-06-10 17:02:09 +00:00
/*
* Get a db path
*/
char * ctdbd_dbpath ( struct ctdbd_connection * conn ,
TALLOC_CTX * mem_ctx , uint32_t db_id )
{
2015-10-02 20:08:53 -07:00
int ret ;
2007-06-10 17:02:09 +00:00
TDB_DATA data ;
2015-03-04 09:43:19 +01:00
TDB_DATA rdata = { 0 } ;
2015-03-03 08:48:00 +01:00
int32_t cstatus = 0 ;
2007-06-10 17:02:09 +00:00
data . dptr = ( uint8_t * ) & db_id ;
data . dsize = sizeof ( db_id ) ;
2016-04-13 18:47:46 +02:00
ret = ctdbd_control_local ( conn , CTDB_CONTROL_GETDBPATH , 0 , 0 , data ,
mem_ctx , & rdata , & cstatus ) ;
2015-10-02 20:08:53 -07:00
if ( ( ret ! = 0 ) | | cstatus ! = 0 ) {
DEBUG ( 0 , ( __location__ " ctdb_control for getdbpath failed: %s \n " ,
strerror ( ret ) ) ) ;
2020-03-30 12:29:38 +02:00
TALLOC_FREE ( rdata . dptr ) ;
2007-06-10 17:02:09 +00:00
}
2015-03-04 09:43:19 +01:00
return ( char * ) rdata . dptr ;
2007-06-10 17:02:09 +00:00
}
/*
* attach to a ctdb database
*/
2015-10-02 20:42:05 -07:00
int ctdbd_db_attach ( struct ctdbd_connection * conn ,
2017-07-11 20:41:43 +02:00
const char * name , uint32_t * db_id , bool persistent )
2007-06-10 17:02:09 +00:00
{
2015-10-02 20:08:53 -07:00
int ret ;
2020-03-30 12:34:12 +02:00
TDB_DATA data = { 0 } ;
2007-06-10 17:02:09 +00:00
int32_t cstatus ;
2012-06-20 11:47:53 +02:00
data = string_term_tdb_data ( name ) ;
2007-06-10 17:02:09 +00:00
2016-04-13 18:47:46 +02:00
ret = ctdbd_control_local ( conn ,
persistent
? CTDB_CONTROL_DB_ATTACH_PERSISTENT
: CTDB_CONTROL_DB_ATTACH ,
2017-07-11 00:38:59 +10:00
0 , 0 , data , NULL , & data , & cstatus ) ;
2015-10-02 20:08:53 -07:00
if ( ret ! = 0 ) {
2007-06-10 17:02:09 +00:00
DEBUG ( 0 , ( __location__ " ctdb_control for db_attach "
2015-10-02 20:08:53 -07:00
" failed: %s \n " , strerror ( ret ) ) ) ;
2015-10-02 20:42:05 -07:00
return ret ;
2007-06-10 17:02:09 +00:00
}
if ( cstatus ! = 0 | | data . dsize ! = sizeof ( uint32_t ) ) {
DEBUG ( 0 , ( __location__ " ctdb_control for db_attach failed \n " ) ) ;
2020-03-30 12:34:12 +02:00
TALLOC_FREE ( data . dptr ) ;
2015-10-02 20:42:05 -07:00
return EIO ;
2007-06-10 17:02:09 +00:00
}
* db_id = * ( uint32_t * ) data . dptr ;
talloc_free ( data . dptr ) ;
2015-10-02 20:42:05 -07:00
return 0 ;
2007-06-10 17:02:09 +00:00
}
/*
* force the migration of a record to this node
*/
2015-10-02 20:42:05 -07:00
int ctdbd_migrate ( struct ctdbd_connection * conn , uint32_t db_id , TDB_DATA key )
2007-06-10 17:02:09 +00:00
{
2015-10-29 16:26:29 +11:00
struct ctdb_req_call_old req ;
2016-10-10 16:26:05 +02:00
struct ctdb_req_header * hdr = NULL ;
2014-05-06 12:21:42 +02:00
struct iovec iov [ 2 ] ;
ssize_t nwritten ;
2015-06-19 11:47:01 +02:00
int ret ;
2007-06-10 17:02:09 +00:00
2017-01-09 08:17:02 +01:00
if ( ctdbd_conn_has_async_reqs ( conn ) ) {
/*
* Can ' t use sync call while an async call is in flight . Adding
* this check as a safety net . We ' ll be using different
* connections for sync and async requests , so this shouldn ' t
* happen , but who knows . . .
*/
DBG_ERR ( " Async ctdb req on sync connection \n " ) ;
return EINVAL ;
}
2007-06-10 17:02:09 +00:00
ZERO_STRUCT ( req ) ;
2009-08-07 12:09:21 +02:00
2015-10-29 16:26:29 +11:00
req . hdr . length = offsetof ( struct ctdb_req_call_old , data ) + key . dsize ;
2007-06-10 17:02:09 +00:00
req . hdr . ctdb_magic = CTDB_MAGIC ;
2014-10-21 11:53:29 +11:00
req . hdr . ctdb_version = CTDB_PROTOCOL ;
2007-06-10 17:02:09 +00:00
req . hdr . operation = CTDB_REQ_CALL ;
2011-10-26 10:56:32 +02:00
req . hdr . reqid = ctdbd_next_reqid ( conn ) ;
2007-06-10 17:02:09 +00:00
req . flags = CTDB_IMMEDIATE_MIGRATION ;
req . callid = CTDB_NULL_FUNC ;
req . db_id = db_id ;
req . keylen = key . dsize ;
2007-08-29 11:46:44 +00:00
DEBUG ( 10 , ( " ctdbd_migrate: Sending ctdb packet \n " ) ) ;
ctdb_packet_dump ( & req . hdr ) ;
2014-05-06 12:21:42 +02:00
iov [ 0 ] . iov_base = & req ;
2015-10-29 16:26:29 +11:00
iov [ 0 ] . iov_len = offsetof ( struct ctdb_req_call_old , data ) ;
2014-05-06 12:21:42 +02:00
iov [ 1 ] . iov_base = key . dptr ;
iov [ 1 ] . iov_len = key . dsize ;
2007-06-10 17:02:09 +00:00
2014-05-06 12:21:42 +02:00
nwritten = write_data_iov ( conn - > fd , iov , ARRAY_SIZE ( iov ) ) ;
if ( nwritten = = - 1 ) {
DEBUG ( 3 , ( " write_data_iov failed: %s \n " , strerror ( errno ) ) ) ;
cluster_fatal ( " cluster dispatch daemon msg write error \n " ) ;
2007-06-10 17:02:09 +00:00
}
2015-06-19 11:47:01 +02:00
ret = ctdb_read_req ( conn , req . hdr . reqid , NULL , & hdr ) ;
if ( ret ! = 0 ) {
DEBUG ( 10 , ( " ctdb_read_req failed: %s \n " , strerror ( ret ) ) ) ;
2007-06-10 17:02:09 +00:00
goto fail ;
}
2014-05-06 12:21:42 +02:00
if ( hdr - > operation ! = CTDB_REPLY_CALL ) {
2016-06-24 19:22:02 +10:00
if ( hdr - > operation = = CTDB_REPLY_ERROR ) {
DBG_ERR ( " received error from ctdb \n " ) ;
} else {
DBG_ERR ( " received invalid reply \n " ) ;
}
ret = EIO ;
2007-06-10 17:02:09 +00:00
goto fail ;
}
fail :
2014-05-06 12:21:42 +02:00
TALLOC_FREE ( hdr ) ;
2015-10-02 20:42:05 -07:00
return ret ;
2007-06-10 17:02:09 +00:00
}
2012-11-23 17:54:57 +01:00
/*
* Fetch a record and parse it
*/
2015-10-02 20:42:05 -07:00
int ctdbd_parse ( struct ctdbd_connection * conn , uint32_t db_id ,
TDB_DATA key , bool local_copy ,
void ( * parser ) ( TDB_DATA key , TDB_DATA data ,
void * private_data ) ,
void * private_data )
2012-11-23 17:54:57 +01:00
{
2015-10-29 16:26:29 +11:00
struct ctdb_req_call_old req ;
2014-05-06 12:21:42 +02:00
struct ctdb_req_header * hdr = NULL ;
2015-10-29 16:29:01 +11:00
struct ctdb_reply_call_old * reply ;
2014-05-06 12:21:42 +02:00
struct iovec iov [ 2 ] ;
ssize_t nwritten ;
2012-11-23 17:54:57 +01:00
uint32_t flags ;
2015-06-19 11:47:01 +02:00
int ret ;
2012-11-23 17:54:57 +01:00
2017-01-09 08:17:02 +01:00
if ( ctdbd_conn_has_async_reqs ( conn ) ) {
/*
* Can ' t use sync call while an async call is in flight . Adding
* this check as a safety net . We ' ll be using different
* connections for sync and async requests , so this shouldn ' t
* happen , but who knows . . .
*/
DBG_ERR ( " Async ctdb req on sync connection \n " ) ;
return EINVAL ;
}
2012-11-23 17:54:57 +01:00
flags = local_copy ? CTDB_WANT_READONLY : 0 ;
ZERO_STRUCT ( req ) ;
2015-10-29 16:26:29 +11:00
req . hdr . length = offsetof ( struct ctdb_req_call_old , data ) + key . dsize ;
2012-11-23 17:54:57 +01:00
req . hdr . ctdb_magic = CTDB_MAGIC ;
2014-10-21 11:53:29 +11:00
req . hdr . ctdb_version = CTDB_PROTOCOL ;
2012-11-23 17:54:57 +01:00
req . hdr . operation = CTDB_REQ_CALL ;
req . hdr . reqid = ctdbd_next_reqid ( conn ) ;
req . flags = flags ;
req . callid = CTDB_FETCH_FUNC ;
req . db_id = db_id ;
req . keylen = key . dsize ;
2014-05-06 12:21:42 +02:00
iov [ 0 ] . iov_base = & req ;
2015-10-29 16:26:29 +11:00
iov [ 0 ] . iov_len = offsetof ( struct ctdb_req_call_old , data ) ;
2014-05-06 12:21:42 +02:00
iov [ 1 ] . iov_base = key . dptr ;
iov [ 1 ] . iov_len = key . dsize ;
2012-11-23 17:54:57 +01:00
2014-05-06 12:21:42 +02:00
nwritten = write_data_iov ( conn - > fd , iov , ARRAY_SIZE ( iov ) ) ;
if ( nwritten = = - 1 ) {
DEBUG ( 3 , ( " write_data_iov failed: %s \n " , strerror ( errno ) ) ) ;
cluster_fatal ( " cluster dispatch daemon msg write error \n " ) ;
2012-11-23 17:54:57 +01:00
}
2015-06-19 11:47:01 +02:00
ret = ctdb_read_req ( conn , req . hdr . reqid , NULL , & hdr ) ;
if ( ret ! = 0 ) {
DEBUG ( 10 , ( " ctdb_read_req failed: %s \n " , strerror ( ret ) ) ) ;
2012-11-23 17:54:57 +01:00
goto fail ;
}
2015-04-23 18:06:17 +02:00
if ( ( hdr = = NULL ) | | ( hdr - > operation ! = CTDB_REPLY_CALL ) ) {
2012-11-23 17:54:57 +01:00
DEBUG ( 0 , ( " received invalid reply \n " ) ) ;
2015-10-02 20:42:05 -07:00
ret = EIO ;
2012-11-23 17:54:57 +01:00
goto fail ;
}
2015-10-29 16:29:01 +11:00
reply = ( struct ctdb_reply_call_old * ) hdr ;
2012-11-23 17:54:57 +01:00
2013-08-28 11:34:08 +00:00
if ( reply - > datalen = = 0 ) {
/*
* Treat an empty record as non - existing
*/
2015-10-02 20:42:05 -07:00
ret = ENOENT ;
2013-08-28 11:34:08 +00:00
goto fail ;
}
2012-11-23 17:54:57 +01:00
parser ( key , make_tdb_data ( & reply - > data [ 0 ] , reply - > datalen ) ,
private_data ) ;
2015-10-02 20:42:05 -07:00
ret = 0 ;
2012-11-23 17:54:57 +01:00
fail :
2014-05-06 12:21:42 +02:00
TALLOC_FREE ( hdr ) ;
2015-10-02 20:42:05 -07:00
return ret ;
2007-06-10 17:02:09 +00:00
}
/*
2016-04-05 17:30:11 +02:00
Traverse a ctdb database . " conn " must be an otherwise unused
ctdb_connection where no other messages but the traverse ones are
expected .
2007-06-10 17:02:09 +00:00
*/
2016-04-05 17:30:11 +02:00
int ctdbd_traverse ( struct ctdbd_connection * conn , uint32_t db_id ,
2007-06-10 17:02:09 +00:00
void ( * fn ) ( TDB_DATA key , TDB_DATA data ,
void * private_data ) ,
void * private_data )
{
2015-10-02 20:08:53 -07:00
int ret ;
2014-05-06 12:21:42 +02:00
TDB_DATA key , data ;
2007-06-10 17:02:09 +00:00
struct ctdb_traverse_start t ;
2019-07-09 16:05:37 +00:00
int32_t cstatus = 0 ;
2007-06-10 17:02:09 +00:00
2017-01-09 08:17:02 +01:00
if ( ctdbd_conn_has_async_reqs ( conn ) ) {
/*
* Can ' t use sync call while an async call is in flight . Adding
* this check as a safety net . We ' ll be using different
* connections for sync and async requests , so this shouldn ' t
* happen , but who knows . . .
*/
DBG_ERR ( " Async ctdb req on sync connection \n " ) ;
return EINVAL ;
}
2007-06-10 17:02:09 +00:00
t . db_id = db_id ;
t . srvid = conn - > rand_srvid ;
2011-10-26 10:56:32 +02:00
t . reqid = ctdbd_next_reqid ( conn ) ;
2007-06-10 17:02:09 +00:00
data . dptr = ( uint8_t * ) & t ;
data . dsize = sizeof ( t ) ;
2016-04-13 18:47:46 +02:00
ret = ctdbd_control_local ( conn , CTDB_CONTROL_TRAVERSE_START ,
conn - > rand_srvid ,
0 , data , NULL , NULL , & cstatus ) ;
2007-06-10 17:02:09 +00:00
2015-10-02 20:08:53 -07:00
if ( ( ret ! = 0 ) | | ( cstatus ! = 0 ) ) {
DEBUG ( 0 , ( " ctdbd_control failed: %s, %d \n " , strerror ( ret ) ,
2007-06-10 17:02:09 +00:00
cstatus ) ) ;
2015-10-02 20:42:05 -07:00
if ( ret = = 0 ) {
2007-06-10 17:02:09 +00:00
/*
* We need a mapping here
*/
2015-10-02 20:42:05 -07:00
ret = EIO ;
2007-06-10 17:02:09 +00:00
}
2015-10-02 20:42:05 -07:00
return ret ;
2007-06-10 17:02:09 +00:00
}
2016-04-08 16:14:33 +02:00
while ( true ) {
2015-03-03 08:48:00 +01:00
struct ctdb_req_header * hdr = NULL ;
2015-10-29 16:36:30 +11:00
struct ctdb_req_message_old * m ;
2015-10-29 17:30:30 +11:00
struct ctdb_rec_data_old * d ;
2007-06-10 17:02:09 +00:00
2015-09-25 14:36:35 -07:00
ret = ctdb_read_packet ( conn - > fd , conn - > timeout , conn , & hdr ) ;
2015-07-11 12:23:22 +02:00
if ( ret ! = 0 ) {
2018-10-05 14:04:18 +10:00
DBG_ERR ( " ctdb_read_packet failed: %s \n " , strerror ( ret ) ) ;
cluster_fatal ( " failed to read data from ctdbd \n " ) ;
2014-05-06 12:21:42 +02:00
}
2019-09-03 15:26:19 +02:00
SMB_ASSERT ( hdr ! = NULL ) ;
2007-06-10 17:02:09 +00:00
2014-05-06 12:21:42 +02:00
if ( hdr - > operation ! = CTDB_REQ_MESSAGE ) {
DEBUG ( 0 , ( " Got operation %u, expected a message \n " ,
( unsigned ) hdr - > operation ) ) ;
2015-10-02 20:42:05 -07:00
return EIO ;
2007-06-10 17:02:09 +00:00
}
2015-10-29 16:36:30 +11:00
m = ( struct ctdb_req_message_old * ) hdr ;
2015-10-29 17:30:30 +11:00
d = ( struct ctdb_rec_data_old * ) & m - > data [ 0 ] ;
2014-05-06 12:21:42 +02:00
if ( m - > datalen < sizeof ( uint32_t ) | | m - > datalen ! = d - > length ) {
DEBUG ( 0 , ( " Got invalid traverse data of length %d \n " ,
( int ) m - > datalen ) ) ;
2015-10-02 20:42:05 -07:00
return EIO ;
2007-06-10 17:02:09 +00:00
}
2014-05-06 12:21:42 +02:00
key . dsize = d - > keylen ;
key . dptr = & d - > data [ 0 ] ;
data . dsize = d - > datalen ;
data . dptr = & d - > data [ d - > keylen ] ;
2007-06-10 17:02:09 +00:00
2014-05-06 12:21:42 +02:00
if ( key . dsize = = 0 & & data . dsize = = 0 ) {
/* end of traverse */
2015-10-02 20:42:05 -07:00
return 0 ;
2007-07-13 10:40:53 +00:00
}
2014-05-06 12:21:42 +02:00
if ( data . dsize < sizeof ( struct ctdb_ltdb_header ) ) {
DEBUG ( 0 , ( " Got invalid ltdb header length %d \n " ,
( int ) data . dsize ) ) ;
2015-10-02 20:42:05 -07:00
return EIO ;
2007-06-10 17:02:09 +00:00
}
2014-05-06 12:21:42 +02:00
data . dsize - = sizeof ( struct ctdb_ltdb_header ) ;
data . dptr + = sizeof ( struct ctdb_ltdb_header ) ;
2007-06-10 17:02:09 +00:00
2014-05-06 12:21:42 +02:00
if ( fn ! = NULL ) {
fn ( key , data , private_data ) ;
2007-06-10 17:02:09 +00:00
}
}
2015-10-02 20:42:05 -07:00
return 0 ;
2007-06-10 17:02:09 +00:00
}
2009-01-28 18:55:13 +01:00
/*
This is used to canonicalize a ctdb_sock_addr structure .
*/
static void smbd_ctdb_canonicalize_ip ( const struct sockaddr_storage * in ,
struct sockaddr_storage * out )
{
memcpy ( out , in , sizeof ( * out ) ) ;
# ifdef HAVE_IPV6
if ( in - > ss_family = = AF_INET6 ) {
const char prefix [ 12 ] = { 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0xff , 0xff } ;
2012-06-20 11:47:32 +02:00
const struct sockaddr_in6 * in6 =
( const struct sockaddr_in6 * ) in ;
2009-01-28 18:55:13 +01:00
struct sockaddr_in * out4 = ( struct sockaddr_in * ) out ;
if ( memcmp ( & in6 - > sin6_addr , prefix , 12 ) = = 0 ) {
memset ( out , 0 , sizeof ( * out ) ) ;
# ifdef HAVE_SOCK_SIN_LEN
out4 - > sin_len = sizeof ( * out ) ;
# endif
out4 - > sin_family = AF_INET ;
out4 - > sin_port = in6 - > sin6_port ;
2012-02-06 16:53:22 +01:00
memcpy ( & out4 - > sin_addr , & in6 - > sin6_addr . s6_addr [ 12 ] , 4 ) ;
2009-01-28 18:55:13 +01:00
}
}
# endif
}
2007-06-10 17:02:09 +00:00
/*
* Register us as a server for a particular tcp connection
*/
2015-10-02 20:42:05 -07:00
int ctdbd_register_ips ( struct ctdbd_connection * conn ,
const struct sockaddr_storage * _server ,
const struct sockaddr_storage * _client ,
2017-06-01 17:45:47 +02:00
int ( * cb ) ( struct tevent_context * ev ,
uint32_t src_vnn , uint32_t dst_vnn ,
2015-10-02 20:42:05 -07:00
uint64_t dst_srvid ,
const uint8_t * msg , size_t msglen ,
void * private_data ) ,
void * private_data )
2007-06-10 17:02:09 +00:00
{
2015-10-29 14:25:34 +11:00
struct ctdb_connection p ;
2015-05-20 08:12:46 +02:00
TDB_DATA data = { . dptr = ( uint8_t * ) & p , . dsize = sizeof ( p ) } ;
2015-10-02 20:08:53 -07:00
int ret ;
2009-01-28 18:55:13 +01:00
struct sockaddr_storage client ;
struct sockaddr_storage server ;
2007-06-10 17:02:09 +00:00
/*
* Only one connection so far
*/
2009-01-28 18:55:13 +01:00
smbd_ctdb_canonicalize_ip ( _client , & client ) ;
smbd_ctdb_canonicalize_ip ( _server , & server ) ;
switch ( client . ss_family ) {
2008-12-18 15:02:42 +01:00
case AF_INET :
2015-10-29 14:25:34 +11:00
memcpy ( & p . dst . ip , & server , sizeof ( p . dst . ip ) ) ;
2015-03-23 17:32:34 +11:00
memcpy ( & p . src . ip , & client , sizeof ( p . src . ip ) ) ;
2008-12-18 15:02:42 +01:00
break ;
case AF_INET6 :
2015-10-29 14:25:34 +11:00
memcpy ( & p . dst . ip6 , & server , sizeof ( p . dst . ip6 ) ) ;
2011-10-21 12:04:59 +02:00
memcpy ( & p . src . ip6 , & client , sizeof ( p . src . ip6 ) ) ;
2008-12-18 15:02:42 +01:00
break ;
default :
2015-10-02 20:42:05 -07:00
return EIO ;
2008-12-18 15:02:42 +01:00
}
2007-06-10 17:02:09 +00:00
/*
* We want to be told about IP releases
*/
2015-10-02 20:42:05 -07:00
ret = register_with_ctdbd ( conn , CTDB_SRVID_RELEASE_IP ,
cb , private_data ) ;
if ( ret ! = 0 ) {
2015-10-02 20:42:05 -07:00
return ret ;
2007-06-10 17:02:09 +00:00
}
/*
* inform ctdb of our tcp connection , so if IP takeover happens ctdb
* can send an extra ack to trigger a reset for our client , so it
* immediately reconnects
*/
2020-06-25 14:00:27 +02:00
ret = ctdbd_control_local ( conn ,
CTDB_CONTROL_TCP_CLIENT , 0 ,
CTDB_CTRL_FLAG_NOREPLY , data , NULL , NULL ,
NULL ) ;
2015-10-02 20:08:53 -07:00
if ( ret ! = 0 ) {
2015-10-02 20:42:05 -07:00
return ret ;
2015-10-02 20:08:53 -07:00
}
2015-10-02 20:42:05 -07:00
return 0 ;
2007-06-10 17:02:09 +00:00
}
2020-09-07 00:17:11 +02:00
static int ctdbd_control_get_public_ips ( struct ctdbd_connection * conn ,
uint32_t flags ,
TALLOC_CTX * mem_ctx ,
struct ctdb_public_ip_list_old * * _ips )
2020-06-25 15:14:04 +02:00
{
struct ctdb_public_ip_list_old * ips = NULL ;
TDB_DATA outdata ;
int32_t cstatus = - 1 ;
size_t min_dsize ;
size_t max_ips ;
int ret ;
* _ips = NULL ;
ret = ctdbd_control_local ( conn ,
CTDB_CONTROL_GET_PUBLIC_IPS ,
0 , /* srvid */
flags ,
tdb_null , /* indata */
mem_ctx ,
& outdata ,
& cstatus ) ;
if ( ret ! = 0 | | cstatus ! = 0 ) {
DBG_ERR ( " ctdb_control for getpublicips failed ret:%d cstatus:%d \n " ,
ret , ( int ) cstatus ) ;
return - 1 ;
}
min_dsize = offsetof ( struct ctdb_public_ip_list_old , ips ) ;
if ( outdata . dsize < min_dsize ) {
DBG_ERR ( " outdata.dsize=%zu < min_dsize=%zu \n " ,
outdata . dsize , min_dsize ) ;
return - 1 ;
}
max_ips = ( outdata . dsize - min_dsize ) / sizeof ( struct ctdb_public_ip ) ;
ips = ( struct ctdb_public_ip_list_old * ) outdata . dptr ;
if ( ( size_t ) ips - > num > max_ips ) {
DBG_ERR ( " ips->num=%zu > max_ips=%zu \n " ,
( size_t ) ips - > num , max_ips ) ;
return - 1 ;
}
* _ips = ips ;
return 0 ;
}
2020-09-07 00:17:11 +02:00
int ctdbd_public_ip_foreach ( struct ctdbd_connection * conn ,
int ( * cb ) ( uint32_t total_ip_count ,
const struct sockaddr_storage * ip ,
bool is_movable_ip ,
void * private_data ) ,
void * private_data )
2020-06-25 15:14:04 +02:00
{
uint32_t i ;
2020-09-07 00:17:11 +02:00
struct ctdb_public_ip_list_old * ips = NULL ;
int ret = ENOMEM ;
TALLOC_CTX * frame = talloc_stackframe ( ) ;
ret = ctdbd_control_get_public_ips ( conn , 0 , frame , & ips ) ;
if ( ret < 0 ) {
ret = EIO ;
goto out_free ;
}
2020-06-25 15:14:04 +02:00
for ( i = 0 ; i < ips - > num ; i + + ) {
struct samba_sockaddr tmp = {
. u = {
2020-09-07 00:17:11 +02:00
. sa = ips - > ips [ i ] . addr . sa ,
2020-06-25 15:14:04 +02:00
} ,
} ;
2020-09-07 00:17:11 +02:00
ret = cb ( ips - > num ,
& tmp . u . ss ,
true , /* all ctdb public ips are movable */
private_data ) ;
if ( ret ! = 0 ) {
goto out_free ;
2020-06-25 15:14:04 +02:00
}
}
2020-09-07 00:17:11 +02:00
ret = 0 ;
out_free :
TALLOC_FREE ( frame ) ;
return ret ;
2020-06-25 15:14:04 +02:00
}
2008-08-07 16:20:05 +10:00
/*
call a control on the local node
*/
2015-10-02 20:42:05 -07:00
int ctdbd_control_local ( struct ctdbd_connection * conn , uint32_t opcode ,
uint64_t srvid , uint32_t flags , TDB_DATA data ,
TALLOC_CTX * mem_ctx , TDB_DATA * outdata ,
2016-04-21 13:31:39 +02:00
int32_t * cstatus )
2008-08-07 16:20:05 +10:00
{
2015-10-02 20:42:05 -07:00
return ctdbd_control ( conn , CTDB_CURRENT_NODE , opcode , srvid , flags , data ,
mem_ctx , outdata , cstatus ) ;
2008-08-07 16:20:05 +10:00
}
2015-10-02 20:42:05 -07:00
int ctdb_watch_us ( struct ctdbd_connection * conn )
2009-10-25 16:12:12 +01:00
{
2015-10-28 17:43:20 +11:00
struct ctdb_notify_data_old reg_data ;
2009-10-25 16:12:12 +01:00
size_t struct_len ;
2015-10-02 20:42:05 -07:00
int ret ;
2016-04-21 13:31:39 +02:00
int32_t cstatus ;
2009-10-25 16:12:12 +01:00
reg_data . srvid = CTDB_SRVID_SAMBA_NOTIFY ;
reg_data . len = 1 ;
reg_data . notify_data [ 0 ] = 0 ;
2015-10-28 17:43:20 +11:00
struct_len = offsetof ( struct ctdb_notify_data_old ,
2009-10-25 16:12:12 +01:00
notify_data ) + reg_data . len ;
2015-10-02 20:42:05 -07:00
ret = ctdbd_control_local (
2009-10-25 16:12:12 +01:00
conn , CTDB_CONTROL_REGISTER_NOTIFY , conn - > rand_srvid , 0 ,
make_tdb_data ( ( uint8_t * ) & reg_data , struct_len ) ,
NULL , NULL , & cstatus ) ;
2015-10-02 20:42:05 -07:00
if ( ret ! = 0 ) {
2009-10-25 16:12:12 +01:00
DEBUG ( 1 , ( " ctdbd_control_local failed: %s \n " ,
2015-10-02 20:42:05 -07:00
strerror ( ret ) ) ) ;
2009-10-25 16:12:12 +01:00
}
2015-10-02 20:42:05 -07:00
return ret ;
2009-10-25 16:12:12 +01:00
}
2015-10-02 20:42:05 -07:00
int ctdb_unwatch ( struct ctdbd_connection * conn )
2009-10-25 16:12:12 +01:00
{
2015-10-28 17:47:03 +11:00
uint64_t srvid = CTDB_SRVID_SAMBA_NOTIFY ;
2015-10-02 20:42:05 -07:00
int ret ;
2016-04-21 13:31:39 +02:00
int32_t cstatus ;
2009-10-25 16:12:12 +01:00
2015-10-02 20:42:05 -07:00
ret = ctdbd_control_local (
2009-10-25 16:12:12 +01:00
conn , CTDB_CONTROL_DEREGISTER_NOTIFY , conn - > rand_srvid , 0 ,
2015-10-28 17:47:03 +11:00
make_tdb_data ( ( uint8_t * ) & srvid , sizeof ( srvid ) ) ,
2009-10-25 16:12:12 +01:00
NULL , NULL , & cstatus ) ;
2015-10-02 20:42:05 -07:00
if ( ret ! = 0 ) {
2009-10-25 16:12:12 +01:00
DEBUG ( 1 , ( " ctdbd_control_local failed: %s \n " ,
2015-10-02 20:42:05 -07:00
strerror ( ret ) ) ) ;
2009-10-25 16:12:12 +01:00
}
2015-10-02 20:42:05 -07:00
return ret ;
2009-10-25 16:12:12 +01:00
}
2015-10-02 20:42:05 -07:00
int ctdbd_probe ( const char * sockname , int timeout )
2013-01-31 11:15:09 +01:00
{
/*
* Do a very early check if ctdbd is around to avoid an abort and core
* later
*/
struct ctdbd_connection * conn = NULL ;
2015-10-02 20:42:05 -07:00
int ret ;
2013-01-31 11:15:09 +01:00
2016-04-05 09:52:01 +02:00
ret = ctdbd_init_connection ( talloc_tos ( ) , sockname , timeout ,
& conn ) ;
2013-01-31 11:15:09 +01:00
/*
* We only care if we can connect .
*/
TALLOC_FREE ( conn ) ;
2015-10-02 20:42:05 -07:00
return ret ;
2013-01-31 11:15:09 +01:00
}
2017-01-09 08:17:02 +01:00
static int ctdbd_connection_destructor ( struct ctdbd_connection * c )
{
if ( c - > fd ! = - 1 ) {
close ( c - > fd ) ;
c - > fd = - 1 ;
}
return 0 ;
}
2020-03-20 13:46:13 +01:00
void ctdbd_prep_hdr_next_reqid (
struct ctdbd_connection * conn , struct ctdb_req_header * hdr )
{
* hdr = ( struct ctdb_req_header ) {
. ctdb_magic = CTDB_MAGIC ,
. ctdb_version = CTDB_PROTOCOL ,
. reqid = ctdbd_next_reqid ( conn ) ,
. destnode = CTDB_CURRENT_NODE ,
} ;
}
2020-03-20 13:58:21 +01:00
struct ctdbd_pkt_read_state {
uint8_t * pkt ;
} ;
static ssize_t ctdbd_pkt_read_more (
uint8_t * buf , size_t buflen , void * private_data ) ;
static void ctdbd_pkt_read_done ( struct tevent_req * subreq ) ;
static struct tevent_req * ctdbd_pkt_read_send (
TALLOC_CTX * mem_ctx , struct tevent_context * ev , int fd )
{
struct tevent_req * req = NULL , * subreq = NULL ;
struct ctdbd_pkt_read_state * state = NULL ;
req = tevent_req_create ( mem_ctx , & state , struct ctdbd_pkt_read_state ) ;
if ( req = = NULL ) {
return NULL ;
}
subreq = read_packet_send ( state , ev , fd , 4 , ctdbd_pkt_read_more , NULL ) ;
if ( tevent_req_nomem ( subreq , req ) ) {
return tevent_req_post ( req , ev ) ;
}
tevent_req_set_callback ( subreq , ctdbd_pkt_read_done , req ) ;
return req ;
}
static ssize_t ctdbd_pkt_read_more (
uint8_t * buf , size_t buflen , void * private_data )
{
uint32_t msglen ;
if ( buflen < 4 ) {
return - 1 ;
}
if ( buflen > 4 ) {
return 0 ; /* Been here, done */
}
memcpy ( & msglen , buf , 4 ) ;
if ( msglen < sizeof ( struct ctdb_req_header ) ) {
return - 1 ;
}
return msglen - sizeof ( msglen ) ;
}
static void ctdbd_pkt_read_done ( struct tevent_req * subreq )
{
struct tevent_req * req = tevent_req_callback_data (
subreq , struct tevent_req ) ;
struct ctdbd_pkt_read_state * state = tevent_req_data (
req , struct ctdbd_pkt_read_state ) ;
ssize_t nread ;
int err ;
nread = read_packet_recv ( subreq , state , & state - > pkt , & err ) ;
TALLOC_FREE ( subreq ) ;
if ( nread = = - 1 ) {
tevent_req_error ( req , err ) ;
return ;
}
tevent_req_done ( req ) ;
}
static int ctdbd_pkt_read_recv (
struct tevent_req * req , TALLOC_CTX * mem_ctx , uint8_t * * pkt )
{
struct ctdbd_pkt_read_state * state = tevent_req_data (
req , struct ctdbd_pkt_read_state ) ;
int err ;
if ( tevent_req_is_unix_error ( req , & err ) ) {
return err ;
}
* pkt = talloc_move ( mem_ctx , & state - > pkt ) ;
tevent_req_received ( req ) ;
return 0 ;
}
static bool ctdbd_conn_receive_next ( struct ctdbd_connection * conn ) ;
static void ctdbd_conn_received ( struct tevent_req * subreq ) ;
struct ctdbd_req_state {
struct ctdbd_connection * conn ;
struct tevent_context * ev ;
uint32_t reqid ;
struct ctdb_req_header * reply ;
} ;
static void ctdbd_req_unset_pending ( struct tevent_req * req )
{
struct ctdbd_req_state * state = tevent_req_data (
req , struct ctdbd_req_state ) ;
struct ctdbd_connection * conn = state - > conn ;
size_t num_pending = talloc_array_length ( conn - > pending ) ;
size_t i , num_after ;
tevent_req_set_cleanup_fn ( req , NULL ) ;
if ( num_pending = = 1 ) {
/*
* conn - > read_req is a child of conn - > pending
*/
TALLOC_FREE ( conn - > pending ) ;
conn - > read_req = NULL ;
return ;
}
for ( i = 0 ; i < num_pending ; i + + ) {
if ( req = = conn - > pending [ i ] ) {
break ;
}
}
if ( i = = num_pending ) {
/*
* Something ' s seriously broken . Just returning here is the
* right thing nevertheless , the point of this routine is to
* remove ourselves from conn - > pending .
*/
return ;
}
num_after = num_pending - i - 1 ;
if ( num_after > 0 ) {
memmove ( & conn - > pending [ i ] ,
& conn - > pending [ i ] + 1 ,
sizeof ( * conn - > pending ) * num_after ) ;
}
conn - > pending = talloc_realloc (
NULL , conn - > pending , struct tevent_req * , num_pending - 1 ) ;
}
static void ctdbd_req_cleanup (
struct tevent_req * req , enum tevent_req_state req_state )
{
ctdbd_req_unset_pending ( req ) ;
}
static bool ctdbd_req_set_pending ( struct tevent_req * req )
{
struct ctdbd_req_state * state = tevent_req_data (
req , struct ctdbd_req_state ) ;
struct ctdbd_connection * conn = state - > conn ;
struct tevent_req * * pending = NULL ;
size_t num_pending = talloc_array_length ( conn - > pending ) ;
bool ok ;
pending = talloc_realloc (
conn , conn - > pending , struct tevent_req * , num_pending + 1 ) ;
if ( pending = = NULL ) {
return false ;
}
pending [ num_pending ] = req ;
conn - > pending = pending ;
tevent_req_set_cleanup_fn ( req , ctdbd_req_cleanup ) ;
ok = ctdbd_conn_receive_next ( conn ) ;
if ( ! ok ) {
ctdbd_req_unset_pending ( req ) ;
return false ;
}
return true ;
}
static bool ctdbd_conn_receive_next ( struct ctdbd_connection * conn )
{
size_t num_pending = talloc_array_length ( conn - > pending ) ;
struct tevent_req * req = NULL ;
struct ctdbd_req_state * state = NULL ;
if ( conn - > read_req ! = NULL ) {
return true ;
}
if ( num_pending = = 0 ) {
/*
* done for now
*/
return true ;
}
req = conn - > pending [ 0 ] ;
state = tevent_req_data ( req , struct ctdbd_req_state ) ;
conn - > read_req = ctdbd_pkt_read_send (
conn - > pending , state - > ev , conn - > fd ) ;
if ( conn - > read_req = = NULL ) {
return false ;
}
tevent_req_set_callback ( conn - > read_req , ctdbd_conn_received , conn ) ;
return true ;
}
static void ctdbd_conn_received ( struct tevent_req * subreq )
{
struct ctdbd_connection * conn = tevent_req_callback_data (
subreq , struct ctdbd_connection ) ;
TALLOC_CTX * frame = talloc_stackframe ( ) ;
uint8_t * pkt = NULL ;
int ret ;
struct ctdb_req_header * hdr = NULL ;
uint32_t reqid ;
struct tevent_req * req = NULL ;
struct ctdbd_req_state * state = NULL ;
size_t i , num_pending ;
bool ok ;
SMB_ASSERT ( subreq = = conn - > read_req ) ;
conn - > read_req = NULL ;
ret = ctdbd_pkt_read_recv ( subreq , frame , & pkt ) ;
TALLOC_FREE ( subreq ) ;
if ( ret ! = 0 ) {
cluster_fatal ( " ctdbd_pkt_read failed \n " ) ;
}
hdr = ( struct ctdb_req_header * ) pkt ;
reqid = hdr - > reqid ;
num_pending = talloc_array_length ( conn - > pending ) ;
for ( i = 0 ; i < num_pending ; i + + ) {
req = conn - > pending [ i ] ;
state = tevent_req_data ( req , struct ctdbd_req_state ) ;
if ( state - > reqid = = reqid ) {
break ;
}
}
if ( i = = num_pending ) {
/* not found */
TALLOC_FREE ( frame ) ;
return ;
}
state - > reply = talloc_move ( state , & hdr ) ;
tevent_req_defer_callback ( req , state - > ev ) ;
tevent_req_done ( req ) ;
TALLOC_FREE ( frame ) ;
ok = ctdbd_conn_receive_next ( conn ) ;
if ( ! ok ) {
cluster_fatal ( " ctdbd_conn_receive_next failed \n " ) ;
}
}
static void ctdbd_req_written ( struct tevent_req * subreq ) ;
struct tevent_req * ctdbd_req_send (
TALLOC_CTX * mem_ctx ,
struct tevent_context * ev ,
struct ctdbd_connection * conn ,
struct iovec * iov ,
size_t num_iov )
{
struct tevent_req * req = NULL , * subreq = NULL ;
struct ctdbd_req_state * state = NULL ;
struct ctdb_req_header * hdr = NULL ;
bool ok ;
req = tevent_req_create ( mem_ctx , & state , struct ctdbd_req_state ) ;
if ( req = = NULL ) {
return NULL ;
}
state - > conn = conn ;
state - > ev = ev ;
if ( ( num_iov = = 0 ) | |
( iov [ 0 ] . iov_len < sizeof ( struct ctdb_req_header ) ) ) {
tevent_req_error ( req , EINVAL ) ;
return tevent_req_post ( req , ev ) ;
}
hdr = iov [ 0 ] . iov_base ;
state - > reqid = hdr - > reqid ;
ok = ctdbd_req_set_pending ( req ) ;
if ( ! ok ) {
tevent_req_oom ( req ) ;
return tevent_req_post ( req , ev ) ;
}
subreq = writev_send (
state , ev , conn - > outgoing , conn - > fd , false , iov , num_iov ) ;
if ( tevent_req_nomem ( subreq , req ) ) {
return tevent_req_post ( req , ev ) ;
}
tevent_req_set_callback ( subreq , ctdbd_req_written , req ) ;
return req ;
}
static void ctdbd_req_written ( struct tevent_req * subreq )
{
struct tevent_req * req = tevent_req_callback_data (
subreq , struct tevent_req ) ;
ssize_t nwritten ;
int err ;
nwritten = writev_recv ( subreq , & err ) ;
TALLOC_FREE ( subreq ) ;
if ( nwritten = = - 1 ) {
tevent_req_error ( req , err ) ;
return ;
}
}
int ctdbd_req_recv (
struct tevent_req * req ,
TALLOC_CTX * mem_ctx ,
struct ctdb_req_header * * reply )
{
struct ctdbd_req_state * state = tevent_req_data (
req , struct ctdbd_req_state ) ;
int err ;
if ( tevent_req_is_unix_error ( req , & err ) ) {
return err ;
}
* reply = talloc_move ( mem_ctx , & state - > reply ) ;
tevent_req_received ( req ) ;
return 0 ;
}
2017-01-09 08:17:02 +01:00
struct ctdbd_parse_state {
struct tevent_context * ev ;
struct ctdbd_connection * conn ;
uint32_t reqid ;
TDB_DATA key ;
uint8_t _keybuf [ 64 ] ;
struct ctdb_req_call_old ctdb_req ;
struct iovec iov [ 2 ] ;
void ( * parser ) ( TDB_DATA key ,
TDB_DATA data ,
void * private_data ) ;
void * private_data ;
} ;
static void ctdbd_parse_done ( struct tevent_req * subreq ) ;
struct tevent_req * ctdbd_parse_send ( TALLOC_CTX * mem_ctx ,
struct tevent_context * ev ,
struct ctdbd_connection * conn ,
uint32_t db_id ,
TDB_DATA key ,
bool local_copy ,
void ( * parser ) ( TDB_DATA key ,
TDB_DATA data ,
void * private_data ) ,
void * private_data ,
enum dbwrap_req_state * req_state )
{
struct tevent_req * req = NULL ;
struct ctdbd_parse_state * state = NULL ;
uint32_t flags ;
uint32_t packet_length ;
struct tevent_req * subreq = NULL ;
req = tevent_req_create ( mem_ctx , & state , struct ctdbd_parse_state ) ;
if ( req = = NULL ) {
* req_state = DBWRAP_REQ_ERROR ;
return NULL ;
}
2020-03-11 11:03:06 +01:00
* req_state = DBWRAP_REQ_DISPATCHED ;
2017-01-09 08:17:02 +01:00
* state = ( struct ctdbd_parse_state ) {
. ev = ev ,
. conn = conn ,
. reqid = ctdbd_next_reqid ( conn ) ,
. parser = parser ,
. private_data = private_data ,
} ;
flags = local_copy ? CTDB_WANT_READONLY : 0 ;
packet_length = offsetof ( struct ctdb_req_call_old , data ) + key . dsize ;
/*
* Copy the key into our state , as ctdb_pkt_send_cleanup ( ) requires that
* all passed iov elements have a lifetime longer that the tevent_req
* returned by ctdb_pkt_send_send ( ) . This is required continue sending a
* the low level request into the ctdb socket , if a higher level
* ( ' this ' ) request is canceled ( or talloc free ' d ) by the application
* layer , without sending invalid packets to ctdb .
*/
if ( key . dsize > sizeof ( state - > _keybuf ) ) {
state - > key . dptr = talloc_memdup ( state , key . dptr , key . dsize ) ;
if ( tevent_req_nomem ( state - > key . dptr , req ) ) {
return tevent_req_post ( req , ev ) ;
}
} else {
memcpy ( state - > _keybuf , key . dptr , key . dsize ) ;
state - > key . dptr = state - > _keybuf ;
}
state - > key . dsize = key . dsize ;
state - > ctdb_req . hdr . length = packet_length ;
state - > ctdb_req . hdr . ctdb_magic = CTDB_MAGIC ;
state - > ctdb_req . hdr . ctdb_version = CTDB_PROTOCOL ;
state - > ctdb_req . hdr . operation = CTDB_REQ_CALL ;
state - > ctdb_req . hdr . reqid = state - > reqid ;
state - > ctdb_req . flags = flags ;
state - > ctdb_req . callid = CTDB_FETCH_FUNC ;
state - > ctdb_req . db_id = db_id ;
state - > ctdb_req . keylen = state - > key . dsize ;
state - > iov [ 0 ] . iov_base = & state - > ctdb_req ;
state - > iov [ 0 ] . iov_len = offsetof ( struct ctdb_req_call_old , data ) ;
state - > iov [ 1 ] . iov_base = state - > key . dptr ;
state - > iov [ 1 ] . iov_len = state - > key . dsize ;
2020-03-11 11:03:06 +01:00
subreq = ctdbd_req_send (
state , ev , conn , state - > iov , ARRAY_SIZE ( state - > iov ) ) ;
2017-01-09 08:17:02 +01:00
if ( tevent_req_nomem ( subreq , req ) ) {
* req_state = DBWRAP_REQ_ERROR ;
return tevent_req_post ( req , ev ) ;
}
2020-03-11 11:03:06 +01:00
tevent_req_set_callback ( subreq , ctdbd_parse_done , req ) ;
2017-01-09 08:17:02 +01:00
return req ;
}
static void ctdbd_parse_done ( struct tevent_req * subreq )
{
struct tevent_req * req = tevent_req_callback_data (
subreq , struct tevent_req ) ;
struct ctdbd_parse_state * state = tevent_req_data (
req , struct ctdbd_parse_state ) ;
struct ctdb_req_header * hdr = NULL ;
struct ctdb_reply_call_old * reply = NULL ;
int ret ;
2020-03-11 11:03:06 +01:00
ret = ctdbd_req_recv ( subreq , state , & hdr ) ;
2017-01-09 08:17:02 +01:00
TALLOC_FREE ( subreq ) ;
2019-09-03 15:26:19 +02:00
if ( tevent_req_error ( req , ret ) ) {
2020-03-11 11:03:06 +01:00
DBG_DEBUG ( " ctdb_req_recv failed %s \n " , strerror ( ret ) ) ;
2017-01-09 08:17:02 +01:00
return ;
}
2019-09-03 15:26:19 +02:00
SMB_ASSERT ( hdr ! = NULL ) ;
2017-01-09 08:17:02 +01:00
if ( hdr - > operation ! = CTDB_REPLY_CALL ) {
DBG_ERR ( " received invalid reply \n " ) ;
ctdb_packet_dump ( hdr ) ;
tevent_req_error ( req , EIO ) ;
return ;
}
reply = ( struct ctdb_reply_call_old * ) hdr ;
if ( reply - > datalen = = 0 ) {
/*
* Treat an empty record as non - existing
*/
tevent_req_error ( req , ENOENT ) ;
return ;
}
state - > parser ( state - > key ,
make_tdb_data ( & reply - > data [ 0 ] , reply - > datalen ) ,
state - > private_data ) ;
tevent_req_done ( req ) ;
return ;
}
int ctdbd_parse_recv ( struct tevent_req * req )
{
2020-03-11 11:03:06 +01:00
return tevent_req_simple_recv_unix ( req ) ;
2017-01-09 08:17:02 +01:00
}