2001-10-09 16:05:34 +00:00
/*
2008-01-30 14:00:02 +00:00
* Copyright ( C ) 2001 - 2004 Sistina Software , Inc . All rights reserved .
2012-01-12 01:51:56 +00:00
* Copyright ( C ) 2004 - 2012 Red Hat , Inc . All rights reserved .
2001-10-09 16:05:34 +00:00
*
2004-03-30 19:35:44 +00:00
* 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
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 ,
* Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
2001-10-09 16:05:34 +00:00
*/
2002-11-18 14:01:16 +00:00
# include "lib.h"
2001-10-16 16:25:28 +00:00
# include "metadata.h"
2001-10-09 16:05:34 +00:00
# include "activate.h"
2003-07-04 22:34:56 +00:00
# include "memlock.h"
2001-11-15 14:27:51 +00:00
# include "display.h"
2001-11-12 12:20:58 +00:00
# include "fs.h"
2005-10-17 18:00:02 +00:00
# include "lvm-exec.h"
2004-04-08 15:23:23 +00:00
# include "lvm-file.h"
2001-12-31 19:09:51 +00:00
# include "lvm-string.h"
2002-02-25 12:56:16 +00:00
# include "toolcontext.h"
2002-02-26 11:49:17 +00:00
# include "dev_manager.h"
2004-03-08 18:13:22 +00:00
# include "str_list.h"
2004-05-04 21:25:57 +00:00
# include "config.h"
2005-10-25 19:08:21 +00:00
# include "filter.h"
2005-12-02 20:35:07 +00:00
# include "segtype.h"
2010-08-16 22:54:35 +00:00
# include "sharedlib.h"
2002-01-21 11:06:32 +00:00
# include <limits.h>
2002-02-18 15:52:48 +00:00
# include <fcntl.h>
2003-07-04 22:34:56 +00:00
# include <unistd.h>
2002-01-21 11:06:32 +00:00
2002-02-11 15:48:34 +00:00
# define _skip(fmt, args...) log_very_verbose("Skipping: " fmt , ## args)
2001-10-09 16:05:34 +00:00
2004-04-08 15:23:23 +00:00
int lvm1_present ( struct cmd_context * cmd )
{
2011-11-18 19:31:09 +00:00
static char path [ PATH_MAX ] ;
2004-04-08 15:23:23 +00:00
2006-08-21 12:54:53 +00:00
if ( dm_snprintf ( path , sizeof ( path ) , " %s/lvm/global " , cmd - > proc_dir )
2004-04-08 15:23:23 +00:00
< 0 ) {
log_error ( " LVM1 proc global snprintf failed " ) ;
return 0 ;
}
if ( path_exists ( path ) )
return 1 ;
else
return 0 ;
}
2006-10-03 17:55:20 +00:00
int list_segment_modules ( struct dm_pool * mem , const struct lv_segment * seg ,
2008-11-03 22:14:30 +00:00
struct dm_list * modules )
2006-10-03 17:55:20 +00:00
{
unsigned int s ;
struct lv_segment * seg2 , * snap_seg ;
2008-11-03 22:14:30 +00:00
struct dm_list * snh ;
2006-10-03 17:55:20 +00:00
if ( seg - > segtype - > ops - > modules_needed & &
! seg - > segtype - > ops - > modules_needed ( mem , seg , modules ) ) {
log_error ( " module string allocation failed " ) ;
return 0 ;
}
if ( lv_is_origin ( seg - > lv ) )
2008-11-03 22:14:30 +00:00
dm_list_iterate ( snh , & seg - > lv - > snapshot_segs )
2006-10-03 17:55:20 +00:00
if ( ! list_lv_modules ( mem ,
2008-11-03 22:14:30 +00:00
dm_list_struct_base ( snh ,
2006-10-03 17:55:20 +00:00
struct lv_segment ,
origin_list ) - > cow ,
modules ) )
return_0 ;
if ( lv_is_cow ( seg - > lv ) ) {
snap_seg = find_cow ( seg - > lv ) ;
if ( snap_seg - > segtype - > ops - > modules_needed & &
! snap_seg - > segtype - > ops - > modules_needed ( mem , snap_seg ,
modules ) ) {
log_error ( " snap_seg module string allocation failed " ) ;
return 0 ;
}
}
for ( s = 0 ; s < seg - > area_count ; s + + ) {
switch ( seg_type ( seg , s ) ) {
case AREA_LV :
seg2 = find_seg_by_le ( seg_lv ( seg , s ) , seg_le ( seg , s ) ) ;
if ( seg2 & & ! list_segment_modules ( mem , seg2 , modules ) )
return_0 ;
break ;
case AREA_PV :
case AREA_UNASSIGNED :
;
}
}
return 1 ;
}
int list_lv_modules ( struct dm_pool * mem , const struct logical_volume * lv ,
2008-11-03 22:14:30 +00:00
struct dm_list * modules )
2006-10-03 17:55:20 +00:00
{
struct lv_segment * seg ;
2008-11-03 22:14:30 +00:00
dm_list_iterate_items ( seg , & lv - > segments )
2006-10-03 17:55:20 +00:00
if ( ! list_segment_modules ( mem , seg , modules ) )
return_0 ;
return 1 ;
}
2003-01-08 22:44:07 +00:00
# ifndef DEVMAPPER_SUPPORT
void set_activation ( int act )
{
2004-03-19 16:26:46 +00:00
static int warned = 0 ;
if ( warned | | ! act )
return ;
log_error ( " Compiled without libdevmapper support. "
" Can't enable activation. " ) ;
warned = 1 ;
2003-01-08 22:44:07 +00:00
}
int activation ( void )
{
return 0 ;
}
int library_version ( char * version , size_t size )
{
return 0 ;
}
int driver_version ( char * version , size_t size )
{
return 0 ;
}
2005-12-19 21:01:39 +00:00
int target_version ( const char * target_name , uint32_t * maj ,
uint32_t * min , uint32_t * patchlevel )
{
return 0 ;
}
2009-02-28 00:54:06 +00:00
int target_present ( struct cmd_context * cmd , const char * target_name ,
int use_modprobe )
2004-03-26 19:52:09 +00:00
{
return 0 ;
}
2012-01-25 13:12:59 +00:00
int lvm_dm_prefix_check ( int major , int minor , const char * prefix )
2011-11-11 15:11:08 +00:00
{
return 0 ;
}
2012-01-25 13:10:26 +00:00
int lv_info ( struct cmd_context * cmd , const struct logical_volume * lv , int use_layer ,
2010-08-17 16:25:32 +00:00
struct lvinfo * info , int with_open_count , int with_read_ahead )
2003-01-08 22:44:07 +00:00
{
return 0 ;
}
2012-01-25 13:10:26 +00:00
int lv_info_by_lvid ( struct cmd_context * cmd , const char * lvid_s , int use_layer ,
2007-11-12 20:51:54 +00:00
struct lvinfo * info , int with_open_count , int with_read_ahead )
2004-03-26 15:35:01 +00:00
{
return 0 ;
}
2012-01-25 13:12:59 +00:00
int lv_check_not_in_use ( struct cmd_context * cmd __attribute__ ( ( unused ) ) ,
struct logical_volume * lv , struct lvinfo * info )
{
return 0 ;
}
2010-11-30 11:53:31 +00:00
int lv_snapshot_percent ( const struct logical_volume * lv , percent_t * percent )
2003-01-08 22:44:07 +00:00
{
return 0 ;
}
2011-02-18 14:47:28 +00:00
int lv_mirror_percent ( struct cmd_context * cmd , const struct logical_volume * lv ,
2010-11-30 11:53:31 +00:00
int wait , percent_t * percent , uint32_t * event_nr )
2003-04-30 15:26:25 +00:00
{
return 0 ;
}
2012-01-25 13:12:59 +00:00
int lv_raid_percent ( const struct logical_volume * lv , percent_t * percent )
{
return 0 ;
}
int lv_thin_pool_percent ( const struct logical_volume * lv , int metadata ,
percent_t * percent )
{
return 0 ;
}
int lv_thin_percent ( const struct logical_volume * lv , int mapped ,
percent_t * percent )
{
return 0 ;
}
int lv_thin_pool_transaction_id ( const struct logical_volume * lv ,
uint64_t * transaction_id )
{
return 0 ;
}
2012-02-23 22:41:57 +00:00
int lvs_in_vg_activated ( const struct volume_group * vg )
2003-01-08 22:44:07 +00:00
{
return 0 ;
}
2011-02-18 14:29:39 +00:00
int lvs_in_vg_opened ( const struct volume_group * vg )
2003-01-08 22:44:07 +00:00
{
return 0 ;
}
2010-08-17 16:25:32 +00:00
/******
2004-03-08 18:54:13 +00:00
int lv_suspend ( struct cmd_context * cmd , const char * lvid_s )
{
return 1 ;
}
2010-08-17 16:25:32 +00:00
* * * * * * */
2012-01-25 13:12:59 +00:00
int lv_suspend_if_active ( struct cmd_context * cmd , const char * lvid_s , unsigned origin_only , unsigned exclusive )
2003-01-08 22:44:07 +00:00
{
return 1 ;
}
2011-02-18 14:29:39 +00:00
int lv_resume ( struct cmd_context * cmd , const char * lvid_s , unsigned origin_only )
2004-03-08 18:54:13 +00:00
{
return 1 ;
}
2011-02-18 14:29:39 +00:00
int lv_resume_if_active ( struct cmd_context * cmd , const char * lvid_s ,
2011-09-27 22:43:40 +00:00
unsigned origin_only , unsigned exclusive , unsigned revert )
2003-01-08 22:44:07 +00:00
{
return 1 ;
}
int lv_deactivate ( struct cmd_context * cmd , const char * lvid_s )
{
return 1 ;
}
2004-03-08 18:54:13 +00:00
int lv_activation_filter ( struct cmd_context * cmd , const char * lvid_s ,
int * activate_lv )
{
return 1 ;
}
2005-08-14 23:18:28 +00:00
int lv_activate ( struct cmd_context * cmd , const char * lvid_s , int exclusive )
2003-01-08 22:44:07 +00:00
{
return 1 ;
}
2005-08-14 23:18:28 +00:00
int lv_activate_with_filter ( struct cmd_context * cmd , const char * lvid_s , int exclusive )
2004-03-08 18:54:13 +00:00
{
return 1 ;
}
2003-11-12 19:16:48 +00:00
int lv_mknodes ( struct cmd_context * cmd , const struct logical_volume * lv )
{
return 1 ;
}
2006-12-20 16:19:01 +00:00
int pv_uses_vg ( struct physical_volume * pv ,
2006-05-12 19:16:48 +00:00
struct volume_group * vg )
2005-10-25 19:08:21 +00:00
{
return 0 ;
}
2006-05-16 16:48:31 +00:00
void activation_release ( void )
{
}
2003-07-04 22:34:56 +00:00
void activation_exit ( void )
{
}
2011-10-06 14:55:39 +00:00
2012-02-23 22:41:57 +00:00
int lv_is_active ( const struct logical_volume * lv )
2011-02-18 14:29:39 +00:00
{
return 0 ;
}
2012-02-23 22:41:57 +00:00
int lv_is_active_but_not_locally ( const struct logical_volume * lv )
2011-10-06 14:55:39 +00:00
{
return 0 ;
}
2012-02-23 22:41:57 +00:00
int lv_is_active_exclusive ( const struct logical_volume * lv )
2011-10-06 14:55:39 +00:00
{
return 0 ;
}
2012-02-23 22:41:57 +00:00
int lv_is_active_exclusive_locally ( const struct logical_volume * lv )
2011-02-18 14:29:39 +00:00
{
return 0 ;
}
2012-02-23 22:41:57 +00:00
int lv_is_active_exclusive_remotely ( const struct logical_volume * lv )
2011-02-18 14:29:39 +00:00
{
return 0 ;
}
2011-10-06 14:55:39 +00:00
2011-02-18 14:29:39 +00:00
int lv_check_transient ( struct logical_volume * lv )
{
return 1 ;
}
int monitor_dev_for_events ( struct cmd_context * cmd , struct logical_volume * lv ,
2012-01-25 13:12:59 +00:00
const struct lv_activate_opts * laopts , int monitor )
2011-02-18 14:29:39 +00:00
{
return 1 ;
}
2012-01-25 13:12:59 +00:00
/* fs.c */
void fs_unlock ( void )
{
}
/* dev_manager.c */
2012-02-23 22:41:57 +00:00
# include "targets.h"
2012-01-25 13:12:59 +00:00
int add_areas_line ( struct dev_manager * dm , struct lv_segment * seg ,
struct dm_tree_node * node , uint32_t start_area ,
uint32_t areas )
{
return 0 ;
}
int device_is_usable ( struct device * dev )
{
return 0 ;
}
int lv_has_target_type ( struct dm_pool * mem , struct logical_volume * lv ,
const char * layer , const char * target_type )
{
return 0 ;
}
2003-01-08 22:44:07 +00:00
# else /* DEVMAPPER_SUPPORT */
2002-11-18 14:01:16 +00:00
static int _activation = 1 ;
2002-12-19 23:25:55 +00:00
void set_activation ( int act )
2002-11-18 14:01:16 +00:00
{
2002-12-19 23:25:55 +00:00
if ( act = = _activation )
2002-11-18 14:01:16 +00:00
return ;
2002-12-19 23:25:55 +00:00
_activation = act ;
2002-11-18 14:01:16 +00:00
if ( _activation )
log_verbose ( " Activation enabled. Device-mapper kernel "
" driver will be used. " ) ;
else
2007-06-28 17:33:44 +00:00
log_warn ( " WARNING: Activation disabled. No device-mapper "
2005-12-22 16:13:38 +00:00
" interaction will be attempted. " ) ;
2002-11-18 14:01:16 +00:00
}
2002-12-19 23:25:55 +00:00
int activation ( void )
2002-11-18 14:01:16 +00:00
{
return _activation ;
}
2012-01-12 01:51:56 +00:00
static int _passes_volumes_filter ( struct cmd_context * cmd ,
struct logical_volume * lv ,
const struct dm_config_node * cn ,
const char * config_path )
2004-03-08 18:13:22 +00:00
{
2011-08-30 14:55:15 +00:00
const struct dm_config_value * cv ;
2010-12-20 13:12:55 +00:00
const char * str ;
2011-11-18 19:31:09 +00:00
static char path [ PATH_MAX ] ;
2004-03-08 18:13:22 +00:00
2012-01-12 01:51:56 +00:00
log_verbose ( " %s configuration setting defined: "
" Checking the list to match %s/%s " ,
config_path , lv - > vg - > name , lv - > name ) ;
2010-07-12 11:37:49 +00:00
2010-11-05 18:18:11 +00:00
for ( cv = cn - > v ; cv ; cv = cv - > next ) {
2011-08-30 14:55:15 +00:00
if ( cv - > type ! = DM_CFG_STRING ) {
2012-01-12 01:51:56 +00:00
log_error ( " Ignoring invalid string in config file %s " ,
config_path ) ;
2004-03-08 18:13:22 +00:00
continue ;
}
str = cv - > v . str ;
if ( ! * str ) {
2012-01-12 01:51:56 +00:00
log_error ( " Ignoring empty string in config file %s " ,
config_path ) ;
2004-03-08 18:13:22 +00:00
continue ;
}
2010-11-05 18:18:11 +00:00
2004-03-08 18:13:22 +00:00
/* Tag? */
if ( * str = = ' @ ' ) {
str + + ;
if ( ! * str ) {
log_error ( " Ignoring empty tag in config file "
2012-01-12 01:51:56 +00:00
" %s " , config_path ) ;
2004-03-08 18:13:22 +00:00
continue ;
}
/* If any host tag matches any LV or VG tag, activate */
if ( ! strcmp ( str , " * " ) ) {
2010-11-09 12:34:40 +00:00
if ( str_list_match_list ( & cmd - > tags , & lv - > tags , NULL )
2004-03-08 18:13:22 +00:00
| | str_list_match_list ( & cmd - > tags ,
2010-11-09 12:34:40 +00:00
& lv - > vg - > tags , NULL ) )
2004-03-08 18:13:22 +00:00
return 1 ;
else
continue ;
}
/* If supplied tag matches LV or VG tag, activate */
if ( str_list_match_item ( & lv - > tags , str ) | |
str_list_match_item ( & lv - > vg - > tags , str ) )
return 1 ;
else
continue ;
}
2007-07-02 11:17:21 +00:00
if ( ! strchr ( str , ' / ' ) ) {
2004-03-08 18:13:22 +00:00
/* vgname supplied */
if ( ! strcmp ( str , lv - > vg - > name ) )
return 1 ;
else
continue ;
}
/* vgname/lvname */
2006-08-21 12:54:53 +00:00
if ( dm_snprintf ( path , sizeof ( path ) , " %s/%s " , lv - > vg - > name ,
2004-03-08 18:13:22 +00:00
lv - > name ) < 0 ) {
2006-08-21 12:54:53 +00:00
log_error ( " dm_snprintf error from %s/%s " , lv - > vg - > name ,
2004-03-08 18:13:22 +00:00
lv - > name ) ;
continue ;
}
if ( ! strcmp ( path , str ) )
return 1 ;
}
2012-01-12 01:51:56 +00:00
log_verbose ( " No item supplied in %s configuration setting "
" matches %s/%s " , config_path , lv - > vg - > name , lv - > name ) ;
2010-07-12 11:37:49 +00:00
2004-03-08 18:13:22 +00:00
return 0 ;
}
2012-01-12 01:51:56 +00:00
static int _passes_activation_filter ( struct cmd_context * cmd ,
struct logical_volume * lv )
{
const struct dm_config_node * cn ;
if ( ! ( cn = find_config_tree_node ( cmd , " activation/volume_list " ) ) ) {
log_verbose ( " activation/volume_list configuration setting "
" not defined: Checking only host tags for %s/%s " ,
lv - > vg - > name , lv - > name ) ;
/* If no host tags defined, activate */
if ( dm_list_empty ( & cmd - > tags ) )
return 1 ;
/* If any host tag matches any LV or VG tag, activate */
if ( str_list_match_list ( & cmd - > tags , & lv - > tags , NULL ) | |
str_list_match_list ( & cmd - > tags , & lv - > vg - > tags , NULL ) )
return 1 ;
log_verbose ( " No host tag matches %s/%s " ,
lv - > vg - > name , lv - > name ) ;
/* Don't activate */
return 0 ;
}
return _passes_volumes_filter ( cmd , lv , cn , " activation/volume_list " ) ;
}
static int _passes_readonly_filter ( struct cmd_context * cmd ,
struct logical_volume * lv )
{
2012-01-12 09:08:55 +00:00
const struct dm_config_node * cn ;
2012-01-12 01:51:56 +00:00
if ( ! ( cn = find_config_tree_node ( cmd , " activation/read_only_volume_list " ) ) )
return 0 ;
return _passes_volumes_filter ( cmd , lv , cn , " activation/read_only_volume_list " ) ;
}
2002-01-17 16:39:24 +00:00
int library_version ( char * version , size_t size )
{
2002-11-18 14:01:16 +00:00
if ( ! activation ( ) )
return 0 ;
2005-10-17 18:00:02 +00:00
return dm_get_library_version ( version , size ) ;
2002-01-17 16:39:24 +00:00
}
int driver_version ( char * version , size_t size )
{
2002-11-18 14:01:16 +00:00
if ( ! activation ( ) )
return 0 ;
2002-01-17 16:39:24 +00:00
log_very_verbose ( " Getting driver version " ) ;
2005-10-17 18:00:02 +00:00
return dm_driver_version ( version , size ) ;
2002-01-17 16:39:24 +00:00
}
2005-12-19 21:01:39 +00:00
int target_version ( const char * target_name , uint32_t * maj ,
uint32_t * min , uint32_t * patchlevel )
2004-03-26 19:52:09 +00:00
{
int r = 0 ;
struct dm_task * dmt ;
struct dm_versions * target , * last_target ;
log_very_verbose ( " Getting target version for %s " , target_name ) ;
2005-11-08 22:52:26 +00:00
if ( ! ( dmt = dm_task_create ( DM_DEVICE_LIST_VERSIONS ) ) )
return_0 ;
2004-03-26 19:52:09 +00:00
2011-07-01 14:09:19 +00:00
if ( activation_checks ( ) & & ! dm_task_enable_checks ( dmt ) )
goto_out ;
2004-03-26 19:52:09 +00:00
if ( ! dm_task_run ( dmt ) ) {
log_debug ( " Failed to get %s target version " , target_name ) ;
/* Assume this was because LIST_VERSIONS isn't supported */
2012-01-25 22:16:04 +00:00
* maj = 0 ;
* min = 0 ;
* patchlevel = 0 ;
r = 1 ;
goto out ;
2004-03-26 19:52:09 +00:00
}
target = dm_task_get_versions ( dmt ) ;
do {
last_target = target ;
if ( ! strcmp ( target_name , target - > name ) ) {
r = 1 ;
2005-12-19 21:01:39 +00:00
* maj = target - > version [ 0 ] ;
* min = target - > version [ 1 ] ;
* patchlevel = target - > version [ 2 ] ;
2004-03-26 19:52:09 +00:00
goto out ;
}
2010-12-20 13:37:26 +00:00
target = ( struct dm_versions * ) ( ( char * ) target + target - > next ) ;
2004-03-26 19:52:09 +00:00
} while ( last_target ! = target ) ;
out :
dm_task_destroy ( dmt ) ;
return r ;
}
2011-11-11 16:41:37 +00:00
int lvm_dm_prefix_check ( int major , int minor , const char * prefix )
2011-11-11 15:11:08 +00:00
{
struct dm_task * dmt ;
const char * uuid ;
int r ;
if ( ! ( dmt = dm_task_create ( DM_DEVICE_STATUS ) ) )
2011-11-11 15:14:05 +00:00
return_0 ;
2011-11-11 15:11:08 +00:00
if ( ! dm_task_set_minor ( dmt , minor ) | |
! dm_task_set_major ( dmt , major ) | |
! dm_task_run ( dmt ) | |
! ( uuid = dm_task_get_uuid ( dmt ) ) ) {
dm_task_destroy ( dmt ) ;
return 0 ;
}
r = strncasecmp ( uuid , prefix , strlen ( prefix ) ) ;
dm_task_destroy ( dmt ) ;
2011-11-11 15:14:05 +00:00
return r ? 0 : 1 ;
2011-11-11 15:11:08 +00:00
}
2009-02-28 00:54:06 +00:00
int module_present ( struct cmd_context * cmd , const char * target_name )
2005-10-17 18:00:02 +00:00
{
2008-04-07 10:23:47 +00:00
int ret = 0 ;
2005-10-19 13:59:18 +00:00
# ifdef MODPROBE_CMD
2005-10-17 18:00:02 +00:00
char module [ 128 ] ;
2009-02-24 15:48:00 +00:00
const char * argv [ 3 ] ;
2008-04-07 10:23:47 +00:00
if ( dm_snprintf ( module , sizeof ( module ) , " dm-%s " , target_name ) < 0 ) {
log_error ( " module_present module name too long: %s " ,
target_name ) ;
return 0 ;
}
2009-02-24 15:48:00 +00:00
argv [ 0 ] = MODPROBE_CMD ;
argv [ 1 ] = module ;
argv [ 2 ] = NULL ;
2011-01-13 14:51:32 +00:00
ret = exec_cmd ( cmd , argv , NULL , 0 ) ;
2005-10-19 13:59:18 +00:00
# endif
2008-04-07 10:23:47 +00:00
return ret ;
}
2009-02-28 00:54:06 +00:00
int target_present ( struct cmd_context * cmd , const char * target_name ,
int use_modprobe )
2008-04-07 10:23:47 +00:00
{
uint32_t maj , min , patchlevel ;
2005-10-17 18:00:02 +00:00
if ( ! activation ( ) )
return 0 ;
# ifdef MODPROBE_CMD
2005-11-08 22:52:26 +00:00
if ( use_modprobe ) {
2005-12-19 21:01:39 +00:00
if ( target_version ( target_name , & maj , & min , & patchlevel ) )
2005-11-08 22:52:26 +00:00
return 1 ;
2005-10-17 18:00:02 +00:00
2009-02-28 00:54:06 +00:00
if ( ! module_present ( cmd , target_name ) )
2005-11-08 22:52:26 +00:00
return_0 ;
2005-10-17 18:00:02 +00:00
}
# endif
2005-12-19 21:01:39 +00:00
return target_version ( target_name , & maj , & min , & patchlevel ) ;
2005-10-17 18:00:02 +00:00
}
2002-02-26 11:49:17 +00:00
/*
* Returns 1 if info structure populated , else 0 on failure .
*/
2012-01-25 13:10:26 +00:00
int lv_info ( struct cmd_context * cmd , const struct logical_volume * lv , int use_layer ,
2010-02-24 20:00:56 +00:00
struct lvinfo * info , int with_open_count , int with_read_ahead )
2001-11-07 11:51:42 +00:00
{
2003-01-08 22:44:07 +00:00
struct dm_info dminfo ;
2012-01-25 13:10:26 +00:00
const char * layer ;
2002-01-10 23:21:07 +00:00
2002-11-18 14:01:16 +00:00
if ( ! activation ( ) )
return 0 ;
2011-02-03 01:16:35 +00:00
/*
* If open_count info is requested and we have to be sure our own udev
* transactions are finished
* For non - clustered locking type we are only interested for non - delete operation
* in progress - as only those could lead to opened files
*/
if ( with_open_count ) {
if ( locking_is_clustered ( ) )
sync_local_dev_names ( cmd ) ; /* Wait to have udev in sync */
2011-02-04 19:14:39 +00:00
else if ( fs_has_non_delete_ops ( ) )
2011-02-03 01:16:35 +00:00
fs_unlock ( ) ; /* For non clustered - wait if there are non-delete ops */
}
2002-11-18 14:01:16 +00:00
2012-01-25 13:10:26 +00:00
if ( use_layer & & lv_is_thin_pool ( lv ) )
layer = " tpool " ;
else if ( use_layer & & lv_is_origin ( lv ) )
layer = " real " ;
else
layer = NULL ;
if ( ! dev_manager_info ( lv - > vg - > cmd - > mem , lv , layer , with_open_count ,
2010-02-24 20:00:56 +00:00
with_read_ahead , & dminfo , & info - > read_ahead ) )
2005-11-08 22:52:26 +00:00
return_0 ;
2002-01-10 23:21:07 +00:00
2003-01-08 22:44:07 +00:00
info - > exists = dminfo . exists ;
info - > suspended = dminfo . suspended ;
info - > open_count = dminfo . open_count ;
info - > major = dminfo . major ;
info - > minor = dminfo . minor ;
info - > read_only = dminfo . read_only ;
2005-11-08 22:52:26 +00:00
info - > live_table = dminfo . live_table ;
info - > inactive_table = dminfo . inactive_table ;
2003-01-08 22:44:07 +00:00
2005-10-17 18:00:02 +00:00
return 1 ;
2002-02-26 11:49:17 +00:00
}
2002-02-11 17:42:02 +00:00
2012-01-25 13:10:26 +00:00
int lv_info_by_lvid ( struct cmd_context * cmd , const char * lvid_s , int use_layer ,
2007-11-12 20:51:54 +00:00
struct lvinfo * info , int with_open_count , int with_read_ahead )
2004-03-26 15:35:01 +00:00
{
2009-12-01 19:10:23 +00:00
int r ;
2004-03-26 15:35:01 +00:00
struct logical_volume * lv ;
2005-10-31 20:15:28 +00:00
if ( ! ( lv = lv_from_lvid ( cmd , lvid_s , 0 ) ) )
2004-03-26 15:35:01 +00:00
return 0 ;
2012-01-25 13:10:26 +00:00
r = lv_info ( cmd , lv , use_layer , info , with_open_count , with_read_ahead ) ;
2011-08-10 20:25:29 +00:00
release_vg ( lv - > vg ) ;
2009-12-01 19:10:23 +00:00
return r ;
2004-03-26 15:35:01 +00:00
}
2011-09-22 17:33:50 +00:00
int lv_check_not_in_use ( struct cmd_context * cmd __attribute__ ( ( unused ) ) ,
struct logical_volume * lv , struct lvinfo * info )
{
if ( ! info - > exists )
return 1 ;
/* If sysfs is not used, use open_count information only. */
2011-09-26 10:17:51 +00:00
if ( ! * dm_sysfs_dir ( ) ) {
if ( info - > open_count ) {
log_error ( " Logical volume %s/%s in use. " ,
lv - > vg - > name , lv - > name ) ;
return 0 ;
}
return 1 ;
}
2011-09-22 17:33:50 +00:00
if ( dm_device_has_holders ( info - > major , info - > minor ) ) {
log_error ( " Logical volume %s/%s is used by another device. " ,
lv - > vg - > name , lv - > name ) ;
return 0 ;
}
if ( dm_device_has_mounted_fs ( info - > major , info - > minor ) ) {
log_error ( " Logical volume %s/%s contains a filesystem in use. " ,
lv - > vg - > name , lv - > name ) ;
return 0 ;
}
return 1 ;
}
2010-05-24 15:32:20 +00:00
/*
* Returns 1 if percent set , else 0 on failure .
*/
int lv_check_transient ( struct logical_volume * lv )
{
int r ;
struct dev_manager * dm ;
if ( ! activation ( ) )
return 0 ;
2011-06-13 22:28:04 +00:00
log_debug ( " Checking transient status for LV %s/%s " , lv - > vg - > name , lv - > name ) ;
2011-06-11 00:03:06 +00:00
if ( ! ( dm = dev_manager_create ( lv - > vg - > cmd , lv - > vg - > name , 1 ) ) )
2010-05-24 15:32:20 +00:00
return_0 ;
if ( ! ( r = dev_manager_transient ( dm , lv ) ) )
stack ;
dev_manager_destroy ( dm ) ;
return r ;
}
2002-05-09 21:17:57 +00:00
/*
* Returns 1 if percent set , else 0 on failure .
*/
2010-11-30 11:53:31 +00:00
int lv_snapshot_percent ( const struct logical_volume * lv , percent_t * percent )
2002-05-09 21:17:57 +00:00
{
int r ;
struct dev_manager * dm ;
2002-11-18 14:01:16 +00:00
if ( ! activation ( ) )
return 0 ;
2011-06-13 22:28:04 +00:00
log_debug ( " Checking snapshot percent for LV %s/%s " , lv - > vg - > name , lv - > name ) ;
2011-06-11 00:03:06 +00:00
if ( ! ( dm = dev_manager_create ( lv - > vg - > cmd , lv - > vg - > name , 1 ) ) )
2005-11-08 22:52:26 +00:00
return_0 ;
2002-05-09 21:17:57 +00:00
2010-11-30 11:53:31 +00:00
if ( ! ( r = dev_manager_snapshot_percent ( dm , lv , percent ) ) )
2002-05-09 21:17:57 +00:00
stack ;
2002-05-22 14:03:45 +00:00
2002-05-09 21:17:57 +00:00
dev_manager_destroy ( dm ) ;
return r ;
}
2003-04-30 15:26:25 +00:00
/* FIXME Merge with snapshot_percent */
2011-02-18 14:47:28 +00:00
int lv_mirror_percent ( struct cmd_context * cmd , const struct logical_volume * lv ,
2010-11-30 11:53:31 +00:00
int wait , percent_t * percent , uint32_t * event_nr )
2003-04-30 15:26:25 +00:00
{
int r ;
struct dev_manager * dm ;
2004-03-26 19:10:48 +00:00
struct lvinfo info ;
2003-04-30 15:26:25 +00:00
2008-01-16 19:18:51 +00:00
/* If mirrored LV is temporarily shrinked to 1 area (= linear),
* it should be considered in - sync . */
2008-11-03 22:14:30 +00:00
if ( dm_list_size ( & lv - > segments ) = = 1 & & first_seg ( lv ) - > area_count = = 1 ) {
2010-11-30 11:53:31 +00:00
* percent = PERCENT_100 ;
2008-01-16 19:18:51 +00:00
return 1 ;
}
2003-04-30 15:26:25 +00:00
if ( ! activation ( ) )
return 0 ;
2011-06-13 22:28:04 +00:00
log_debug ( " Checking mirror percent for LV %s/%s " , lv - > vg - > name , lv - > name ) ;
2010-08-17 16:25:32 +00:00
if ( ! lv_info ( cmd , lv , 0 , & info , 0 , 0 ) )
2005-11-08 22:52:26 +00:00
return_0 ;
2004-03-26 19:10:48 +00:00
if ( ! info . exists )
return 0 ;
2011-06-11 00:03:06 +00:00
if ( ! ( dm = dev_manager_create ( lv - > vg - > cmd , lv - > vg - > name , 1 ) ) )
2005-11-08 22:52:26 +00:00
return_0 ;
2003-04-30 15:26:25 +00:00
2010-11-30 11:53:31 +00:00
if ( ! ( r = dev_manager_mirror_percent ( dm , lv , wait , percent , event_nr ) ) )
2003-04-30 15:26:25 +00:00
stack ;
dev_manager_destroy ( dm ) ;
return r ;
}
2011-08-11 18:24:40 +00:00
int lv_raid_percent ( const struct logical_volume * lv , percent_t * percent )
{
return lv_mirror_percent ( lv - > vg - > cmd , lv , 0 , percent , NULL ) ;
}
2011-12-21 13:10:05 +00:00
/*
2012-01-19 15:25:37 +00:00
* Returns data or metadata percent usage , depends on metadata 0 / 1.
2011-12-21 13:10:05 +00:00
* Returns 1 if percent set , else 0 on failure .
*/
2012-01-19 15:25:37 +00:00
int lv_thin_pool_percent ( const struct logical_volume * lv , int metadata ,
percent_t * percent )
2011-12-21 13:10:05 +00:00
{
int r ;
struct dev_manager * dm ;
if ( ! activation ( ) )
return 0 ;
2012-01-19 15:25:37 +00:00
log_debug ( " Checking thin %sdata percent for LV %s/%s " ,
( metadata ) ? " meta " : " " , lv - > vg - > name , lv - > name ) ;
2011-12-21 13:10:05 +00:00
if ( ! ( dm = dev_manager_create ( lv - > vg - > cmd , lv - > vg - > name , 1 ) ) )
return_0 ;
2012-01-19 15:25:37 +00:00
if ( ! ( r = dev_manager_thin_pool_percent ( dm , lv , metadata , percent ) ) )
2011-12-21 13:10:05 +00:00
stack ;
dev_manager_destroy ( dm ) ;
return r ;
}
2012-01-19 15:27:54 +00:00
/*
* Returns 1 if percent set , else 0 on failure .
*/
int lv_thin_percent ( const struct logical_volume * lv ,
int mapped , percent_t * percent )
{
int r ;
struct dev_manager * dm ;
if ( ! activation ( ) )
return 0 ;
log_debug ( " Checking thin percent for LV %s/%s " ,
lv - > vg - > name , lv - > name ) ;
if ( ! ( dm = dev_manager_create ( lv - > vg - > cmd , lv - > vg - > name , 1 ) ) )
return_0 ;
if ( ! ( r = dev_manager_thin_percent ( dm , lv , mapped , percent ) ) )
stack ;
dev_manager_destroy ( dm ) ;
return r ;
}
2012-01-25 08:48:42 +00:00
/*
* Returns 1 if transaction_id set , else 0 on failure .
*/
int lv_thin_pool_transaction_id ( const struct logical_volume * lv ,
uint64_t * transaction_id )
{
int r ;
struct dev_manager * dm ;
struct dm_status_thin_pool * status ;
if ( ! activation ( ) )
return 0 ;
log_debug ( " Checking thin percent for LV %s/%s " ,
lv - > vg - > name , lv - > name ) ;
if ( ! ( dm = dev_manager_create ( lv - > vg - > cmd , lv - > vg - > name , 1 ) ) )
return_0 ;
if ( ! ( r = dev_manager_thin_pool_status ( dm , lv , & status ) ) )
stack ;
else
* transaction_id = status - > transaction_id ;
dev_manager_destroy ( dm ) ;
return r ;
}
2012-02-23 22:41:57 +00:00
static int _lv_active ( struct cmd_context * cmd , const struct logical_volume * lv )
2001-11-07 15:02:07 +00:00
{
2003-01-08 22:44:07 +00:00
struct lvinfo info ;
2001-11-07 15:02:07 +00:00
2010-08-17 16:25:32 +00:00
if ( ! lv_info ( cmd , lv , 0 , & info , 0 , 0 ) ) {
2001-11-07 15:02:07 +00:00
stack ;
2002-02-26 11:49:17 +00:00
return - 1 ;
2001-11-07 15:02:07 +00:00
}
2002-02-26 11:49:17 +00:00
return info . exists ;
2001-11-07 15:02:07 +00:00
}
2005-10-17 18:00:02 +00:00
static int _lv_open_count ( struct cmd_context * cmd , struct logical_volume * lv )
2002-02-18 15:52:48 +00:00
{
2003-01-08 22:44:07 +00:00
struct lvinfo info ;
2002-02-18 15:52:48 +00:00
2010-08-17 16:25:32 +00:00
if ( ! lv_info ( cmd , lv , 0 , & info , 1 , 0 ) ) {
2002-02-18 15:52:48 +00:00
stack ;
2002-02-26 11:49:17 +00:00
return - 1 ;
2002-02-18 15:52:48 +00:00
}
2002-02-26 11:49:17 +00:00
return info . open_count ;
2002-02-18 15:52:48 +00:00
}
2011-06-17 14:14:19 +00:00
static int _lv_activate_lv ( struct logical_volume * lv , struct lv_activate_opts * laopts )
2001-10-09 16:05:34 +00:00
{
2002-02-11 15:48:34 +00:00
int r ;
2002-02-26 11:49:17 +00:00
struct dev_manager * dm ;
2001-10-09 16:05:34 +00:00
2011-06-11 00:03:06 +00:00
if ( ! ( dm = dev_manager_create ( lv - > vg - > cmd , lv - > vg - > name , ( lv - > status & PVMOVE ) ? 0 : 1 ) ) )
2005-11-08 22:52:26 +00:00
return_0 ;
2001-11-02 13:45:05 +00:00
2011-06-17 14:14:19 +00:00
if ( ! ( r = dev_manager_activate ( dm , lv , laopts ) ) )
2002-02-11 15:48:34 +00:00
stack ;
2001-11-02 13:45:05 +00:00
2002-02-26 11:49:17 +00:00
dev_manager_destroy ( dm ) ;
2001-11-02 13:45:05 +00:00
return r ;
2001-10-09 16:05:34 +00:00
}
2001-10-16 16:25:28 +00:00
2011-06-17 14:14:19 +00:00
static int _lv_preload ( struct logical_volume * lv , struct lv_activate_opts * laopts ,
int * flush_required )
2001-10-31 17:59:52 +00:00
{
2012-01-12 01:51:56 +00:00
int r = 0 ;
2002-02-26 11:49:17 +00:00
struct dev_manager * dm ;
2012-01-12 01:51:56 +00:00
int old_readonly = laopts - > read_only ;
laopts - > read_only = _passes_readonly_filter ( lv - > vg - > cmd , lv ) ;
2001-11-07 11:51:42 +00:00
2011-06-11 00:03:06 +00:00
if ( ! ( dm = dev_manager_create ( lv - > vg - > cmd , lv - > vg - > name , ( lv - > status & PVMOVE ) ? 0 : 1 ) ) )
2012-01-12 01:51:56 +00:00
goto_out ;
2005-11-08 22:52:26 +00:00
2011-06-17 14:14:19 +00:00
if ( ! ( r = dev_manager_preload ( dm , lv , laopts , flush_required ) ) )
2002-02-11 15:48:34 +00:00
stack ;
2005-11-08 22:52:26 +00:00
dev_manager_destroy ( dm ) ;
2012-01-12 01:51:56 +00:00
laopts - > read_only = old_readonly ;
out :
2005-11-08 22:52:26 +00:00
return r ;
}
static int _lv_deactivate ( struct logical_volume * lv )
{
int r ;
struct dev_manager * dm ;
2011-06-11 00:03:06 +00:00
if ( ! ( dm = dev_manager_create ( lv - > vg - > cmd , lv - > vg - > name , 1 ) ) )
2005-11-08 22:52:26 +00:00
return_0 ;
2001-11-07 11:51:42 +00:00
2002-02-26 11:49:17 +00:00
if ( ! ( r = dev_manager_deactivate ( dm , lv ) ) )
2001-11-07 11:51:42 +00:00
stack ;
2002-02-26 11:49:17 +00:00
dev_manager_destroy ( dm ) ;
return r ;
2001-11-07 11:51:42 +00:00
}
2011-06-17 14:14:19 +00:00
static int _lv_suspend_lv ( struct logical_volume * lv , struct lv_activate_opts * laopts ,
int lockfs , int flush_required )
2002-01-10 23:21:07 +00:00
{
2002-03-14 15:36:07 +00:00
int r ;
struct dev_manager * dm ;
2001-11-28 18:03:11 +00:00
2012-01-12 01:51:56 +00:00
laopts - > read_only = _passes_readonly_filter ( lv - > vg - > cmd , lv ) ;
2011-06-11 00:03:06 +00:00
/*
* When we are asked to manipulate ( normally suspend / resume ) the PVMOVE
* device directly , we don ' t want to touch the devices that use it .
*/
if ( ! ( dm = dev_manager_create ( lv - > vg - > cmd , lv - > vg - > name , ( lv - > status & PVMOVE ) ? 0 : 1 ) ) )
2005-11-08 22:52:26 +00:00
return_0 ;
2001-10-31 17:59:52 +00:00
2011-06-17 14:14:19 +00:00
if ( ! ( r = dev_manager_suspend ( dm , lv , laopts , lockfs , flush_required ) ) )
2001-11-07 11:51:42 +00:00
stack ;
2001-10-31 17:59:52 +00:00
2002-03-14 15:36:07 +00:00
dev_manager_destroy ( dm ) ;
return r ;
2002-02-11 15:48:34 +00:00
}
2002-01-10 23:21:07 +00:00
2002-03-01 19:08:11 +00:00
/*
2004-05-11 18:18:14 +00:00
* These two functions return the number of visible LVs in the state ,
2011-06-13 22:28:04 +00:00
* or - 1 on error . FIXME Check this .
2002-03-01 19:08:11 +00:00
*/
2012-02-23 22:41:57 +00:00
int lvs_in_vg_activated ( const struct volume_group * vg )
2001-11-02 16:28:04 +00:00
{
2005-06-01 16:51:55 +00:00
struct lv_list * lvl ;
2001-11-08 16:15:58 +00:00
int count = 0 ;
2001-11-07 11:51:42 +00:00
2002-11-18 14:01:16 +00:00
if ( ! activation ( ) )
return 0 ;
2011-06-13 22:28:04 +00:00
dm_list_iterate_items ( lvl , & vg - > lvs )
2009-05-13 21:26:45 +00:00
if ( lv_is_visible ( lvl - > lv ) )
2010-02-24 20:00:56 +00:00
count + = ( _lv_active ( vg - > cmd , lvl - > lv ) = = 1 ) ;
2011-06-13 22:28:04 +00:00
log_debug ( " Counted %d active LVs in VG %s " , count , vg - > name ) ;
2001-11-07 11:51:42 +00:00
return count ;
2001-11-02 16:28:04 +00:00
}
2001-11-07 15:02:07 +00:00
2007-08-07 09:06:05 +00:00
int lvs_in_vg_opened ( const struct volume_group * vg )
2001-11-07 15:02:07 +00:00
{
2007-08-07 09:06:05 +00:00
const struct lv_list * lvl ;
2001-11-08 16:15:58 +00:00
int count = 0 ;
2001-11-07 15:02:07 +00:00
2002-11-18 14:01:16 +00:00
if ( ! activation ( ) )
return 0 ;
2011-06-13 22:28:04 +00:00
dm_list_iterate_items ( lvl , & vg - > lvs )
2011-11-07 10:58:13 +00:00
if ( lv_is_visible ( lvl - > lv ) )
2005-10-17 18:00:02 +00:00
count + = ( _lv_open_count ( vg - > cmd , lvl - > lv ) > 0 ) ;
2011-06-13 22:28:04 +00:00
log_debug ( " Counted %d open LVs in VG %s " , count , vg - > name ) ;
2001-11-07 15:02:07 +00:00
return count ;
}
2002-02-25 12:56:16 +00:00
2008-04-10 21:34:18 +00:00
/*
2011-02-04 20:30:17 +00:00
* _lv_is_active
* @ lv : logical volume being queried
* @ locally : set if active locally ( when provided )
* @ exclusive : set if active exclusively ( when provided )
*
2008-04-10 21:34:18 +00:00
* Determine whether an LV is active locally or in a cluster .
2011-02-04 20:30:17 +00:00
* In addition to the return code which indicates whether or
* not the LV is active somewhere , two other values are set
* to yield more information about the status of the activation :
* return locally exclusively status
* = = = = = = = = = = = = = = = = = = = = = = = = = = = = = =
* 0 0 0 not active
* 1 0 0 active remotely
* 1 0 1 exclusive remotely
* 1 1 0 active locally and possibly remotely
* 1 1 1 exclusive locally ( or local & & ! cluster )
* The VG lock must be held to call this function .
*
* Returns : 0 or 1
2008-04-10 21:34:18 +00:00
*/
2012-02-23 22:41:57 +00:00
static int _lv_is_active ( const struct logical_volume * lv ,
2011-02-04 20:30:17 +00:00
int * locally , int * exclusive )
2008-04-10 21:34:18 +00:00
{
2011-02-04 20:30:17 +00:00
int r , l , e ; /* remote, local, and exclusive */
r = l = e = 0 ;
2009-05-20 12:58:03 +00:00
2010-02-24 20:00:56 +00:00
if ( _lv_active ( lv - > vg - > cmd , lv ) )
2011-02-04 20:30:17 +00:00
l = 1 ;
2008-04-10 21:34:18 +00:00
2011-02-04 20:30:17 +00:00
if ( ! vg_is_clustered ( lv - > vg ) ) {
2012-01-21 05:29:51 +00:00
if ( l )
e = 1 ; /* exclusive by definition */
2011-02-04 20:30:17 +00:00
goto out ;
}
/* Active locally, and the caller doesn't care about exclusive */
if ( l & & ! exclusive )
goto out ;
2008-04-10 21:34:18 +00:00
2011-02-04 20:30:17 +00:00
if ( ( r = remote_lock_held ( lv - > lvid . s , & e ) ) > = 0 )
goto out ;
2009-05-20 12:58:03 +00:00
/*
2011-02-04 20:30:17 +00:00
* If lock query is not supported ( due to interfacing with old
* code ) , then we cannot evaluate exclusivity properly .
*
* Old users of this function will never be affected by this ,
* since they are only concerned about active vs . not active .
* New users of this function who specifically ask for ' exclusive '
* will be given an error message .
2009-05-20 12:58:03 +00:00
*/
2012-01-21 05:29:51 +00:00
log_error ( " Unable to determine exclusivity of %s " , lv - > name ) ;
e = 0 ;
/*
* We used to attempt activate_lv_excl_local ( lv - > vg - > cmd , lv ) here ,
* but it ' s unreliable .
*/
2011-02-04 20:30:17 +00:00
out :
if ( locally )
* locally = l ;
if ( exclusive )
* exclusive = e ;
log_very_verbose ( " %s/%s is %sactive%s%s " ,
lv - > vg - > name , lv - > name ,
( r | | l ) ? " " : " not " ,
( exclusive & & e ) ? " exclusive " : " " ,
e ? ( l ? " locally " : " remotely " ) : " " ) ;
return r | | l ;
}
2012-02-23 22:41:57 +00:00
int lv_is_active ( const struct logical_volume * lv )
2011-02-04 20:30:17 +00:00
{
return _lv_is_active ( lv , NULL , NULL ) ;
}
2012-02-23 22:41:57 +00:00
int lv_is_active_but_not_locally ( const struct logical_volume * lv )
2011-02-04 20:30:17 +00:00
{
int l ;
2011-06-22 21:31:21 +00:00
return _lv_is_active ( lv , & l , NULL ) & & ! l ;
2011-02-04 20:30:17 +00:00
}
2012-02-23 22:41:57 +00:00
int lv_is_active_exclusive ( const struct logical_volume * lv )
2011-10-06 14:55:39 +00:00
{
int e ;
return _lv_is_active ( lv , NULL , & e ) & & e ;
}
2012-02-23 22:41:57 +00:00
int lv_is_active_exclusive_locally ( const struct logical_volume * lv )
2011-02-04 20:30:17 +00:00
{
int l , e ;
2011-06-13 22:28:04 +00:00
2011-02-04 20:30:17 +00:00
return _lv_is_active ( lv , & l , & e ) & & l & & e ;
}
2012-02-23 22:41:57 +00:00
int lv_is_active_exclusive_remotely ( const struct logical_volume * lv )
2011-02-04 20:30:17 +00:00
{
int l , e ;
2011-06-13 22:28:04 +00:00
2011-02-04 20:30:17 +00:00
return _lv_is_active ( lv , & l , & e ) & & ! l & & e ;
2008-04-10 21:34:18 +00:00
}
2010-08-16 22:54:35 +00:00
# ifdef DMEVENTD
static struct dm_event_handler * _create_dm_event_handler ( struct cmd_context * cmd , const char * dmuuid , const char * dso ,
const int timeout , enum dm_event_mask mask )
{
struct dm_event_handler * dmevh ;
if ( ! ( dmevh = dm_event_handler_create ( ) ) )
return_NULL ;
if ( dm_event_handler_set_dmeventd_path ( dmevh , find_config_tree_str ( cmd , " dmeventd/executable " , NULL ) ) )
goto_bad ;
if ( dm_event_handler_set_dso ( dmevh , dso ) )
goto_bad ;
if ( dm_event_handler_set_uuid ( dmevh , dmuuid ) )
goto_bad ;
dm_event_handler_set_timeout ( dmevh , timeout ) ;
dm_event_handler_set_event_mask ( dmevh , mask ) ;
return dmevh ;
bad :
dm_event_handler_destroy ( dmevh ) ;
return NULL ;
}
char * get_monitor_dso_path ( struct cmd_context * cmd , const char * libpath )
{
char * path ;
if ( ! ( path = dm_pool_alloc ( cmd - > mem , PATH_MAX ) ) ) {
log_error ( " Failed to allocate dmeventd library path. " ) ;
return NULL ;
}
get_shared_library_path ( cmd , libpath , path , PATH_MAX ) ;
return path ;
}
2011-12-21 13:08:11 +00:00
static char * _build_target_uuid ( struct cmd_context * cmd , struct logical_volume * lv )
{
const char * layer ;
if ( lv_is_thin_pool ( lv ) )
layer = " tpool " ; /* Monitor "tpool" for the "thin pool". */
else if ( lv_is_origin ( lv ) )
layer = " real " ; /* Monitor "real" for "snapshot-origin". */
else
layer = NULL ;
return build_dm_uuid ( cmd - > mem , lv - > lvid . s , layer ) ;
}
2010-08-17 01:16:41 +00:00
int target_registered_with_dmeventd ( struct cmd_context * cmd , const char * dso ,
struct logical_volume * lv , int * pending )
2010-08-16 22:54:35 +00:00
{
char * uuid ;
enum dm_event_mask evmask = 0 ;
struct dm_event_handler * dmevh ;
* pending = 0 ;
if ( ! dso )
return_0 ;
2011-12-21 13:08:11 +00:00
if ( ! ( uuid = _build_target_uuid ( cmd , lv ) ) )
2010-08-16 22:54:35 +00:00
return_0 ;
if ( ! ( dmevh = _create_dm_event_handler ( cmd , uuid , dso , 0 , DM_EVENT_ALL_ERRORS ) ) )
return_0 ;
if ( dm_event_get_registered_device ( dmevh , 0 ) ) {
dm_event_handler_destroy ( dmevh ) ;
return 0 ;
}
evmask = dm_event_handler_get_event_mask ( dmevh ) ;
if ( evmask & DM_EVENT_REGISTRATION_PENDING ) {
* pending = 1 ;
evmask & = ~ DM_EVENT_REGISTRATION_PENDING ;
}
dm_event_handler_destroy ( dmevh ) ;
return evmask ;
}
2010-08-17 01:16:41 +00:00
int target_register_events ( struct cmd_context * cmd , const char * dso , struct logical_volume * lv ,
2010-08-16 22:54:35 +00:00
int evmask __attribute__ ( ( unused ) ) , int set , int timeout )
{
char * uuid ;
struct dm_event_handler * dmevh ;
int r ;
if ( ! dso )
return_0 ;
2010-08-17 01:16:41 +00:00
/* We always monitor the "real" device, never the "snapshot-origin" itself. */
2011-12-21 13:08:11 +00:00
if ( ! ( uuid = _build_target_uuid ( cmd , lv ) ) )
2010-08-16 22:54:35 +00:00
return_0 ;
if ( ! ( dmevh = _create_dm_event_handler ( cmd , uuid , dso , timeout ,
DM_EVENT_ALL_ERRORS | ( timeout ? DM_EVENT_TIMEOUT : 0 ) ) ) )
return_0 ;
r = set ? dm_event_register_handler ( dmevh ) : dm_event_unregister_handler ( dmevh ) ;
dm_event_handler_destroy ( dmevh ) ;
if ( ! r )
return_0 ;
log_info ( " %s %s for events " , set ? " Monitored " : " Unmonitored " , uuid ) ;
return 1 ;
}
# endif
2006-05-12 19:16:48 +00:00
/*
2007-01-19 22:21:45 +00:00
* Returns 0 if an attempt to ( un ) monitor the device failed .
* Returns 1 otherwise .
2006-05-12 19:16:48 +00:00
*/
2010-08-17 16:25:32 +00:00
int monitor_dev_for_events ( struct cmd_context * cmd , struct logical_volume * lv ,
2011-06-17 14:14:19 +00:00
const struct lv_activate_opts * laopts , int monitor )
2005-12-02 20:35:07 +00:00
{
2006-01-27 18:38:14 +00:00
# ifdef DMEVENTD
2007-01-19 22:21:45 +00:00
int i , pending = 0 , monitored ;
int r = 1 ;
2008-11-03 22:14:30 +00:00
struct dm_list * tmp , * snh , * snht ;
2005-12-02 20:35:07 +00:00
struct lv_segment * seg ;
2010-03-26 22:15:43 +00:00
struct lv_segment * log_seg ;
2007-01-24 22:06:11 +00:00
int ( * monitor_fn ) ( struct lv_segment * s , int e ) ;
2008-01-31 12:19:36 +00:00
uint32_t s ;
2011-06-17 14:14:19 +00:00
static const struct lv_activate_opts zlaopts = { 0 } ;
2012-03-23 09:58:04 +00:00
static const struct lv_activate_opts thinopts = { . skip_in_use = 1 } ;
struct lvinfo info ;
2011-06-17 14:14:19 +00:00
if ( ! laopts )
laopts = & zlaopts ;
2005-12-02 20:35:07 +00:00
2007-01-24 23:43:27 +00:00
/* skip dmeventd code altogether */
if ( dmeventd_monitor_mode ( ) = = DMEVENTD_MONITOR_IGNORE )
return 1 ;
2007-01-19 22:21:45 +00:00
/*
* Nothing to do if dmeventd configured not to be used .
*/
if ( monitor & & ! dmeventd_monitor_mode ( ) )
2006-05-12 19:16:48 +00:00
return 1 ;
2012-03-23 09:58:04 +00:00
/*
* Allow to unmonitor thin pool via explicit pool unmonitor
* or unmonitor before the last thin pool user deactivation
* Skip unmonitor , if invoked via unmonitor of thin volume
* and there is another thin pool user ( open_count > 1 )
*/
if ( laopts - > skip_in_use & & lv_info ( lv - > vg - > cmd , lv , 1 , & info , 1 , 0 ) & &
( info . open_count ! = 1 ) ) {
log_debug ( " Skipping unmonitor of opened %s (open:%d) " ,
lv - > name , info . open_count ) ;
return 1 ;
}
2008-01-09 15:32:19 +00:00
/*
* In case of a snapshot device , we monitor lv - > snapshot - > lv ,
* not the actual LV itself .
*/
2011-06-17 14:22:48 +00:00
if ( lv_is_cow ( lv ) & & ( laopts - > no_merging | | ! lv_is_merging_cow ( lv ) ) )
return monitor_dev_for_events ( cmd , lv - > snapshot - > lv , NULL , monitor ) ;
2008-01-09 15:32:19 +00:00
/*
* In case this LV is a snapshot origin , we instead monitor
2010-08-17 01:16:41 +00:00
* each of its respective snapshots . The origin itself may
* also need to be monitored if it is a mirror , for example .
2008-01-09 15:32:19 +00:00
*/
2011-06-17 14:14:19 +00:00
if ( ! laopts - > origin_only & & lv_is_origin ( lv ) )
2008-11-03 22:14:30 +00:00
dm_list_iterate_safe ( snh , snht , & lv - > snapshot_segs )
if ( ! monitor_dev_for_events ( cmd , dm_list_struct_base ( snh ,
2011-06-17 14:14:19 +00:00
struct lv_segment , origin_list ) - > cow , NULL , monitor ) )
2008-01-17 17:17:09 +00:00
r = 0 ;
2008-01-09 15:32:19 +00:00
2010-03-26 22:15:43 +00:00
/*
* If the volume is mirrored and its log is also mirrored , monitor
* the log volume as well .
*/
if ( ( seg = first_seg ( lv ) ) ! = NULL & & seg - > log_lv ! = NULL & &
( log_seg = first_seg ( seg - > log_lv ) ) ! = NULL & &
seg_is_mirrored ( log_seg ) )
2011-06-17 14:14:19 +00:00
if ( ! monitor_dev_for_events ( cmd , seg - > log_lv , NULL , monitor ) )
2010-03-26 22:15:43 +00:00
r = 0 ;
2008-11-03 22:14:30 +00:00
dm_list_iterate ( tmp , & lv - > segments ) {
seg = dm_list_item ( tmp , struct lv_segment ) ;
2005-12-02 20:35:07 +00:00
2008-01-31 12:19:36 +00:00
/* Recurse for AREA_LV */
for ( s = 0 ; s < seg - > area_count ; s + + ) {
if ( seg_type ( seg , s ) ! = AREA_LV )
continue ;
2011-06-17 14:14:19 +00:00
if ( ! monitor_dev_for_events ( cmd , seg_lv ( seg , s ) , NULL ,
2008-01-31 12:19:36 +00:00
monitor ) ) {
log_error ( " Failed to %smonitor %s " ,
monitor ? " " : " un " ,
seg_lv ( seg , s ) - > name ) ;
r = 0 ;
}
}
2012-03-23 09:58:04 +00:00
/*
* If requested unmonitoring of thin volume , request test
* if there is no other thin pool user
*
* FIXME : code here looks like _lv_postorder ( )
*/
if ( seg - > pool_lv & &
! monitor_dev_for_events ( cmd , seg - > pool_lv ,
( ! monitor ) ? & thinopts : NULL , monitor ) )
r = 0 ;
if ( seg - > metadata_lv & &
! monitor_dev_for_events ( cmd , seg - > metadata_lv , NULL , monitor ) )
r = 0 ;
2007-01-12 20:38:30 +00:00
if ( ! seg_monitored ( seg ) | | ( seg - > status & PVMOVE ) )
continue ;
2007-01-19 22:21:45 +00:00
monitor_fn = NULL ;
2006-01-27 18:38:14 +00:00
2007-01-12 20:38:30 +00:00
/* Check monitoring status */
2007-01-19 22:21:45 +00:00
if ( seg - > segtype - > ops - > target_monitored )
monitored = seg - > segtype - > ops - > target_monitored ( seg , & pending ) ;
2007-01-12 20:38:30 +00:00
else
continue ; /* segtype doesn't support registration */
/*
* FIXME : We should really try again if pending
*/
2007-01-19 22:21:45 +00:00
monitored = ( pending ) ? 0 : monitored ;
2007-01-12 20:38:30 +00:00
2007-01-19 22:21:45 +00:00
if ( monitor ) {
if ( monitored )
2007-01-12 20:38:30 +00:00
log_verbose ( " %s/%s already monitored. " , lv - > vg - > name , lv - > name ) ;
2007-01-19 22:21:45 +00:00
else if ( seg - > segtype - > ops - > target_monitor_events )
monitor_fn = seg - > segtype - > ops - > target_monitor_events ;
2007-01-12 20:38:30 +00:00
} else {
2007-01-19 22:21:45 +00:00
if ( ! monitored )
2007-01-12 20:38:30 +00:00
log_verbose ( " %s/%s already not monitored. " , lv - > vg - > name , lv - > name ) ;
2007-01-19 22:21:45 +00:00
else if ( seg - > segtype - > ops - > target_unmonitor_events )
monitor_fn = seg - > segtype - > ops - > target_unmonitor_events ;
2007-01-12 20:38:30 +00:00
}
2005-12-02 20:35:07 +00:00
2007-01-12 20:38:30 +00:00
/* Do [un]monitor */
2007-01-19 22:21:45 +00:00
if ( ! monitor_fn )
2006-05-12 19:16:48 +00:00
continue ;
2006-04-28 14:06:06 +00:00
2010-08-16 23:29:09 +00:00
log_verbose ( " %sonitoring %s/%s%s " , monitor ? " M " : " Not m " , lv - > vg - > name , lv - > name ,
test_mode ( ) ? " [Test mode: skipping this] " : " " ) ;
/* FIXME Test mode should really continue a bit further. */
if ( test_mode ( ) )
continue ;
2007-01-19 22:21:45 +00:00
2006-05-12 19:16:48 +00:00
/* FIXME specify events */
2007-01-24 16:51:24 +00:00
if ( ! monitor_fn ( seg , 0 ) ) {
2007-01-19 22:21:45 +00:00
log_error ( " %s/%s: %s segment monitoring function failed. " ,
lv - > vg - > name , lv - > name , seg - > segtype - > name ) ;
return 0 ;
2006-04-28 14:06:06 +00:00
}
2006-05-12 19:16:48 +00:00
2007-01-12 20:38:30 +00:00
/* Check [un]monitor results */
/* Try a couple times if pending, but not forever... */
for ( i = 0 ; i < 10 ; i + + ) {
pending = 0 ;
2007-01-19 22:21:45 +00:00
monitored = seg - > segtype - > ops - > target_monitored ( seg , & pending ) ;
2007-01-12 20:38:30 +00:00
if ( pending | |
2007-01-19 22:21:45 +00:00
( ! monitored & & monitor ) | |
( monitored & & ! monitor ) )
log_very_verbose ( " %s/%s %smonitoring still pending: waiting... " ,
lv - > vg - > name , lv - > name , monitor ? " " : " un " ) ;
2007-01-12 20:38:30 +00:00
else
break ;
sleep ( 1 ) ;
}
2010-08-16 23:29:09 +00:00
if ( r )
r = ( monitored & & monitor ) | | ( ! monitored & & ! monitor ) ;
2005-12-02 20:35:07 +00:00
}
2005-12-08 17:49:34 +00:00
2006-05-12 19:16:48 +00:00
return r ;
# else
2005-12-02 20:35:07 +00:00
return 1 ;
2006-05-12 19:16:48 +00:00
# endif
2005-12-02 20:35:07 +00:00
}
2011-06-30 18:25:18 +00:00
struct detached_lv_data {
struct logical_volume * lv_pre ;
struct lv_activate_opts * laopts ;
int * flush_required ;
} ;
static int _preload_detached_lv ( struct cmd_context * cmd , struct logical_volume * lv , void * data )
{
struct detached_lv_data * detached = data ;
struct lv_list * lvl_pre ;
if ( ( lvl_pre = find_lv_in_vg ( detached - > lv_pre - > vg , lv - > name ) ) ) {
2011-07-08 12:48:41 +00:00
if ( lv_is_visible ( lvl_pre - > lv ) & & lv_is_active ( lv ) & & ( ! lv_is_cow ( lv ) | | ! lv_is_cow ( lvl_pre - > lv ) ) & &
2011-06-30 18:25:18 +00:00
! _lv_preload ( lvl_pre - > lv , detached - > laopts , detached - > flush_required ) )
return_0 ;
}
return 1 ;
}
2004-03-08 18:54:13 +00:00
static int _lv_suspend ( struct cmd_context * cmd , const char * lvid_s ,
2011-06-17 14:14:19 +00:00
struct lv_activate_opts * laopts , int error_if_not_suspended )
2002-02-25 12:56:16 +00:00
{
2011-06-11 00:03:06 +00:00
struct logical_volume * lv = NULL , * lv_pre = NULL , * pvmove_lv = NULL ;
struct lv_list * lvl_pre ;
struct seg_list * sl ;
2011-07-08 12:48:41 +00:00
struct lv_segment * snap_seg ;
2003-01-08 22:44:07 +00:00
struct lvinfo info ;
2009-05-20 09:52:37 +00:00
int r = 0 , lockfs = 0 , flush_required = 0 ;
2011-06-30 18:25:18 +00:00
struct detached_lv_data detached ;
2002-02-25 12:56:16 +00:00
2002-11-18 14:01:16 +00:00
if ( ! activation ( ) )
return 1 ;
2006-01-27 20:39:37 +00:00
if ( ! ( lv = lv_from_lvid ( cmd , lvid_s , 0 ) ) )
2009-04-10 10:00:04 +00:00
goto_out ;
2006-01-27 20:39:37 +00:00
2005-10-31 20:15:28 +00:00
/* Use precommitted metadata if present */
2006-01-27 20:39:37 +00:00
if ( ! ( lv_pre = lv_from_lvid ( cmd , lvid_s , 1 ) ) )
2009-04-10 10:00:04 +00:00
goto_out ;
2002-02-25 12:56:16 +00:00
2010-08-17 16:25:32 +00:00
/* Ignore origin_only unless LV is origin in both old and new metadata */
2012-01-25 09:10:13 +00:00
if ( ! lv_is_thin_volume ( lv ) & & ! ( lv_is_origin ( lv ) & & lv_is_origin ( lv_pre ) ) )
2011-06-17 14:14:19 +00:00
laopts - > origin_only = 0 ;
2010-08-17 16:25:32 +00:00
2002-03-14 15:36:07 +00:00
if ( test_mode ( ) ) {
2011-06-17 14:14:19 +00:00
_skip ( " Suspending %s%s. " , lv - > name , laopts - > origin_only ? " origin without snapshots " : " " ) ;
2009-04-10 10:00:04 +00:00
r = 1 ;
goto out ;
2002-03-14 15:36:07 +00:00
}
2011-06-17 14:14:19 +00:00
if ( ! lv_info ( cmd , lv , laopts - > origin_only , & info , 0 , 0 ) )
2009-04-10 10:00:04 +00:00
goto_out ;
2002-03-11 19:02:28 +00:00
2009-04-10 10:00:04 +00:00
if ( ! info . exists | | info . suspended ) {
2009-12-03 19:23:40 +00:00
if ( ! error_if_not_suspended ) {
r = 1 ;
if ( info . suspended )
2011-06-11 00:03:06 +00:00
critical_section_inc ( cmd , " already suspended " ) ;
2009-12-03 19:23:40 +00:00
}
2009-04-10 10:00:04 +00:00
goto out ;
}
2003-07-04 22:34:56 +00:00
2010-05-21 14:34:01 +00:00
if ( ! lv_read_replicator_vgs ( lv ) )
goto_out ;
2009-06-01 12:43:31 +00:00
lv_calculate_readahead ( lv , NULL ) ;
2011-06-11 00:03:06 +00:00
/*
2011-07-05 18:36:37 +00:00
* Preload devices for the LV .
2011-06-11 00:03:06 +00:00
* If the PVMOVE LV is being removed , it ' s only present in the old
* metadata and not the new , so we must explicitly add the new
* tables for all the changed LVs here , as the relationships
* are not found by walking the new metadata .
*/
2011-07-05 18:36:37 +00:00
if ( ! ( lv_pre - > status & LOCKED ) & &
( lv - > status & LOCKED ) & &
( pvmove_lv = find_pvmove_lv_in_lv ( lv ) ) ) {
/* Preload all the LVs above the PVMOVE LV */
dm_list_iterate_items ( sl , & pvmove_lv - > segs_using_this_lv ) {
if ( ! ( lvl_pre = find_lv_in_vg ( lv_pre - > vg , sl - > seg - > lv - > name ) ) ) {
2011-07-08 12:48:41 +00:00
log_error ( INTERNAL_ERROR " LV %s missing from preload metadata " , sl - > seg - > lv - > name ) ;
2011-06-11 00:03:06 +00:00
goto out ;
}
2011-06-17 14:14:19 +00:00
if ( ! _lv_preload ( lvl_pre - > lv , laopts , & flush_required ) )
2011-06-11 00:03:06 +00:00
goto_out ;
2011-07-05 18:36:37 +00:00
}
/* Now preload the PVMOVE LV itself */
if ( ! ( lvl_pre = find_lv_in_vg ( lv_pre - > vg , pvmove_lv - > name ) ) ) {
2011-07-08 12:48:41 +00:00
log_error ( INTERNAL_ERROR " LV %s missing from preload metadata " , pvmove_lv - > name ) ;
2011-07-05 18:36:37 +00:00
goto out ;
}
if ( ! _lv_preload ( lvl_pre - > lv , laopts , & flush_required ) )
goto_out ;
} else {
if ( ! _lv_preload ( lv_pre , laopts , & flush_required ) )
/* FIXME Revert preloading */
goto_out ;
2011-06-30 18:25:18 +00:00
2011-07-05 18:36:37 +00:00
/*
* Search for existing LVs that have become detached and preload them .
*/
detached . lv_pre = lv_pre ;
detached . laopts = laopts ;
detached . flush_required = & flush_required ;
2011-06-30 18:25:18 +00:00
2011-07-05 18:36:37 +00:00
if ( ! for_each_sub_lv ( cmd , lv , & _preload_detached_lv , & detached ) )
goto_out ;
2011-07-08 12:48:41 +00:00
/*
* Preload any snapshots that are being removed .
*/
if ( ! laopts - > origin_only & & lv_is_origin ( lv ) ) {
dm_list_iterate_items_gen ( snap_seg , & lv - > snapshot_segs , origin_list ) {
2011-07-08 15:35:50 +00:00
if ( ! ( lvl_pre = find_lv_in_vg_by_lvid ( lv_pre - > vg , & snap_seg - > cow - > lvid ) ) ) {
log_error ( INTERNAL_ERROR " LV %s (%s) missing from preload metadata " ,
snap_seg - > cow - > name , snap_seg - > cow - > lvid . id [ 1 ] . uuid ) ;
2011-07-08 12:48:41 +00:00
goto out ;
}
if ( ! lv_is_cow ( lvl_pre - > lv ) & &
! _lv_preload ( lvl_pre - > lv , laopts , & flush_required ) )
goto_out ;
}
}
2005-11-08 22:52:26 +00:00
}
2011-06-17 14:14:19 +00:00
if ( ! monitor_dev_for_events ( cmd , lv , laopts , 0 ) )
2006-04-28 14:06:06 +00:00
/* FIXME Consider aborting here */
2006-01-27 18:38:14 +00:00
stack ;
2011-06-11 00:03:06 +00:00
critical_section_inc ( cmd , " suspending " ) ;
if ( pvmove_lv )
critical_section_inc ( cmd , " suspending pvmove LV " ) ;
2006-08-08 21:20:00 +00:00
2011-06-17 14:14:19 +00:00
if ( ! laopts - > origin_only & &
2010-08-17 16:25:32 +00:00
( lv_is_origin ( lv_pre ) | | lv_is_cow ( lv_pre ) ) )
2006-08-08 21:20:00 +00:00
lockfs = 1 ;
2011-06-11 00:03:06 +00:00
/*
* Suspending an LV directly above a PVMOVE LV also
* suspends other LVs using that same PVMOVE LV .
* FIXME Remove this and delay the ' clear node ' until
* after the code knows whether there ' s a different
* inactive table to load or not instead so lv_suspend
* can be called separately for each LV safely .
*/
if ( ( lv_pre - > vg - > status & PRECOMMITTED ) & &
( lv_pre - > status & LOCKED ) & & find_pvmove_lv_in_lv ( lv_pre ) ) {
2011-06-17 14:14:19 +00:00
if ( ! _lv_suspend_lv ( lv_pre , laopts , lockfs , flush_required ) ) {
2011-06-11 00:03:06 +00:00
critical_section_dec ( cmd , " failed precommitted suspend " ) ;
if ( pvmove_lv )
critical_section_dec ( cmd , " failed precommitted suspend (pvmove) " ) ;
goto_out ;
}
} else {
/* Normal suspend */
2011-06-17 14:14:19 +00:00
if ( ! _lv_suspend_lv ( lv , laopts , lockfs , flush_required ) ) {
2011-06-11 00:03:06 +00:00
critical_section_dec ( cmd , " failed suspend " ) ;
if ( pvmove_lv )
critical_section_dec ( cmd , " failed suspend (pvmove) " ) ;
goto_out ;
}
2003-07-04 22:34:56 +00:00
}
2002-03-01 19:08:11 +00:00
2009-04-10 10:00:04 +00:00
r = 1 ;
out :
if ( lv_pre )
2011-08-10 20:25:29 +00:00
release_vg ( lv_pre - > vg ) ;
2010-05-21 14:34:01 +00:00
if ( lv ) {
lv_release_replicator_vgs ( lv ) ;
2011-08-10 20:25:29 +00:00
release_vg ( lv - > vg ) ;
2010-05-21 14:34:01 +00:00
}
2009-04-10 10:00:04 +00:00
return r ;
2002-02-25 12:56:16 +00:00
}
2012-01-20 03:46:52 +00:00
/*
* In a cluster , set exclusive to indicate that only one node is using the
* device . Any preloaded tables may then use non - clustered targets .
*
* Returns success if the device is not active
*/
2012-01-20 00:27:18 +00:00
int lv_suspend_if_active ( struct cmd_context * cmd , const char * lvid_s , unsigned origin_only , unsigned exclusive )
2004-03-08 18:54:13 +00:00
{
2012-01-20 00:27:18 +00:00
struct lv_activate_opts laopts = {
. origin_only = origin_only ,
. exclusive = exclusive
} ;
2011-06-17 14:14:19 +00:00
return _lv_suspend ( cmd , lvid_s , & laopts , 0 ) ;
2004-03-08 18:54:13 +00:00
}
2010-08-17 16:25:32 +00:00
/* No longer used */
/***********
2004-03-08 18:54:13 +00:00
int lv_suspend ( struct cmd_context * cmd , const char * lvid_s )
{
return _lv_suspend ( cmd , lvid_s , 1 ) ;
}
2010-08-17 16:25:32 +00:00
* * * * * * * * * * */
2004-03-08 18:54:13 +00:00
static int _lv_resume ( struct cmd_context * cmd , const char * lvid_s ,
2011-06-17 14:14:19 +00:00
struct lv_activate_opts * laopts , int error_if_not_active )
2002-02-25 12:56:16 +00:00
{
struct logical_volume * lv ;
2003-01-08 22:44:07 +00:00
struct lvinfo info ;
2009-04-10 10:00:04 +00:00
int r = 0 ;
2012-01-25 09:13:10 +00:00
int messages_only = 0 ;
2002-02-25 12:56:16 +00:00
2002-11-18 14:01:16 +00:00
if ( ! activation ( ) )
return 1 ;
2005-10-31 20:15:28 +00:00
if ( ! ( lv = lv_from_lvid ( cmd , lvid_s , 0 ) ) )
2009-10-30 13:07:49 +00:00
goto_out ;
2002-02-25 12:56:16 +00:00
2012-01-25 09:13:10 +00:00
if ( lv_is_thin_pool ( lv ) & & laopts - > origin_only )
messages_only = 1 ;
2010-08-17 16:25:32 +00:00
if ( ! lv_is_origin ( lv ) )
2011-06-17 14:14:19 +00:00
laopts - > origin_only = 0 ;
2010-08-17 16:25:32 +00:00
2002-03-14 15:36:07 +00:00
if ( test_mode ( ) ) {
2011-09-27 22:43:40 +00:00
_skip ( " Resuming %s%s%s. " , lv - > name , laopts - > origin_only ? " without snapshots " : " " ,
laopts - > revert ? " (reverting) " : " " ) ;
2009-04-10 10:00:04 +00:00
r = 1 ;
goto out ;
2002-03-14 15:36:07 +00:00
}
2011-09-27 22:43:40 +00:00
log_debug ( " Resuming LV %s/%s%s%s%s. " , lv - > vg - > name , lv - > name ,
2011-06-13 22:28:04 +00:00
error_if_not_active ? " " : " if active " ,
2011-09-27 22:43:40 +00:00
laopts - > origin_only ? " without snapshots " : " " ,
laopts - > revert ? " (reverting) " : " " ) ;
2011-06-13 22:28:04 +00:00
2011-06-17 14:14:19 +00:00
if ( ! lv_info ( cmd , lv , laopts - > origin_only , & info , 0 , 0 ) )
2009-04-10 10:00:04 +00:00
goto_out ;
2002-03-11 19:02:28 +00:00
2012-01-25 09:13:10 +00:00
if ( ! info . exists | | ! ( info . suspended | | messages_only ) ) {
2010-07-08 12:24:04 +00:00
if ( error_if_not_active )
goto_out ;
r = 1 ;
2011-06-11 00:03:06 +00:00
if ( ! info . suspended )
critical_section_dec ( cmd , " already resumed " ) ;
2010-07-08 12:24:04 +00:00
goto out ;
2009-04-10 10:00:04 +00:00
}
2003-07-04 22:34:56 +00:00
2012-01-12 01:51:56 +00:00
laopts - > read_only = _passes_readonly_filter ( cmd , lv ) ;
2011-06-17 14:14:19 +00:00
if ( ! _lv_activate_lv ( lv , laopts ) )
2009-10-30 13:07:49 +00:00
goto_out ;
2003-07-04 22:34:56 +00:00
2011-06-11 00:03:06 +00:00
critical_section_dec ( cmd , " resumed " ) ;
2002-02-25 12:56:16 +00:00
2011-06-17 14:14:19 +00:00
if ( ! monitor_dev_for_events ( cmd , lv , laopts , 1 ) )
2006-01-27 18:38:14 +00:00
stack ;
2005-12-02 20:35:07 +00:00
2009-04-10 10:00:04 +00:00
r = 1 ;
out :
if ( lv )
2011-08-10 20:25:29 +00:00
release_vg ( lv - > vg ) ;
2009-04-10 10:00:04 +00:00
return r ;
2002-02-25 12:56:16 +00:00
}
2012-01-20 03:46:52 +00:00
/*
* In a cluster , set exclusive to indicate that only one node is using the
* device . Any tables loaded may then use non - clustered targets .
*
2012-01-25 08:51:29 +00:00
* @ origin_only
* @ exclusive This parameter only has an affect in cluster - context .
* It forces local target type to be used ( instead of
* cluster - aware type ) .
2012-01-20 03:46:52 +00:00
* Returns success if the device is not active
*/
2011-02-18 00:36:04 +00:00
int lv_resume_if_active ( struct cmd_context * cmd , const char * lvid_s ,
2012-01-20 03:46:52 +00:00
unsigned origin_only , unsigned exclusive ,
unsigned revert )
2004-03-08 18:54:13 +00:00
{
2011-06-17 14:14:19 +00:00
struct lv_activate_opts laopts = {
. origin_only = origin_only ,
2011-09-27 22:43:40 +00:00
. exclusive = exclusive ,
. revert = revert
2011-06-17 14:14:19 +00:00
} ;
return _lv_resume ( cmd , lvid_s , & laopts , 0 ) ;
2004-03-08 18:54:13 +00:00
}
2010-08-17 16:25:32 +00:00
int lv_resume ( struct cmd_context * cmd , const char * lvid_s , unsigned origin_only )
2004-03-08 18:54:13 +00:00
{
2011-06-17 14:14:19 +00:00
struct lv_activate_opts laopts = { . origin_only = origin_only , } ;
return _lv_resume ( cmd , lvid_s , & laopts , 1 ) ;
2004-03-08 18:54:13 +00:00
}
2009-09-29 18:50:28 +00:00
static int _lv_has_open_snapshots ( struct logical_volume * lv )
{
struct lv_segment * snap_seg ;
struct lvinfo info ;
int r = 0 ;
dm_list_iterate_items_gen ( snap_seg , & lv - > snapshot_segs , origin_list ) {
2010-08-17 16:25:32 +00:00
if ( ! lv_info ( lv - > vg - > cmd , snap_seg - > cow , 0 , & info , 1 , 0 ) ) {
2009-09-29 18:50:28 +00:00
r = 1 ;
continue ;
}
if ( info . exists & & info . open_count ) {
log_error ( " LV %s/%s has open snapshot %s: "
" not deactivating " , lv - > vg - > name , lv - > name ,
snap_seg - > cow - > name ) ;
r = 1 ;
}
}
return r ;
}
2002-03-11 15:08:39 +00:00
int lv_deactivate ( struct cmd_context * cmd , const char * lvid_s )
2002-02-27 12:26:41 +00:00
{
struct logical_volume * lv ;
2003-01-08 22:44:07 +00:00
struct lvinfo info ;
2009-04-10 10:00:04 +00:00
int r = 0 ;
2002-02-27 12:26:41 +00:00
2002-11-18 14:01:16 +00:00
if ( ! activation ( ) )
return 1 ;
2005-10-31 20:15:28 +00:00
if ( ! ( lv = lv_from_lvid ( cmd , lvid_s , 0 ) ) )
2009-04-10 10:00:04 +00:00
goto out ;
2002-02-27 12:26:41 +00:00
2002-03-14 15:36:07 +00:00
if ( test_mode ( ) ) {
_skip ( " Deactivating '%s'. " , lv - > name ) ;
2009-04-10 10:00:04 +00:00
r = 1 ;
goto out ;
2002-03-14 15:36:07 +00:00
}
2011-06-13 22:28:04 +00:00
log_debug ( " Deactivating %s/%s. " , lv - > vg - > name , lv - > name ) ;
2010-08-17 16:25:32 +00:00
if ( ! lv_info ( cmd , lv , 0 , & info , 1 , 0 ) )
2009-04-10 10:00:04 +00:00
goto_out ;
2002-03-11 19:02:28 +00:00
2009-04-10 10:00:04 +00:00
if ( ! info . exists ) {
r = 1 ;
goto out ;
}
2002-02-27 12:26:41 +00:00
2009-09-29 18:50:28 +00:00
if ( lv_is_visible ( lv ) ) {
2011-09-22 17:33:50 +00:00
if ( ! lv_check_not_in_use ( cmd , lv , & info ) )
goto_out ;
2009-09-29 18:50:28 +00:00
if ( lv_is_origin ( lv ) & & _lv_has_open_snapshots ( lv ) )
goto_out ;
2003-10-21 22:00:36 +00:00
}
2010-05-21 14:34:01 +00:00
if ( ! lv_read_replicator_vgs ( lv ) )
goto_out ;
2009-06-01 12:43:31 +00:00
lv_calculate_readahead ( lv , NULL ) ;
2011-06-17 14:14:19 +00:00
if ( ! monitor_dev_for_events ( cmd , lv , NULL , 0 ) )
2006-01-27 18:38:14 +00:00
stack ;
2005-12-02 20:35:07 +00:00
2011-06-11 00:03:06 +00:00
critical_section_inc ( cmd , " deactivating " ) ;
2003-07-04 22:34:56 +00:00
r = _lv_deactivate ( lv ) ;
2011-06-11 00:03:06 +00:00
critical_section_dec ( cmd , " deactivated " ) ;
2003-07-04 22:34:56 +00:00
2011-02-03 01:24:46 +00:00
if ( ! lv_info ( cmd , lv , 0 , & info , 0 , 0 ) | | info . exists )
2009-09-29 15:17:54 +00:00
r = 0 ;
2009-04-10 10:00:04 +00:00
out :
2010-05-21 14:34:01 +00:00
if ( lv ) {
lv_release_replicator_vgs ( lv ) ;
2011-08-10 20:25:29 +00:00
release_vg ( lv - > vg ) ;
2010-05-21 14:34:01 +00:00
}
2009-04-10 10:00:04 +00:00
2003-07-04 22:34:56 +00:00
return r ;
2002-02-27 12:26:41 +00:00
}
2004-03-08 18:54:13 +00:00
/* Test if LV passes filter */
int lv_activation_filter ( struct cmd_context * cmd , const char * lvid_s ,
int * activate_lv )
{
struct logical_volume * lv ;
2009-04-10 10:00:04 +00:00
int r = 0 ;
2004-03-08 18:54:13 +00:00
2009-04-10 10:00:04 +00:00
if ( ! activation ( ) ) {
* activate_lv = 1 ;
return 1 ;
}
2004-03-08 18:54:13 +00:00
2008-06-08 11:33:15 +00:00
if ( ! ( lv = lv_from_lvid ( cmd , lvid_s , 0 ) ) )
2009-04-10 10:00:04 +00:00
goto out ;
2004-03-08 18:54:13 +00:00
if ( ! _passes_activation_filter ( cmd , lv ) ) {
2010-11-05 18:18:11 +00:00
log_verbose ( " Not activating %s/%s since it does not pass "
" activation filter. " , lv - > vg - > name , lv - > name ) ;
2004-03-08 18:54:13 +00:00
* activate_lv = 0 ;
2009-04-10 10:00:04 +00:00
} else
* activate_lv = 1 ;
r = 1 ;
out :
if ( lv )
2011-08-10 20:25:29 +00:00
release_vg ( lv - > vg ) ;
2004-03-08 18:54:13 +00:00
2009-04-10 10:00:04 +00:00
return r ;
2004-03-08 18:54:13 +00:00
}
2005-08-14 23:18:28 +00:00
static int _lv_activate ( struct cmd_context * cmd , const char * lvid_s ,
2011-06-17 14:14:19 +00:00
struct lv_activate_opts * laopts , int filter )
2002-02-27 12:26:41 +00:00
{
struct logical_volume * lv ;
2003-01-08 22:44:07 +00:00
struct lvinfo info ;
2009-04-10 10:00:04 +00:00
int r = 0 ;
2002-02-27 12:26:41 +00:00
2002-11-18 14:01:16 +00:00
if ( ! activation ( ) )
return 1 ;
2008-06-08 11:33:15 +00:00
if ( ! ( lv = lv_from_lvid ( cmd , lvid_s , 0 ) ) )
2009-04-10 10:00:04 +00:00
goto out ;
2002-02-27 12:26:41 +00:00
2004-03-08 18:54:13 +00:00
if ( filter & & ! _passes_activation_filter ( cmd , lv ) ) {
2010-11-05 18:18:11 +00:00
log_error ( " Not activating %s/%s since it does not pass "
" activation filter. " , lv - > vg - > name , lv - > name ) ;
2009-04-10 10:00:04 +00:00
goto out ;
2004-03-08 18:13:22 +00:00
}
2008-09-19 07:12:45 +00:00
if ( ( ! lv - > vg - > cmd - > partial_activation ) & & ( lv - > status & PARTIAL_LV ) ) {
2008-09-19 06:42:00 +00:00
log_error ( " Refusing activation of partial LV %s. Use --partial to override. " ,
lv - > name ) ;
2009-04-10 10:00:04 +00:00
goto_out ;
2008-09-19 06:42:00 +00:00
}
2009-10-16 17:41:49 +00:00
if ( lv_has_unknown_segments ( lv ) ) {
log_error ( " Refusing activation of LV %s containing "
" an unrecognised segment. " , lv - > name ) ;
goto_out ;
}
2002-03-14 15:36:07 +00:00
if ( test_mode ( ) ) {
_skip ( " Activating '%s'. " , lv - > name ) ;
2009-04-10 10:00:04 +00:00
r = 1 ;
goto out ;
2002-03-14 15:36:07 +00:00
}
2012-01-12 01:51:56 +00:00
if ( filter )
laopts - > read_only = _passes_readonly_filter ( cmd , lv ) ;
log_debug ( " Activating %s/%s%s%s. " , lv - > vg - > name , lv - > name ,
laopts - > exclusive ? " exclusively " : " " ,
laopts - > read_only ? " read-only " : " " ) ;
2011-06-13 22:28:04 +00:00
2010-08-17 16:25:32 +00:00
if ( ! lv_info ( cmd , lv , 0 , & info , 0 , 0 ) )
2009-04-10 10:00:04 +00:00
goto_out ;
2002-03-01 19:08:11 +00:00
2012-01-12 01:51:56 +00:00
/*
* Nothing to do ?
*/
if ( info . exists & & ! info . suspended & & info . live_table & &
( info . read_only = = read_only_lv ( lv , laopts ) ) ) {
2009-04-10 10:00:04 +00:00
r = 1 ;
goto out ;
}
2002-02-27 12:26:41 +00:00
2010-05-21 14:34:01 +00:00
if ( ! lv_read_replicator_vgs ( lv ) )
goto_out ;
2009-06-01 12:43:31 +00:00
lv_calculate_readahead ( lv , NULL ) ;
2011-06-11 00:03:06 +00:00
critical_section_inc ( cmd , " activating " ) ;
2011-06-17 14:14:19 +00:00
if ( ! ( r = _lv_activate_lv ( lv , laopts ) ) )
2009-10-30 13:07:49 +00:00
stack ;
2011-06-11 00:03:06 +00:00
critical_section_dec ( cmd , " activated " ) ;
2003-07-04 22:34:56 +00:00
2011-06-17 14:14:19 +00:00
if ( r & & ! monitor_dev_for_events ( cmd , lv , laopts , 1 ) )
2006-01-27 18:38:14 +00:00
stack ;
2005-12-02 20:35:07 +00:00
2009-04-10 10:00:04 +00:00
out :
2010-05-21 14:34:01 +00:00
if ( lv ) {
lv_release_replicator_vgs ( lv ) ;
2011-08-10 20:25:29 +00:00
release_vg ( lv - > vg ) ;
2010-05-21 14:34:01 +00:00
}
2009-04-10 10:00:04 +00:00
2003-07-04 22:34:56 +00:00
return r ;
2002-02-27 12:26:41 +00:00
}
2003-01-08 22:44:07 +00:00
2004-03-08 18:54:13 +00:00
/* Activate LV */
2005-08-14 23:18:28 +00:00
int lv_activate ( struct cmd_context * cmd , const char * lvid_s , int exclusive )
2004-03-08 18:54:13 +00:00
{
2011-06-17 14:14:19 +00:00
struct lv_activate_opts laopts = { . exclusive = exclusive } ;
if ( ! _lv_activate ( cmd , lvid_s , & laopts , 0 ) )
2009-10-30 13:07:49 +00:00
return_0 ;
return 1 ;
2004-03-08 18:54:13 +00:00
}
/* Activate LV only if it passes filter */
2005-08-14 23:18:28 +00:00
int lv_activate_with_filter ( struct cmd_context * cmd , const char * lvid_s , int exclusive )
2004-03-08 18:54:13 +00:00
{
2011-06-17 14:14:19 +00:00
struct lv_activate_opts laopts = { . exclusive = exclusive } ;
if ( ! _lv_activate ( cmd , lvid_s , & laopts , 1 ) )
2009-10-30 13:07:49 +00:00
return_0 ;
return 1 ;
2004-03-08 18:54:13 +00:00
}
2003-11-12 19:16:48 +00:00
int lv_mknodes ( struct cmd_context * cmd , const struct logical_volume * lv )
{
int r = 1 ;
2004-03-30 14:40:03 +00:00
if ( ! lv ) {
2005-10-16 23:03:59 +00:00
r = dm_mknodes ( NULL ) ;
2004-03-30 14:40:03 +00:00
fs_unlock ( ) ;
return r ;
}
2010-02-24 20:00:56 +00:00
if ( ! activation ( ) )
return 1 ;
2003-11-12 19:16:48 +00:00
2010-02-24 20:00:56 +00:00
r = dev_manager_mknodes ( lv ) ;
2003-11-12 19:16:48 +00:00
fs_unlock ( ) ;
return r ;
}
2005-10-25 19:08:21 +00:00
/*
* Does PV use VG somewhere in its construction ?
* Returns 1 on failure .
*/
2006-05-11 17:58:58 +00:00
int pv_uses_vg ( struct physical_volume * pv ,
2006-05-12 19:16:48 +00:00
struct volume_group * vg )
2005-10-25 19:08:21 +00:00
{
2012-02-23 13:11:07 +00:00
if ( ! activation ( ) | | ! pv - > dev )
2005-10-25 19:08:21 +00:00
return 0 ;
if ( ! dm_is_dm_major ( MAJOR ( pv - > dev - > dev ) ) )
return 0 ;
2006-05-11 17:58:58 +00:00
return dev_manager_device_uses_vg ( pv - > dev , vg ) ;
2005-10-25 19:08:21 +00:00
}
2006-05-16 16:48:31 +00:00
void activation_release ( void )
{
dev_manager_release ( ) ;
}
2003-07-04 22:34:56 +00:00
void activation_exit ( void )
{
dev_manager_exit ( ) ;
}
2003-01-08 22:44:07 +00:00
# endif