2002-02-11 15:42:34 +00:00
/*
* Copyright ( C ) 2001 Sistina Software ( UK ) Limited .
*
* This file is released under the LGPL .
*
*/
2002-11-18 14:01:16 +00:00
# include "lib.h"
2002-02-11 15:42:34 +00:00
# include "locking.h"
# include "locking_types.h"
# include "lvm-string.h"
# include "activate.h"
2002-11-01 19:57:25 +00:00
# include "toolcontext.h"
2002-02-11 15:42:34 +00:00
# include <signal.h>
2002-11-01 19:57:25 +00:00
# include <sys/stat.h>
2003-01-03 21:10:28 +00:00
# include <sys/mman.h>
2002-11-01 19:57:25 +00:00
# include <limits.h>
2002-02-11 15:42:34 +00:00
static struct locking_type _locking ;
2002-03-15 16:07:38 +00:00
static sigset_t _oldset ;
2002-02-11 15:42:34 +00:00
2002-03-15 16:07:38 +00:00
static int _lock_count = 0 ; /* Number of locks held */
2003-01-03 21:10:28 +00:00
static int _write_lock_held = 0 ;
2002-03-15 16:07:38 +00:00
static int _signals_blocked = 0 ;
2002-02-11 15:42:34 +00:00
2003-01-03 21:10:28 +00:00
static void _block_signals ( int flags )
2002-02-11 15:42:34 +00:00
{
2002-03-15 16:07:38 +00:00
sigset_t set ;
2002-02-11 15:42:34 +00:00
2003-01-03 21:10:28 +00:00
if ( ! _write_lock_held & & ( flags & LCK_SCOPE_MASK ) = = LCK_LV & &
( flags & LCK_TYPE_MASK ) = = LCK_WRITE ) {
if ( mlockall ( MCL_CURRENT | MCL_FUTURE ) )
log_sys_error ( " mlockall " , " " ) ;
else {
log_very_verbose ( " Locking memory " ) ;
_write_lock_held = 1 ;
}
}
2002-03-15 16:07:38 +00:00
if ( _signals_blocked )
2002-02-11 15:42:34 +00:00
return ;
2002-03-15 16:07:38 +00:00
if ( sigfillset ( & set ) ) {
log_sys_error ( " sigfillset " , " _block_signals " ) ;
return ;
}
if ( sigprocmask ( SIG_SETMASK , & set , & _oldset ) ) {
log_sys_error ( " sigprocmask " , " _block_signals " ) ;
return ;
}
2002-02-11 15:42:34 +00:00
2002-03-15 16:07:38 +00:00
_signals_blocked = 1 ;
2002-02-11 15:42:34 +00:00
return ;
}
2002-03-15 16:07:38 +00:00
static void _unblock_signals ( void )
2002-02-11 15:42:34 +00:00
{
2003-01-03 21:10:28 +00:00
if ( ! _lock_count & & _write_lock_held ) {
if ( munlockall ( ) ) {
log_very_verbose ( " Unlocking memory " ) ;
log_sys_error ( " munlockall " , " " ) ;
}
_write_lock_held = 0 ;
}
2002-03-15 16:07:38 +00:00
/* Don't unblock signals while any locks are held */
if ( ! _signals_blocked | | _lock_count )
2002-02-11 15:42:34 +00:00
return ;
2002-03-15 16:07:38 +00:00
if ( sigprocmask ( SIG_SETMASK , & _oldset , NULL ) ) {
log_sys_error ( " sigprocmask " , " _block_signals " ) ;
return ;
}
2002-02-11 15:42:34 +00:00
2002-03-15 16:07:38 +00:00
_signals_blocked = 0 ;
2002-02-11 15:42:34 +00:00
return ;
}
static inline void _update_lock_count ( int flags )
{
2002-04-04 11:18:45 +00:00
if ( ( flags & LCK_TYPE_MASK ) = = LCK_UNLOCK )
2002-02-11 15:42:34 +00:00
_lock_count - - ;
else
_lock_count + + ;
}
/*
* Select a locking type
*/
2002-11-18 14:01:16 +00:00
int init_locking ( int type , struct config_tree * cf )
2002-02-11 15:42:34 +00:00
{
switch ( type ) {
case 0 :
2002-04-11 14:10:32 +00:00
init_no_locking ( & _locking , cf ) ;
2002-03-25 18:50:37 +00:00
log_print ( " WARNING: Locking disabled. Be careful! "
2002-02-11 15:42:34 +00:00
" This could corrupt your metadata. " ) ;
2002-07-10 20:43:32 +00:00
return 1 ;
2002-02-11 15:42:34 +00:00
case 1 :
if ( ! init_file_locking ( & _locking , cf ) )
2002-07-10 20:43:32 +00:00
break ;
2002-02-11 15:42:34 +00:00
log_very_verbose ( " File-based locking enabled. " ) ;
2002-07-10 20:43:32 +00:00
return 1 ;
2002-04-08 16:04:50 +00:00
2003-03-24 18:08:53 +00:00
# ifdef HAVE_LIBDL
2002-02-11 15:42:34 +00:00
case 2 :
2002-04-08 16:04:50 +00:00
if ( ! init_external_locking ( & _locking , cf ) )
2002-07-10 20:43:32 +00:00
break ;
2002-04-08 16:04:50 +00:00
log_very_verbose ( " External locking enabled. " ) ;
2002-07-10 20:43:32 +00:00
return 1 ;
2003-03-24 18:08:53 +00:00
# endif
2002-04-08 16:04:50 +00:00
2002-02-11 15:42:34 +00:00
default :
log_error ( " Unknown locking type requested. " ) ;
return 0 ;
}
2002-07-10 20:43:32 +00:00
if ( ! ignorelockingfailure ( ) )
return 0 ;
2002-11-01 19:57:25 +00:00
/* FIXME Ensure only read ops are permitted */
2002-07-10 20:43:32 +00:00
log_verbose ( " Locking disabled - only read operations permitted. " ) ;
init_no_locking ( & _locking , cf ) ;
2002-02-11 15:42:34 +00:00
return 1 ;
}
void fin_locking ( void )
{
_locking . fin_locking ( ) ;
}
2002-11-01 19:57:25 +00:00
/*
* Does the LVM1 driver know of this VG name ?
*/
int check_lvm1_vg_inactive ( struct cmd_context * cmd , const char * vgname )
{
struct stat info ;
char path [ PATH_MAX ] ;
2002-11-18 14:01:16 +00:00
/* We'll allow operations on orphans */
if ( ! * vgname )
return 1 ;
if ( lvm_snprintf ( path , sizeof ( path ) , " %s/lvm/VGs/%s " , cmd - > proc_dir ,
vgname ) < 0 ) {
2002-11-01 19:57:25 +00:00
log_error ( " LVM1 proc VG pathname too long for %s " , vgname ) ;
return 0 ;
}
if ( stat ( path , & info ) = = 0 ) {
log_error ( " %s exists: Is the original LVM driver using "
" this volume group? " , path ) ;
return 0 ;
} else if ( errno ! = ENOENT & & errno ! = ENOTDIR ) {
log_sys_error ( " stat " , path ) ;
return 0 ;
}
return 1 ;
}
2002-02-11 15:42:34 +00:00
/*
2002-03-15 16:07:38 +00:00
* VG locking is by VG name .
* FIXME This should become VG uuid .
2002-02-11 15:42:34 +00:00
*/
2002-12-19 23:25:55 +00:00
static int _lock_vol ( struct cmd_context * cmd , const char * resource , int flags )
2002-03-05 20:03:09 +00:00
{
2003-01-03 21:10:28 +00:00
_block_signals ( flags ) ;
2002-03-05 20:03:09 +00:00
if ( ! ( _locking . lock_resource ( cmd , resource , flags ) ) ) {
2002-03-15 16:07:38 +00:00
_unblock_signals ( ) ;
2002-03-05 20:03:09 +00:00
return 0 ;
}
_update_lock_count ( flags ) ;
2002-03-15 16:07:38 +00:00
_unblock_signals ( ) ;
2002-03-05 20:03:09 +00:00
return 1 ;
}
2002-02-25 12:56:16 +00:00
int lock_vol ( struct cmd_context * cmd , const char * vol , int flags )
2002-02-11 15:42:34 +00:00
{
char resource [ 258 ] ;
switch ( flags & LCK_SCOPE_MASK ) {
2002-11-01 19:57:25 +00:00
case LCK_VG :
/* Lock VG to change on-disk metadata. */
/* If LVM1 driver knows about the VG, it can't be accessed. */
if ( ! check_lvm1_vg_inactive ( cmd , vol ) )
return 0 ;
case LCK_LV :
/* Suspend LV if it's active. */
2002-12-19 23:25:55 +00:00
strncpy ( resource , vol , sizeof ( resource ) ) ;
2002-02-11 15:42:34 +00:00
break ;
default :
log_error ( " Unrecognised lock scope: %d " ,
flags & LCK_SCOPE_MASK ) ;
return 0 ;
}
2002-03-05 20:03:09 +00:00
if ( ! _lock_vol ( cmd , resource , flags ) )
2002-02-11 15:42:34 +00:00
return 0 ;
2002-03-05 20:03:09 +00:00
/* Perform immediate unlock unless LCK_HOLD set */
2002-04-04 11:18:45 +00:00
if ( ! ( flags & LCK_HOLD ) & & ( ( flags & LCK_TYPE_MASK ) ! = LCK_UNLOCK ) ) {
2002-03-15 16:07:38 +00:00
if ( ! _lock_vol ( cmd , resource ,
2002-04-04 11:18:45 +00:00
( flags & ~ LCK_TYPE_MASK ) | LCK_UNLOCK ) )
2002-03-05 20:03:09 +00:00
return 0 ;
}
2002-02-11 15:42:34 +00:00
return 1 ;
}