2010-03-18 09:19:30 +00:00
/*
* Copyright ( C ) 2009 Red Hat , Inc . All rights reserved .
*
* This file is part of LVM2 .
*
* This copyrighted material is made available to anyone wishing to use ,
* modify , copy , or redistribute it subject to the terms and conditions
* of the GNU Lesser General Public License v .2 .1 .
*
* You should have received a copy of the GNU Lesser General Public License
* along with this program ; if not , write to the Free Software Foundation ,
* Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*/
# define _GNU_SOURCE
# define _FILE_OFFSET_BITS 64
2010-06-15 11:00:44 +00:00
# include "configure.h"
2010-03-18 09:19:30 +00:00
# include <netinet/in.h>
# include <sys/un.h>
2010-03-26 15:45:36 +00:00
# include <sys/socket.h>
2010-03-18 09:19:30 +00:00
# include <unistd.h>
# include <fcntl.h>
2010-06-15 11:00:44 +00:00
# include "libdevmapper.h"
2010-03-18 09:19:30 +00:00
# include <pthread.h>
# include "locking.h"
# include "lvm-logging.h"
# include "clvm.h"
# include "clvmd-comms.h"
# include "lvm-functions.h"
# include "clvmd.h"
2010-03-26 15:45:36 +00:00
static const char SINGLENODE_CLVMD_SOCKNAME [ ] = " \0 singlenode_clvmd " ;
2010-03-18 09:19:30 +00:00
static int listen_fd = - 1 ;
static int init_comms ( )
{
2010-03-26 15:45:36 +00:00
struct sockaddr_un addr ;
2010-03-18 09:19:30 +00:00
2010-03-26 15:45:36 +00:00
listen_fd = socket ( PF_UNIX , SOCK_STREAM , 0 ) ;
if ( listen_fd < 0 ) {
DEBUGLOG ( " Can't create local socket: %s \n " , strerror ( errno ) ) ;
2010-03-18 09:19:30 +00:00
return - 1 ;
2010-03-26 15:45:36 +00:00
}
2010-03-18 09:19:30 +00:00
/* Set Close-on-exec */
fcntl ( listen_fd , F_SETFD , 1 ) ;
2010-03-26 15:45:36 +00:00
memset ( & addr , 0 , sizeof ( addr ) ) ;
memcpy ( addr . sun_path , SINGLENODE_CLVMD_SOCKNAME ,
sizeof ( SINGLENODE_CLVMD_SOCKNAME ) ) ;
addr . sun_family = AF_UNIX ;
if ( bind ( listen_fd , ( struct sockaddr * ) & addr , sizeof ( addr ) ) < 0 ) {
DEBUGLOG ( " Can't bind local socket: %s \n " , strerror ( errno ) ) ;
close ( listen_fd ) ;
return - 1 ;
}
if ( listen ( listen_fd , 10 ) < 0 ) {
DEBUGLOG ( " Can't listen local socket: %s \n " , strerror ( errno ) ) ;
close ( listen_fd ) ;
return - 1 ;
}
2010-03-18 09:19:30 +00:00
return 0 ;
}
static int _init_cluster ( void )
{
int r ;
r = init_comms ( ) ;
if ( r )
return r ;
DEBUGLOG ( " Single-node cluster initialised. \n " ) ;
return 0 ;
}
static void _cluster_closedown ( void )
{
close ( listen_fd ) ;
DEBUGLOG ( " cluster_closedown \n " ) ;
destroy_lvhash ( ) ;
}
static void _get_our_csid ( char * csid )
{
int nodeid = 1 ;
memcpy ( csid , & nodeid , sizeof ( int ) ) ;
}
static int _csid_from_name ( char * csid , const char * name )
{
return 1 ;
}
static int _name_from_csid ( const char * csid , char * name )
{
sprintf ( name , " %x " , 0xdead ) ;
return 0 ;
}
static int _get_num_nodes ( )
{
return 1 ;
}
/* Node is now known to be running a clvmd */
static void _add_up_node ( const char * csid )
{
}
/* Call a callback for each node, so the caller knows whether it's up or down */
static int _cluster_do_node_callback ( struct local_client * master_client ,
void ( * callback ) ( struct local_client * ,
const char * csid , int node_up ) )
{
return 0 ;
}
int _lock_file ( const char * file , uint32_t flags ) ;
static int * _locks = NULL ;
static char * * _resources = NULL ;
static int _lock_max = 1 ;
static pthread_mutex_t _lock_mutex = PTHREAD_MUTEX_INITIALIZER ;
/* Real locking */
static int _lock_resource ( const char * resource , int mode , int flags , int * lockid )
{
int * _locks_1 ;
char * * _resources_1 ;
int i , j ;
DEBUGLOG ( " lock_resource '%s', flags=%d, mode=%d \n " ,
resource , flags , mode ) ;
retry :
pthread_mutex_lock ( & _lock_mutex ) ;
/* look for an existing lock for this resource */
for ( i = 1 ; i < _lock_max ; + + i ) {
if ( ! _resources [ i ] )
break ;
if ( ! strcmp ( _resources [ i ] , resource ) ) {
if ( ( _locks [ i ] & LCK_WRITE ) | | ( _locks [ i ] & LCK_EXCL ) ) {
DEBUGLOG ( " %s already write/exclusively locked... \n " , resource ) ;
goto maybe_retry ;
}
if ( ( mode & LCK_WRITE ) | | ( mode & LCK_EXCL ) ) {
DEBUGLOG ( " %s already locked and WRITE/EXCL lock requested... \n " ,
resource ) ;
goto maybe_retry ;
}
}
}
if ( i = = _lock_max ) { /* out of lock slots, extend */
_locks_1 = dm_realloc ( _locks , 2 * _lock_max * sizeof ( int ) ) ;
if ( ! _locks_1 )
return 1 ; /* fail */
_locks = _locks_1 ;
_resources_1 = dm_realloc ( _resources , 2 * _lock_max * sizeof ( char * ) ) ;
if ( ! _resources_1 ) {
/* _locks may get realloc'd twice, but that should be safe */
return 1 ; /* fail */
}
_resources = _resources_1 ;
/* clear the new resource entries */
for ( j = _lock_max ; j < 2 * _lock_max ; + + j )
_resources [ j ] = NULL ;
_lock_max = 2 * _lock_max ;
}
/* resource is not currently locked, grab it */
* lockid = i ;
_locks [ i ] = mode ;
_resources [ i ] = dm_strdup ( resource ) ;
DEBUGLOG ( " %s locked -> %d \n " , resource , i ) ;
pthread_mutex_unlock ( & _lock_mutex ) ;
return 0 ;
maybe_retry :
pthread_mutex_unlock ( & _lock_mutex ) ;
if ( ! ( flags & LCK_NONBLOCK ) ) {
usleep ( 10000 ) ;
goto retry ;
}
return 1 ; /* fail */
}
static int _unlock_resource ( const char * resource , int lockid )
{
DEBUGLOG ( " unlock_resource: %s lockid: %x \n " , resource , lockid ) ;
if ( ! _resources [ lockid ] ) {
DEBUGLOG ( " (%s) %d not locked \n " , resource , lockid ) ;
return 1 ;
}
if ( strcmp ( _resources [ lockid ] , resource ) ) {
DEBUGLOG ( " %d has wrong resource (requested %s, got %s) \n " ,
lockid , resource , _resources [ lockid ] ) ;
return 1 ;
}
dm_free ( _resources [ lockid ] ) ;
_resources [ lockid ] = 0 ;
return 0 ;
}
static int _is_quorate ( )
{
return 1 ;
}
static int _get_main_cluster_fd ( void )
{
return listen_fd ;
}
static int _cluster_fd_callback ( struct local_client * fd , char * buf , int len ,
const char * csid ,
struct local_client * * new_client )
{
return 1 ;
}
static int _cluster_send_message ( const void * buf , int msglen ,
const char * csid ,
const char * errtext )
{
return 0 ;
}
static int _get_cluster_name ( char * buf , int buflen )
{
strncpy ( buf , " localcluster " , buflen ) ;
buf [ buflen - 1 ] = 0 ;
return 0 ;
}
static struct cluster_ops _cluster_singlenode_ops = {
. cluster_init_completed = NULL ,
. cluster_send_message = _cluster_send_message ,
. name_from_csid = _name_from_csid ,
. csid_from_name = _csid_from_name ,
. get_num_nodes = _get_num_nodes ,
. cluster_fd_callback = _cluster_fd_callback ,
. get_main_cluster_fd = _get_main_cluster_fd ,
. cluster_do_node_callback = _cluster_do_node_callback ,
. is_quorate = _is_quorate ,
. get_our_csid = _get_our_csid ,
. add_up_node = _add_up_node ,
. reread_config = NULL ,
. cluster_closedown = _cluster_closedown ,
. get_cluster_name = _get_cluster_name ,
. sync_lock = _lock_resource ,
. sync_unlock = _unlock_resource ,
} ;
struct cluster_ops * init_singlenode_cluster ( void )
{
if ( ! _init_cluster ( ) )
return & _cluster_singlenode_ops ;
else
return NULL ;
}