2009-07-28 18:46:53 +04:00
/*
2017-05-25 06:18:18 +03:00
Copyright Red Hat , Inc . 2006 - 2017
2009-07-28 18:46:53 +04: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 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 >
*/
# include <stdio.h>
# include <stdlib.h>
# include <string.h>
# include <unistd.h>
# include <signal.h>
# include <string.h>
# include <sys/types.h>
# include <sys/stat.h>
# include <sys/wait.h>
# include <sys/un.h>
# include <sys/socket.h>
# include <sys/select.h>
# include <sys/ioctl.h>
# include <arpa/inet.h>
# include <net/if.h>
# include <netinet/in.h>
# include <netdb.h>
# include <sys/time.h>
# include <fcntl.h>
# include <errno.h>
# include <pthread.h>
2009-09-15 21:27:03 +04:00
# include <libvirt/virterror.h>
2009-07-28 18:46:53 +04:00
# include <nss.h>
# include <libgen.h>
2009-09-01 23:13:53 +04:00
# include <syslog.h>
2009-08-18 00:53:58 +04:00
# include <simpleconfig.h>
2010-01-14 21:38:37 +03:00
# include <static_map.h>
2009-07-28 18:46:53 +04:00
# include <server_plugin.h>
/* Local includes */
# include "xvm.h"
# include "simple_auth.h"
# include "options.h"
# include "mcast.h"
# include "tcp.h"
# include "virt.h"
# include "debug.h"
2010-01-09 02:20:33 +03:00
# include "uuid-test.h"
2009-07-28 18:46:53 +04:00
2009-08-12 21:20:50 +04:00
# define NAME "libvirt"
2017-05-28 17:07:54 +03:00
# define VERSION "0.3"
2009-08-12 21:20:50 +04:00
2009-09-01 23:13:53 +04:00
# define MAGIC 0x1e19317a
struct libvirt_info {
int magic ;
2017-05-28 17:07:54 +03:00
config_object_t * config ;
2014-06-27 08:07:46 +04:00
int vp_count ;
virConnectPtr * vp ;
2009-09-01 23:13:53 +04:00
} ;
2009-09-03 18:55:30 +04:00
# define VALIDATE(arg) \
do { \
if ( ! arg | | ( ( struct libvirt_info * ) arg ) - > magic ! = MAGIC ) { \
errno = EINVAL ; \
return - 1 ; \
} \
} while ( 0 )
2009-09-01 23:13:53 +04:00
2017-05-28 17:07:54 +03:00
static void
libvirt_init_libvirt_conf ( struct libvirt_info * info ) {
config_object_t * config = info - > config ;
int i = 0 ;
if ( info - > vp ) {
dbg_printf ( 2 , " Lost libvirtd connection. Reinitializing. \n " ) ;
for ( i = 0 ; i < info - > vp_count ; i + + )
virConnectClose ( info - > vp [ i ] ) ;
free ( info - > vp ) ;
info - > vp = NULL ;
}
info - > vp_count = 0 ;
do {
virConnectPtr vp ;
virConnectPtr * vpl = NULL ;
char conf_attr [ 256 ] ;
char value [ 1024 ] ;
char * uri ;
if ( i ! = 0 ) {
snprintf ( conf_attr , sizeof ( conf_attr ) ,
" backends/libvirt/@uri%d " , i ) ;
} else
snprintf ( conf_attr , sizeof ( conf_attr ) , " backends/libvirt/@uri " ) ;
+ + i ;
if ( sc_get ( config , conf_attr , value , sizeof ( value ) ) ! = 0 )
break ;
uri = value ;
vp = virConnectOpen ( uri ) ;
if ( ! vp ) {
dbg_printf ( 1 , " [libvirt:INIT] Failed to connect to URI: %s \n " , uri ) ;
continue ;
}
vpl = realloc ( info - > vp , sizeof ( * info - > vp ) * ( info - > vp_count + 1 ) ) ;
if ( ! vpl ) {
dbg_printf ( 1 , " [libvirt:INIT] Out of memory allocating URI: %s \n " ,
uri ) ;
virConnectClose ( vp ) ;
continue ;
}
info - > vp = vpl ;
info - > vp [ info - > vp_count + + ] = vp ;
if ( i > 1 )
dbg_printf ( 1 , " [libvirt:INIT] Added URI%d %s \n " , i - 1 , uri ) ;
else
dbg_printf ( 1 , " [libvirt:INIT] Added URI %s \n " , uri ) ;
} while ( 1 ) ;
}
static int
libvirt_bad_connections ( struct libvirt_info * info ) {
int bad = 0 ;
int i ;
for ( i = 0 ; i < info - > vp_count ; i + + ) {
/*
* * Send a dummy command to trigger an error if libvirtd
* * died or restarted
*/
virConnectNumOfDomains ( info - > vp [ i ] ) ;
if ( ! virConnectIsAlive ( info - > vp [ i ] ) ) {
dbg_printf ( 1 , " libvirt connection %d is dead \n " , i ) ;
bad + + ;
}
}
if ( info - > vp_count < 1 | | bad )
libvirt_init_libvirt_conf ( info ) ;
return bad | | info - > vp_count < 1 ;
}
static void
libvirt_validate_connections ( struct libvirt_info * info ) {
while ( 1 ) {
if ( libvirt_bad_connections ( info ) )
sleep ( 1 ) ;
else
break ;
}
}
2009-07-28 18:46:53 +04:00
static int
libvirt_null ( const char * vm_name , void * priv )
{
2014-06-27 08:07:46 +04:00
dbg_printf ( 5 , " ENTER %s %s \n " , __FUNCTION__ , vm_name ) ;
2009-07-28 18:46:53 +04:00
printf ( " NULL operation: returning failure \n " ) ;
return 1 ;
}
static int
2014-06-27 08:07:46 +04:00
libvirt_off ( const char * vm_name , const char * src , uint32_t seqno , void * priv )
2009-07-28 18:46:53 +04:00
{
2009-09-01 23:13:53 +04:00
struct libvirt_info * info = ( struct libvirt_info * ) priv ;
2009-07-28 18:46:53 +04:00
2014-06-27 08:07:46 +04:00
dbg_printf ( 5 , " ENTER %s %s %u \n " , __FUNCTION__ , vm_name , seqno ) ;
2009-09-01 23:13:53 +04:00
VALIDATE ( info ) ;
2009-07-28 18:46:53 +04:00
2017-05-28 17:07:54 +03:00
libvirt_validate_connections ( info ) ;
2017-05-25 06:18:18 +03:00
return vm_off ( info - > vp , info - > vp_count , vm_name ) ;
2009-07-28 18:46:53 +04:00
}
static int
2014-06-27 08:07:46 +04:00
libvirt_on ( const char * vm_name , const char * src , uint32_t seqno , void * priv )
2009-07-28 18:46:53 +04:00
{
2009-09-15 23:24:56 +04:00
struct libvirt_info * info = ( struct libvirt_info * ) priv ;
2014-06-27 08:07:46 +04:00
dbg_printf ( 5 , " ENTER %s %s %u \n " , __FUNCTION__ , vm_name , seqno ) ;
2009-09-15 23:24:56 +04:00
VALIDATE ( info ) ;
2017-05-28 17:07:54 +03:00
libvirt_validate_connections ( info ) ;
2017-05-25 06:18:18 +03:00
return vm_on ( info - > vp , info - > vp_count , vm_name ) ;
2009-07-28 18:46:53 +04:00
}
static int
libvirt_devstatus ( void * priv )
{
dbg_printf ( 5 , " %s --- \n " , __FUNCTION__ ) ;
if ( priv )
return 0 ;
return 1 ;
}
static int
libvirt_status ( const char * vm_name , void * priv )
{
2009-09-01 23:13:53 +04:00
struct libvirt_info * info = ( struct libvirt_info * ) priv ;
2009-07-28 18:46:53 +04:00
2014-06-27 08:07:46 +04:00
dbg_printf ( 5 , " ENTER %s %s \n " , __FUNCTION__ , vm_name ) ;
2009-09-01 23:13:53 +04:00
VALIDATE ( info ) ;
2009-07-28 18:46:53 +04:00
2017-05-28 17:07:54 +03:00
libvirt_validate_connections ( info ) ;
2017-05-25 06:18:18 +03:00
return vm_status ( info - > vp , info - > vp_count , vm_name ) ;
2009-07-28 18:46:53 +04:00
}
static int
2014-06-27 08:07:46 +04:00
libvirt_reboot ( const char * vm_name , const char * src , uint32_t seqno , void * priv )
2009-07-28 18:46:53 +04:00
{
2009-09-01 23:13:53 +04:00
struct libvirt_info * info = ( struct libvirt_info * ) priv ;
2009-07-28 18:46:53 +04:00
2014-06-27 08:07:46 +04:00
dbg_printf ( 5 , " ENTER %s %s %u \n " , __FUNCTION__ , vm_name , seqno ) ;
2009-09-01 23:13:53 +04:00
VALIDATE ( info ) ;
2014-06-27 08:07:46 +04:00
2017-05-28 17:07:54 +03:00
libvirt_validate_connections ( info ) ;
2017-05-25 06:18:18 +03:00
return vm_reboot ( info - > vp , info - > vp_count , vm_name ) ;
2009-07-28 18:46:53 +04:00
}
2009-12-07 20:55:48 +03:00
static int
libvirt_hostlist ( hostlist_callback callback , void * arg , void * priv )
{
struct libvirt_info * info = ( struct libvirt_info * ) priv ;
2017-05-25 06:18:18 +03:00
virt_list_t * vl ;
int x ;
2009-12-07 20:55:48 +03:00
2014-06-27 08:07:46 +04:00
dbg_printf ( 5 , " ENTER %s \n " , __FUNCTION__ ) ;
2009-12-07 20:55:48 +03:00
VALIDATE ( info ) ;
2017-05-28 17:07:54 +03:00
libvirt_validate_connections ( info ) ;
2017-05-25 06:18:18 +03:00
vl = vl_get ( info - > vp , info - > vp_count , 1 ) ;
if ( ! vl )
return 0 ;
2009-12-07 20:55:48 +03:00
2017-05-25 06:18:18 +03:00
for ( x = 0 ; x < vl - > vm_count ; x + + ) {
callback ( vl - > vm_states [ x ] . v_name ,
vl - > vm_states [ x ] . v_uuid ,
vl - > vm_states [ x ] . v_state . s_state , arg ) ;
2014-06-27 08:07:46 +04:00
2017-05-25 06:18:18 +03:00
dbg_printf ( 10 , " [libvirt:HOSTLIST] Sent %s %s %d \n " ,
vl - > vm_states [ x ] . v_name ,
vl - > vm_states [ x ] . v_uuid ,
vl - > vm_states [ x ] . v_state . s_state ) ;
2014-06-27 08:07:46 +04:00
}
2009-12-07 20:55:48 +03:00
2017-05-25 06:18:18 +03:00
vl_free ( vl ) ;
2009-12-07 20:55:48 +03:00
return 0 ;
}
2009-08-12 21:20:50 +04:00
static int
2009-08-28 22:12:46 +04:00
libvirt_init ( backend_context_t * c , config_object_t * config )
2009-07-28 18:46:53 +04:00
{
2009-08-18 00:53:58 +04:00
char value [ 256 ] ;
2009-09-01 23:13:53 +04:00
struct libvirt_info * info = NULL ;
2009-07-28 18:46:53 +04:00
2017-05-28 17:07:54 +03:00
dbg_printf ( 5 , " ENTER [%s:%d %s] \n " , __FILE__ , __LINE__ , __FUNCTION__ ) ;
info = calloc ( 1 , sizeof ( * info ) ) ;
2009-09-01 23:13:53 +04:00
if ( ! info )
return - 1 ;
2017-05-28 17:07:54 +03:00
info - > magic = MAGIC ;
info - > config = config ;
2009-09-01 23:13:53 +04:00
2017-05-28 17:07:54 +03:00
libvirt_init_libvirt_conf ( info ) ;
2009-09-01 23:13:53 +04:00
2009-09-15 23:17:14 +04:00
# ifdef _MODULE
2014-06-27 08:07:46 +04:00
if ( sc_get ( config , " fence_virtd/@debug " , value , sizeof ( value ) ) = = 0 )
2009-09-04 01:45:38 +04:00
dset ( atoi ( value ) ) ;
2009-09-15 23:17:14 +04:00
# endif
2009-09-04 01:45:38 +04:00
2014-06-27 08:07:46 +04:00
if ( info - > vp_count < 1 ) {
dbg_printf ( 1 , " [libvirt:INIT] Could not connect to any hypervisors \n " ) ;
if ( info - > vp )
free ( info - > vp ) ;
2009-09-01 23:13:53 +04:00
free ( info ) ;
2009-07-28 18:46:53 +04:00
return - 1 ;
2009-09-01 23:13:53 +04:00
}
2017-05-28 17:07:54 +03:00
* c = ( void * ) info ;
2009-07-28 18:46:53 +04:00
return 0 ;
}
2009-08-12 21:20:50 +04:00
static int
2009-08-28 22:12:46 +04:00
libvirt_shutdown ( backend_context_t c )
2009-07-28 18:46:53 +04:00
{
2009-09-01 23:13:53 +04:00
struct libvirt_info * info = ( struct libvirt_info * ) c ;
2014-06-27 08:07:46 +04:00
int i ;
int ret = 0 ;
2009-09-01 23:13:53 +04:00
VALIDATE ( info ) ;
2014-06-27 08:07:46 +04:00
for ( i = 0 ; i < info - > vp_count ; i + + ) {
if ( virConnectClose ( info - > vp [ i ] ) < 0 )
ret = - errno ;
2009-09-01 23:13:53 +04:00
}
2014-06-27 08:07:46 +04:00
free ( info - > vp ) ;
2010-01-15 02:46:21 +03:00
free ( info ) ;
2014-06-27 08:07:46 +04:00
return ret ;
2009-07-28 18:46:53 +04:00
}
2009-08-12 21:20:50 +04:00
static fence_callbacks_t libvirt_callbacks = {
2009-07-28 18:46:53 +04:00
. null = libvirt_null ,
. off = libvirt_off ,
. on = libvirt_on ,
. reboot = libvirt_reboot ,
. status = libvirt_status ,
2009-12-07 20:55:48 +03:00
. devstatus = libvirt_devstatus ,
. hostlist = libvirt_hostlist
2009-07-28 18:46:53 +04:00
} ;
2009-08-12 21:20:50 +04:00
2009-09-02 00:22:30 +04:00
static backend_plugin_t libvirt_plugin = {
2009-08-12 21:20:50 +04:00
. name = NAME ,
. version = VERSION ,
. callbacks = & libvirt_callbacks ,
. init = libvirt_init ,
. cleanup = libvirt_shutdown ,
} ;
2009-08-17 17:58:06 +04:00
# ifdef _MODULE
double
BACKEND_VER_SYM ( void )
{
return PLUGIN_VERSION_BACKEND ;
}
2009-09-02 00:22:30 +04:00
const backend_plugin_t *
2009-08-17 17:58:06 +04:00
BACKEND_INFO_SYM ( void )
{
return & libvirt_plugin ;
}
# else
2009-08-12 21:20:50 +04:00
static void __attribute__ ( ( constructor ) )
2009-08-17 17:58:06 +04:00
libvirt_register_plugin ( void )
2009-08-12 21:20:50 +04:00
{
2009-09-15 23:17:14 +04:00
plugin_reg_backend ( & libvirt_plugin ) ;
2009-08-12 21:20:50 +04:00
}
# endif