2002-02-11 15:42:34 +00:00
/*
2008-01-30 14:00:02 +00:00
* Copyright ( C ) 2001 - 2004 Sistina Software , Inc . All rights reserved .
2014-01-27 12:11:09 +01:00
* Copyright ( C ) 2004 - 2014 Red Hat , Inc . All rights reserved .
2002-02-11 15:42:34 +00:00
*
2004-03-30 19:35:44 +00:00
* This file is part of LVM2 .
2002-02-11 15:42:34 +00:00
*
2004-03-30 19:35:44 +00:00
* This copyrighted material is made available to anyone wishing to use ,
* modify , copy , or redistribute it subject to the terms and conditions
2007-08-20 20:55:30 +00:00
* of the GNU Lesser General Public License v .2 .1 .
2004-03-30 19:35:44 +00:00
*
2007-08-20 20:55:30 +00:00
* You should have received a copy of the GNU Lesser General Public License
2004-03-30 19:35:44 +00:00
* along with this program ; if not , write to the Free Software Foundation ,
2016-01-21 11:49:46 +01:00
* Inc . , 51 Franklin Street , Fifth Floor , Boston , MA 02110 - 1301 USA
2002-02-11 15:42:34 +00:00
*/
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 "activate.h"
# include "config.h"
# include "defaults.h"
# include "lvm-string.h"
2014-05-01 17:37:14 +01:00
# include "lvm-flock.h"
2008-04-15 14:46:19 +00:00
# include "lvmcache.h"
2002-02-11 15:42:34 +00:00
# include <limits.h>
# include <unistd.h>
# include <sys/stat.h>
# include <fcntl.h>
# include <signal.h>
2014-03-19 00:22:18 +01:00
static char _lock_dir [ PATH_MAX ] ;
2002-02-11 15:42:34 +00:00
2002-12-19 23:25:55 +00:00
static void _fin_file_locking ( void )
2002-02-11 15:42:34 +00:00
{
2014-05-01 17:37:14 +01:00
release_flocks ( 1 ) ;
2003-05-06 12:03:13 +00:00
}
static void _reset_file_locking ( void )
{
2014-05-01 17:37:14 +01:00
release_flocks ( 0 ) ;
2002-02-11 15:42:34 +00:00
}
2002-12-19 23:25:55 +00:00
static int _file_lock_resource ( struct cmd_context * cmd , const char * resource ,
2014-09-22 15:50:07 +02:00
uint32_t flags , const struct logical_volume * lv )
2002-02-11 15:42:34 +00:00
{
char lockfile [ PATH_MAX ] ;
2010-08-17 19:25:05 +00:00
unsigned origin_only = ( flags & LCK_ORIGIN_ONLY ) ? 1 : 0 ;
2011-09-27 22:43:40 +00:00
unsigned revert = ( flags & LCK_REVERT ) ? 1 : 0 ;
2002-02-11 15:42:34 +00:00
switch ( flags & LCK_SCOPE_MASK ) {
2014-06-20 13:24:02 +01:00
case LCK_ACTIVATION :
if ( dm_snprintf ( lockfile , sizeof ( lockfile ) ,
" %s/A_%s " , _lock_dir , resource + 1 ) < 0 ) {
log_error ( " Too long locking filename %s/A_%s. " , _lock_dir , resource + 1 ) ;
return 0 ;
}
if ( ! lock_file ( lockfile , flags ) )
return_0 ;
break ;
2002-02-11 15:42:34 +00:00
case LCK_VG :
2014-10-21 14:47:46 +02:00
if ( ! strcmp ( resource , VG_SYNC_NAMES ) ) {
2011-01-12 20:42:50 +00:00
fs_unlock ( ) ;
2014-10-21 14:47:46 +02:00
} else if ( strcmp ( resource , VG_GLOBAL ) )
/* Skip cache refresh for VG_GLOBAL - the caller handles it */
lvmcache_drop_metadata ( resource , 0 ) ;
2011-01-12 20:42:50 +00:00
2008-05-09 19:26:58 +00:00
/* LCK_CACHE does not require a real lock */
if ( flags & LCK_CACHE )
2008-04-15 14:46:19 +00:00
break ;
2008-05-09 19:26:58 +00:00
2011-01-05 15:10:30 +00:00
if ( is_orphan_vg ( resource ) | | is_global_vg ( resource ) ) {
if ( dm_snprintf ( lockfile , sizeof ( lockfile ) ,
" %s/P_%s " , _lock_dir , resource + 1 ) < 0 ) {
log_error ( " Too long locking filename %s/P_%s. " ,
_lock_dir , resource + 1 ) ;
return 0 ;
}
} else
if ( dm_snprintf ( lockfile , sizeof ( lockfile ) ,
" %s/V_%s " , _lock_dir , resource ) < 0 ) {
log_error ( " Too long locking filename %s/V_%s. " ,
_lock_dir , resource ) ;
return 0 ;
}
2003-07-04 22:34:56 +00:00
2014-05-01 17:37:14 +01:00
if ( ! lock_file ( lockfile , flags ) )
2008-04-03 18:56:40 +00:00
return_0 ;
2002-02-11 15:42:34 +00:00
break ;
case LCK_LV :
2002-02-20 21:30:27 +00:00
switch ( flags & LCK_TYPE_MASK ) {
2002-04-04 11:18:45 +00:00
case LCK_UNLOCK :
2011-09-27 22:43:40 +00:00
log_very_verbose ( " Unlocking LV %s%s%s " , resource , origin_only ? " without snapshots " : " " , revert ? " (reverting) " : " " ) ;
2015-11-25 11:10:32 +01:00
if ( ! lv_resume_if_active ( cmd , resource , origin_only , 0 , revert , lv_committed ( lv ) ) )
2002-02-25 12:56:16 +00:00
return 0 ;
2002-02-20 21:30:27 +00:00
break ;
2004-05-05 12:03:07 +00:00
case LCK_NULL :
2007-11-16 21:16:20 +00:00
log_very_verbose ( " Locking LV %s (NL) " , resource ) ;
2015-11-25 11:10:32 +01:00
if ( ! lv_deactivate ( cmd , resource , lv_committed ( lv ) ) )
2004-05-05 12:03:07 +00:00
return 0 ;
break ;
2002-02-27 12:26:41 +00:00
case LCK_READ :
2007-11-16 21:16:20 +00:00
log_very_verbose ( " Locking LV %s (R) " , resource ) ;
2015-11-25 11:10:32 +01:00
if ( ! lv_activate_with_filter ( cmd , resource , 0 , ( lv - > status & LV_NOSCAN ) ? 1 : 0 ,
( lv - > status & LV_TEMPORARY ) ? 1 : 0 , lv_committed ( lv ) ) )
2002-02-27 12:26:41 +00:00
return 0 ;
break ;
2007-11-16 21:16:20 +00:00
case LCK_PREAD :
log_very_verbose ( " Locking LV %s (PR) - ignored " , resource ) ;
break ;
2002-02-20 21:30:27 +00:00
case LCK_WRITE :
2010-08-17 19:25:05 +00:00
log_very_verbose ( " Locking LV %s (W)%s " , resource , origin_only ? " without snapshots " : " " ) ;
2015-11-25 11:10:32 +01:00
if ( ! lv_suspend_if_active ( cmd , resource , origin_only , 0 , lv_committed ( lv ) , lv ) )
2002-02-25 12:56:16 +00:00
return 0 ;
2002-02-20 21:30:27 +00:00
break ;
2002-02-27 12:26:41 +00:00
case LCK_EXCL :
2007-11-16 21:16:20 +00:00
log_very_verbose ( " Locking LV %s (EX) " , resource ) ;
2015-11-25 11:10:32 +01:00
if ( ! lv_activate_with_filter ( cmd , resource , 1 , ( lv - > status & LV_NOSCAN ) ? 1 : 0 ,
( lv - > status & LV_TEMPORARY ) ? 1 : 0 , lv_committed ( lv ) ) )
2002-02-27 12:26:41 +00:00
return 0 ;
break ;
2002-02-20 21:30:27 +00:00
default :
break ;
}
2002-02-25 12:56:16 +00:00
break ;
default :
log_error ( " Unrecognised lock scope: %d " ,
flags & LCK_SCOPE_MASK ) ;
return 0 ;
2002-02-20 21:30:27 +00:00
}
2002-02-25 12:56:16 +00:00
return 1 ;
}
2011-08-09 11:44:57 +00:00
int init_file_locking ( struct locking_type * locking , struct cmd_context * cmd ,
int suppress_messages )
2002-02-11 15:42:34 +00:00
{
2011-04-08 14:13:08 +00:00
int r ;
2012-02-08 11:17:34 +00:00
const char * locking_dir ;
2011-04-08 14:13:08 +00:00
2014-05-01 17:37:14 +01:00
init_flock ( cmd ) ;
2002-12-19 23:25:55 +00:00
locking - > lock_resource = _file_lock_resource ;
2003-05-06 12:03:13 +00:00
locking - > reset_locking = _reset_file_locking ;
2002-12-19 23:25:55 +00:00
locking - > fin_locking = _fin_file_locking ;
2004-03-26 20:17:11 +00:00
locking - > flags = 0 ;
2002-02-11 15:42:34 +00:00
/* Get lockfile directory from config file */
2013-06-25 12:29:54 +02:00
locking_dir = find_config_tree_str ( cmd , global_locking_dir_CFG , NULL ) ;
2012-02-08 11:17:34 +00:00
if ( strlen ( locking_dir ) > = sizeof ( _lock_dir ) ) {
log_error ( " Path for locking_dir %s is invalid. " , locking_dir ) ;
return 0 ;
}
strcpy ( _lock_dir , locking_dir ) ;
2002-02-11 15:42:34 +00:00
2010-12-13 10:43:56 +00:00
( void ) dm_prepare_selinux_context ( _lock_dir , S_IFDIR ) ;
r = dm_create_dir ( _lock_dir ) ;
( void ) dm_prepare_selinux_context ( NULL , 0 ) ;
if ( ! r )
2002-02-11 15:42:34 +00:00
return 0 ;
2002-07-10 20:43:32 +00:00
/* Trap a read-only file system */
if ( ( access ( _lock_dir , R_OK | W_OK | X_OK ) = = - 1 ) & & ( errno = = EROFS ) )
return 0 ;
2002-02-11 15:42:34 +00:00
return 1 ;
}