2004-06-24 12:02:38 +04:00
/******************************************************************************
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
* *
* * Copyright ( C ) Sistina Software , Inc . 2002 - 2003 All rights reserved .
2005-02-22 19:26:21 +03:00
* * Copyright ( C ) 2004 - 2005 Red Hat , Inc . All rights reserved .
2004-06-24 12:02:38 +04:00
* *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
/* This provides the inter-clvmd communications for a system without CMAN.
There is a listening TCP socket which accepts new connections in the
normal way .
It can also make outgoing connnections to the other clvmd nodes .
*/
# include <pthread.h>
# include <sys/types.h>
# include <sys/utsname.h>
# include <sys/ioctl.h>
# include <sys/socket.h>
# include <sys/stat.h>
# include <sys/socket.h>
# include <netinet/in.h>
# include <stdio.h>
# include <stdlib.h>
# include <stdint.h>
# include <fcntl.h>
# include <string.h>
# include <stddef.h>
# include <stdint.h>
# include <unistd.h>
# include <errno.h>
# include <syslog.h>
# include <netdb.h>
# include <assert.h>
2005-11-09 12:24:10 +03:00
# include "libdevmapper.h"
2004-06-24 12:02:38 +04:00
# include "clvm.h"
# include "clvmd-comms.h"
# include "clvmd.h"
# include "clvmd-gulm.h"
# define DEFAULT_TCP_PORT 21064
static int listen_fd = - 1 ;
static int tcp_port ;
2005-11-09 12:24:10 +03:00
struct dm_hash_table * sock_hash ;
2004-06-24 12:02:38 +04:00
static int get_our_ip_address ( char * addr , int * family ) ;
static int read_from_tcpsock ( struct local_client * fd , char * buf , int len , char * csid ,
struct local_client * * new_client ) ;
/* Called by init_cluster() to open up the listening socket */
2004-11-03 13:45:07 +03:00
int init_comms ( unsigned short port )
2004-06-24 12:02:38 +04:00
{
2004-11-03 13:45:07 +03:00
struct sockaddr_in6 addr ;
2004-06-24 12:02:38 +04:00
2005-11-09 12:24:10 +03:00
sock_hash = dm_hash_create ( 100 ) ;
2006-05-10 01:23:51 +04:00
tcp_port = port ? : DEFAULT_TCP_PORT ;
2004-06-24 12:02:38 +04:00
2004-11-03 13:45:07 +03:00
listen_fd = socket ( AF_INET6 , SOCK_STREAM , 0 ) ;
2004-06-24 12:02:38 +04:00
if ( listen_fd < 0 )
{
return - 1 ;
}
else
{
int one = 1 ;
setsockopt ( listen_fd , SOL_SOCKET , SO_REUSEADDR , & one , sizeof ( int ) ) ;
2005-04-01 20:03:00 +04:00
setsockopt ( listen_fd , SOL_SOCKET , SO_KEEPALIVE , & one , sizeof ( int ) ) ;
2004-06-24 12:02:38 +04:00
}
2004-11-03 13:45:07 +03:00
memset ( & addr , 0 , sizeof ( addr ) ) ; // Bind to INADDR_ANY
addr . sin6_family = AF_INET6 ;
addr . sin6_port = htons ( tcp_port ) ;
2004-06-24 12:02:38 +04:00
2004-11-03 13:45:07 +03:00
if ( bind ( listen_fd , ( struct sockaddr * ) & addr , sizeof ( addr ) ) < 0 )
2004-06-24 12:02:38 +04:00
{
2004-11-03 13:45:07 +03:00
DEBUGLOG ( " Can't bind to port: %s \n " , strerror ( errno ) ) ;
2004-06-24 12:02:38 +04:00
syslog ( LOG_ERR , " Can't bind to port %d, is clvmd already running ? " , tcp_port ) ;
close ( listen_fd ) ;
return - 1 ;
}
listen ( listen_fd , 5 ) ;
2005-03-07 20:03:44 +03:00
/* Set Close-on-exec */
fcntl ( listen_fd , F_SETFD , 1 ) ;
2004-06-24 12:02:38 +04:00
return 0 ;
}
void tcp_remove_client ( char * csid )
{
struct local_client * client ;
DEBUGLOG ( " tcp_remove_client \n " ) ;
/* Don't actually close the socket here - that's the
job of clvmd . c whch will do the job when it notices the
other end has gone . We just need to remove the client ( s ) from
the hash table so we don ' t try to use it for sending any more */
2005-11-09 12:24:10 +03:00
client = dm_hash_lookup_binary ( sock_hash , csid , GULM_MAX_CSID_LEN ) ;
2004-06-24 12:02:38 +04:00
if ( client )
{
2005-11-09 12:24:10 +03:00
dm_hash_remove_binary ( sock_hash , csid , GULM_MAX_CSID_LEN ) ;
2005-04-13 17:50:07 +04:00
client - > removeme = 1 ;
2005-06-13 14:16:21 +04:00
close ( client - > fd ) ;
2004-06-24 12:02:38 +04:00
}
/* Look for a mangled one too */
csid [ 0 ] ^ = 0x80 ;
2005-11-09 12:24:10 +03:00
client = dm_hash_lookup_binary ( sock_hash , csid , GULM_MAX_CSID_LEN ) ;
2004-06-24 12:02:38 +04:00
if ( client )
{
2005-11-09 12:24:10 +03:00
dm_hash_remove_binary ( sock_hash , csid , GULM_MAX_CSID_LEN ) ;
2005-04-13 17:50:07 +04:00
client - > removeme = 1 ;
2005-06-13 14:16:21 +04:00
close ( client - > fd ) ;
2004-06-24 12:02:38 +04:00
}
/* Put it back as we found it */
csid [ 0 ] ^ = 0x80 ;
}
int alloc_client ( int fd , char * csid , struct local_client * * new_client )
{
struct local_client * client ;
2004-11-03 13:45:07 +03:00
DEBUGLOG ( " alloc_client %d csid = %s \n " , fd , print_csid ( csid ) ) ;
2004-06-24 12:02:38 +04:00
/* Create a local_client and return it */
client = malloc ( sizeof ( struct local_client ) ) ;
if ( ! client )
{
DEBUGLOG ( " malloc failed \n " ) ;
return - 1 ;
}
memset ( client , 0 , sizeof ( struct local_client ) ) ;
client - > fd = fd ;
client - > type = CLUSTER_DATA_SOCK ;
client - > callback = read_from_tcpsock ;
if ( new_client )
* new_client = client ;
/* Add to our list of node sockets */
2005-11-09 12:24:10 +03:00
if ( dm_hash_lookup_binary ( sock_hash , csid , GULM_MAX_CSID_LEN ) )
2004-06-24 12:02:38 +04:00
{
DEBUGLOG ( " alloc_client mangling CSID for second connection \n " ) ;
/* This is a duplicate connection but we can't close it because
the other end may already have started sending .
So , we mangle the IP address and keep it , all sending will
go out of the main FD
*/
csid [ 0 ] ^ = 0x80 ;
client - > bits . net . flags = 1 ; /* indicate mangled CSID */
/* If it still exists then kill the connection as we should only
ever have one incoming connection from each node */
2005-11-09 12:24:10 +03:00
if ( dm_hash_lookup_binary ( sock_hash , csid , GULM_MAX_CSID_LEN ) )
2004-06-24 12:02:38 +04:00
{
DEBUGLOG ( " Multiple incoming connections from node \n " ) ;
syslog ( LOG_ERR , " Bogus incoming connection from %d.%d.%d.%d \n " , csid [ 0 ] , csid [ 1 ] , csid [ 2 ] , csid [ 3 ] ) ;
free ( client ) ;
errno = ECONNREFUSED ;
return - 1 ;
}
}
2005-11-09 12:24:10 +03:00
dm_hash_insert_binary ( sock_hash , csid , GULM_MAX_CSID_LEN , client ) ;
2004-06-24 12:02:38 +04:00
return 0 ;
}
2005-01-13 16:24:02 +03:00
int get_main_gulm_cluster_fd ( )
2004-06-24 12:02:38 +04:00
{
return listen_fd ;
}
/* Read on main comms (listen) socket, accept it */
2005-01-13 16:24:02 +03:00
int cluster_fd_gulm_callback ( struct local_client * fd , char * buf , int len , char * csid ,
2004-06-24 12:02:38 +04:00
struct local_client * * new_client )
{
int newfd ;
2004-11-03 13:45:07 +03:00
struct sockaddr_in6 addr ;
2004-06-24 12:02:38 +04:00
socklen_t addrlen = sizeof ( addr ) ;
int status ;
2005-01-13 16:24:02 +03:00
char name [ GULM_MAX_CLUSTER_MEMBER_NAME_LEN ] ;
2004-06-24 12:02:38 +04:00
DEBUGLOG ( " cluster_fd_callback \n " ) ;
* new_client = NULL ;
newfd = accept ( listen_fd , ( struct sockaddr * ) & addr , & addrlen ) ;
DEBUGLOG ( " cluster_fd_callback, newfd=%d (errno=%d) \n " , newfd , errno ) ;
if ( ! newfd )
{
syslog ( LOG_ERR , " error in accept: %m " ) ;
errno = EAGAIN ;
return - 1 ; /* Don't return an error or clvmd will close the listening FD */
}
/* Check that the client is a member of the cluster
and reject if not .
*/
2005-01-13 16:24:02 +03:00
if ( gulm_name_from_csid ( ( char * ) & addr . sin6_addr , name ) < 0 )
2004-06-24 12:02:38 +04:00
{
2004-11-03 13:45:07 +03:00
syslog ( LOG_ERR , " Got connect from non-cluster node %s \n " ,
print_csid ( ( char * ) & addr . sin6_addr ) ) ;
DEBUGLOG ( " Got connect from non-cluster node %s \n " ,
print_csid ( ( char * ) & addr . sin6_addr ) ) ;
2004-06-24 12:02:38 +04:00
close ( newfd ) ;
errno = EAGAIN ;
return - 1 ;
}
2004-11-03 13:45:07 +03:00
status = alloc_client ( newfd , ( char * ) & addr . sin6_addr , new_client ) ;
2004-06-24 12:02:38 +04:00
if ( status )
{
DEBUGLOG ( " cluster_fd_callback, alloc_client failed, status = %d \n " , status ) ;
close ( newfd ) ;
/* See above... */
errno = EAGAIN ;
return - 1 ;
}
DEBUGLOG ( " cluster_fd_callback, returning %d, %p \n " , newfd , * new_client ) ;
return newfd ;
}
2005-06-10 13:11:01 +04:00
/* Try to get at least 'len' bytes from the socket */
static int really_read ( int fd , char * buf , int len )
{
int got , offset ;
got = offset = 0 ;
do {
got = read ( fd , buf + offset , len - offset ) ;
DEBUGLOG ( " really_read. got %d bytes \n " , got ) ;
offset + = got ;
} while ( got > 0 & & offset < len ) ;
if ( got < 0 )
return got ;
else
return offset ;
}
2004-06-24 12:02:38 +04:00
static int read_from_tcpsock ( struct local_client * client , char * buf , int len , char * csid ,
struct local_client * * new_client )
{
2004-11-03 13:45:07 +03:00
struct sockaddr_in6 addr ;
2004-06-24 12:02:38 +04:00
socklen_t slen = sizeof ( addr ) ;
2005-02-18 18:31:32 +03:00
struct clvm_header * header = ( struct clvm_header * ) buf ;
2004-06-24 12:02:38 +04:00
int status ;
2005-02-18 18:31:32 +03:00
uint32_t arglen ;
2004-06-24 12:02:38 +04:00
DEBUGLOG ( " read_from_tcpsock fd %d \n " , client - > fd ) ;
* new_client = NULL ;
/* Get "csid" */
getpeername ( client - > fd , ( struct sockaddr * ) & addr , & slen ) ;
2005-01-13 16:24:02 +03:00
memcpy ( csid , & addr . sin6_addr , GULM_MAX_CSID_LEN ) ;
2004-06-24 12:02:38 +04:00
2005-02-18 18:31:32 +03:00
/* Read just the header first, then get the rest if there is any.
* Stream sockets , sigh .
*/
2005-06-10 13:11:01 +04:00
status = really_read ( client - > fd , buf , sizeof ( struct clvm_header ) ) ;
2005-02-18 18:31:32 +03:00
if ( status > 0 )
{
int status2 ;
arglen = ntohl ( header - > arglen ) ;
/* Get the rest */
if ( arglen & & arglen < GULM_MAX_CLUSTER_MESSAGE )
{
2005-06-10 13:11:01 +04:00
status2 = really_read ( client - > fd , buf + status , arglen ) ;
2005-02-18 18:31:32 +03:00
if ( status2 > 0 )
status + = status2 ;
else
status = status2 ;
}
}
2004-06-24 12:02:38 +04:00
DEBUGLOG ( " read_from_tcpsock, status = %d(errno = %d) \n " , status , errno ) ;
/* Remove it from the hash table if there's an error, clvmd will
remove the socket from its lists and free the client struct */
if ( status = = 0 | |
( status < 0 & & errno ! = EAGAIN & & errno ! = EINTR ) )
{
2005-01-13 16:24:02 +03:00
char remcsid [ GULM_MAX_CSID_LEN ] ;
2004-06-24 12:02:38 +04:00
2005-01-13 16:24:02 +03:00
memcpy ( remcsid , csid , GULM_MAX_CSID_LEN ) ;
2004-06-24 12:02:38 +04:00
close ( client - > fd ) ;
/* If the csid was mangled, then make sure we remove the right entry */
if ( client - > bits . net . flags )
remcsid [ 0 ] ^ = 0x80 ;
2005-11-09 12:24:10 +03:00
dm_hash_remove_binary ( sock_hash , remcsid , GULM_MAX_CSID_LEN ) ;
2004-06-24 12:02:38 +04:00
/* Tell cluster manager layer */
add_down_node ( remcsid ) ;
}
2004-11-16 13:55:01 +03:00
else {
2005-02-22 19:26:21 +03:00
gulm_add_up_node ( csid ) ;
2004-11-16 13:55:01 +03:00
/* Send it back to clvmd */
2005-04-01 20:03:00 +04:00
process_message ( client , buf , status , csid ) ;
2004-11-16 13:55:01 +03:00
}
2004-06-24 12:02:38 +04:00
return status ;
}
2005-02-22 19:26:21 +03:00
int gulm_connect_csid ( char * csid , struct local_client * * newclient )
2004-06-24 12:02:38 +04:00
{
int fd ;
2004-11-03 13:45:07 +03:00
struct sockaddr_in6 addr ;
2004-06-24 12:02:38 +04:00
int status ;
2005-04-01 20:03:00 +04:00
int one = 1 ;
2004-06-24 12:02:38 +04:00
DEBUGLOG ( " Connecting socket \n " ) ;
2004-11-03 13:45:07 +03:00
fd = socket ( PF_INET6 , SOCK_STREAM , 0 ) ;
2004-06-24 12:02:38 +04:00
if ( fd < 0 )
{
syslog ( LOG_ERR , " Unable to create new socket: %m " ) ;
return - 1 ;
}
2004-11-03 13:45:07 +03:00
addr . sin6_family = AF_INET6 ;
2005-01-13 16:24:02 +03:00
memcpy ( & addr . sin6_addr , csid , GULM_MAX_CSID_LEN ) ;
2004-11-03 13:45:07 +03:00
addr . sin6_port = htons ( tcp_port ) ;
2004-06-24 12:02:38 +04:00
DEBUGLOG ( " Connecting socket %d \n " , fd ) ;
2004-11-03 13:45:07 +03:00
if ( connect ( fd , ( struct sockaddr * ) & addr , sizeof ( struct sockaddr_in6 ) ) < 0 )
2004-06-24 12:02:38 +04:00
{
2005-02-02 12:17:56 +03:00
/* "Connection refused" is "normal" because clvmd may not yet be running
* on that node .
*/
if ( errno ! = ECONNREFUSED )
{
syslog ( LOG_ERR , " Unable to connect to remote node: %m " ) ;
}
2004-06-24 12:02:38 +04:00
DEBUGLOG ( " Unable to connect to remote node: %s \n " , strerror ( errno ) ) ;
close ( fd ) ;
return - 1 ;
}
2005-03-07 20:03:44 +03:00
/* Set Close-on-exec */
fcntl ( fd , F_SETFD , 1 ) ;
2005-04-01 20:03:00 +04:00
setsockopt ( fd , SOL_SOCKET , SO_KEEPALIVE , & one , sizeof ( int ) ) ;
2005-03-07 20:03:44 +03:00
2004-06-24 12:02:38 +04:00
status = alloc_client ( fd , csid , newclient ) ;
if ( status )
close ( fd ) ;
else
add_client ( * newclient ) ;
/* If we can connect to it, it must be running a clvmd */
2005-01-13 16:24:02 +03:00
gulm_add_up_node ( csid ) ;
2004-06-24 12:02:38 +04:00
return status ;
}
/* Send a message to a known CSID */
static int tcp_send_message ( void * buf , int msglen , unsigned char * csid , const char * errtext )
{
int status ;
struct local_client * client ;
2005-01-13 16:24:02 +03:00
char ourcsid [ GULM_MAX_CSID_LEN ] ;
2004-06-24 12:02:38 +04:00
assert ( csid ) ;
2004-11-03 13:45:07 +03:00
DEBUGLOG ( " tcp_send_message, csid = %s, msglen = %d \n " , print_csid ( csid ) , msglen ) ;
2004-06-24 12:02:38 +04:00
/* Don't connect to ourself */
2005-01-13 16:24:02 +03:00
get_our_gulm_csid ( ourcsid ) ;
if ( memcmp ( csid , ourcsid , GULM_MAX_CSID_LEN ) = = 0 )
2004-06-24 12:02:38 +04:00
return msglen ;
2005-11-09 12:24:10 +03:00
client = dm_hash_lookup_binary ( sock_hash , csid , GULM_MAX_CSID_LEN ) ;
2004-06-24 12:02:38 +04:00
if ( ! client )
{
2005-02-22 19:26:21 +03:00
status = gulm_connect_csid ( csid , & client ) ;
2004-06-24 12:02:38 +04:00
if ( status )
return - 1 ;
}
DEBUGLOG ( " tcp_send_message, fd = %d \n " , client - > fd ) ;
return write ( client - > fd , buf , msglen ) ;
}
2005-01-13 16:24:02 +03:00
int gulm_cluster_send_message ( void * buf , int msglen , char * csid , const char * errtext )
2004-06-24 12:02:38 +04:00
{
int status = 0 ;
DEBUGLOG ( " cluster send message, csid = %p, msglen = %d \n " , csid , msglen ) ;
/* If csid is NULL then send to all known (not just connected) nodes */
if ( ! csid )
{
void * context = NULL ;
2005-01-13 16:24:02 +03:00
char loop_csid [ GULM_MAX_CSID_LEN ] ;
2004-06-24 12:02:38 +04:00
/* Loop round all gulm-known nodes */
while ( get_next_node_csid ( & context , loop_csid ) )
{
status = tcp_send_message ( buf , msglen , loop_csid , errtext ) ;
if ( status = = 0 | |
( status < 0 & & ( errno = = EAGAIN | | errno = = EINTR ) ) )
break ;
}
}
else
{
status = tcp_send_message ( buf , msglen , csid , errtext ) ;
}
return status ;
}
/* To get our own IP address we get the locally bound address of the
socket that ' s talking to GULM in the assumption ( eek ) that it will
be on the " right " network in a multi - homed system */
static int get_our_ip_address ( char * addr , int * family )
{
2004-11-03 13:45:07 +03:00
struct utsname info ;
uname ( & info ) ;
get_ip_address ( info . nodename , addr ) ;
2004-06-24 12:02:38 +04:00
return 0 ;
}
/* Public version of above for those that don't care what protocol
we ' re using */
2005-01-13 16:24:02 +03:00
void get_our_gulm_csid ( char * csid )
2004-06-24 12:02:38 +04:00
{
2005-01-13 16:24:02 +03:00
static char our_csid [ GULM_MAX_CSID_LEN ] ;
2004-06-24 12:02:38 +04:00
static int got_csid = 0 ;
if ( ! got_csid )
{
int family ;
memset ( our_csid , 0 , sizeof ( our_csid ) ) ;
if ( get_our_ip_address ( our_csid , & family ) )
{
got_csid = 1 ;
}
}
2005-01-13 16:24:02 +03:00
memcpy ( csid , our_csid , GULM_MAX_CSID_LEN ) ;
2004-06-24 12:02:38 +04:00
}
2004-11-03 13:45:07 +03:00
static void map_v4_to_v6 ( struct in_addr * ip4 , struct in6_addr * ip6 )
{
ip6 - > s6_addr32 [ 0 ] = 0 ;
ip6 - > s6_addr32 [ 1 ] = 0 ;
ip6 - > s6_addr32 [ 2 ] = htonl ( 0xffff ) ;
ip6 - > s6_addr32 [ 3 ] = ip4 - > s_addr ;
}
2004-06-24 12:02:38 +04:00
/* Get someone else's IP address from DNS */
int get_ip_address ( char * node , char * addr )
{
struct hostent * he ;
2005-01-13 16:24:02 +03:00
memset ( addr , 0 , GULM_MAX_CSID_LEN ) ;
2004-06-24 12:02:38 +04:00
// TODO: what do we do about multi-homed hosts ???
// CCSs ip_interfaces solved this but some bugger removed it.
/* Try IPv6 first. The man page for gethostbyname implies that
it will lookup ip6 & ip4 names , but it seems not to */
he = gethostbyname2 ( node , AF_INET6 ) ;
2004-11-03 13:45:07 +03:00
if ( he )
{
memcpy ( addr , he - > h_addr_list [ 0 ] ,
he - > h_length ) ;
}
else
{
2004-06-24 12:02:38 +04:00
he = gethostbyname2 ( node , AF_INET ) ;
2004-11-03 13:45:07 +03:00
if ( ! he )
return - 1 ;
map_v4_to_v6 ( ( struct in_addr * ) he - > h_addr_list [ 0 ] , ( struct in6_addr * ) addr ) ;
}
2004-06-24 12:02:38 +04:00
return 0 ;
}
2004-11-03 13:45:07 +03:00
char * print_csid ( char * csid )
{
static char buf [ 128 ] ;
int * icsid = ( int * ) csid ;
sprintf ( buf , " [%x.%x.%x.%x] " ,
icsid [ 0 ] , icsid [ 1 ] , icsid [ 2 ] , icsid [ 3 ] ) ;
return buf ;
}