2009-09-15 10:29:08 -04:00
/*
Copyright Red Hat , Inc . 2009
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 the
Free Software Foundation ; either version 2 , or ( at your option ) any
later version .
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 .
You should have received a copy of the GNU General Public License
along with this program ; see the file COPYING . If not , write to the
Free Software Foundation , Inc . , 675 Mass Ave , Cambridge ,
MA 0213 9 , USA .
*/
/*
* Author : Lon Hohberger < lhh at redhat . com >
*/
2009-09-15 11:25:02 -04:00
# include <config.h>
2009-09-15 10:29:08 -04:00
# include <stdio.h>
# include <simpleconfig.h>
2010-01-14 13:38:37 -05:00
# include <static_map.h>
2009-09-15 10:29:08 -04:00
# include <sys/types.h>
# include <stdint.h>
# include <time.h>
# include <server_plugin.h>
# include <string.h>
# include <malloc.h>
2009-09-16 20:07:46 -04:00
# include <syslog.h>
2009-09-15 10:29:08 -04:00
# include <errno.h>
2009-09-16 20:07:46 -04:00
# include <unistd.h>
2009-09-15 13:27:03 -04:00
# include <libvirt/libvirt.h>
2009-09-16 20:07:46 -04:00
# include <pthread.h>
2009-09-15 11:25:02 -04:00
# ifdef HAVE_OPENAIS_CPG_H
2009-09-15 10:29:08 -04:00
# include <openais/cpg.h>
# else
2009-09-15 11:25:02 -04:00
# ifdef HAVE_COROSYNC_CPG_H
2009-09-15 10:29:08 -04:00
# include <corosync/cpg.h>
# endif
2009-09-15 11:25:02 -04:00
# endif
2009-09-15 10:29:08 -04:00
2009-09-16 20:07:46 -04:00
# include <libcman.h>
# include <debug.h>
# include "virt.h"
# include "xvm.h"
# include "checkpoint.h"
2009-09-15 10:29:08 -04:00
# define NAME "checkpoint"
# define VERSION "0.8"
# define MAGIC 0x1e017afe
struct check_info {
int magic ;
int pad ;
} ;
# define VALIDATE(arg) \
do { \
if ( ! arg | | ( ( struct check_info * ) arg ) - > magic ! = MAGIC ) { \
errno = EINVAL ; \
return - 1 ; \
} \
} while ( 0 )
2009-09-16 20:07:46 -04:00
static void * checkpoint_handle = NULL ;
static virt_list_t * local_vms = NULL ;
static char * uri = NULL ;
static int use_uuid = 0 ;
static int
virt_list_update ( virConnectPtr vp , virt_list_t * * vl , int my_id )
{
virt_list_t * list = NULL ;
list = vl_get ( vp , my_id ) ;
if ( ! list )
return - 1 ;
if ( * vl )
vl_free ( * vl ) ;
* vl = list ;
return 0 ;
}
static int
get_cman_ids ( cman_handle_t ch , uint32_t * my_id , uint32_t * high_id )
{
int max_nodes ;
int actual ;
cman_node_t * nodes = NULL ;
cman_node_t me ;
uint32_t high = 0 ;
int ret = - 1 , x , _local = 0 ;
if ( ! my_id & & ! high_id )
return 0 ;
if ( ! ch ) {
_local = 1 ;
ch = cman_init ( NULL ) ;
}
if ( ! ch )
return - 1 ;
max_nodes = cman_get_node_count ( ch ) ;
if ( max_nodes < = 0 )
goto out ;
if ( my_id ) {
memset ( & me , 0 , sizeof ( me ) ) ;
if ( cman_get_node ( ch , CMAN_NODEID_US , & me ) < 0 )
goto out ;
* my_id = me . cn_nodeid ;
}
if ( ! high_id ) {
ret = 0 ;
goto out ;
}
nodes = malloc ( sizeof ( cman_node_t ) * max_nodes ) ;
if ( ! nodes )
goto out ;
memset ( nodes , 0 , sizeof ( cman_node_t ) * max_nodes ) ;
if ( cman_get_nodes ( ch , max_nodes , & actual , nodes ) < 0 )
goto out ;
for ( x = 0 ; x < actual ; x + + )
if ( nodes [ x ] . cn_nodeid > high & & nodes [ x ] . cn_member )
high = nodes [ x ] . cn_nodeid ;
* high_id = high ;
ret = 0 ;
out :
if ( nodes )
free ( nodes ) ;
if ( ch & & _local )
cman_finish ( ch ) ;
return ret ;
}
static int
node_operational ( uint32_t nodeid )
{
cman_handle_t ch ;
cman_node_t node ;
ch = cman_init ( NULL ) ;
if ( ! ch )
return - 1 ;
memset ( & node , 0 , sizeof ( node ) ) ;
if ( cman_get_node ( ch , nodeid , & node ) = = 0 ) {
cman_finish ( ch ) ;
return ! ! node . cn_member ;
}
cman_finish ( ch ) ;
return 0 ;
}
static int
get_domain_state_ckpt ( void * hp , const char * domain , vm_state_t * state )
{
errno = EINVAL ;
if ( ! hp | | ! domain | | ! state | | ! strlen ( ( char * ) domain ) )
return - 1 ;
if ( ! strcmp ( DOMAIN0NAME , ( char * ) domain ) )
return - 1 ;
return ckpt_read ( hp , domain , state , sizeof ( * state ) ) ;
}
static inline int
wait_domain ( const char * vm_name , virConnectPtr vp , int timeout )
{
int tries = 0 ;
int response = 1 ;
2011-09-20 11:15:34 -04:00
int ret ;
2009-09-16 20:07:46 -04:00
virDomainPtr vdp ;
virDomainInfo vdi ;
if ( use_uuid ) {
2011-06-20 14:09:27 -04:00
vdp = virDomainLookupByUUIDString ( vp , ( const char * ) vm_name ) ;
2009-09-16 20:07:46 -04:00
} else {
vdp = virDomainLookupByName ( vp , vm_name ) ;
}
if ( ! vdp )
return 0 ;
/* Check domain liveliness. If the domain is still here,
we return failure , and the client must then retry */
/* XXX On the xen 3.0.4 API, we will be able to guarantee
synchronous virDomainDestroy , so this check will not
be necessary */
do {
2011-09-20 11:15:34 -04:00
if ( + + tries > timeout )
break ;
2009-09-16 20:07:46 -04:00
sleep ( 1 ) ;
if ( use_uuid ) {
2011-06-20 14:09:27 -04:00
vdp = virDomainLookupByUUIDString ( vp ,
( const char * ) vm_name ) ;
2009-09-16 20:07:46 -04:00
} else {
vdp = virDomainLookupByName ( vp , vm_name ) ;
}
if ( ! vdp ) {
dbg_printf ( 2 , " Domain no longer exists \n " ) ;
response = 0 ;
break ;
}
memset ( & vdi , 0 , sizeof ( vdi ) ) ;
2011-09-20 11:15:34 -04:00
ret = virDomainGetInfo ( vdp , & vdi ) ;
2009-09-16 20:07:46 -04:00
virDomainFree ( vdp ) ;
2011-09-20 11:15:34 -04:00
if ( ret < 0 )
continue ;
2009-09-16 20:07:46 -04:00
if ( vdi . state = = VIR_DOMAIN_SHUTOFF ) {
dbg_printf ( 2 , " Domain has been shut off \n " ) ;
response = 0 ;
break ;
}
dbg_printf ( 4 , " Domain still exists (state %d) "
" after %d seconds \n " ,
vdi . state , tries ) ;
} while ( 1 ) ;
return response ;
}
/*
Returns : 0 - operational
1 - dead or presumed so
2 - VM not local and I am not the right node to deal with it
3 - VM status unknown ; cannot operate on it
*/
2009-09-15 10:29:08 -04:00
static int
2009-09-16 20:07:46 -04:00
cluster_virt_status ( const char * vm_name , uint32_t * owner )
{
vm_state_t chk_state ;
virt_state_t * vs ;
uint32_t me , high_id ;
int ret = 0 ;
dbg_printf ( 80 , " %s %s \n " , __FUNCTION__ , vm_name ) ;
/* if we can't find the high ID, we can't do anything useful */
/* This should be cpg_get_ids() but it's segfaulting for some
reason : ( */
if ( get_cman_ids ( NULL , & me , & high_id ) ! = 0 )
return 2 ;
if ( use_uuid ) {
vs = vl_find_uuid ( local_vms , vm_name ) ;
} else {
vs = vl_find_name ( local_vms , vm_name ) ;
}
if ( ! vs ) {
ret = 2 ; /* not found locally */
if ( me ! = high_id )
goto out ;
if ( get_domain_state_ckpt ( checkpoint_handle ,
vm_name , & chk_state ) ) {
dbg_printf ( 2 , " High ID: Unknown VM \n " ) ;
ret = 3 ;
goto out ;
}
if ( node_operational ( chk_state . s_owner ) ) {
* owner = chk_state . s_owner ;
dbg_printf ( 2 , " High ID: Owner is operational \n " ) ;
ret = 2 ;
} else {
dbg_printf ( 2 , " High ID: Owner is dead; returning 'off' \n " ) ;
ret = 1 ;
}
} else if ( vs - > v_state . s_state = = VIR_DOMAIN_SHUTOFF ) {
ret = 1 ; /* local and off */
}
out :
2011-06-28 17:09:02 -04:00
dbg_printf ( 80 , " %s %s \n " , __FUNCTION__ , vm_name ) ;
2009-09-16 20:07:46 -04:00
return ret ;
}
static void
store_domains_by_name ( void * hp , virt_list_t * vl )
{
int x ;
if ( ! vl )
return ;
for ( x = 0 ; x < vl - > vm_count ; x + + ) {
if ( ! strcmp ( DOMAIN0NAME , vl - > vm_states [ x ] . v_name ) )
continue ;
dbg_printf ( 2 , " Storing %s \n " , vl - > vm_states [ x ] . v_name ) ;
ckpt_write ( hp , vl - > vm_states [ x ] . v_name ,
& vl - > vm_states [ x ] . v_state ,
sizeof ( vm_state_t ) ) ;
}
}
static void
store_domains_by_uuid ( void * hp , virt_list_t * vl )
{
int x ;
if ( ! vl )
return ;
for ( x = 0 ; x < vl - > vm_count ; x + + ) {
if ( ! strcmp ( DOMAIN0UUID , vl - > vm_states [ x ] . v_uuid ) )
continue ;
dbg_printf ( 2 , " Storing %s \n " , vl - > vm_states [ x ] . v_uuid ) ;
ckpt_write ( hp , vl - > vm_states [ x ] . v_uuid ,
& vl - > vm_states [ x ] . v_state ,
sizeof ( vm_state_t ) ) ;
}
}
static void
update_local_vms ( void )
{
virConnectPtr vp = NULL ;
uint32_t my_id = 0 ;
cpg_get_ids ( & my_id , NULL ) ;
vp = virConnectOpen ( uri ) ;
if ( ! vp ) {
syslog ( LOG_ERR , " Failed to connect to hypervisor \n " ) ;
}
virt_list_update ( vp , & local_vms , my_id ) ;
vl_print ( local_vms ) ;
if ( use_uuid )
store_domains_by_uuid ( checkpoint_handle , local_vms ) ;
else
store_domains_by_name ( checkpoint_handle , local_vms ) ;
virConnectClose ( vp ) ;
}
static int
do_off ( const char * vm_name )
{
virConnectPtr vp ;
virDomainPtr vdp ;
virDomainInfo vdi ;
int ret = - 1 ;
2011-06-28 17:09:02 -04:00
dbg_printf ( 5 , " %s %s \n " , __FUNCTION__ , vm_name ) ;
2009-09-16 20:07:46 -04:00
vp = virConnectOpen ( uri ) ;
if ( ! vp )
return 1 ;
if ( use_uuid ) {
2011-06-20 14:09:27 -04:00
vdp = virDomainLookupByUUIDString ( vp ,
( const char * ) vm_name ) ;
2009-09-16 20:07:46 -04:00
} else {
vdp = virDomainLookupByName ( vp , vm_name ) ;
}
if ( ! vdp | |
( ( virDomainGetInfo ( vdp , & vdi ) = = 0 ) & &
( vdi . state = = VIR_DOMAIN_SHUTOFF ) ) ) {
dbg_printf ( 2 , " Nothing to do - domain does not exist \n " ) ;
if ( vdp )
virDomainFree ( vdp ) ;
return 0 ;
}
syslog ( LOG_NOTICE , " Destroying domain %s \n " , vm_name ) ;
dbg_printf ( 2 , " [OFF] Calling virDomainDestroy \n " ) ;
ret = virDomainDestroy ( vdp ) ;
if ( ret < 0 ) {
syslog ( LOG_NOTICE , " Failed to destroy domain: %d \n " , ret ) ;
printf ( " virDomainDestroy() failed: %d \n " , ret ) ;
ret = 1 ;
goto out ;
}
if ( ret ) {
syslog ( LOG_NOTICE ,
" Domain %s still exists; fencing failed \n " ,
vm_name ) ;
printf ( " Domain %s still exists; fencing failed \n " , vm_name ) ;
ret = 1 ;
goto out ;
}
ret = 0 ;
out :
virConnectClose ( vp ) ;
return ret ;
}
static int
do_reboot ( const char * vm_name )
{
virConnectPtr vp ;
virDomainPtr vdp , nvdp ;
virDomainInfo vdi ;
char * domain_desc ;
int ret ;
//uuid_unparse(vm_uuid, uu_string);
dbg_printf ( 5 , " %s %s \n " , __FUNCTION__ , vm_name ) ;
vp = virConnectOpen ( uri ) ;
if ( ! vp )
return 1 ;
if ( use_uuid ) {
2011-06-20 14:09:27 -04:00
vdp = virDomainLookupByUUIDString ( vp ,
( const char * ) vm_name ) ;
2009-09-16 20:07:46 -04:00
} else {
vdp = virDomainLookupByName ( vp , vm_name ) ;
}
if ( ! vdp | | ( ( virDomainGetInfo ( vdp , & vdi ) = = 0 ) & &
( vdi . state = = VIR_DOMAIN_SHUTOFF ) ) ) {
dbg_printf ( 2 , " [libvirt:REBOOT] Nothing to "
" do - domain does not exist \n " ) ;
if ( vdp )
virDomainFree ( vdp ) ;
return 0 ;
}
syslog ( LOG_NOTICE , " Rebooting domain %s \n " , vm_name ) ;
printf ( " Rebooting domain %s... \n " , vm_name ) ;
domain_desc = virDomainGetXMLDesc ( vdp , 0 ) ;
if ( ! domain_desc ) {
printf ( " Failed getting domain description from "
" libvirt \n " ) ;
}
dbg_printf ( 2 , " [REBOOT] Calling virDomainDestroy(%p) \n " , vdp ) ;
ret = virDomainDestroy ( vdp ) ;
if ( ret < 0 ) {
printf ( " virDomainDestroy() failed: %d/%d \n " , ret , errno ) ;
free ( domain_desc ) ;
virDomainFree ( vdp ) ;
ret = 1 ;
goto out ;
}
ret = wait_domain ( vm_name , vp , 15 ) ;
if ( ret ) {
syslog ( LOG_NOTICE , " Domain %s still exists; fencing failed \n " ,
vm_name ) ;
printf ( " Domain %s still exists; fencing failed \n " , vm_name ) ;
if ( domain_desc )
free ( domain_desc ) ;
ret = 1 ;
goto out ;
}
if ( ! domain_desc ) {
ret = 0 ;
goto out ;
}
/* 'on' is not a failure */
ret = 0 ;
dbg_printf ( 3 , " [[ XML Domain Info ]] \n " ) ;
dbg_printf ( 3 , " %s \n [[ XML END ]] \n " , domain_desc ) ;
dbg_printf ( 2 , " Calling virDomainCreateLinux()... \n " ) ;
nvdp = virDomainCreateLinux ( vp , domain_desc , 0 ) ;
if ( nvdp = = NULL ) {
/* More recent versions of libvirt or perhaps the
* KVM back - end do not let you create a domain from
* XML if there is already a defined domain description
* with the same name that it knows about . You must
* then call virDomainCreate ( ) */
dbg_printf ( 2 , " Failed; Trying virDomainCreate()... \n " ) ;
if ( virDomainCreate ( vdp ) < 0 ) {
syslog ( LOG_NOTICE ,
" Could not restart %s \n " ,
vm_name ) ;
dbg_printf ( 1 , " Failed to recreate guest "
" %s! \n " , vm_name ) ;
}
}
free ( domain_desc ) ;
out :
virConnectClose ( vp ) ;
return ret ;
}
static void
do_real_work ( void * data , size_t len , uint32_t nodeid , uint32_t seqno )
{
struct ckpt_fence_req * req = data ;
struct ckpt_fence_req reply ;
uint32_t owner ;
2010-01-15 08:06:34 -05:00
int ret = 1 ;
2009-09-16 20:07:46 -04:00
memcpy ( & reply , req , sizeof ( reply ) ) ;
update_local_vms ( ) ;
switch ( req - > request ) {
case FENCE_STATUS :
ret = cluster_virt_status ( req - > vm_name , & owner ) ;
2011-06-20 14:09:27 -04:00
if ( ret = = 3 ) {
ret = RESP_OFF ;
break ;
}
2009-09-16 20:07:46 -04:00
if ( ret = = 2 ) {
return ;
}
2010-01-15 08:06:34 -05:00
if ( ret = = 1 ) {
ret = RESP_OFF ;
}
2009-09-16 20:07:46 -04:00
break ;
case FENCE_OFF :
ret = cluster_virt_status ( req - > vm_name , & owner ) ;
2011-06-20 14:09:27 -04:00
if ( ret = = 3 ) {
/* No record of this VM in the checkpoint. */
ret = 0 ;
break ;
}
2009-09-16 20:07:46 -04:00
if ( ret ! = 0 ) {
return ;
}
/* Must be running locally to perform 'off' */
ret = do_off ( req - > vm_name ) ;
break ;
case FENCE_REBOOT :
ret = cluster_virt_status ( req - > vm_name , & owner ) ;
if ( ret ! = 0 ) {
return ;
}
/* Must be running locally to perform 'reboot' */
ret = do_reboot ( req - > vm_name ) ;
break ;
}
2010-01-15 08:06:34 -05:00
reply . response = ret ;
2009-09-16 20:07:46 -04:00
cpg_send_reply ( & reply , sizeof ( reply ) , nodeid , seqno ) ;
}
static int
do_request ( const char * vm_name , int request , uint32_t seqno )
{
struct ckpt_fence_req freq , * frp ;
size_t retlen ;
uint32_t seq ;
int ret ;
memset ( & freq , 0 , sizeof ( freq ) ) ;
snprintf ( freq . vm_name , sizeof ( freq . vm_name ) , vm_name ) ;
freq . request = request ;
freq . seqno = seqno ;
if ( cpg_send_req ( & freq , sizeof ( freq ) , & seq ) ! = 0 ) {
printf ( " Failed to send \n " ) ;
return 1 ;
}
if ( cpg_wait_reply ( ( void * ) & frp , & retlen , seq ) ! = 0 ) {
printf ( " Failed to receive \n " ) ;
return 1 ;
}
ret = frp - > response ;
free ( frp ) ;
return ret ;
}
static int
checkpoint_null ( const char * vm_name , void * priv )
2009-09-15 10:29:08 -04:00
{
VALIDATE ( priv ) ;
printf ( " [CKPT] Null operation on %s \n " , vm_name ) ;
return 1 ;
}
static int
2010-01-13 13:52:55 -05:00
checkpoint_off ( const char * vm_name , const char * src ,
uint32_t seqno , void * priv )
2009-09-15 10:29:08 -04:00
{
VALIDATE ( priv ) ;
printf ( " [CKPT] OFF operation on %s seq %d \n " , vm_name , seqno ) ;
2009-09-16 20:07:46 -04:00
return do_request ( vm_name , FENCE_OFF , seqno ) ;
2009-09-15 10:29:08 -04:00
}
static int
2010-01-13 13:52:55 -05:00
checkpoint_on ( const char * vm_name , const char * src ,
uint32_t seqno , void * priv )
2009-09-15 10:29:08 -04:00
{
VALIDATE ( priv ) ;
printf ( " [CKPT] ON operation on %s seq %d \n " , vm_name , seqno ) ;
return 1 ;
}
static int
2009-09-16 20:07:46 -04:00
checkpoint_devstatus ( void * priv )
2009-09-15 10:29:08 -04:00
{
printf ( " [CKPT] Device status \n " ) ;
VALIDATE ( priv ) ;
return 0 ;
}
static int
2009-09-16 20:07:46 -04:00
checkpoint_status ( const char * vm_name , void * priv )
2009-09-15 10:29:08 -04:00
{
VALIDATE ( priv ) ;
printf ( " [CKPT] STATUS operation on %s \n " , vm_name ) ;
2009-09-16 20:07:46 -04:00
return do_request ( vm_name , FENCE_STATUS , 0 ) ;
2009-09-15 10:29:08 -04:00
}
static int
2010-01-13 13:52:55 -05:00
checkpoint_reboot ( const char * vm_name , const char * src ,
uint32_t seqno , void * priv )
2009-09-15 10:29:08 -04:00
{
VALIDATE ( priv ) ;
printf ( " [CKPT] REBOOT operation on %s seq %d \n " , vm_name , seqno ) ;
2009-09-16 20:07:46 -04:00
return do_request ( vm_name , FENCE_REBOOT , 0 ) ;
2009-09-15 10:29:08 -04:00
}
2009-12-07 12:55:48 -05:00
static int
checkpoint_hostlist ( hostlist_callback callback , void * arg , void * priv )
{
VALIDATE ( priv ) ;
printf ( " [CKPT] HOSTLIST operation \n " ) ;
return 1 ;
}
2009-09-15 10:29:08 -04:00
static int
2009-09-16 20:07:46 -04:00
checkpoint_init ( backend_context_t * c , config_object_t * config )
2009-09-15 10:29:08 -04:00
{
2009-09-16 20:07:46 -04:00
char value [ 1024 ] ;
2009-09-15 10:29:08 -04:00
struct check_info * info = NULL ;
2009-09-16 20:07:46 -04:00
int x ;
2009-09-15 10:29:08 -04:00
2009-09-16 20:07:46 -04:00
# ifdef _MODULE
if ( sc_get ( config , " fence_virtd/@debug " , value , sizeof ( value ) ) = = 0 )
dset ( atoi ( value ) ) ;
# endif
if ( sc_get ( config , " backends/libvirt/@uri " ,
value , sizeof ( value ) ) = = 0 ) {
uri = strdup ( value ) ;
if ( ! uri ) {
free ( info ) ;
return - 1 ;
}
dbg_printf ( 1 , " Using %s \n " , uri ) ;
}
if ( sc_get ( config , " backends/checkpoint/@uri " ,
value , sizeof ( value ) ) = = 0 ) {
if ( uri )
free ( uri ) ;
uri = strdup ( value ) ;
if ( ! uri ) {
free ( info ) ;
return - 1 ;
}
dbg_printf ( 1 , " Using %s \n " , uri ) ;
}
2010-01-04 14:12:21 -05:00
/* Naming scheme is no longer a top-level config option.
* However , we retain it here for configuration compatibility with
* versions 0.1 .3 and previous .
*/
2009-09-16 20:07:46 -04:00
if ( sc_get ( config , " fence_virtd/@name_mode " ,
value , sizeof ( value ) - 1 ) = = 0 ) {
dbg_printf ( 1 , " Got %s for name_mode \n " , value ) ;
if ( ! strcasecmp ( value , " uuid " ) ) {
use_uuid = 1 ;
} else if ( ! strcasecmp ( value , " name " ) ) {
use_uuid = 0 ;
} else {
dbg_printf ( 1 , " Unsupported name_mode: %s \n " , value ) ;
}
}
2010-01-04 14:12:21 -05:00
if ( sc_get ( config , " backends/checkpoint/@name_mode " ,
value , sizeof ( value ) - 1 ) = = 0 ) {
dbg_printf ( 1 , " Got %s for name_mode \n " , value ) ;
if ( ! strcasecmp ( value , " uuid " ) ) {
use_uuid = 1 ;
} else if ( ! strcasecmp ( value , " name " ) ) {
use_uuid = 0 ;
} else {
dbg_printf ( 1 , " Unsupported name_mode: %s \n " , value ) ;
}
}
2009-09-16 20:07:46 -04:00
if ( cpg_start ( PACKAGE_NAME , do_real_work ) < 0 ) {
return - 1 ;
}
2009-09-15 10:29:08 -04:00
info = malloc ( sizeof ( * info ) ) ;
if ( ! info )
return - 1 ;
memset ( info , 0 , sizeof ( * info ) ) ;
info - > magic = MAGIC ;
2009-09-16 20:07:46 -04:00
x = 0 ;
while ( ( checkpoint_handle = ckpt_init (
" vm_states " , 262144 , 4096 , 64 , 10
) ) = = NULL ) {
if ( ! x ) {
dbg_printf ( 1 , " Could not initialize "
" saCkPt; retrying... \n " ) ;
x = 1 ;
}
sleep ( 3 ) ;
}
if ( x )
dbg_printf ( 1 , " Checkpoint initialized \n " ) ;
update_local_vms ( ) ;
2009-09-15 10:29:08 -04:00
* c = ( void * ) info ;
return 0 ;
}
static int
2009-09-16 20:07:46 -04:00
checkpoint_shutdown ( backend_context_t c )
2009-09-15 10:29:08 -04:00
{
struct check_info * info = ( struct check_info * ) c ;
VALIDATE ( info ) ;
info - > magic = 0 ;
free ( info ) ;
2009-09-16 20:07:46 -04:00
cpg_stop ( ) ;
2009-09-15 10:29:08 -04:00
return 0 ;
}
2009-09-16 20:07:46 -04:00
static fence_callbacks_t checkpoint_callbacks = {
. null = checkpoint_null ,
. off = checkpoint_off ,
. on = checkpoint_on ,
. reboot = checkpoint_reboot ,
. status = checkpoint_status ,
2009-12-07 12:55:48 -05:00
. devstatus = checkpoint_devstatus ,
. hostlist = checkpoint_hostlist
2009-09-15 10:29:08 -04:00
} ;
2009-09-16 20:07:46 -04:00
static backend_plugin_t checkpoint_plugin = {
2009-09-15 10:29:08 -04:00
. name = NAME ,
. version = VERSION ,
2009-09-16 20:07:46 -04:00
. callbacks = & checkpoint_callbacks ,
. init = checkpoint_init ,
. cleanup = checkpoint_shutdown ,
2009-09-15 10:29:08 -04:00
} ;
# ifdef _MODULE
double
BACKEND_VER_SYM ( void )
{
return PLUGIN_VERSION_BACKEND ;
}
const backend_plugin_t *
BACKEND_INFO_SYM ( void )
{
2009-09-16 20:07:46 -04:00
return & checkpoint_plugin ;
2009-09-15 10:29:08 -04:00
}
# else
static void __attribute__ ( ( constructor ) )
2009-09-16 20:07:46 -04:00
checkpoint_register_plugin ( void )
2009-09-15 10:29:08 -04:00
{
2009-09-16 20:07:46 -04:00
plugin_reg_backend ( & checkpoint_plugin ) ;
2009-09-15 10:29:08 -04:00
}
# endif