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 .
2018-02-27 14:13:00 +01:00
* Copyright ( C ) 2004 - 2018 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 ,
2016-01-21 11:49:46 +01:00
* Inc . , 51 Franklin Street , Fifth Floor , Boston , MA 02110 - 1301 USA
2001-10-09 16:05:34 +00:00
*/
2018-05-14 10:30:20 +01:00
# include "lib/misc/lib.h"
# include "lib/metadata/metadata.h"
# include "lib/activate/activate.h"
# include "lib/mm/memlock.h"
# include "lib/display/display.h"
2001-11-12 12:20:58 +00:00
# include "fs.h"
2018-05-14 10:30:20 +01:00
# include "lib/misc/lvm-exec.h"
# include "lib/misc/lvm-file.h"
# include "lib/misc/lvm-string.h"
# include "lib/commands/toolcontext.h"
2002-02-26 11:49:17 +00:00
# include "dev_manager.h"
2018-05-14 10:30:20 +01:00
# include "lib/datastruct/str_list.h"
# include "lib/config/config.h"
# include "lib/metadata/segtype.h"
# include "lib/misc/sharedlib.h"
# include "lib/cache/lvmcache.h"
# include "lib/metadata/metadata.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
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 ) ) {
2013-07-02 16:26:03 -04:00
snap_seg = find_snapshot ( seg - > lv ) ;
2006-10-03 17:55:20 +00:00
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 ;
}
2014-09-22 15:50:07 +02:00
static int _lv_passes_volumes_filter ( struct cmd_context * cmd , const struct logical_volume * lv ,
2013-09-27 13:58:55 +02:00
const struct dm_config_node * cn , const int cfg_id )
{
const struct dm_config_value * cv ;
const char * str ;
static char config_path [ PATH_MAX ] ;
2014-09-21 01:08:43 +02:00
size_t len = strlen ( lv - > vg - > name ) ;
2013-09-27 13:58:55 +02:00
config_def_get_path ( config_path , sizeof ( config_path ) , cfg_id ) ;
log_verbose ( " %s configuration setting defined: "
2015-11-25 16:06:31 +01:00
" Checking the list to match %s. " ,
config_path , display_lvname ( lv ) ) ;
2013-09-27 13:58:55 +02:00
for ( cv = cn - > v ; cv ; cv = cv - > next ) {
if ( cv - > type = = DM_CFG_EMPTY_ARRAY )
goto out ;
if ( cv - > type ! = DM_CFG_STRING ) {
2014-09-20 22:05:29 +02:00
log_print_unless_silent ( " Ignoring invalid string in config file %s. " ,
config_path ) ;
2013-09-27 13:58:55 +02:00
continue ;
}
str = cv - > v . str ;
if ( ! * str ) {
2014-09-20 22:05:29 +02:00
log_print_unless_silent ( " Ignoring empty string in config file %s. " ,
config_path ) ;
2013-09-27 13:58:55 +02:00
continue ;
}
/* Tag? */
if ( * str = = ' @ ' ) {
str + + ;
if ( ! * str ) {
2014-09-20 22:05:29 +02:00
log_print_unless_silent ( " Ignoring empty tag in config file %s " ,
config_path ) ;
2013-09-27 13:58:55 +02:00
continue ;
}
/* If any host tag matches any LV or VG tag, activate */
if ( ! strcmp ( str , " * " ) ) {
if ( str_list_match_list ( & cmd - > tags , & lv - > tags , NULL )
| | str_list_match_list ( & cmd - > tags ,
& lv - > vg - > tags , NULL ) )
return 1 ;
2017-07-19 23:12:48 +02:00
continue ;
2013-09-27 13:58:55 +02:00
}
/* 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 ;
2017-07-19 23:12:48 +02:00
continue ;
2013-09-27 13:58:55 +02:00
}
2014-09-21 01:08:43 +02:00
/* If supplied name is vgname[/lvname] */
if ( ( strncmp ( str , lv - > vg - > name , len ) = = 0 ) & &
( ! str [ len ] | |
( ( str [ len ] = = ' / ' ) & &
! strcmp ( str + len + 1 , lv - > name ) ) ) )
2013-09-27 13:58:55 +02:00
return 1 ;
}
out :
2015-11-25 16:06:31 +01:00
log_verbose ( " No item supplied in %s configuration setting matches %s. " ,
config_path , display_lvname ( lv ) ) ;
2013-09-27 13:58:55 +02:00
return 0 ;
}
int lv_passes_auto_activation_filter ( struct cmd_context * cmd , struct logical_volume * lv )
{
const struct dm_config_node * cn ;
2015-07-08 11:22:24 +02:00
if ( ! ( cn = find_config_tree_array ( cmd , activation_auto_activation_volume_list_CFG , NULL ) ) ) {
2013-09-27 13:58:55 +02:00
log_verbose ( " activation/auto_activation_volume_list configuration setting "
" not defined: All logical volumes will be auto-activated. " ) ;
return 1 ;
}
return _lv_passes_volumes_filter ( cmd , lv , cn , activation_auto_activation_volume_list_CFG ) ;
}
2003-01-08 22:44:07 +00:00
# ifndef DEVMAPPER_SUPPORT
2014-04-29 13:22:10 +02:00
void set_activation ( int act , int silent )
2003-01-08 22:44:07 +00:00
{
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 ;
}
2015-01-14 13:03:52 +01:00
int lv_info_with_seg_status ( struct cmd_context * cmd , const struct logical_volume * lv ,
const struct lv_segment * lv_seg , int use_layer ,
struct lv_with_info_and_seg_status * status ,
int with_open_count , int with_read_ahead )
{
return 0 ;
}
int lv_status ( struct cmd_context * cmd , const struct lv_segment * lv_seg ,
2015-01-20 13:16:41 +01:00
int use_layer , struct lv_seg_status * lv_seg_status )
2015-01-14 13:03:52 +01:00
{
return 0 ;
}
int lv_cache_status ( const struct logical_volume * cache_lv ,
struct lv_status_cache * * status )
{
2016-06-03 12:38:46 +02:00
return 0 ;
2015-01-14 13:03:52 +01:00
}
2016-06-03 12:38:46 +02:00
int lv_check_not_in_use ( const struct logical_volume * lv , int error_if_used )
2012-01-25 13:12:59 +00:00
{
return 0 ;
}
2014-06-09 12:08:27 +02:00
int lv_snapshot_percent ( const struct logical_volume * lv , dm_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 ,
2014-06-09 12:08:27 +02:00
int wait , dm_percent_t * percent , uint32_t * event_nr )
2003-04-30 15:26:25 +00:00
{
return 0 ;
}
2014-06-09 12:08:27 +02:00
int lv_raid_percent ( const struct logical_volume * lv , dm_percent_t * percent )
2012-01-25 13:12:59 +00:00
{
return 0 ;
}
2017-02-24 00:50:00 +01:00
int lv_raid_data_offset ( const struct logical_volume * lv , uint64_t * data_offset )
{
return 0 ;
}
2013-02-01 11:31:47 -06:00
int lv_raid_dev_health ( const struct logical_volume * lv , char * * dev_health )
{
return 0 ;
}
2017-02-24 00:50:00 +01:00
int lv_raid_dev_count ( const struct logical_volume * lv , uint32_t * dev_cnt )
{
return 0 ;
}
2013-04-11 15:33:59 -05:00
int lv_raid_mismatch_count ( const struct logical_volume * lv , uint64_t * cnt )
{
return 0 ;
}
int lv_raid_sync_action ( const struct logical_volume * lv , char * * sync_action )
{
return 0 ;
}
int lv_raid_message ( const struct logical_volume * lv , const char * msg )
{
return 0 ;
}
2012-01-25 13:12:59 +00:00
int lv_thin_pool_percent ( const struct logical_volume * lv , int metadata ,
2014-06-09 12:08:27 +02:00
dm_percent_t * percent )
2012-01-25 13:12:59 +00:00
{
return 0 ;
}
int lv_thin_percent ( const struct logical_volume * lv , int mapped ,
2014-06-09 12:08:27 +02:00
dm_percent_t * percent )
2012-01-25 13:12:59 +00:00
{
return 0 ;
}
int lv_thin_pool_transaction_id ( const struct logical_volume * lv ,
uint64_t * transaction_id )
{
return 0 ;
}
2014-04-29 13:22:10 +02:00
int lv_thin_device_id ( const struct logical_volume * lv , uint32_t * device_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 ;
}
2013-09-27 13:58:55 +02:00
int lv_suspend_if_active ( struct cmd_context * cmd , const char * lvid_s , unsigned origin_only , unsigned exclusive ,
2015-11-25 10:52:22 +01:00
const struct logical_volume * lv , const struct logical_volume * lv_pre )
2003-01-08 22:44:07 +00:00
{
return 1 ;
}
2014-09-22 15:50:07 +02:00
int lv_resume ( struct cmd_context * cmd , const char * lvid_s , unsigned origin_only , const struct logical_volume * lv )
2004-03-08 18:54:13 +00:00
{
return 1 ;
}
2013-09-27 13:58:55 +02:00
int lv_resume_if_active ( struct cmd_context * cmd , const char * lvid_s , unsigned origin_only ,
2014-09-22 15:50:07 +02:00
unsigned exclusive , unsigned revert , const struct logical_volume * lv )
2003-01-08 22:44:07 +00:00
{
return 1 ;
}
2014-09-22 15:50:07 +02:00
int lv_deactivate ( struct cmd_context * cmd , const char * lvid_s , const struct logical_volume * lv )
2003-01-08 22:44:07 +00:00
{
return 1 ;
}
2004-03-08 18:54:13 +00:00
int lv_activation_filter ( struct cmd_context * cmd , const char * lvid_s ,
2014-09-22 15:50:07 +02:00
int * activate_lv , const struct logical_volume * lv )
2004-03-08 18:54:13 +00:00
{
return 1 ;
}
2013-10-08 13:27:21 +02:00
int lv_activate ( struct cmd_context * cmd , const char * lvid_s , int exclusive , int noscan ,
2014-09-22 15:50:07 +02:00
int temporary , const struct logical_volume * lv )
2003-01-08 22:44:07 +00:00
{
return 1 ;
}
2013-09-27 13:58:55 +02:00
int lv_activate_with_filter ( struct cmd_context * cmd , const char * lvid_s , int exclusive ,
2014-09-22 15:50:07 +02:00
int noscan , int temporary , const struct logical_volume * lv )
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 ;
}
2016-12-23 03:35:13 +01:00
int lv_deactivate_any_missing_subdevs ( 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
2016-10-27 11:38:16 +02:00
int raid4_is_supported ( struct cmd_context * cmd , const struct segment_type * segtype )
{
return 1 ;
}
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 ;
}
2013-05-15 02:13:31 +01:00
int lv_is_active_locally ( const struct logical_volume * lv )
{
return 0 ;
}
2016-01-19 22:01:59 +00:00
int lv_is_active_remotely ( const struct logical_volume * lv )
{
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 ;
}
2014-09-22 15:50:07 +02:00
int monitor_dev_for_events ( struct cmd_context * cmd , const 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 */
2018-05-14 10:30:20 +01:00
# include "lib/activate/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 ;
}
2014-09-23 12:47:11 +02:00
int device_is_usable ( struct device * dev , struct dev_usable_check_params check )
2012-01-25 13:12:59 +00:00
{
return 0 ;
}
2014-09-22 15:50:07 +02:00
int lv_has_target_type ( struct dm_pool * mem , const struct logical_volume * lv ,
2012-01-25 13:12:59 +00:00
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 ;
2014-04-18 02:46:34 +01:00
void set_activation ( int act , int silent )
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. " ) ;
2014-04-18 02:46:34 +01:00
else if ( ! silent )
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. " ) ;
2014-04-18 02:46:34 +01:00
else
log_verbose ( " Activation disabled. No device-mapper "
" 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_activation_filter ( struct cmd_context * cmd ,
2014-09-22 15:50:07 +02:00
const struct logical_volume * lv )
2012-01-12 01:51:56 +00:00
{
const struct dm_config_node * cn ;
2015-07-08 11:22:24 +02:00
if ( ! ( cn = find_config_tree_array ( cmd , activation_volume_list_CFG , NULL ) ) ) {
2012-01-12 01:51:56 +00:00
log_verbose ( " activation/volume_list configuration setting "
2015-11-25 16:06:31 +01:00
" not defined: Checking only host tags for %s. " ,
display_lvname ( lv ) ) ;
2012-01-12 01:51:56 +00:00
/* 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 ;
2015-11-25 16:06:31 +01:00
log_verbose ( " No host tag matches %s " , display_lvname ( lv ) ) ;
2012-01-12 01:51:56 +00:00
/* Don't activate */
return 0 ;
}
2013-03-05 17:00:43 +01:00
return _lv_passes_volumes_filter ( cmd , lv , cn , activation_volume_list_CFG ) ;
2012-01-12 01:51:56 +00:00
}
static int _passes_readonly_filter ( struct cmd_context * cmd ,
2014-09-22 15:50:07 +02:00
const struct logical_volume * lv )
2012-01-12 01:51:56 +00:00
{
2012-01-12 09:08:55 +00:00
const struct dm_config_node * cn ;
2012-01-12 01:51:56 +00:00
2015-07-08 11:22:24 +02:00
if ( ! ( cn = find_config_tree_array ( cmd , activation_read_only_volume_list_CFG , NULL ) ) )
2012-01-12 01:51:56 +00:00
return 0 ;
2013-03-05 17:00:43 +01:00
return _lv_passes_volumes_filter ( cmd , lv , cn , activation_read_only_volume_list_CFG ) ;
2012-06-27 08:59:34 -04:00
}
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 ) ) {
2013-01-07 22:30:29 +00:00
log_debug_activation ( " Failed to get %s target version " , target_name ) ;
2004-03-26 19:52:09 +00:00
/* 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 :
2012-08-07 18:47:33 +01:00
if ( r )
log_very_verbose ( " Found %s target "
" v% " PRIu32 " .% " PRIu32 " .% " PRIu32 " . " ,
target_name , * maj , * min , * patchlevel ) ;
2004-03-26 19:52:09 +00:00
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 ] ;
2014-09-23 16:47:27 +02:00
const char * argv [ ] = { MODPROBE_CMD , module , NULL } ;
2016-04-26 21:41:04 +02:00
# endif
struct stat st ;
char path [ PATH_MAX ] ;
2018-02-12 21:50:07 +01:00
int i = dm_snprintf ( path , sizeof ( path ) , " %smodule/dm_%s " ,
2016-04-26 21:41:04 +02:00
dm_sysfs_dir ( ) , target_name ) ;
if ( i > 0 ) {
while ( path [ - - i ] ! = ' / ' ) /* stop on dm_ */
if ( path [ i ] = = ' - ' )
path [ i ] = ' _ ' ; /* replace '-' with '_' */
if ( ( lstat ( path , & st ) = = 0 ) & & S_ISDIR ( st . st_mode ) ) {
2016-04-27 00:01:08 +02:00
log_debug_activation ( " Module directory %s exists. " , path ) ;
2016-04-26 21:41:04 +02:00
return 1 ;
}
}
2008-04-07 10:23:47 +00:00
2016-04-26 21:41:04 +02:00
# ifdef MODPROBE_CMD
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 ;
}
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 ;
}
2016-04-27 11:11:58 +02:00
int target_present_version ( struct cmd_context * cmd , const char * target_name ,
int use_modprobe ,
uint32_t * maj , uint32_t * min , uint32_t * patchlevel )
2008-04-07 10:23:47 +00:00
{
2016-05-06 13:59:50 +02:00
if ( ! activation ( ) ) {
log_error ( INTERNAL_ERROR " Target present version called when activation is disabled. " ) ;
return 0 ;
}
2005-10-17 18:00:02 +00:00
# ifdef MODPROBE_CMD
2005-11-08 22:52:26 +00:00
if ( use_modprobe ) {
2016-04-27 11:11:58 +02: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
2016-04-27 11:11:58 +02:00
return target_version ( target_name , maj , min , patchlevel ) ;
}
int target_present ( struct cmd_context * cmd , const char * target_name ,
int use_modprobe )
{
uint32_t maj , min , patchlevel ;
2005-10-17 18:00:02 +00:00
2016-04-27 11:11:58 +02:00
return target_present_version ( cmd , target_name , use_modprobe ,
& maj , & min , & patchlevel ) ;
2005-10-17 18:00:02 +00:00
}
2016-12-01 14:53:35 +01:00
/*
* When ' * info ' is NULL , returns 1 only when LV is active .
* When ' * info ' ! = NULL , returns 1 when info structure is populated .
*/
2014-11-04 15:00:32 +01:00
static int _lv_info ( struct cmd_context * cmd , const struct logical_volume * lv ,
2014-11-13 11:41:49 +01:00
int use_layer , struct lvinfo * info ,
2015-01-14 10:31:24 +01:00
const struct lv_segment * seg ,
struct lv_seg_status * seg_status ,
2014-11-04 15:00:32 +01:00
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 ;
2002-01-10 23:21:07 +00:00
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 ) {
2015-06-30 18:54:38 +01:00
if ( locking_is_clustered ( ) & & ! sync_local_dev_names ( cmd ) ) /* Wait to have udev in sync */
return_0 ;
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
2014-11-04 10:33:35 +01:00
/* New thin-pool has no layer, but -tpool suffix needs to be queried */
2015-01-30 16:22:11 +01:00
if ( ! use_layer & & lv_is_new_thin_pool ( lv ) ) {
/* Check if there isn't existing old thin pool mapping in the table */
2016-04-08 16:27:12 +01:00
if ( ! dev_manager_info ( cmd , lv , NULL , 0 , 0 , & dminfo , NULL , NULL ) )
2015-01-30 16:22:11 +01:00
return_0 ;
if ( ! dminfo . exists )
use_layer = 1 ;
}
2014-11-04 10:33:35 +01:00
2016-05-25 16:14:46 +02:00
if ( seg_status ) {
/* TODO: for now it's mess with seg_status */
2014-11-13 11:41:49 +01:00
seg_status - > seg = seg ;
2016-05-25 16:14:46 +02:00
}
2014-11-13 11:41:49 +01:00
2016-04-08 16:27:12 +01:00
if ( ! dev_manager_info ( cmd , lv ,
2013-02-01 11:09:34 +01:00
( use_layer ) ? lv_layer ( lv ) : NULL ,
with_open_count , with_read_ahead ,
2014-11-04 15:00:32 +01:00
& dminfo , ( info ) ? & info - > read_ahead : NULL ,
seg_status ) )
2005-11-08 22:52:26 +00:00
return_0 ;
2002-01-10 23:21:07 +00:00
2013-09-19 14:05:55 +02:00
if ( ! info )
return dminfo . exists ;
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
2014-11-04 15:00:32 +01:00
/*
* Returns 1 if info structure populated , else 0 on failure .
* When lvinfo * is NULL , it returns 1 if the device is locally active , 0 otherwise .
*/
int lv_info ( struct cmd_context * cmd , const struct logical_volume * lv , int use_layer ,
struct lvinfo * info , int with_open_count , int with_read_ahead )
{
if ( ! activation ( ) )
return 0 ;
2014-11-13 11:41:49 +01:00
return _lv_info ( cmd , lv , use_layer , info , NULL , NULL , with_open_count , with_read_ahead ) ;
2014-11-04 15:00:32 +01: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
}
2014-11-13 11:41:49 +01:00
/*
2016-12-05 14:31:25 +01:00
* Returns 1 if lv_with_info_and_seg_status info structure populated ,
2014-11-13 11:41:49 +01:00
* else 0 on failure or if device not active locally .
*
2016-12-05 10:20:42 +01:00
* When seg_status parsing had troubles it will set type to SEG_STATUS_UNKNOWN .
*
2016-12-05 14:31:25 +01:00
* Using usually one ioctl to obtain info and status .
* More complex segment do collect info from one device ,
* but status from another device .
*
* TODO : further improve with more statuses ( i . e . snapshot ' s origin / merge )
2014-11-13 11:41:49 +01:00
*/
2016-12-05 14:31:25 +01:00
int lv_info_with_seg_status ( struct cmd_context * cmd ,
const struct lv_segment * lv_seg ,
2015-01-14 10:31:24 +01:00
struct lv_with_info_and_seg_status * status ,
int with_open_count , int with_read_ahead )
2014-11-04 15:00:32 +01:00
{
2016-12-05 15:23:18 +01:00
const struct logical_volume * olv , * lv = status - > lv = lv_seg - > lv ;
2016-12-05 14:31:25 +01:00
2014-11-04 15:00:32 +01:00
if ( ! activation ( ) )
return 0 ;
2016-12-05 10:20:42 +01:00
if ( lv_is_used_cache_pool ( lv ) ) {
/* INFO is not set as cache-pool cannot be active.
* STATUS is collected from cache LV */
2017-06-27 00:24:34 +02:00
if ( ! ( lv_seg = get_only_segment_using_this_lv ( lv ) ) )
return_0 ;
2016-12-17 21:52:27 +01:00
( void ) _lv_info ( cmd , lv_seg - > lv , 1 , NULL , lv_seg , & status - > seg_status , 0 , 0 ) ;
2016-12-05 10:20:42 +01:00
return 1 ;
}
if ( lv_is_thin_pool ( lv ) ) {
/* Always collect status for '-tpool' */
if ( _lv_info ( cmd , lv , 1 , & status - > info , lv_seg , & status - > seg_status , 0 , 0 ) & &
( status - > seg_status . type = = SEG_STATUS_THIN_POOL ) ) {
/* There is -tpool device, but query 'active' state of 'fake' thin-pool */
if ( ! _lv_info ( cmd , lv , 0 , NULL , NULL , NULL , 0 , 0 ) & &
! status - > seg_status . thin_pool - > needs_check )
status - > info . exists = 0 ; /* So pool LV is not active */
}
return 1 ;
2017-07-19 16:16:12 +02:00
}
if ( lv_is_external_origin ( lv ) ) {
2016-12-17 21:54:51 +01:00
if ( ! _lv_info ( cmd , lv , 0 , & status - > info , NULL , NULL ,
with_open_count , with_read_ahead ) )
return_0 ;
( void ) _lv_info ( cmd , lv , 1 , NULL , lv_seg , & status - > seg_status , 0 , 0 ) ;
return 1 ;
2017-07-19 16:16:12 +02:00
}
if ( lv_is_origin ( lv ) ) {
2016-12-05 10:20:42 +01:00
/* Query segment status for 'layered' (-real) device most of the time,
* only for merging snapshot , query its progress .
* TODO : single LV may need couple status to be exposed at once . . . .
* but this needs more logical background
*/
2016-12-05 15:23:18 +01:00
/* Show INFO for actual origin and grab status for merging origin */
if ( ! _lv_info ( cmd , lv , 0 , & status - > info , lv_seg ,
lv_is_merging_origin ( lv ) ? & status - > seg_status : NULL ,
with_open_count , with_read_ahead ) )
2016-12-05 10:20:42 +01:00
return_0 ;
2016-12-05 15:23:18 +01:00
if ( status - > info . exists & &
( status - > seg_status . type ! = SEG_STATUS_SNAPSHOT ) ) /* Not merging */
2016-12-05 10:20:42 +01:00
/* Grab STATUS from layered -real */
( void ) _lv_info ( cmd , lv , 1 , NULL , lv_seg , & status - > seg_status , 0 , 0 ) ;
return 1 ;
2017-07-19 16:16:12 +02:00
}
if ( lv_is_cow ( lv ) ) {
2016-12-05 15:23:18 +01:00
if ( lv_is_merging_cow ( lv ) ) {
olv = origin_from_cow ( lv ) ;
if ( ! _lv_info ( cmd , olv , 0 , & status - > info , first_seg ( olv ) , & status - > seg_status ,
with_open_count , with_read_ahead ) )
return_0 ;
if ( status - > seg_status . type = = SEG_STATUS_SNAPSHOT ) {
log_debug_activation ( " Snapshot merge is in progress, querying status of %s instead. " ,
display_lvname ( lv ) ) ;
/*
* When merge is in progress , query merging origin LV instead .
* COW volume is already mapped as error target in this case .
*/
return 1 ;
}
/* Merge not yet started, still a snapshot... */
}
/* Hadle fictional lvm2 snapshot and query snapshotX volume */
lv_seg = find_snapshot ( lv ) ;
2016-12-05 10:20:42 +01:00
}
2016-12-05 14:31:25 +01:00
return _lv_info ( cmd , lv , 0 , & status - > info , lv_seg , & status - > seg_status ,
2016-11-30 13:43:43 +01:00
with_open_count , with_read_ahead ) ;
2014-11-04 15:00:32 +01:00
}
2013-10-15 12:44:42 +02:00
# define OPEN_COUNT_CHECK_RETRIES 25
# define OPEN_COUNT_CHECK_USLEEP_DELAY 200000
2016-04-21 22:14:10 +01:00
/* Only report error if error_if_used is set */
int lv_check_not_in_use ( const struct logical_volume * lv , int error_if_used )
2011-09-22 17:33:50 +00:00
{
2014-09-24 10:05:26 +02:00
struct lvinfo info ;
2013-10-15 12:44:42 +02:00
unsigned int open_count_check_retries ;
2014-09-24 10:05:26 +02:00
if ( ! lv_info ( lv - > vg - > cmd , lv , 0 , & info , 1 , 0 ) | | ! info . exists | | ! info . open_count )
2011-09-22 17:33:50 +00:00
return 1 ;
/* If sysfs is not used, use open_count information only. */
2013-10-15 12:44:42 +02:00
if ( dm_sysfs_dir ( ) ) {
2014-09-24 10:05:26 +02:00
if ( dm_device_has_holders ( info . major , info . minor ) ) {
2016-04-21 22:14:10 +01:00
if ( error_if_used )
log_error ( " Logical volume %s is used by another device. " ,
display_lvname ( lv ) ) ;
else
log_debug_activation ( " Logical volume %s is used by another device. " ,
display_lvname ( lv ) ) ;
2011-09-26 10:17:51 +00:00
return 0 ;
}
2014-09-24 10:05:26 +02:00
if ( dm_device_has_mounted_fs ( info . major , info . minor ) ) {
2016-04-21 22:14:10 +01:00
if ( error_if_used )
log_error ( " Logical volume %s contains a filesystem in use. " ,
display_lvname ( lv ) ) ;
else
log_debug_activation ( " Logical volume %s contains a filesystem in use. " ,
display_lvname ( lv ) ) ;
2013-10-15 12:44:42 +02:00
return 0 ;
}
2011-09-22 17:33:50 +00:00
}
2013-10-15 12:44:42 +02:00
open_count_check_retries = retry_deactivation ( ) ? OPEN_COUNT_CHECK_RETRIES : 1 ;
2014-09-24 10:05:26 +02:00
while ( info . open_count > 0 & & open_count_check_retries - - ) {
2014-05-27 17:07:04 +02:00
if ( ! open_count_check_retries ) {
2016-04-21 22:14:10 +01:00
if ( error_if_used )
log_error ( " Logical volume %s in use. " , display_lvname ( lv ) ) ;
else
log_debug_activation ( " Logical volume %s in use. " , display_lvname ( lv ) ) ;
2013-10-15 12:44:42 +02:00
return 0 ;
2014-05-27 17:07:04 +02:00
}
2014-05-28 15:27:14 +02:00
usleep ( OPEN_COUNT_CHECK_USLEEP_DELAY ) ;
2015-11-25 16:06:31 +01:00
log_debug_activation ( " Retrying open_count check for %s. " ,
display_lvname ( lv ) ) ;
2014-09-24 10:05:26 +02:00
if ( ! lv_info ( lv - > vg - > cmd , lv , 0 , & info , 1 , 0 ) ) {
2014-05-27 17:07:04 +02:00
stack ; /* device dissappeared? */
2013-10-15 12:44:42 +02:00
break ;
2014-05-27 17:07:04 +02:00
}
2011-09-22 17:33:50 +00:00
}
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 ;
2015-11-25 16:06:31 +01:00
log_debug_activation ( " Checking transient status for LV %s. " ,
display_lvname ( lv ) ) ;
2011-06-13 22:28:04 +00:00
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 .
*/
2014-06-09 12:08:27 +02:00
int lv_snapshot_percent ( const struct logical_volume * lv , dm_percent_t * percent )
2002-05-09 21:17:57 +00:00
{
int r ;
struct dev_manager * dm ;
2013-09-19 22:18:16 +02:00
if ( ! lv_info ( lv - > vg - > cmd , lv , 0 , NULL , 0 , 0 ) )
2002-11-18 14:01:16 +00:00
return 0 ;
2015-11-25 16:06:31 +01:00
log_debug_activation ( " Checking snapshot percent for LV %s. " ,
display_lvname ( lv ) ) ;
2011-06-13 22:28:04 +00:00
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 ,
2014-06-09 12:08:27 +02:00
int wait , dm_percent_t * percent , uint32_t * event_nr )
2003-04-30 15:26:25 +00:00
{
int r ;
struct dev_manager * dm ;
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 ) {
2014-06-09 12:08:27 +02:00
* percent = DM_PERCENT_100 ;
2008-01-16 19:18:51 +00:00
return 1 ;
}
2013-09-19 22:18:16 +02:00
if ( ! lv_info ( cmd , lv , 0 , NULL , 0 , 0 ) )
2003-04-30 15:26:25 +00:00
return 0 ;
2015-11-25 16:06:31 +01:00
log_debug_activation ( " Checking mirror percent for LV %s. " ,
display_lvname ( lv ) ) ;
2011-06-13 22:28:04 +00:00
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 ;
}
2014-06-09 12:08:27 +02:00
int lv_raid_percent ( const struct logical_volume * lv , dm_percent_t * percent )
2011-08-11 18:24:40 +00:00
{
return lv_mirror_percent ( lv - > vg - > cmd , lv , 0 , percent , NULL ) ;
}
2017-02-24 00:50:00 +01:00
int lv_raid_data_offset ( const struct logical_volume * lv , uint64_t * data_offset )
{
int r ;
struct dev_manager * dm ;
struct dm_status_raid * status ;
if ( ! lv_info ( lv - > vg - > cmd , lv , 0 , NULL , 0 , 0 ) )
return 0 ;
log_debug_activation ( " Checking raid data offset and dev sectors for LV %s/%s " ,
lv - > vg - > name , lv - > name ) ;
if ( ! ( dm = dev_manager_create ( lv - > vg - > cmd , lv - > vg - > name , 1 ) ) )
return_0 ;
2018-03-02 15:49:12 +01:00
if ( ! ( r = dev_manager_raid_status ( dm , lv , & status ) ) ) {
dev_manager_destroy ( dm ) ;
return_0 ;
}
2017-02-24 00:50:00 +01:00
* data_offset = status - > data_offset ;
dev_manager_destroy ( dm ) ;
return r ;
}
2013-02-01 11:31:47 -06:00
int lv_raid_dev_health ( const struct logical_volume * lv , char * * dev_health )
{
int r ;
struct dev_manager * dm ;
struct dm_status_raid * status ;
2013-02-01 11:32:18 -06:00
2013-02-01 11:31:47 -06:00
* dev_health = NULL ;
2013-09-19 22:18:16 +02:00
if ( ! lv_info ( lv - > vg - > cmd , lv , 0 , NULL , 0 , 0 ) )
return 0 ;
2013-02-01 11:31:47 -06:00
2015-11-25 16:06:31 +01:00
log_debug_activation ( " Checking raid device health for LV %s. " ,
display_lvname ( lv ) ) ;
2013-02-01 11:31:47 -06:00
if ( ! ( dm = dev_manager_create ( lv - > vg - > cmd , lv - > vg - > name , 1 ) ) )
return_0 ;
if ( ! ( r = dev_manager_raid_status ( dm , lv , & status ) ) | |
! ( * dev_health = dm_pool_strdup ( lv - > vg - > cmd - > mem ,
2013-04-08 15:04:08 -05:00
status - > dev_health ) ) ) {
dev_manager_destroy ( dm ) ;
return_0 ;
}
2013-02-01 11:31:47 -06:00
dev_manager_destroy ( dm ) ;
return r ;
}
2017-02-24 00:50:00 +01:00
int lv_raid_dev_count ( const struct logical_volume * lv , uint32_t * dev_cnt )
{
struct dev_manager * dm ;
struct dm_status_raid * status ;
* dev_cnt = 0 ;
if ( ! lv_info ( lv - > vg - > cmd , lv , 0 , NULL , 0 , 0 ) )
return 0 ;
log_debug_activation ( " Checking raid device count for LV %s/%s " ,
lv - > vg - > name , lv - > name ) ;
if ( ! ( dm = dev_manager_create ( lv - > vg - > cmd , lv - > vg - > name , 1 ) ) )
return_0 ;
if ( ! dev_manager_raid_status ( dm , lv , & status ) ) {
dev_manager_destroy ( dm ) ;
return_0 ;
}
* dev_cnt = status - > dev_count ;
dev_manager_destroy ( dm ) ;
return 1 ;
}
2013-04-11 15:33:59 -05:00
int lv_raid_mismatch_count ( const struct logical_volume * lv , uint64_t * cnt )
{
struct dev_manager * dm ;
struct dm_status_raid * status ;
* cnt = 0 ;
2013-09-19 22:18:16 +02:00
if ( ! lv_info ( lv - > vg - > cmd , lv , 0 , NULL , 0 , 0 ) )
2013-04-11 15:33:59 -05:00
return 0 ;
2015-11-25 16:06:31 +01:00
log_debug_activation ( " Checking raid mismatch count for LV %s. " ,
display_lvname ( lv ) ) ;
2013-04-11 15:33:59 -05:00
if ( ! ( dm = dev_manager_create ( lv - > vg - > cmd , lv - > vg - > name , 1 ) ) )
return_0 ;
if ( ! dev_manager_raid_status ( dm , lv , & status ) ) {
dev_manager_destroy ( dm ) ;
return_0 ;
}
* cnt = status - > mismatch_count ;
dev_manager_destroy ( dm ) ;
return 1 ;
}
int lv_raid_sync_action ( const struct logical_volume * lv , char * * sync_action )
{
struct dev_manager * dm ;
struct dm_status_raid * status ;
char * action ;
* sync_action = NULL ;
2013-09-19 22:18:16 +02:00
if ( ! lv_info ( lv - > vg - > cmd , lv , 0 , NULL , 0 , 0 ) )
2013-04-11 15:33:59 -05:00
return 0 ;
2015-11-25 16:06:31 +01:00
log_debug_activation ( " Checking raid sync_action for LV %s. " ,
display_lvname ( lv ) ) ;
2013-04-11 15:33:59 -05:00
if ( ! ( dm = dev_manager_create ( lv - > vg - > cmd , lv - > vg - > name , 1 ) ) )
return_0 ;
2013-07-19 10:01:48 -05:00
/* status->sync_action can be NULL if dm-raid version < 1.5.0 */
2013-04-11 15:33:59 -05:00
if ( ! dev_manager_raid_status ( dm , lv , & status ) | |
2013-07-19 10:01:48 -05:00
! status - > sync_action | |
2013-04-11 15:33:59 -05:00
! ( action = dm_pool_strdup ( lv - > vg - > cmd - > mem ,
status - > sync_action ) ) ) {
dev_manager_destroy ( dm ) ;
return_0 ;
}
* sync_action = action ;
dev_manager_destroy ( dm ) ;
return 1 ;
}
int lv_raid_message ( const struct logical_volume * lv , const char * msg )
{
int r = 0 ;
struct dev_manager * dm ;
struct dm_status_raid * status ;
2013-09-19 22:33:01 -05:00
if ( ! seg_is_raid ( first_seg ( lv ) ) ) {
2013-10-14 15:14:16 -05:00
/*
* Make it easier for user to know what to do when
* they are using thinpool .
*/
if ( lv_is_thin_pool ( lv ) & &
( lv_is_raid ( seg_lv ( first_seg ( lv ) , 0 ) ) | |
lv_is_raid ( first_seg ( lv ) - > metadata_lv ) ) ) {
2013-11-13 13:56:29 +00:00
log_error ( " Thin pool data or metadata volume "
2015-11-25 16:06:31 +01:00
" must be specified. (E.g. \" %s_tdata \" ) " ,
display_lvname ( lv ) ) ;
2013-10-14 15:14:16 -05:00
return 0 ;
}
2015-11-25 16:06:31 +01:00
log_error ( " %s must be a RAID logical volume to perform this action. " ,
display_lvname ( lv ) ) ;
2013-09-19 22:33:01 -05:00
return 0 ;
}
2013-09-19 22:18:16 +02:00
if ( ! lv_info ( lv - > vg - > cmd , lv , 0 , NULL , 0 , 0 ) ) {
2013-04-11 15:33:59 -05:00
log_error ( " Unable to send message to an inactive logical volume. " ) ;
return 0 ;
}
if ( ! ( dm = dev_manager_create ( lv - > vg - > cmd , lv - > vg - > name , 1 ) ) )
return_0 ;
if ( ! ( r = dev_manager_raid_status ( dm , lv , & status ) ) ) {
2015-11-25 16:06:31 +01:00
log_error ( " Failed to retrieve status of %s. " ,
display_lvname ( lv ) ) ;
2013-04-11 15:33:59 -05:00
goto out ;
}
if ( ! status - > sync_action ) {
log_error ( " Kernel driver does not support this action: %s " , msg ) ;
goto out ;
}
/*
* Note that ' dev_manager_raid_message ' allows us to pass down any
* currently valid message . However , this function restricts the
* number of user available combinations to a minimum . Specifically ,
* " idle " - > " check "
* " idle " - > " repair "
* ( The state automatically switches to " idle " when a sync process is
* complete . )
*/
if ( strcmp ( msg , " check " ) & & strcmp ( msg , " repair " ) ) {
/*
* MD allows " frozen " to operate in a toggling fashion .
* We could allow this if we like . . .
*/
log_error ( " \" %s \" is not a supported sync operation. " , msg ) ;
goto out ;
}
if ( strcmp ( status - > sync_action , " idle " ) ) {
2015-11-25 16:06:31 +01:00
log_error ( " %s state is currently \" %s \" . Unable to switch to \" %s \" . " ,
display_lvname ( lv ) , status - > sync_action , msg ) ;
2013-04-11 15:33:59 -05:00
goto out ;
}
r = dev_manager_raid_message ( dm , lv , msg ) ;
out :
dev_manager_destroy ( dm ) ;
return r ;
}
2014-11-03 12:52:29 +01:00
/*
* Return dm_status_cache for cache volume , accept also cache pool
*
* As there are too many variable for cache volumes , and it hard
* to make good API - so let ' s obtain dm_status_cache and return
* all info we have - user just has to release struct after its use .
*/
int lv_cache_status ( const struct logical_volume * cache_lv ,
struct lv_status_cache * * status )
2014-01-28 12:24:51 -06:00
{
struct dev_manager * dm ;
struct lv_segment * cache_seg ;
2016-05-25 16:27:12 +02:00
if ( lv_is_cache_pool ( cache_lv ) ) {
if ( dm_list_empty ( & cache_lv - > segs_using_this_lv ) | |
! ( cache_seg = get_only_segment_using_this_lv ( cache_lv ) ) ) {
log_error ( INTERNAL_ERROR " Cannot check status for unused cache pool %s. " ,
display_lvname ( cache_lv ) ) ;
return 0 ;
}
2014-01-28 12:24:51 -06:00
cache_lv = cache_seg - > lv ;
}
2016-05-25 16:27:12 +02:00
if ( lv_is_pending_delete ( cache_lv ) ) {
log_error ( " Cannot check status for deleted cache volume %s. " ,
display_lvname ( cache_lv ) ) ;
2014-11-11 11:00:35 +01:00
return 0 ;
2016-05-25 16:27:12 +02:00
}
2014-11-11 11:00:35 +01:00
2016-12-17 21:52:27 +01:00
if ( ! lv_info ( cache_lv - > vg - > cmd , cache_lv , 1 , NULL , 0 , 0 ) ) {
2016-05-25 16:27:12 +02:00
log_error ( " Cannot check status for locally inactive cache volume %s. " ,
display_lvname ( cache_lv ) ) ;
2014-11-03 12:52:29 +01:00
return 0 ;
2016-05-25 16:27:12 +02:00
}
2014-01-28 12:24:51 -06:00
2016-05-25 16:27:12 +02:00
log_debug_activation ( " Checking status for cache volume %s. " ,
2014-11-03 12:52:29 +01:00
display_lvname ( cache_lv ) ) ;
2014-01-28 12:24:51 -06:00
if ( ! ( dm = dev_manager_create ( cache_lv - > vg - > cmd , cache_lv - > vg - > name , 1 ) ) )
return_0 ;
2014-11-03 12:52:29 +01:00
if ( ! dev_manager_cache_status ( dm , cache_lv , status ) ) {
2014-01-28 12:24:51 -06:00
dev_manager_destroy ( dm ) ;
return_0 ;
}
2014-11-03 12:52:29 +01:00
/* User has to call dm_pool_destroy(status->mem)! */
2014-01-28 12:24:51 -06:00
return 1 ;
}
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 ,
2014-06-09 12:08:27 +02:00
dm_percent_t * percent )
2011-12-21 13:10:05 +00:00
{
int r ;
struct dev_manager * dm ;
2014-11-03 12:38:24 +01:00
if ( ! lv_info ( lv - > vg - > cmd , lv , 1 , NULL , 0 , 0 ) )
2011-12-21 13:10:05 +00:00
return 0 ;
2015-11-25 16:06:31 +01:00
log_debug_activation ( " Checking thin %sdata percent for LV %s. " ,
( metadata ) ? " meta " : " " , display_lvname ( lv ) ) ;
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 ,
2014-06-09 12:08:27 +02:00
int mapped , dm_percent_t * percent )
2012-01-19 15:27:54 +00:00
{
int r ;
struct dev_manager * dm ;
2014-11-03 12:38:24 +01:00
if ( ! lv_info ( lv - > vg - > cmd , lv , 0 , NULL , 0 , 0 ) )
2012-01-19 15:27:54 +00:00
return 0 ;
2015-11-25 16:06:31 +01:00
log_debug_activation ( " Checking thin percent for LV %s. " ,
display_lvname ( lv ) ) ;
2012-01-19 15:27:54 +00:00
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 ;
2014-11-03 12:38:24 +01:00
if ( ! lv_info ( lv - > vg - > cmd , lv , 1 , NULL , 0 , 0 ) )
2012-01-25 08:48:42 +00:00
return 0 ;
2016-03-29 15:57:12 +02:00
log_debug_activation ( " Checking thin-pool transaction id for LV %s. " ,
2015-11-25 16:06:31 +01:00
display_lvname ( lv ) ) ;
2012-01-25 08:48:42 +00:00
if ( ! ( dm = dev_manager_create ( lv - > vg - > cmd , lv - > vg - > name , 1 ) ) )
return_0 ;
2016-05-27 13:45:47 +02:00
if ( ! ( r = dev_manager_thin_pool_status ( dm , lv , & status , 0 ) ) )
2012-01-25 08:48:42 +00:00
stack ;
else
* transaction_id = status - > transaction_id ;
dev_manager_destroy ( dm ) ;
return r ;
}
2013-12-04 13:57:27 +01:00
int lv_thin_device_id ( const struct logical_volume * lv , uint32_t * device_id )
{
int r ;
struct dev_manager * dm ;
2014-11-03 12:38:24 +01:00
if ( ! lv_info ( lv - > vg - > cmd , lv , 0 , NULL , 0 , 0 ) )
2013-12-04 13:57:27 +01:00
return 0 ;
2015-11-25 16:06:31 +01:00
log_debug_activation ( " Checking device id for LV %s. " ,
display_lvname ( lv ) ) ;
2013-12-04 13:57:27 +01:00
if ( ! ( dm = dev_manager_create ( lv - > vg - > cmd , lv - > vg - > name , 1 ) ) )
return_0 ;
if ( ! ( r = dev_manager_thin_device_id ( dm , lv , device_id ) ) )
stack ;
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 ) ) {
2016-06-22 23:04:53 +02:00
log_debug ( " Cannot determine activation status of %s%s. " ,
display_lvname ( lv ) ,
activation ( ) ? " " : " (no device driver) " ) ;
return 0 ;
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
}
2014-09-22 15:50:07 +02:00
static int _lv_open_count ( struct cmd_context * cmd , const 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
}
2014-09-22 15:50:07 +02:00
static int _lv_activate_lv ( const 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
2014-09-15 21:33:53 +01:00
if ( ! ( dm = dev_manager_create ( lv - > vg - > cmd , lv - > vg - > name , ! lv_is_pvmove ( lv ) ) ) )
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
2014-09-22 15:50:07 +02:00
static int _lv_preload ( const struct logical_volume * lv , struct lv_activate_opts * laopts ,
2011-06-17 14:14:19 +00:00
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 ;
2014-09-15 21:33:53 +01:00
if ( ! ( dm = dev_manager_create ( lv - > vg - > cmd , lv - > vg - > name , ! lv_is_pvmove ( lv ) ) ) )
2012-01-12 01:51:56 +00:00
goto_out ;
2005-11-08 22:52:26 +00:00
2013-11-22 10:00:00 +01:00
laopts - > read_only = _passes_readonly_filter ( lv - > vg - > cmd , lv ) ;
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 ;
}
2014-09-22 15:50:07 +02:00
static int _lv_deactivate ( const struct logical_volume * lv )
2005-11-08 22:52:26 +00:00
{
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
}
2014-09-22 15:50:07 +02:00
static int _lv_suspend_lv ( const struct logical_volume * lv , struct lv_activate_opts * laopts ,
2011-06-17 14:14:19 +00:00
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 .
*/
2014-09-15 21:33:53 +01:00
if ( ! ( dm = dev_manager_create ( lv - > vg - > cmd , lv - > vg - > name , ! lv_is_pvmove ( lv ) ) ) )
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
2013-01-07 22:30:29 +00:00
log_debug_activation ( " 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
2016-06-14 14:56:17 +02:00
log_debug_activation ( " 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 )
2016-01-19 22:01:59 +00:00
* @ remotely : set if active remotely ( when provided )
2011-02-04 20:30:17 +00:00
* @ 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 :
2016-01-19 22:01:59 +00:00
*
2011-02-04 20:30:17 +00:00
* 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 ,
2016-01-19 22:01:59 +00:00
int * locally , int * remotely , 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 */
2016-01-19 22:01:59 +00:00
int skip_cluster_query = 0 ;
2011-02-04 20:30:17 +00:00
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 ;
}
2016-01-19 22:01:59 +00:00
/* Active locally, and the caller doesn't care about exclusive or remotely */
if ( l & & ! exclusive & & ! remotely )
skip_cluster_query = 1 ;
if ( skip_cluster_query )
2011-02-04 20:30:17 +00:00
goto out ;
2008-04-10 21:34:18 +00:00
2018-02-14 23:06:49 +01:00
if ( ( r = cluster_lock_held ( lv - > lvid . s , " " , & e ) ) > = 0 ) {
if ( l & & e )
r = 0 ; /* exclusive locally */
2011-02-04 20:30:17 +00:00
goto out ;
2018-02-14 23:06:49 +01:00
}
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 '
2016-06-21 15:16:03 +02:00
* will be given a warning message .
2009-05-20 12:58:03 +00:00
*/
2016-06-21 15:16:03 +02:00
log_warn ( " WARNING: Unable to determine exclusivity of %s. " , display_lvname ( lv ) ) ;
2012-01-21 05:29:51 +00:00
e = 0 ;
2016-01-19 22:01:59 +00:00
/* Also set remotely as a precaution, as we don't know */
r = 1 ;
2012-01-21 05:29:51 +00:00
/*
* 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 ;
2016-01-19 22:01:59 +00:00
if ( remotely )
* remotely = r ;
2011-02-04 20:30:17 +00:00
2016-01-19 22:01:59 +00:00
log_very_verbose ( " %s is %sactive%s%s%s%s " ,
2015-11-25 16:06:31 +01:00
display_lvname ( lv ) ,
2011-02-04 20:30:17 +00:00
( r | | l ) ? " " : " not " ,
( exclusive & & e ) ? " exclusive " : " " ,
2016-01-19 22:01:59 +00:00
l ? " locally " : " " ,
( ! skip_cluster_query & & l & & r ) ? " and " : " " ,
( ! skip_cluster_query & & r ) ? " remotely " : " " ) ;
2011-02-04 20:30:17 +00:00
return r | | l ;
}
2016-10-27 11:38:16 +02:00
/*
* Check if " raid4 " @ segtype is supported by kernel .
*
* if segment type is not raid4 , return 1.
*/
int raid4_is_supported ( struct cmd_context * cmd , const struct segment_type * segtype )
{
unsigned attrs ;
if ( segtype_is_raid4 ( segtype ) & &
( ! segtype - > ops - > target_present | |
! segtype - > ops - > target_present ( cmd , NULL , & attrs ) | |
! ( attrs & RAID_FEATURE_RAID4 ) ) ) {
log_error ( " RAID module does not support RAID4. " ) ;
return 0 ;
}
return 1 ;
}
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
{
2016-01-19 22:01:59 +00:00
return _lv_is_active ( lv , NULL , NULL , NULL ) ;
2011-02-04 20:30:17 +00:00
}
2013-05-15 02:13:31 +01:00
int lv_is_active_locally ( const struct logical_volume * lv )
{
int l ;
2016-01-19 22:01:59 +00:00
return _lv_is_active ( lv , & l , NULL , NULL ) & & l ;
}
int lv_is_active_remotely ( const struct logical_volume * lv )
{
int r ;
return _lv_is_active ( lv , NULL , & r , NULL ) & & r ;
2013-05-15 02:13:31 +01:00
}
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 ;
2016-01-19 22:01:59 +00:00
return _lv_is_active ( lv , & l , NULL , 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 ;
2016-01-19 22:01:59 +00:00
return _lv_is_active ( lv , NULL , NULL , & e ) & & e ;
2011-10-06 14:55:39 +00:00
}
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
2016-01-19 22:01:59 +00:00
return _lv_is_active ( lv , & l , NULL , & e ) & & l & & e ;
2011-02-04 20:30:17 +00:00
}
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
2016-01-19 22:01:59 +00:00
return _lv_is_active ( lv , & l , NULL , & 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 ;
2018-02-09 23:38:02 +01:00
if ( ! cmd - > default_settings . dmeventd_executable )
cmd - > default_settings . dmeventd_executable = find_config_tree_str ( cmd , dmeventd_executable_CFG , NULL ) ;
if ( dm_event_handler_set_dmeventd_path ( dmevh , cmd - > default_settings . dmeventd_executable ) )
2010-08-16 22:54:35 +00:00
goto_bad ;
2016-08-23 02:24:30 +01:00
if ( dso & & dm_event_handler_set_dso ( dmevh , dso ) )
2010-08-16 22:54:35 +00:00
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 ) ;
2018-01-29 16:28:57 +01:00
2010-08-16 22:54:35 +00:00
return NULL ;
}
2018-02-10 20:22:32 +01:00
char * get_monitor_dso_path ( struct cmd_context * cmd , int id )
2010-08-16 22:54:35 +00:00
{
2018-02-10 20:22:32 +01:00
const char * libpath = find_config_tree_str ( cmd , id , NULL ) ;
char path [ PATH_MAX ] ;
2010-08-16 22:54:35 +00:00
2018-02-10 20:22:32 +01:00
get_shared_library_path ( cmd , libpath , path , sizeof ( path ) ) ;
2010-08-16 22:54:35 +00:00
2018-02-13 19:00:47 +01:00
return dm_strdup ( path ) ;
2010-08-16 22:54:35 +00:00
}
2014-09-22 15:50:07 +02:00
static char * _build_target_uuid ( struct cmd_context * cmd , const struct logical_volume * lv )
2011-12-21 13:08:11 +00:00
{
const char * layer ;
if ( lv_is_thin_pool ( lv ) )
layer = " tpool " ; /* Monitor "tpool" for the "thin pool". */
2017-10-16 15:05:57 +02:00
else if ( lv_is_origin ( lv ) | | lv_is_external_origin ( lv ) )
2011-12-21 13:08:11 +00:00
layer = " real " ; /* Monitor "real" for "snapshot-origin". */
else
layer = NULL ;
2014-03-11 17:13:47 +01:00
return build_dm_uuid ( cmd - > mem , lv , layer ) ;
2011-12-21 13:08:11 +00:00
}
2018-01-29 16:28:57 +01:00
static int _device_registered_with_dmeventd ( struct cmd_context * cmd ,
const struct logical_volume * lv ,
const char * * dso ,
int * pending , int * monitored )
2016-08-23 02:24:30 +01:00
{
char * uuid ;
2018-01-29 16:28:57 +01:00
enum dm_event_mask evmask ;
2016-08-23 02:24:30 +01:00
struct dm_event_handler * dmevh ;
2018-01-29 16:28:57 +01:00
int r ;
2016-08-23 02:24:30 +01:00
* pending = 0 ;
2018-01-29 16:28:57 +01:00
* monitored = 0 ;
2016-08-23 02:24:30 +01:00
if ( ! ( uuid = _build_target_uuid ( cmd , lv ) ) )
return_0 ;
if ( ! ( dmevh = _create_dm_event_handler ( cmd , uuid , NULL , 0 , DM_EVENT_ALL_ERRORS ) ) )
return_0 ;
2018-01-29 16:28:57 +01:00
if ( ( r = dm_event_get_registered_device ( dmevh , 0 ) ) ) {
if ( r = = - ENOENT ) {
r = 1 ;
goto out ;
}
r = 0 ;
goto_out ;
}
/* FIXME: why do we care which 'dso' is monitoring? */
if ( dso & & ( * dso = dm_event_handler_get_dso ( dmevh ) ) & &
! ( * dso = dm_pool_strdup ( cmd - > mem , * dso ) ) ) {
r = 0 ;
goto_out ;
2016-08-23 02:24:30 +01:00
}
evmask = dm_event_handler_get_event_mask ( dmevh ) ;
if ( evmask & DM_EVENT_REGISTRATION_PENDING ) {
* pending = 1 ;
evmask & = ~ DM_EVENT_REGISTRATION_PENDING ;
}
2018-01-29 16:28:57 +01:00
* monitored = evmask ;
r = 1 ;
out :
2016-08-23 02:24:30 +01:00
dm_event_handler_destroy ( dmevh ) ;
2018-01-29 16:28:57 +01:00
return r ;
2016-08-23 02:24:30 +01:00
}
2010-08-17 01:16:41 +00:00
int target_registered_with_dmeventd ( struct cmd_context * cmd , const char * dso ,
2018-01-29 16:28:57 +01:00
const struct logical_volume * lv ,
int * pending , int * monitored )
2010-08-16 22:54:35 +00:00
{
char * uuid ;
2018-01-29 16:28:57 +01:00
enum dm_event_mask evmask ;
2010-08-16 22:54:35 +00:00
struct dm_event_handler * dmevh ;
2018-01-29 16:28:57 +01:00
int r ;
2010-08-16 22:54:35 +00:00
* pending = 0 ;
2018-01-29 16:28:57 +01:00
* monitored = 0 ;
2010-08-16 22:54:35 +00:00
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 ;
2018-01-29 16:28:57 +01:00
if ( ( r = dm_event_get_registered_device ( dmevh , 0 ) ) ) {
if ( r = = - ENOENT ) {
r = 1 ;
goto out ;
}
r = 0 ;
goto_out ;
2010-08-16 22:54:35 +00:00
}
evmask = dm_event_handler_get_event_mask ( dmevh ) ;
if ( evmask & DM_EVENT_REGISTRATION_PENDING ) {
* pending = 1 ;
evmask & = ~ DM_EVENT_REGISTRATION_PENDING ;
}
2018-01-29 16:28:57 +01:00
* monitored = evmask ;
r = 1 ;
out :
2010-08-16 22:54:35 +00:00
dm_event_handler_destroy ( dmevh ) ;
2018-01-29 16:28:57 +01:00
return r ;
2010-08-16 22:54:35 +00:00
}
2014-09-22 15:50:07 +02:00
int target_register_events ( struct cmd_context * cmd , const char * dso , const 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 ;
2018-02-09 23:40:37 +01:00
log_verbose ( " %s %s for events " , set ? " Monitored " : " Unmonitored " , uuid ) ;
2010-08-16 22:54:35 +00:00
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
*/
2014-09-22 15:50:07 +02:00
int monitor_dev_for_events ( struct cmd_context * cmd , const 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
2018-01-29 16:28:57 +01:00
int i , pending = 0 , monitored = 0 ;
2007-01-19 22:21:45 +00:00
int r = 1 ;
2013-04-25 11:46:17 +02:00
struct dm_list * 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 } ;
2018-04-28 22:14:47 +02:00
struct lv_activate_opts mirr_laopts = { . origin_only = 1 } ;
2012-03-23 09:58:04 +00:00
struct lvinfo info ;
2016-09-12 16:37:31 +02:00
const char * dso = NULL ;
2016-08-23 11:30:34 +01:00
int new_unmonitor ;
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 ;
2016-08-24 10:05:09 +02:00
/*
* Activation of unused cache - pool activates metadata device as
* a public LV for clearing purpose .
* FIXME :
* As VG lock is held across whole operation unmonitored volume
* is usually OK since dmeventd couldn ' t do anything .
* However in case command would have crashed , such LV is
* left unmonitored and may potentially require dmeventd .
*/
2017-06-27 10:16:13 +02:00
if ( lv_is_cache_pool_data ( lv ) | | lv_is_cache_pool_metadata ( lv ) ) {
if ( ! ( seg = find_pool_seg ( first_seg ( lv ) ) ) )
return_0 ;
if ( ! lv_is_used_cache_pool ( seg - > lv ) ) {
log_debug_activation ( " Skipping %smonitor of %s.%s " ,
( monitor ) ? " " : " un " , display_lvname ( lv ) ,
( monitor ) ? " Cache pool activation for clearing only. " : " " ) ;
return 1 ;
}
2016-08-24 10:05:09 +02:00
}
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
2013-09-07 02:46:48 +02:00
* Skip unmonitor , if invoked via deactivation of thin volume
2012-03-23 09:58:04 +00:00
* and there is another thin pool user ( open_count > 1 )
2013-09-07 02:46:48 +02:00
* FIXME think about watch ruler influence .
2012-03-23 09:58:04 +00:00
*/
2013-09-07 02:46:48 +02:00
if ( laopts - > skip_in_use & & lv_is_thin_pool ( lv ) & &
lv_info ( lv - > vg - > cmd , lv , 1 , & info , 1 , 0 ) & & ( info . open_count > 1 ) ) {
2013-01-07 22:30:29 +00:00
log_debug_activation ( " Skipping unmonitor of opened %s (open:%d) " ,
2015-11-25 16:06:31 +01:00
display_lvname ( lv ) , info . open_count ) ;
2012-03-23 09:58:04 +00:00
return 1 ;
}
2013-05-27 10:20:06 +02:00
/* Do not monitor snapshot that already covers origin */
if ( monitor & & lv_is_cow_covering_origin ( lv ) ) {
log_debug_activation ( " Skipping monitor of snapshot larger "
2015-11-25 16:06:31 +01:00
" then origin %s. " , display_lvname ( lv ) ) ;
2013-05-27 10:20:06 +02:00
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 .
*/
2014-09-24 09:58:04 +02:00
if ( lv_is_cow ( lv ) & & ( laopts - > no_merging | | ! lv_is_merging_cow ( lv ) ) ) {
if ( ! ( r = monitor_dev_for_events ( cmd , lv - > snapshot - > lv , NULL , monitor ) ) )
stack ;
return r ;
}
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
2016-09-20 02:30:58 +01:00
* also need to be monitored if it is a mirror , for example ,
* so fall through to process it afterwards .
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 ,
2014-09-24 09:58:04 +02:00
struct lv_segment , origin_list ) - > cow , NULL , monitor ) ) {
stack ;
2008-01-17 17:17:09 +00:00
r = 0 ;
2014-09-24 09:58:04 +02:00
}
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 ) )
2014-09-24 09:58:04 +02:00
if ( ! monitor_dev_for_events ( cmd , seg - > log_lv , NULL , monitor ) ) {
stack ;
2010-03-26 22:15:43 +00:00
r = 0 ;
2014-09-24 09:58:04 +02:00
}
2010-03-26 22:15:43 +00:00
2013-04-25 11:46:17 +02:00
dm_list_iterate_items ( seg , & lv - > segments ) {
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 ) ) {
2018-02-12 16:15:35 +01:00
stack ;
2008-01-31 12:19:36 +00:00
r = 0 ;
}
}
2012-03-23 09:58:04 +00:00
/*
2013-09-07 02:46:48 +02:00
* If requested unmonitoring of thin volume , preserve skip_in_use flag .
2012-03-23 09:58:04 +00:00
*
* FIXME : code here looks like _lv_postorder ( )
*/
if ( seg - > pool_lv & &
! monitor_dev_for_events ( cmd , seg - > pool_lv ,
2014-09-24 09:58:04 +02:00
( ! monitor ) ? laopts : NULL , monitor ) ) {
stack ;
2012-03-23 09:58:04 +00:00
r = 0 ;
2014-09-24 09:58:04 +02:00
}
2012-03-23 09:58:04 +00:00
2017-10-16 15:05:57 +02:00
if ( seg - > external_lv & &
! monitor_dev_for_events ( cmd , seg - > external_lv ,
( ! monitor ) ? laopts : NULL , monitor ) ) {
stack ;
r = 0 ;
}
2012-03-23 09:58:04 +00:00
if ( seg - > metadata_lv & &
2014-09-24 09:58:04 +02:00
! monitor_dev_for_events ( cmd , seg - > metadata_lv , NULL , monitor ) ) {
stack ;
2012-03-23 09:58:04 +00:00
r = 0 ;
2014-09-24 09:58:04 +02:00
}
2012-03-23 09:58:04 +00:00
2013-09-05 11:32:42 +02:00
if ( ! seg_monitored ( seg ) | |
( seg - > status & PVMOVE ) | |
! seg - > segtype - > ops - > target_monitored ) /* doesn't support registration */
2007-01-12 20:38:30 +00:00
continue ;
2007-01-19 22:21:45 +00:00
2018-01-29 16:28:57 +01:00
if ( ! monitor ) {
2016-08-23 02:24:30 +01:00
/* When unmonitoring, obtain existing dso being used. */
2018-01-29 16:28:57 +01:00
if ( ! _device_registered_with_dmeventd ( cmd , seg_is_snapshot ( seg ) ? seg - > cow : seg - > lv ,
& dso , & pending , & monitored ) ) {
log_warn ( " WARNING: Failed to %smonitor %s. " ,
monitor ? " " : " un " ,
display_lvname ( seg_is_snapshot ( seg ) ? seg - > cow : seg - > lv ) ) ;
return 0 ;
}
} else if ( ! seg - > segtype - > ops - > target_monitored ( seg , & pending , & monitored ) ) {
log_warn ( " WARNING: Failed to %smonitor %s. " ,
monitor ? " " : " un " ,
display_lvname ( seg - > lv ) ) ;
return 0 ;
}
2007-01-12 20:38:30 +00:00
2013-09-05 11:32:42 +02:00
/* 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
2013-09-05 11:32:42 +02:00
monitor_fn = NULL ;
2016-08-23 11:30:34 +01:00
new_unmonitor = 0 ;
2013-09-05 11:32:42 +02:00
2007-01-19 22:21:45 +00:00
if ( monitor ) {
if ( monitored )
2015-11-25 16:06:31 +01:00
log_verbose ( " %s already monitored. " , display_lvname ( lv ) ) ;
2016-08-23 12:16:39 +01:00
else if ( seg - > segtype - > ops - > target_monitor_events ) {
2018-02-09 23:40:37 +01:00
log_very_verbose ( " Monitoring %s with %s.%s " , display_lvname ( lv ) ,
seg - > segtype - > dso ,
test_mode ( ) ? " [Test mode: skipping this] " : " " ) ;
2007-01-19 22:21:45 +00:00
monitor_fn = seg - > segtype - > ops - > target_monitor_events ;
2016-08-23 12:16:39 +01:00
}
2007-01-12 20:38:30 +00:00
} else {
2007-01-19 22:21:45 +00:00
if ( ! monitored )
2015-11-25 16:06:31 +01:00
log_verbose ( " %s already not monitored. " , display_lvname ( lv ) ) ;
2016-09-12 16:37:31 +02:00
else if ( dso & & * dso ) {
2016-08-23 02:24:30 +01:00
/*
2016-08-23 11:30:34 +01:00
* Divert unmonitor away from code that depends on the new segment
2016-08-23 02:24:30 +01:00
* type instead of the existing one if it ' s changing .
*/
2016-08-23 12:16:39 +01:00
log_verbose ( " Not monitoring %s with %s%s " , display_lvname ( lv ) , dso , test_mode ( ) ? " [Test mode: skipping this] " : " " ) ;
2016-08-23 11:30:34 +01:00
new_unmonitor = 1 ;
2016-08-23 12:16:39 +01:00
}
2007-01-12 20:38:30 +00:00
}
2005-12-02 20:35:07 +00:00
2016-08-23 12:16:39 +01:00
/* FIXME Test mode should really continue a bit further. */
if ( test_mode ( ) )
continue ;
2010-08-16 23:29:09 +00:00
2016-08-23 12:16:39 +01:00
if ( new_unmonitor ) {
2016-09-20 02:30:58 +01:00
if ( ! target_register_events ( cmd , dso , seg_is_snapshot ( seg ) ? seg - > cow : lv , 0 , 0 , 10 ) ) {
2018-02-12 16:15:35 +01:00
log_warn ( " WARNING: %s: segment unmonitoring failed. " ,
display_lvname ( lv ) ) ;
2016-08-23 11:30:34 +01:00
return 0 ;
}
2016-08-23 12:16:39 +01:00
} else if ( monitor_fn ) {
2016-08-23 11:30:34 +01:00
/* FIXME specify events */
if ( ! monitor_fn ( seg , 0 ) ) {
2018-02-12 16:15:35 +01:00
log_warn ( " WARNING: %s: %s segment monitoring function failed. " ,
display_lvname ( lv ) , lvseg_name ( seg ) ) ;
2016-08-23 11:30:34 +01:00
return 0 ;
}
2016-08-23 12:16:39 +01:00
} else
continue ;
2006-05-12 19:16:48 +00:00
2018-04-28 22:14:47 +02:00
if ( ! vg_write_lock_held ( ) & & lv_is_mirror ( lv ) ) {
mirr_laopts . exclusive = lv_is_active_exclusive_locally ( lv ) ? 1 : 0 ;
/*
* Commands vgchange and lvchange do use read - only lock when changing
* monitoring ( - - monitor y | n ) . All other use cases hold ' write - lock '
* so they skip this dm mirror table refreshing step .
*/
if ( ! _lv_activate_lv ( lv , & mirr_laopts ) ) {
stack ;
r = 0 ;
}
}
2007-01-12 20:38:30 +00:00
/* Check [un]monitor results */
/* Try a couple times if pending, but not forever... */
2017-02-13 22:50:19 +01:00
for ( i = 0 ; ; i + + ) {
2007-01-12 20:38:30 +00:00
pending = 0 ;
2018-01-29 16:28:57 +01:00
if ( ! seg - > segtype - > ops - > target_monitored ( seg , & pending , & monitored ) ) {
stack ;
r = 0 ;
break ;
}
2017-02-13 22:50:19 +01:00
if ( ! pending | | i > = 40 )
2007-01-12 20:38:30 +00:00
break ;
2017-02-13 22:50:19 +01:00
log_very_verbose ( " %s %smonitoring still pending: waiting... " ,
display_lvname ( lv ) , monitor ? " " : " un " ) ;
2014-11-08 01:29:01 +01:00
usleep ( 10000 * i ) ;
2007-01-12 20:38:30 +00:00
}
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
2012-06-29 10:18:28 +02:00
if ( ! r & & ! error_message_produced ( ) )
2018-02-12 16:15:35 +01:00
log_warn ( " WARNING: %sonitoring %s failed. " , monitor ? " M " : " Not m " ,
display_lvname ( lv ) ) ;
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 {
2014-09-22 15:50:07 +02:00
const struct logical_volume * lv_pre ;
2011-06-30 18:25:18 +00:00
struct lv_activate_opts * laopts ;
int * flush_required ;
} ;
2014-03-27 10:35:07 +01:00
static int _preload_detached_lv ( struct logical_volume * lv , void * data )
2011-06-30 18:25:18 +00:00
{
struct detached_lv_data * detached = data ;
2015-11-21 23:31:44 +01:00
struct logical_volume * lv_pre ;
2011-06-30 18:25:18 +00:00
2015-01-28 18:30:08 +01:00
/* Check and preload removed raid image leg or metadata */
2015-01-28 13:36:25 +01:00
if ( lv_is_raid_image ( lv ) ) {
2015-11-21 23:31:44 +01:00
if ( ( lv_pre = find_lv_in_vg_by_lvid ( detached - > lv_pre - > vg , & lv - > lvid ) ) & &
! lv_is_raid_image ( lv_pre ) & & lv_is_active ( lv ) & &
! _lv_preload ( lv_pre , detached - > laopts , detached - > flush_required ) )
2015-01-28 13:36:25 +01:00
return_0 ;
2015-01-28 18:30:08 +01:00
} else if ( lv_is_raid_metadata ( lv ) ) {
2015-11-21 23:31:44 +01:00
if ( ( lv_pre = find_lv_in_vg_by_lvid ( detached - > lv_pre - > vg , & lv - > lvid ) ) & &
! lv_is_raid_metadata ( lv_pre ) & & lv_is_active ( lv ) & &
! _lv_preload ( lv_pre , detached - > laopts , detached - > flush_required ) )
2015-01-28 13:36:25 +01:00
return_0 ;
}
2018-04-20 11:55:22 +02:00
if ( ! lv_is_visible ( lv ) & & ( lv_pre = find_lv ( detached - > lv_pre - > vg , lv - > name ) ) & &
lv_is_visible ( lv_pre ) ) {
if ( ! _lv_preload ( lv_pre , detached - > laopts , detached - > flush_required ) )
return_0 ;
}
2015-11-22 23:04:11 +01:00
/* FIXME: condition here should be far more limiting to really
* detect detached LVs */
2015-11-21 23:31:44 +01:00
if ( ( lv_pre = find_lv ( detached - > lv_pre - > vg , lv - > name ) ) ) {
if ( lv_is_visible ( lv_pre ) & & lv_is_active ( lv ) & &
2015-11-22 23:04:11 +01:00
! lv_is_pool ( lv ) & &
2015-11-21 23:31:44 +01:00
( ! lv_is_cow ( lv ) | | ! lv_is_cow ( lv_pre ) ) & &
! _lv_preload ( lv_pre , detached - > laopts , detached - > flush_required ) )
2011-06-30 18:25:18 +00:00
return_0 ;
}
return 1 ;
}
2004-03-08 18:54:13 +00:00
static int _lv_suspend ( struct cmd_context * cmd , const char * lvid_s ,
2013-03-17 21:29:58 +01:00
struct lv_activate_opts * laopts , int error_if_not_suspended ,
2015-11-25 10:52:22 +01:00
const struct logical_volume * lv , const struct logical_volume * lv_pre )
2002-02-25 12:56:16 +00:00
{
2014-09-22 15:50:07 +02:00
const struct logical_volume * pvmove_lv = NULL ;
2015-11-25 10:52:22 +01:00
const struct logical_volume * lv_to_free = NULL ;
const struct logical_volume * lv_pre_to_free = NULL ;
2017-11-30 13:24:41 +01:00
struct logical_volume * lv_pre_tmp , * lv_tmp ;
2011-06-11 00:03:06 +00:00
struct seg_list * sl ;
2016-04-06 10:28:02 +02: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 ;
2017-11-24 13:51:17 +01:00
struct dm_pool * mem = NULL ;
struct dm_list suspend_lvs ;
struct lv_list * lvl ;
2018-04-24 13:19:48 -05:00
const union lvid * lvid = ( const union lvid * ) lvid_s ;
const char * vgid = ( const char * ) lvid - > id [ 0 ] . uuid ;
struct volume_group * vg ;
struct volume_group * vg_pre ;
2017-11-30 13:24:41 +01:00
int found ;
2002-02-25 12:56:16 +00:00
2002-11-18 14:01:16 +00:00
if ( ! activation ( ) )
return 1 ;
2018-04-25 15:08:54 -05:00
if ( ! cmd - > is_clvmd )
goto skip_read ;
2018-04-24 13:19:48 -05:00
if ( lv & & lv_pre )
goto skip_read ;
2018-04-30 16:48:53 -05:00
if ( ! ( vg = lvmcache_get_saved_vg ( vgid , 0 ) ) ) {
log_debug ( " lv_suspend did not find saved_vg %.8s so reading " , vgid ) ;
if ( ! ( vg = vg_read_by_vgid ( cmd , vgid , 0 ) ) ) {
log_error ( " lv_suspend could not read vgid %.8s " , vgid ) ;
2018-04-24 13:19:48 -05:00
goto out ;
}
2018-04-30 16:48:53 -05:00
log_debug ( " lv_suspend using read vg %s %d %p " , vg - > name , vg - > seqno , vg ) ;
} else {
log_debug ( " lv_suspend using saved_vg %s %d %p " , vg - > name , vg - > seqno , vg ) ;
}
2018-04-24 13:19:48 -05:00
2018-04-30 16:48:53 -05:00
if ( ! ( vg_pre = lvmcache_get_saved_vg ( vgid , 1 ) ) ) {
log_debug ( " lv_suspend did not find pre saved_vg %.8s so reading " , vgid ) ;
if ( ! ( vg_pre = vg_read_by_vgid ( cmd , vgid , 1 ) ) ) {
log_error ( " lv_suspend could not read pre vgid %.8s " , vgid ) ;
2018-04-24 13:19:48 -05:00
goto out ;
}
2018-04-30 16:48:53 -05:00
log_debug ( " lv_suspend using pre read vg %s %d %p " , vg_pre - > name , vg_pre - > seqno , vg_pre ) ;
} else {
log_debug ( " lv_suspend using pre saved_vg %s %d %p " , vg_pre - > name , vg_pre - > seqno , vg_pre ) ;
}
/*
* Note that vg and vg_pre returned by vg_read_by_vgid will
* not be the same as saved_vg_old / saved_vg_new that would
* be returned by lvmcache_get_saved_vg ( ) because the saved_vg ' s
* are copies of the vg struct that is created by _vg_read .
* ( Should we grab and use the saved_vg to use here instead of
* the vg returned by vg_read_by_vgid ? )
*/
if ( ( vg - > status & EXPORTED_VG ) | | ( vg_pre - > status & EXPORTED_VG ) ) {
log_error ( " Volume group \" %s \" is exported " , vg - > name ) ;
goto out ;
2018-04-24 13:19:48 -05:00
}
lv = lv_to_free = find_lv_in_vg_by_lvid ( vg , lvid ) ;
lv_pre = lv_pre_to_free = find_lv_in_vg_by_lvid ( vg_pre , lvid ) ;
if ( ! lv | | ! lv_pre ) {
log_error ( " lv_suspend could not find lv %p lv_pre %p vg %p vg_pre %p vgid %s " ,
lv , lv_pre , vg , vg_pre , vgid ) ;
goto out ;
}
skip_read :
2015-11-25 10:52:22 +01:00
/* lv comes from committed metadata */
if ( ! lv & & ! ( lv_to_free = 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 */
2015-11-25 10:52:22 +01:00
if ( ! lv_pre & & ! ( lv_pre_to_free = 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 */
thin: move pool messaging from resume to suspend
Existing messaging intarface for thin-pool has a few 'weak' points:
* Message were posted with each 'resume' operation, thus not allowing
activation of thin-pool with the existing state.
* Acceleration skipped suspend step has not worked in cluster,
since clvmd resumes only nodes which are suspended (have proper lock
state).
* Resume may fail and code is not really designed to 'fail' in this
phase (generic rule here is resume DOES NOT fail unless something serious
is wrong and lvm2 tool usually doesn't handle recovery path in this case.)
* Full thin-pool suspend happened, when taken a thin-volume snapshot.
With this patch the new method relocates message passing into suspend
state.
This has a few drawbacks with current API, but overal it performs
better and gives are more posibilities to deal with errors.
Patch introduces a new logic for 'origin-only' suspend of thin-pool and
this also relates to thin-volume when taking snapshot.
When suspend_origin_only operation is invoked on a pool with
queued messages then only those messages are posted to thin-pool and
actual suspend of thin pool and data and metadata volume is skipped.
This makes taking a snapshot of thin-volume lighter operation and
avoids blocking of other unrelated active thin volumes.
Also fail now happens in 'suspend' state where the 'Fail' is more expected
and it is better handled through error paths.
Activation of thin-pool is now not sending any message and leaves upto a tool
to decided later how to finish unfinished double-commit transaction.
Problem which needs some API improvements relates to the lvm2 tree
construction. For the suspend tree we do not add target table line
into the tree, but only a device is inserted into a tree.
Current mechanism to attach messages for thin-pool requires the libdm
to know about thin-pool target, so lvm2 currently takes assumption, node
is really a thin-pool and fills in the table line for this node (which
should be ensured by the PRELOAD phase, but it's a misuse of internal API)
we would possibly need to be able to attach message to 'any' node.
Other thing to notice - current messaging interface in thin-pool
target requires to suspend thin volume origin first and then send
a create message, but this could not have any 'nice' solution on lvm2
side and IMHO we should introduce something like 'create_after_resume'
message.
Patch also changes the moment, where lvm2 transaction id is increased.
Now it happens only after successful finish of kernel transaction id
change. This change was needed to handle properly activation of pool,
which is in the middle of unfinished transaction, and also this corrects
usage of thin-pool by external apps like Docker.
2015-07-01 13:31:37 +02:00
/* or LV is thin or thin pool volume */
2015-11-25 10:52:22 +01:00
if ( ! lv_is_thin_volume ( lv ) & & ! lv_is_thin_pool ( 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 ( ) ) {
2015-11-25 10:52:22 +01:00
_skip ( " Suspending %s%s. " , display_lvname ( lv ) ,
2013-03-20 00:00:11 +01:00
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
}
2015-11-25 10:52:22 +01: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
lvmcache: simplify metadata cache
The copy of VG metadata stored in lvmcache was not being used
in general. It pretended to be a generic VG metadata cache,
but was not being used except for clvmd activation. There
it was used to avoid reading from disk while devices were
suspended, i.e. in resume.
This removes the code that attempted to make this look
like a generic metadata cache, and replaces with with
something narrowly targetted to what it's actually used for.
This is a way of passing the VG from suspend to resume in
clvmd. Since in the case of clvmd one caller can't simply
pass the same VG to both suspend and resume, suspend needs
to stash the VG somewhere that resume can grab it from.
(resume doesn't want to read it from disk since devices
are suspended.) The lvmcache vginfo struct is used as a
convenient place to stash the VG to pass it from suspend
to resume, even though it isn't related to the lvmcache
or vginfo. These suspended_vg* vginfo fields should
not be used or touched anywhere else, they are only to
be used for passing the VG data from suspend to resume
in clvmd. The VG data being passed between suspend and
resume is never modified, and will only exist in the
brief period between suspend and resume in clvmd.
suspend has both old (current) and new (precommitted)
copies of the VG metadata. It stashes both of these in
the vginfo prior to suspending devices. When vg_commit
is successful, it sets a flag in vginfo as before,
signaling the transition from old to new metadata.
resume grabs the VG stashed by suspend. If the vg_commit
happened, it grabs the new VG, and if the vg_commit didn't
happen it grabs the old VG. The VG is then used to resume
LVs.
This isolates clvmd-specific code and usage from the
normal lvm vg_read code, making the code simpler and
the behavior easier to verify.
Sequence of operations:
- lv_suspend() has both vg_old and vg_new
and stashes a copy of each onto the vginfo:
lvmcache_save_suspended_vg(vg_old);
lvmcache_save_suspended_vg(vg_new);
- vg_commit() happens, which causes all clvmd
instances to call lvmcache_commit_metadata(vg).
A flag is set in the vginfo indicating the
transition from the old to new VG:
vginfo->suspended_vg_committed = 1;
- lv_resume() needs either vg_old or vg_new
to use in resuming LVs. It doesn't want to
read the VG from disk since devices are
suspended, so it gets the VG stashed by
lv_suspend:
vg = lvmcache_get_suspended_vg(vgid);
If the vg_commit did not happen, suspended_vg_committed
will not be set, and in this case, lvmcache_get_suspended_vg()
will return the old VG instead of the new VG, and it will
resume LVs based on the old metadata.
2017-11-01 09:35:40 -05:00
/*
* Save old and new ( current and precommitted ) versions of the
* VG metadata for lv_resume ( ) to use , since lv_resume can ' t
* read metadata given that devices are suspended . lv_resume ( )
* will resume LVs using the old / current metadata if the vg_commit
* did happen ( or failed ) , and it will resume LVs using the
* new / precommitted metadata if the vg_commit succeeded .
*/
2018-04-25 15:08:54 -05:00
if ( cmd - > is_clvmd ) {
lvmcache_save_vg ( lv - > vg , 0 ) ;
lvmcache_save_vg ( lv_pre - > vg , 1 ) ;
}
lvmcache: simplify metadata cache
The copy of VG metadata stored in lvmcache was not being used
in general. It pretended to be a generic VG metadata cache,
but was not being used except for clvmd activation. There
it was used to avoid reading from disk while devices were
suspended, i.e. in resume.
This removes the code that attempted to make this look
like a generic metadata cache, and replaces with with
something narrowly targetted to what it's actually used for.
This is a way of passing the VG from suspend to resume in
clvmd. Since in the case of clvmd one caller can't simply
pass the same VG to both suspend and resume, suspend needs
to stash the VG somewhere that resume can grab it from.
(resume doesn't want to read it from disk since devices
are suspended.) The lvmcache vginfo struct is used as a
convenient place to stash the VG to pass it from suspend
to resume, even though it isn't related to the lvmcache
or vginfo. These suspended_vg* vginfo fields should
not be used or touched anywhere else, they are only to
be used for passing the VG data from suspend to resume
in clvmd. The VG data being passed between suspend and
resume is never modified, and will only exist in the
brief period between suspend and resume in clvmd.
suspend has both old (current) and new (precommitted)
copies of the VG metadata. It stashes both of these in
the vginfo prior to suspending devices. When vg_commit
is successful, it sets a flag in vginfo as before,
signaling the transition from old to new metadata.
resume grabs the VG stashed by suspend. If the vg_commit
happened, it grabs the new VG, and if the vg_commit didn't
happen it grabs the old VG. The VG is then used to resume
LVs.
This isolates clvmd-specific code and usage from the
normal lvm vg_read code, making the code simpler and
the behavior easier to verify.
Sequence of operations:
- lv_suspend() has both vg_old and vg_new
and stashes a copy of each onto the vginfo:
lvmcache_save_suspended_vg(vg_old);
lvmcache_save_suspended_vg(vg_new);
- vg_commit() happens, which causes all clvmd
instances to call lvmcache_commit_metadata(vg).
A flag is set in the vginfo indicating the
transition from the old to new VG:
vginfo->suspended_vg_committed = 1;
- lv_resume() needs either vg_old or vg_new
to use in resuming LVs. It doesn't want to
read the VG from disk since devices are
suspended, so it gets the VG stashed by
lv_suspend:
vg = lvmcache_get_suspended_vg(vgid);
If the vg_commit did not happen, suspended_vg_committed
will not be set, and in this case, lvmcache_get_suspended_vg()
will return the old VG instead of the new VG, and it will
resume LVs based on the old metadata.
2017-11-01 09:35:40 -05: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
2015-11-25 10:52:22 +01:00
lv_calculate_readahead ( lv , NULL ) ;
2009-06-01 12:43:31 +00:00
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 .
*/
2015-11-25 10:52:22 +01:00
if ( lv_is_locked ( lv ) & & ! lv_is_locked ( lv_pre ) & &
( pvmove_lv = find_pvmove_lv_in_lv ( lv ) ) ) {
2011-07-05 18:36:37 +00:00
/* Preload all the LVs above the PVMOVE LV */
dm_list_iterate_items ( sl , & pvmove_lv - > segs_using_this_lv ) {
2015-11-25 10:52:22 +01:00
if ( ! ( lv_pre_tmp = find_lv ( lv_pre - > vg , sl - > seg - > lv - > name ) ) ) {
2015-11-25 16:06:31 +01:00
log_error ( INTERNAL_ERROR " LV %s missing from preload metadata. " ,
display_lvname ( sl - > seg - > lv ) ) ;
2011-06-11 00:03:06 +00:00
goto out ;
}
2015-11-25 10:52:22 +01:00
if ( ! _lv_preload ( lv_pre_tmp , 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 */
2015-11-25 10:52:22 +01:00
if ( ! ( lv_pre_tmp = find_lv ( lv_pre - > vg , pvmove_lv - > name ) ) ) {
2015-11-25 16:06:31 +01:00
log_error ( INTERNAL_ERROR " LV %s missing from preload metadata. " ,
display_lvname ( pvmove_lv ) ) ;
2011-07-05 18:36:37 +00:00
goto out ;
}
2015-11-25 10:52:22 +01:00
if ( ! _lv_preload ( lv_pre_tmp , laopts , & flush_required ) )
2011-07-05 18:36:37 +00:00
goto_out ;
2017-11-15 12:08:33 +01:00
/* Suspending 1st. LV above PVMOVE suspends whole tree */
dm_list_iterate_items ( sl , & pvmove_lv - > segs_using_this_lv ) {
lv = sl - > seg - > lv ;
break ;
}
2011-07-05 18:36:37 +00:00
} else {
2015-11-25 10:52:22 +01:00
if ( ! _lv_preload ( lv_pre , laopts , & flush_required ) )
2011-07-05 18:36:37 +00:00
/* 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 .
*/
2015-11-25 10:52:22 +01:00
detached . lv_pre = lv_pre ;
2011-07-05 18:36:37 +00:00
detached . laopts = laopts ;
detached . flush_required = & flush_required ;
2011-06-30 18:25:18 +00:00
2015-11-25 10:52:22 +01:00
if ( ! for_each_sub_lv ( ( struct logical_volume * ) lv , & _preload_detached_lv , & detached ) )
2011-07-05 18:36:37 +00:00
goto_out ;
2011-07-08 12:48:41 +00:00
/*
* Preload any snapshots that are being removed .
*/
2015-11-25 10:52:22 +01:00
if ( ! laopts - > origin_only & & lv_is_origin ( lv ) ) {
dm_list_iterate_items_gen ( snap_seg , & lv - > snapshot_segs , origin_list ) {
if ( ! ( lv_pre_tmp = find_lv_in_vg_by_lvid ( lv_pre - > vg , & snap_seg - > cow - > lvid ) ) ) {
2015-11-25 16:06:31 +01:00
log_error ( INTERNAL_ERROR " LV %s (%s) missing from preload metadata. " ,
display_lvname ( snap_seg - > cow ) ,
snap_seg - > cow - > lvid . id [ 1 ] . uuid ) ;
2011-07-08 12:48:41 +00:00
goto out ;
}
2015-11-25 10:52:22 +01:00
if ( ! lv_is_cow ( lv_pre_tmp ) & &
! _lv_preload ( lv_pre_tmp , laopts , & flush_required ) )
2011-07-08 12:48:41 +00:00
goto_out ;
}
}
2005-11-08 22:52:26 +00:00
}
2016-04-06 10:29:05 +02:00
/* Flush is ATM required for the tested cases
* NOTE : Mirror repair requires noflush for proper repair !
* TODO : Relax this limiting condition further */
if ( ! flush_required & &
2017-11-15 12:07:47 +01:00
( lv_is_pvmove ( lv ) | | pvmove_lv | |
2016-04-06 10:29:05 +02:00
( ! lv_is_mirror ( lv ) & & ! lv_is_thin_pool ( lv ) & & ! lv_is_thin_volume ( lv ) ) ) ) {
log_debug ( " Requiring flush for LV %s. " , display_lvname ( lv ) ) ;
flush_required = 1 ;
}
2015-11-25 10:52:22 +01: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-17 14:14:19 +00:00
if ( ! laopts - > origin_only & &
2015-11-25 10:52:22 +01:00
( lv_is_origin ( lv_pre ) | | lv_is_cow ( lv_pre ) ) )
2006-08-08 21:20:00 +00:00
lockfs = 1 ;
2013-02-05 11:26:27 +01:00
/* Converting non-thin LV to thin external origin ? */
2015-11-25 10:52:22 +01:00
if ( ! lv_is_thin_volume ( lv ) & & lv_is_thin_volume ( lv_pre ) )
2013-02-05 11:26:27 +01:00
lockfs = 1 ; /* Sync before conversion */
2015-11-25 10:52:22 +01:00
if ( laopts - > origin_only & & lv_is_thin_volume ( lv ) & & lv_is_thin_volume ( lv_pre ) )
2012-06-05 11:26:54 +02:00
lockfs = 1 ;
2017-11-15 14:26:10 +01:00
critical_section_inc ( cmd , " suspending " ) ;
2017-11-24 13:51:17 +01:00
if ( ! lv_is_locked ( lv ) & & lv_is_locked ( lv_pre ) & &
( pvmove_lv = find_pvmove_lv_in_lv ( lv_pre ) ) ) {
/*
* When starting PVMOVE , suspend participating LVs first
* with committed metadata by looking at precommited pvmove list .
* In committed metadata these LVs are not connected in any way .
*
* TODO : prepare list of LVs needed to be suspended and pass them
* via ' struct laopts ' directly to _lv_suspend_lv ( ) and handle this
* with a single ' dmtree ' call .
*/
if ( ! ( mem = dm_pool_create ( " suspend_lvs " , 128 ) ) )
goto_out ;
/* Prepare list of all LVs for suspend ahead */
dm_list_init ( & suspend_lvs ) ;
dm_list_iterate_items ( sl , & pvmove_lv - > segs_using_this_lv ) {
2017-11-30 13:24:41 +01:00
lv_tmp = sl - > seg - > lv ;
if ( lv_is_cow ( lv_tmp ) )
/* Never suspend COW, always has to be origin */
lv_tmp = origin_from_cow ( lv_tmp ) ;
found = 0 ;
dm_list_iterate_items ( lvl , & suspend_lvs )
if ( strcmp ( lvl - > lv - > name , lv_tmp - > name ) = = 0 ) {
found = 1 ;
break ;
}
if ( found )
continue ; /* LV is already in the list */
2017-11-24 13:51:17 +01:00
if ( ! ( lvl = dm_pool_alloc ( mem , sizeof ( * lvl ) ) ) ) {
log_error ( " lv_list alloc failed. " ) ;
goto out ;
}
/* Look for precommitted LV name in commmitted VG */
2017-11-30 13:24:41 +01:00
if ( ! ( lvl - > lv = find_lv ( lv - > vg , lv_tmp - > name ) ) ) {
2017-11-24 13:51:17 +01:00
log_error ( INTERNAL_ERROR " LV %s missing from preload metadata. " ,
2017-11-30 13:24:41 +01:00
display_lvname ( lv_tmp ) ) ;
2017-11-24 13:51:17 +01:00
goto out ;
}
dm_list_add ( & suspend_lvs , & lvl - > list ) ;
}
dm_list_iterate_items ( lvl , & suspend_lvs )
if ( ! _lv_suspend_lv ( lvl - > lv , laopts , lockfs , 1 ) ) {
critical_section_dec ( cmd , " failed suspend " ) ;
goto_out ; /* FIXME: resume on recovery path? */
}
} else /* Standard suspend */
if ( ! _lv_suspend_lv ( lv , laopts , lockfs , flush_required ) ) {
critical_section_dec ( cmd , " failed suspend " ) ;
goto_out ;
}
2002-03-01 19:08:11 +00:00
2009-04-10 10:00:04 +00:00
r = 1 ;
out :
2017-11-24 13:51:17 +01:00
if ( mem )
dm_pool_destroy ( mem ) ;
2015-11-25 10:52:22 +01:00
if ( lv_pre_to_free )
release_vg ( lv_pre_to_free - > vg ) ;
2017-10-13 14:43:15 -05:00
if ( lv_to_free )
2015-11-25 10:52:22 +01:00
release_vg ( lv_to_free - > 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 preloaded tables may then use non - clustered targets .
*
* Returns success if the device is not active
*/
2014-09-22 15:50:07 +02:00
int lv_suspend_if_active ( struct cmd_context * cmd , const char * lvid_s , unsigned origin_only , unsigned exclusive ,
2015-11-25 10:52:22 +01:00
const struct logical_volume * lv , const struct logical_volume * lv_pre )
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
2015-11-25 10:52:22 +01:00
return _lv_suspend ( cmd , lvid_s , & laopts , 0 , lv , lv_pre ) ;
2004-03-08 18:54:13 +00:00
}
2017-11-10 21:15:50 +01:00
static int _check_suspended_lv ( struct logical_volume * lv , void * data )
{
struct lvinfo info ;
if ( lv_info ( lv - > vg - > cmd , lv , 0 , & info , 0 , 0 ) & & info . exists & & info . suspended ) {
log_debug ( " Found suspended LV %s in critical section(). " , display_lvname ( lv ) ) ;
return 0 ; /* There is suspended subLV in the tree */
}
2017-11-28 23:11:20 +01:00
if ( lv_layer ( lv ) & & lv_info ( lv - > vg - > cmd , lv , 1 , & info , 0 , 0 ) & & info . exists & & info . suspended ) {
log_debug ( " Found suspended layered LV %s in critical section(). " , display_lvname ( lv ) ) ;
return 0 ; /* There is suspended subLV in the tree */
}
2017-11-10 21:15:50 +01:00
return 1 ;
}
2004-03-08 18:54:13 +00:00
static int _lv_resume ( struct cmd_context * cmd , const char * lvid_s ,
2013-03-17 21:29:58 +01:00
struct lv_activate_opts * laopts , int error_if_not_active ,
2014-09-22 15:50:07 +02:00
const struct logical_volume * lv )
2002-02-25 12:56:16 +00:00
{
2017-11-28 23:11:20 +01:00
struct dm_list * snh ;
lvmcache: simplify metadata cache
The copy of VG metadata stored in lvmcache was not being used
in general. It pretended to be a generic VG metadata cache,
but was not being used except for clvmd activation. There
it was used to avoid reading from disk while devices were
suspended, i.e. in resume.
This removes the code that attempted to make this look
like a generic metadata cache, and replaces with with
something narrowly targetted to what it's actually used for.
This is a way of passing the VG from suspend to resume in
clvmd. Since in the case of clvmd one caller can't simply
pass the same VG to both suspend and resume, suspend needs
to stash the VG somewhere that resume can grab it from.
(resume doesn't want to read it from disk since devices
are suspended.) The lvmcache vginfo struct is used as a
convenient place to stash the VG to pass it from suspend
to resume, even though it isn't related to the lvmcache
or vginfo. These suspended_vg* vginfo fields should
not be used or touched anywhere else, they are only to
be used for passing the VG data from suspend to resume
in clvmd. The VG data being passed between suspend and
resume is never modified, and will only exist in the
brief period between suspend and resume in clvmd.
suspend has both old (current) and new (precommitted)
copies of the VG metadata. It stashes both of these in
the vginfo prior to suspending devices. When vg_commit
is successful, it sets a flag in vginfo as before,
signaling the transition from old to new metadata.
resume grabs the VG stashed by suspend. If the vg_commit
happened, it grabs the new VG, and if the vg_commit didn't
happen it grabs the old VG. The VG is then used to resume
LVs.
This isolates clvmd-specific code and usage from the
normal lvm vg_read code, making the code simpler and
the behavior easier to verify.
Sequence of operations:
- lv_suspend() has both vg_old and vg_new
and stashes a copy of each onto the vginfo:
lvmcache_save_suspended_vg(vg_old);
lvmcache_save_suspended_vg(vg_new);
- vg_commit() happens, which causes all clvmd
instances to call lvmcache_commit_metadata(vg).
A flag is set in the vginfo indicating the
transition from the old to new VG:
vginfo->suspended_vg_committed = 1;
- lv_resume() needs either vg_old or vg_new
to use in resuming LVs. It doesn't want to
read the VG from disk since devices are
suspended, so it gets the VG stashed by
lv_suspend:
vg = lvmcache_get_suspended_vg(vgid);
If the vg_commit did not happen, suspended_vg_committed
will not be set, and in this case, lvmcache_get_suspended_vg()
will return the old VG instead of the new VG, and it will
resume LVs based on the old metadata.
2017-11-01 09:35:40 -05:00
struct volume_group * vg = NULL ;
struct logical_volume * lv_found = NULL ;
const union lvid * lvid ;
const char * vgid ;
2003-01-08 22:44:07 +00:00
struct lvinfo info ;
2009-04-10 10:00:04 +00:00
int r = 0 ;
2002-02-25 12:56:16 +00:00
2002-11-18 14:01:16 +00:00
if ( ! activation ( ) )
return 1 ;
lvmcache: simplify metadata cache
The copy of VG metadata stored in lvmcache was not being used
in general. It pretended to be a generic VG metadata cache,
but was not being used except for clvmd activation. There
it was used to avoid reading from disk while devices were
suspended, i.e. in resume.
This removes the code that attempted to make this look
like a generic metadata cache, and replaces with with
something narrowly targetted to what it's actually used for.
This is a way of passing the VG from suspend to resume in
clvmd. Since in the case of clvmd one caller can't simply
pass the same VG to both suspend and resume, suspend needs
to stash the VG somewhere that resume can grab it from.
(resume doesn't want to read it from disk since devices
are suspended.) The lvmcache vginfo struct is used as a
convenient place to stash the VG to pass it from suspend
to resume, even though it isn't related to the lvmcache
or vginfo. These suspended_vg* vginfo fields should
not be used or touched anywhere else, they are only to
be used for passing the VG data from suspend to resume
in clvmd. The VG data being passed between suspend and
resume is never modified, and will only exist in the
brief period between suspend and resume in clvmd.
suspend has both old (current) and new (precommitted)
copies of the VG metadata. It stashes both of these in
the vginfo prior to suspending devices. When vg_commit
is successful, it sets a flag in vginfo as before,
signaling the transition from old to new metadata.
resume grabs the VG stashed by suspend. If the vg_commit
happened, it grabs the new VG, and if the vg_commit didn't
happen it grabs the old VG. The VG is then used to resume
LVs.
This isolates clvmd-specific code and usage from the
normal lvm vg_read code, making the code simpler and
the behavior easier to verify.
Sequence of operations:
- lv_suspend() has both vg_old and vg_new
and stashes a copy of each onto the vginfo:
lvmcache_save_suspended_vg(vg_old);
lvmcache_save_suspended_vg(vg_new);
- vg_commit() happens, which causes all clvmd
instances to call lvmcache_commit_metadata(vg).
A flag is set in the vginfo indicating the
transition from the old to new VG:
vginfo->suspended_vg_committed = 1;
- lv_resume() needs either vg_old or vg_new
to use in resuming LVs. It doesn't want to
read the VG from disk since devices are
suspended, so it gets the VG stashed by
lv_suspend:
vg = lvmcache_get_suspended_vg(vgid);
If the vg_commit did not happen, suspended_vg_committed
will not be set, and in this case, lvmcache_get_suspended_vg()
will return the old VG instead of the new VG, and it will
resume LVs based on the old metadata.
2017-11-01 09:35:40 -05:00
/*
* When called in clvmd , lvid_s is set and lv is not . We need to
* get the VG metadata without reading disks because devs are
* suspended . lv_suspend ( ) saved old and new VG metadata for us
2018-04-24 13:19:48 -05:00
* to use here . If vg_commit ( ) happened , lvmcache_get_saved_vg_latest
lvmcache: simplify metadata cache
The copy of VG metadata stored in lvmcache was not being used
in general. It pretended to be a generic VG metadata cache,
but was not being used except for clvmd activation. There
it was used to avoid reading from disk while devices were
suspended, i.e. in resume.
This removes the code that attempted to make this look
like a generic metadata cache, and replaces with with
something narrowly targetted to what it's actually used for.
This is a way of passing the VG from suspend to resume in
clvmd. Since in the case of clvmd one caller can't simply
pass the same VG to both suspend and resume, suspend needs
to stash the VG somewhere that resume can grab it from.
(resume doesn't want to read it from disk since devices
are suspended.) The lvmcache vginfo struct is used as a
convenient place to stash the VG to pass it from suspend
to resume, even though it isn't related to the lvmcache
or vginfo. These suspended_vg* vginfo fields should
not be used or touched anywhere else, they are only to
be used for passing the VG data from suspend to resume
in clvmd. The VG data being passed between suspend and
resume is never modified, and will only exist in the
brief period between suspend and resume in clvmd.
suspend has both old (current) and new (precommitted)
copies of the VG metadata. It stashes both of these in
the vginfo prior to suspending devices. When vg_commit
is successful, it sets a flag in vginfo as before,
signaling the transition from old to new metadata.
resume grabs the VG stashed by suspend. If the vg_commit
happened, it grabs the new VG, and if the vg_commit didn't
happen it grabs the old VG. The VG is then used to resume
LVs.
This isolates clvmd-specific code and usage from the
normal lvm vg_read code, making the code simpler and
the behavior easier to verify.
Sequence of operations:
- lv_suspend() has both vg_old and vg_new
and stashes a copy of each onto the vginfo:
lvmcache_save_suspended_vg(vg_old);
lvmcache_save_suspended_vg(vg_new);
- vg_commit() happens, which causes all clvmd
instances to call lvmcache_commit_metadata(vg).
A flag is set in the vginfo indicating the
transition from the old to new VG:
vginfo->suspended_vg_committed = 1;
- lv_resume() needs either vg_old or vg_new
to use in resuming LVs. It doesn't want to
read the VG from disk since devices are
suspended, so it gets the VG stashed by
lv_suspend:
vg = lvmcache_get_suspended_vg(vgid);
If the vg_commit did not happen, suspended_vg_committed
will not be set, and in this case, lvmcache_get_suspended_vg()
will return the old VG instead of the new VG, and it will
resume LVs based on the old metadata.
2017-11-01 09:35:40 -05:00
* will return the new metadata for us to use in resuming LVs .
2018-04-24 13:19:48 -05:00
* If vg_commit ( ) did not happen , lvmcache_get_saved_vg_latest
lvmcache: simplify metadata cache
The copy of VG metadata stored in lvmcache was not being used
in general. It pretended to be a generic VG metadata cache,
but was not being used except for clvmd activation. There
it was used to avoid reading from disk while devices were
suspended, i.e. in resume.
This removes the code that attempted to make this look
like a generic metadata cache, and replaces with with
something narrowly targetted to what it's actually used for.
This is a way of passing the VG from suspend to resume in
clvmd. Since in the case of clvmd one caller can't simply
pass the same VG to both suspend and resume, suspend needs
to stash the VG somewhere that resume can grab it from.
(resume doesn't want to read it from disk since devices
are suspended.) The lvmcache vginfo struct is used as a
convenient place to stash the VG to pass it from suspend
to resume, even though it isn't related to the lvmcache
or vginfo. These suspended_vg* vginfo fields should
not be used or touched anywhere else, they are only to
be used for passing the VG data from suspend to resume
in clvmd. The VG data being passed between suspend and
resume is never modified, and will only exist in the
brief period between suspend and resume in clvmd.
suspend has both old (current) and new (precommitted)
copies of the VG metadata. It stashes both of these in
the vginfo prior to suspending devices. When vg_commit
is successful, it sets a flag in vginfo as before,
signaling the transition from old to new metadata.
resume grabs the VG stashed by suspend. If the vg_commit
happened, it grabs the new VG, and if the vg_commit didn't
happen it grabs the old VG. The VG is then used to resume
LVs.
This isolates clvmd-specific code and usage from the
normal lvm vg_read code, making the code simpler and
the behavior easier to verify.
Sequence of operations:
- lv_suspend() has both vg_old and vg_new
and stashes a copy of each onto the vginfo:
lvmcache_save_suspended_vg(vg_old);
lvmcache_save_suspended_vg(vg_new);
- vg_commit() happens, which causes all clvmd
instances to call lvmcache_commit_metadata(vg).
A flag is set in the vginfo indicating the
transition from the old to new VG:
vginfo->suspended_vg_committed = 1;
- lv_resume() needs either vg_old or vg_new
to use in resuming LVs. It doesn't want to
read the VG from disk since devices are
suspended, so it gets the VG stashed by
lv_suspend:
vg = lvmcache_get_suspended_vg(vgid);
If the vg_commit did not happen, suspended_vg_committed
will not be set, and in this case, lvmcache_get_suspended_vg()
will return the old VG instead of the new VG, and it will
resume LVs based on the old metadata.
2017-11-01 09:35:40 -05:00
* returns the old metadata which we use to resume LVs .
*/
if ( ! lv & & lvid_s ) {
lvid = ( const union lvid * ) lvid_s ;
vgid = ( const char * ) lvid - > id [ 0 ] . uuid ;
2018-04-24 13:19:48 -05:00
if ( ( vg = lvmcache_get_saved_vg_latest ( vgid ) ) ) {
lvmcache: simplify metadata cache
The copy of VG metadata stored in lvmcache was not being used
in general. It pretended to be a generic VG metadata cache,
but was not being used except for clvmd activation. There
it was used to avoid reading from disk while devices were
suspended, i.e. in resume.
This removes the code that attempted to make this look
like a generic metadata cache, and replaces with with
something narrowly targetted to what it's actually used for.
This is a way of passing the VG from suspend to resume in
clvmd. Since in the case of clvmd one caller can't simply
pass the same VG to both suspend and resume, suspend needs
to stash the VG somewhere that resume can grab it from.
(resume doesn't want to read it from disk since devices
are suspended.) The lvmcache vginfo struct is used as a
convenient place to stash the VG to pass it from suspend
to resume, even though it isn't related to the lvmcache
or vginfo. These suspended_vg* vginfo fields should
not be used or touched anywhere else, they are only to
be used for passing the VG data from suspend to resume
in clvmd. The VG data being passed between suspend and
resume is never modified, and will only exist in the
brief period between suspend and resume in clvmd.
suspend has both old (current) and new (precommitted)
copies of the VG metadata. It stashes both of these in
the vginfo prior to suspending devices. When vg_commit
is successful, it sets a flag in vginfo as before,
signaling the transition from old to new metadata.
resume grabs the VG stashed by suspend. If the vg_commit
happened, it grabs the new VG, and if the vg_commit didn't
happen it grabs the old VG. The VG is then used to resume
LVs.
This isolates clvmd-specific code and usage from the
normal lvm vg_read code, making the code simpler and
the behavior easier to verify.
Sequence of operations:
- lv_suspend() has both vg_old and vg_new
and stashes a copy of each onto the vginfo:
lvmcache_save_suspended_vg(vg_old);
lvmcache_save_suspended_vg(vg_new);
- vg_commit() happens, which causes all clvmd
instances to call lvmcache_commit_metadata(vg).
A flag is set in the vginfo indicating the
transition from the old to new VG:
vginfo->suspended_vg_committed = 1;
- lv_resume() needs either vg_old or vg_new
to use in resuming LVs. It doesn't want to
read the VG from disk since devices are
suspended, so it gets the VG stashed by
lv_suspend:
vg = lvmcache_get_suspended_vg(vgid);
If the vg_commit did not happen, suspended_vg_committed
will not be set, and in this case, lvmcache_get_suspended_vg()
will return the old VG instead of the new VG, and it will
resume LVs based on the old metadata.
2017-11-01 09:35:40 -05:00
log_debug_activation ( " Resuming LVID %s found saved vg seqno %d %s " , lvid_s , vg - > seqno , vg - > name ) ;
if ( ( lv_found = find_lv_in_vg_by_lvid ( vg , lvid ) ) ) {
log_debug_activation ( " Resuming LVID %s found saved LV %s " , lvid_s , display_lvname ( lv_found ) ) ;
lv = lv_found ;
} else
log_debug_activation ( " Resuming LVID %s did not find saved LV " , lvid_s ) ;
} else
log_debug_activation ( " Resuming LVID %s did not find saved VG " , lvid_s ) ;
/*
* resume must have been called without a preceding suspend ,
* so we need to read the vg .
*/
if ( ! lv ) {
log_debug_activation ( " Resuming LVID %s reading VG " , lvid_s ) ;
if ( ! ( lv_found = lv_from_lvid ( cmd , lvid_s , 0 ) ) ) {
log_debug_activation ( " Resuming LVID %s failed to read VG " , lvid_s ) ;
goto out ;
}
lv = lv_found ;
}
}
2002-02-25 12:56:16 +00:00
thin: move pool messaging from resume to suspend
Existing messaging intarface for thin-pool has a few 'weak' points:
* Message were posted with each 'resume' operation, thus not allowing
activation of thin-pool with the existing state.
* Acceleration skipped suspend step has not worked in cluster,
since clvmd resumes only nodes which are suspended (have proper lock
state).
* Resume may fail and code is not really designed to 'fail' in this
phase (generic rule here is resume DOES NOT fail unless something serious
is wrong and lvm2 tool usually doesn't handle recovery path in this case.)
* Full thin-pool suspend happened, when taken a thin-volume snapshot.
With this patch the new method relocates message passing into suspend
state.
This has a few drawbacks with current API, but overal it performs
better and gives are more posibilities to deal with errors.
Patch introduces a new logic for 'origin-only' suspend of thin-pool and
this also relates to thin-volume when taking snapshot.
When suspend_origin_only operation is invoked on a pool with
queued messages then only those messages are posted to thin-pool and
actual suspend of thin pool and data and metadata volume is skipped.
This makes taking a snapshot of thin-volume lighter operation and
avoids blocking of other unrelated active thin volumes.
Also fail now happens in 'suspend' state where the 'Fail' is more expected
and it is better handled through error paths.
Activation of thin-pool is now not sending any message and leaves upto a tool
to decided later how to finish unfinished double-commit transaction.
Problem which needs some API improvements relates to the lvm2 tree
construction. For the suspend tree we do not add target table line
into the tree, but only a device is inserted into a tree.
Current mechanism to attach messages for thin-pool requires the libdm
to know about thin-pool target, so lvm2 currently takes assumption, node
is really a thin-pool and fills in the table line for this node (which
should be ensured by the PRELOAD phase, but it's a misuse of internal API)
we would possibly need to be able to attach message to 'any' node.
Other thing to notice - current messaging interface in thin-pool
target requires to suspend thin volume origin first and then send
a create message, but this could not have any 'nice' solution on lvm2
side and IMHO we should introduce something like 'create_after_resume'
message.
Patch also changes the moment, where lvm2 transaction id is increased.
Now it happens only after successful finish of kernel transaction id
change. This change was needed to handle properly activation of pool,
which is in the middle of unfinished transaction, and also this corrects
usage of thin-pool by external apps like Docker.
2015-07-01 13:31:37 +02:00
if ( ! lv_is_origin ( lv ) & & ! lv_is_thin_volume ( lv ) & & ! lv_is_thin_pool ( 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 ( ) ) {
2015-11-25 16:06:31 +01:00
_skip ( " Resuming %s%s%s. " , display_lvname ( lv ) ,
laopts - > origin_only ? " without snapshots " : " " ,
2011-09-27 22:43:40 +00:00
laopts - > revert ? " (reverting) " : " " ) ;
2009-04-10 10:00:04 +00:00
r = 1 ;
goto out ;
2002-03-14 15:36:07 +00:00
}
2015-11-25 16:06:31 +01:00
log_debug_activation ( " Resuming LV %s%s%s%s. " , display_lvname ( lv ) ,
2013-01-07 22:30:29 +00:00
error_if_not_active ? " " : " if active " ,
2015-06-15 14:33:29 +02:00
laopts - > origin_only ?
( lv_is_thin_pool ( lv ) ? " pool only " :
lv_is_thin_volume ( lv ) ? " thin only " : " without snapshots " ) : " " ,
2013-01-07 22:30:29 +00:00
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
thin: move pool messaging from resume to suspend
Existing messaging intarface for thin-pool has a few 'weak' points:
* Message were posted with each 'resume' operation, thus not allowing
activation of thin-pool with the existing state.
* Acceleration skipped suspend step has not worked in cluster,
since clvmd resumes only nodes which are suspended (have proper lock
state).
* Resume may fail and code is not really designed to 'fail' in this
phase (generic rule here is resume DOES NOT fail unless something serious
is wrong and lvm2 tool usually doesn't handle recovery path in this case.)
* Full thin-pool suspend happened, when taken a thin-volume snapshot.
With this patch the new method relocates message passing into suspend
state.
This has a few drawbacks with current API, but overal it performs
better and gives are more posibilities to deal with errors.
Patch introduces a new logic for 'origin-only' suspend of thin-pool and
this also relates to thin-volume when taking snapshot.
When suspend_origin_only operation is invoked on a pool with
queued messages then only those messages are posted to thin-pool and
actual suspend of thin pool and data and metadata volume is skipped.
This makes taking a snapshot of thin-volume lighter operation and
avoids blocking of other unrelated active thin volumes.
Also fail now happens in 'suspend' state where the 'Fail' is more expected
and it is better handled through error paths.
Activation of thin-pool is now not sending any message and leaves upto a tool
to decided later how to finish unfinished double-commit transaction.
Problem which needs some API improvements relates to the lvm2 tree
construction. For the suspend tree we do not add target table line
into the tree, but only a device is inserted into a tree.
Current mechanism to attach messages for thin-pool requires the libdm
to know about thin-pool target, so lvm2 currently takes assumption, node
is really a thin-pool and fills in the table line for this node (which
should be ensured by the PRELOAD phase, but it's a misuse of internal API)
we would possibly need to be able to attach message to 'any' node.
Other thing to notice - current messaging interface in thin-pool
target requires to suspend thin volume origin first and then send
a create message, but this could not have any 'nice' solution on lvm2
side and IMHO we should introduce something like 'create_after_resume'
message.
Patch also changes the moment, where lvm2 transaction id is increased.
Now it happens only after successful finish of kernel transaction id
change. This change was needed to handle properly activation of pool,
which is in the middle of unfinished transaction, and also this corrects
usage of thin-pool by external apps like Docker.
2015-07-01 13:31:37 +02:00
if ( ! info . exists | | ! info . suspended ) {
2010-07-08 12:24:04 +00:00
if ( error_if_not_active )
goto_out ;
2017-11-10 21:15:50 +01:00
2017-11-24 13:53:02 +01:00
/* ATM only thin-pool with origin-only suspend does not really suspend anything
* it ' s used only for message passing to thin - pool */
if ( laopts - > origin_only & & lv_is_thin_pool ( lv ) )
critical_section_dec ( cmd , " resumed " ) ;
2017-11-10 21:15:50 +01:00
if ( ! info . suspended & & critical_section ( ) ) {
2017-11-28 23:11:20 +01:00
/* Validation check if any subLV is suspended */
if ( ! laopts - > origin_only & & lv_is_origin ( lv ) ) {
/* Check all snapshots for this origin LV */
dm_list_iterate ( snh , & lv - > snapshot_segs )
if ( ! _check_suspended_lv ( dm_list_struct_base ( snh , struct lv_segment , origin_list ) - > cow , NULL ) )
goto needs_resume ; /* Found suspended snapshot */
2017-11-10 21:15:50 +01:00
}
2017-11-28 23:11:20 +01:00
if ( ( r = for_each_sub_lv ( ( struct logical_volume * ) lv , & _check_suspended_lv , NULL ) ) )
goto out ; /* Nothing was found suspended */
2017-11-10 21:15:50 +01:00
} else {
r = 1 ;
goto out ;
}
2009-04-10 10:00:04 +00:00
}
2017-11-28 23:11:20 +01:00
needs_resume :
2012-01-12 01:51:56 +00:00
laopts - > read_only = _passes_readonly_filter ( cmd , lv ) ;
2016-04-15 02:21:27 +01:00
laopts - > resuming = 1 ;
2012-01-12 01:51:56 +00:00
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 :
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 ,
2014-09-22 15:50:07 +02:00
unsigned revert , const struct logical_volume * lv )
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
} ;
2013-03-17 21:29:58 +01:00
return _lv_resume ( cmd , lvid_s , & laopts , 0 , lv ) ;
2004-03-08 18:54:13 +00:00
}
2014-09-22 15:50:07 +02:00
int lv_resume ( struct cmd_context * cmd , const char * lvid_s , unsigned origin_only ,
const struct logical_volume * lv )
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 , } ;
2013-03-17 21:29:58 +01:00
return _lv_resume ( cmd , lvid_s , & laopts , 1 , lv ) ;
2004-03-08 18:54:13 +00:00
}
2014-09-22 15:50:07 +02:00
static int _lv_has_open_snapshots ( const struct logical_volume * lv )
2009-09-29 18:50:28 +00:00
{
struct lv_segment * snap_seg ;
int r = 0 ;
2013-11-29 15:02:57 +01:00
dm_list_iterate_items_gen ( snap_seg , & lv - > snapshot_segs , origin_list )
2016-04-21 22:14:10 +01:00
if ( ! lv_check_not_in_use ( snap_seg - > cow , 1 ) )
2013-11-29 15:02:57 +01:00
r + + ;
2009-09-29 18:50:28 +00:00
2013-11-29 15:02:57 +01:00
if ( r )
2015-11-25 16:06:31 +01:00
log_error ( " LV %s has open %d snapshot(s), not deactivating. " ,
display_lvname ( lv ) , r ) ;
2009-09-29 18:50:28 +00:00
return r ;
}
2014-09-22 15:50:07 +02:00
int lv_deactivate ( struct cmd_context * cmd , const char * lvid_s , const struct logical_volume * lv )
2002-02-27 12:26:41 +00:00
{
2014-09-22 15:50:07 +02:00
const struct logical_volume * lv_to_free = NULL ;
2003-01-08 22:44:07 +00:00
struct lvinfo info ;
2013-09-07 02:46:48 +02:00
static const struct lv_activate_opts laopts = { . skip_in_use = 1 } ;
2014-11-05 11:53:11 +01:00
struct dm_list * snh ;
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 ;
2013-03-17 21:29:58 +01:00
if ( ! lv & & ! ( lv_to_free = 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 ( ) ) {
2015-11-25 16:06:31 +01:00
_skip ( " Deactivating %s. " , display_lvname ( lv ) ) ;
2009-04-10 10:00:04 +00:00
r = 1 ;
goto out ;
2002-03-14 15:36:07 +00:00
}
2015-11-25 16:06:31 +01:00
log_debug_activation ( " Deactivating %s. " , display_lvname ( lv ) ) ;
2011-06-13 22:28:04 +00:00
2014-09-24 10:05:26 +02:00
if ( ! lv_info ( cmd , lv , 0 , & 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 ) {
r = 1 ;
2014-11-05 11:53:11 +01:00
/* Check attached snapshot segments are also inactive */
dm_list_iterate ( snh , & lv - > snapshot_segs ) {
if ( ! lv_info ( cmd , dm_list_struct_base ( snh , struct lv_segment , origin_list ) - > cow ,
0 , & info , 0 , 0 ) )
goto_out ;
if ( info . exists ) {
r = 0 ; /* Snapshot left in table? */
break ;
}
}
if ( r )
goto out ;
2009-04-10 10:00:04 +00:00
}
2002-02-27 12:26:41 +00:00
2013-11-29 21:22:55 +01:00
if ( lv_is_visible ( lv ) | | lv_is_virtual_origin ( lv ) | |
lv_is_merging_thin_snapshot ( lv ) ) {
2016-04-21 22:14:10 +01:00
if ( ! lv_check_not_in_use ( lv , 1 ) )
2011-09-22 17:33:50 +00:00
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
}
2013-09-07 02:46:48 +02:00
if ( ! monitor_dev_for_events ( cmd , lv , & laopts , 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 ) ;
2017-12-05 18:48:06 +01:00
/*
* Remove any transiently activated error
* devices which arean ' t used any more .
*/
if ( r & & lv_is_raid ( lv ) & & ! lv_deactivate_any_missing_subdevs ( lv ) ) {
log_error ( " Failed to remove temporary SubLVs from %s " ,
display_lvname ( lv ) ) ;
r = 0 ;
}
2011-06-11 00:03:06 +00:00
critical_section_dec ( cmd , " deactivated " ) ;
2003-07-04 22:34:56 +00:00
2014-09-24 10:03:55 +02:00
if ( ! lv_info ( cmd , lv , 0 , & info , 0 , 0 ) | | info . exists ) {
/* Turn into log_error, but we do not log error */
log_debug_activation ( " Deactivated volume is still %s present. " ,
display_lvname ( lv ) ) ;
2009-09-29 15:17:54 +00:00
r = 0 ;
2014-09-24 10:03:55 +02:00
}
2009-04-10 10:00:04 +00:00
out :
2017-10-13 14:43:15 -05:00
if ( lv_to_free )
2013-03-17 21:29:58 +01:00
release_vg ( lv_to_free - > vg ) ;
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 ,
2014-09-22 15:50:07 +02:00
int * activate_lv , const struct logical_volume * lv )
2004-03-08 18:54:13 +00:00
{
2014-09-22 15:50:07 +02:00
const struct logical_volume * lv_to_free = NULL ;
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
2013-03-17 21:29:58 +01:00
if ( ! lv & & ! ( lv_to_free = lv = lv_from_lvid ( cmd , lvid_s , 0 ) ) )
2015-11-25 16:06:31 +01:00
goto_out ;
2004-03-08 18:54:13 +00:00
if ( ! _passes_activation_filter ( cmd , lv ) ) {
2015-11-25 16:06:31 +01:00
log_verbose ( " Not activating %s since it does not pass "
" activation filter. " , display_lvname ( lv ) ) ;
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 :
2013-03-17 21:29:58 +01:00
if ( lv_to_free )
release_vg ( lv_to_free - > 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 ,
2013-03-17 21:29:58 +01:00
struct lv_activate_opts * laopts , int filter ,
2014-09-22 15:50:07 +02:00
const struct logical_volume * lv )
2002-02-27 12:26:41 +00:00
{
2014-09-22 15:50:07 +02:00
const struct logical_volume * lv_to_free = NULL ;
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
2018-01-17 15:15:43 +01:00
if ( ! activation ( ) )
return 1 ;
if ( ! lv & & ! ( lv_to_free = lv = lv_from_lvid ( cmd , lvid_s , 0 ) ) )
goto out ;
2018-01-15 16:26:00 +01:00
if ( ! laopts - > exclusive & &
( lv_is_origin ( lv ) | |
seg_only_exclusive ( first_seg ( lv ) ) ) ) {
log_error ( INTERNAL_ERROR " Trying non-exlusive activation of %s with "
" a volume type %s requiring exclusive activation. " ,
display_lvname ( lv ) , lvseg_name ( first_seg ( lv ) ) ) ;
return 0 ;
}
2004-03-08 18:54:13 +00:00
if ( filter & & ! _passes_activation_filter ( cmd , lv ) ) {
2015-11-25 16:06:31 +01:00
log_verbose ( " Not activating %s since it does not pass "
" activation filter. " , display_lvname ( lv ) ) ;
2013-11-01 10:28:42 +01:00
r = 1 ;
2009-04-10 10:00:04 +00:00
goto out ;
2004-03-08 18:13:22 +00:00
}
2016-03-02 20:59:03 +01:00
if ( ( ! lv - > vg - > cmd - > partial_activation ) & & lv_is_partial ( lv ) ) {
2014-07-23 16:13:12 +01:00
if ( ! lv_is_raid_type ( lv ) | | ! partial_raid_lv_supports_degraded_activation ( lv ) ) {
activation: Add "degraded" activation mode
Currently, we have two modes of activation, an unnamed nominal mode
(which I will refer to as "complete") and "partial" mode. The
"complete" mode requires that a volume group be 'complete' - that
is, no missing PVs. If there are any missing PVs, no affected LVs
are allowed to activate - even RAID LVs which might be able to
tolerate a failure. The "partial" mode allows anything to be
activated (or at least attempted). If a non-redundant LV is
missing a portion of its addressable space due to a device failure,
it will be replaced with an error target. RAID LVs will either
activate or fail to activate depending on how badly their
redundancy is compromised.
This patch adds a third option, "degraded" mode. This mode can
be selected via the '--activationmode {complete|degraded|partial}'
option to lvchange/vgchange. It can also be set in lvm.conf.
The "degraded" activation mode allows RAID LVs with a sufficient
level of redundancy to activate (e.g. a RAID5 LV with one device
failure, a RAID6 with two device failures, or RAID1 with n-1
failures). RAID LVs with too many device failures are not allowed
to activate - nor are any non-redundant LVs that may have been
affected. This patch also makes the "degraded" mode the default
activation mode.
The degraded activation mode does not yet work in a cluster. A
new cluster lock flag (LCK_DEGRADED_MODE) will need to be created
to make that work. Currently, there is limited space for this
extra flag and I am looking for possible solutions. One possible
solution is to usurp LCK_CONVERT, as it is not used. When the
locking_type is 3, the degraded mode flag simply gets dropped and
the old ("complete") behavior is exhibited.
2014-07-09 22:56:11 -05:00
log_error ( " Refusing activation of partial LV %s. "
" Use '--activationmode partial' to override. " ,
2014-07-22 20:50:29 +01:00
display_lvname ( lv ) ) ;
activation: Add "degraded" activation mode
Currently, we have two modes of activation, an unnamed nominal mode
(which I will refer to as "complete") and "partial" mode. The
"complete" mode requires that a volume group be 'complete' - that
is, no missing PVs. If there are any missing PVs, no affected LVs
are allowed to activate - even RAID LVs which might be able to
tolerate a failure. The "partial" mode allows anything to be
activated (or at least attempted). If a non-redundant LV is
missing a portion of its addressable space due to a device failure,
it will be replaced with an error target. RAID LVs will either
activate or fail to activate depending on how badly their
redundancy is compromised.
This patch adds a third option, "degraded" mode. This mode can
be selected via the '--activationmode {complete|degraded|partial}'
option to lvchange/vgchange. It can also be set in lvm.conf.
The "degraded" activation mode allows RAID LVs with a sufficient
level of redundancy to activate (e.g. a RAID5 LV with one device
failure, a RAID6 with two device failures, or RAID1 with n-1
failures). RAID LVs with too many device failures are not allowed
to activate - nor are any non-redundant LVs that may have been
affected. This patch also makes the "degraded" mode the default
activation mode.
The degraded activation mode does not yet work in a cluster. A
new cluster lock flag (LCK_DEGRADED_MODE) will need to be created
to make that work. Currently, there is limited space for this
extra flag and I am looking for possible solutions. One possible
solution is to usurp LCK_CONVERT, as it is not used. When the
locking_type is 3, the degraded mode flag simply gets dropped and
the old ("complete") behavior is exhibited.
2014-07-09 22:56:11 -05:00
goto out ;
2014-07-22 20:50:29 +01:00
}
if ( ! lv - > vg - > cmd - > degraded_activation ) {
activation: Add "degraded" activation mode
Currently, we have two modes of activation, an unnamed nominal mode
(which I will refer to as "complete") and "partial" mode. The
"complete" mode requires that a volume group be 'complete' - that
is, no missing PVs. If there are any missing PVs, no affected LVs
are allowed to activate - even RAID LVs which might be able to
tolerate a failure. The "partial" mode allows anything to be
activated (or at least attempted). If a non-redundant LV is
missing a portion of its addressable space due to a device failure,
it will be replaced with an error target. RAID LVs will either
activate or fail to activate depending on how badly their
redundancy is compromised.
This patch adds a third option, "degraded" mode. This mode can
be selected via the '--activationmode {complete|degraded|partial}'
option to lvchange/vgchange. It can also be set in lvm.conf.
The "degraded" activation mode allows RAID LVs with a sufficient
level of redundancy to activate (e.g. a RAID5 LV with one device
failure, a RAID6 with two device failures, or RAID1 with n-1
failures). RAID LVs with too many device failures are not allowed
to activate - nor are any non-redundant LVs that may have been
affected. This patch also makes the "degraded" mode the default
activation mode.
The degraded activation mode does not yet work in a cluster. A
new cluster lock flag (LCK_DEGRADED_MODE) will need to be created
to make that work. Currently, there is limited space for this
extra flag and I am looking for possible solutions. One possible
solution is to usurp LCK_CONVERT, as it is not used. When the
locking_type is 3, the degraded mode flag simply gets dropped and
the old ("complete") behavior is exhibited.
2014-07-09 22:56:11 -05:00
log_error ( " Refusing activation of partial LV %s. "
" Try '--activationmode degraded'. " ,
2014-07-22 20:50:29 +01:00
display_lvname ( lv ) ) ;
activation: Add "degraded" activation mode
Currently, we have two modes of activation, an unnamed nominal mode
(which I will refer to as "complete") and "partial" mode. The
"complete" mode requires that a volume group be 'complete' - that
is, no missing PVs. If there are any missing PVs, no affected LVs
are allowed to activate - even RAID LVs which might be able to
tolerate a failure. The "partial" mode allows anything to be
activated (or at least attempted). If a non-redundant LV is
missing a portion of its addressable space due to a device failure,
it will be replaced with an error target. RAID LVs will either
activate or fail to activate depending on how badly their
redundancy is compromised.
This patch adds a third option, "degraded" mode. This mode can
be selected via the '--activationmode {complete|degraded|partial}'
option to lvchange/vgchange. It can also be set in lvm.conf.
The "degraded" activation mode allows RAID LVs with a sufficient
level of redundancy to activate (e.g. a RAID5 LV with one device
failure, a RAID6 with two device failures, or RAID1 with n-1
failures). RAID LVs with too many device failures are not allowed
to activate - nor are any non-redundant LVs that may have been
affected. This patch also makes the "degraded" mode the default
activation mode.
The degraded activation mode does not yet work in a cluster. A
new cluster lock flag (LCK_DEGRADED_MODE) will need to be created
to make that work. Currently, there is limited space for this
extra flag and I am looking for possible solutions. One possible
solution is to usurp LCK_CONVERT, as it is not used. When the
locking_type is 3, the degraded mode flag simply gets dropped and
the old ("complete") behavior is exhibited.
2014-07-09 22:56:11 -05: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 "
2015-11-25 16:06:31 +01:00
" an unrecognised segment. " , display_lvname ( lv ) ) ;
2013-05-23 16:17:08 +02:00
goto out ;
2009-10-16 17:41:49 +00:00
}
2015-01-05 16:45:30 +01:00
/*
2015-04-13 11:00:07 +02:00
* Check if cmirrord is running for clustered mirrors .
2015-01-05 16:45:30 +01:00
*/
if ( ! laopts - > exclusive & & vg_is_clustered ( lv - > vg ) & &
lv_is_mirror ( lv ) & & ! lv_is_raid ( lv ) & &
! cluster_mirror_is_available ( lv - > vg - > cmd ) ) {
log_error ( " Shared cluster mirrors are not available. " ) ;
goto out ;
}
2002-03-14 15:36:07 +00:00
if ( test_mode ( ) ) {
2015-11-25 16:06:31 +01:00
_skip ( " Activating %s. " , display_lvname ( lv ) ) ;
2009-04-10 10:00:04 +00:00
r = 1 ;
goto out ;
2002-03-14 15:36:07 +00:00
}
2018-02-28 17:16:17 +01:00
/* Component LV activation is enforced to be 'read-only' */
/* TODO: should not apply for LVs in maintenance mode */
if ( ! lv_is_visible ( lv ) & & lv_is_component ( lv ) ) {
laopts - > read_only = 1 ;
2018-02-28 17:22:09 +01:00
laopts - > component_lv = lv ;
2018-02-28 17:16:17 +01:00
} else if ( filter )
2012-01-12 01:51:56 +00:00
laopts - > read_only = _passes_readonly_filter ( cmd , lv ) ;
2015-11-25 16:06:31 +01:00
log_debug_activation ( " Activating %s%s%s%s%s. " , display_lvname ( lv ) ,
2013-01-07 22:30:29 +00:00
laopts - > exclusive ? " exclusively " : " " ,
2013-10-08 13:27:21 +02:00
laopts - > read_only ? " read-only " : " " ,
activation: flag temporary LVs internally
Add LV_TEMPORARY flag for LVs with limited existence during command
execution. Such LVs are temporary in way that they need to be activated,
some action done and then removed immediately. Such LVs are just like
any normal LV - the only difference is that they are removed during
LVM command execution. This is also the case for LVs representing
future pool metadata spare LVs which we need to initialize by using
the usual LV before they are declared as pool metadata spare.
We can optimize some other parts like udev to do a better job if
it knows that the LV is temporary and any processing on it is just
useless.
This flag is orthogonal to LV_NOSCAN flag introduced recently
as LV_NOSCAN flag is primarily used to mark an LV for the scanning
to be avoided before the zeroing of the device happens. The LV_TEMPORARY
flag makes a difference between a full-fledged LV visible in the system
and the LV just used as a temporary overlay for some action that needs to
be done on underlying PVs.
For example: lvcreate --thinpool POOL --zero n -L 1G vg
- first, the usual LV is created to do a clean up for pool metadata
spare. The LV is activated, zeroed, deactivated.
- between "activated" and "zeroed" stage, the LV_NOSCAN flag is used
to avoid any scanning in udev
- betwen "zeroed" and "deactivated" stage, we need to avoid the WATCH
udev rule, but since the LV is just a usual LV, we can't make a
difference. The LV_TEMPORARY internal LV flag helps here. If we
create the LV with this flag, the DM_UDEV_DISABLE_DISK_RULES
and DM_UDEV_DISABLE_OTHER_RULES flag are set (just like as it is
with "invisible" and non-top-level LVs) - udev is directed to
skip WATCH rule use.
- if the LV_TEMPORARY flag was not used, there would normally be
a WATCH event generated once the LV is closed after "zeroed"
stage. This will make problems with immediated deactivation that
follows.
2013-10-23 14:06:39 +02:00
laopts - > noscan ? " noscan " : " " ,
laopts - > temporary ? " temporary " : " " ) ;
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 & &
2018-03-08 10:27:04 +01:00
( info . read_only = = read_only_lv ( lv , laopts , NULL ) ) ) {
2009-04-10 10:00:04 +00:00
r = 1 ;
2016-04-14 22:41:25 +01:00
log_debug_activation ( " LV %s is already active. " , display_lvname ( lv ) ) ;
2009-04-10 10:00:04 +00:00
goto out ;
}
2002-02-27 12:26:41 +00:00
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 :
2017-10-13 14:43:15 -05:00
if ( lv_to_free )
2013-03-17 21:29:58 +01:00
release_vg ( lv_to_free - > vg ) ;
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 */
2013-10-08 13:27:21 +02:00
int lv_activate ( struct cmd_context * cmd , const char * lvid_s , int exclusive ,
2014-09-22 15:50:07 +02:00
int noscan , int temporary , const struct logical_volume * lv )
2004-03-08 18:54:13 +00:00
{
activation: flag temporary LVs internally
Add LV_TEMPORARY flag for LVs with limited existence during command
execution. Such LVs are temporary in way that they need to be activated,
some action done and then removed immediately. Such LVs are just like
any normal LV - the only difference is that they are removed during
LVM command execution. This is also the case for LVs representing
future pool metadata spare LVs which we need to initialize by using
the usual LV before they are declared as pool metadata spare.
We can optimize some other parts like udev to do a better job if
it knows that the LV is temporary and any processing on it is just
useless.
This flag is orthogonal to LV_NOSCAN flag introduced recently
as LV_NOSCAN flag is primarily used to mark an LV for the scanning
to be avoided before the zeroing of the device happens. The LV_TEMPORARY
flag makes a difference between a full-fledged LV visible in the system
and the LV just used as a temporary overlay for some action that needs to
be done on underlying PVs.
For example: lvcreate --thinpool POOL --zero n -L 1G vg
- first, the usual LV is created to do a clean up for pool metadata
spare. The LV is activated, zeroed, deactivated.
- between "activated" and "zeroed" stage, the LV_NOSCAN flag is used
to avoid any scanning in udev
- betwen "zeroed" and "deactivated" stage, we need to avoid the WATCH
udev rule, but since the LV is just a usual LV, we can't make a
difference. The LV_TEMPORARY internal LV flag helps here. If we
create the LV with this flag, the DM_UDEV_DISABLE_DISK_RULES
and DM_UDEV_DISABLE_OTHER_RULES flag are set (just like as it is
with "invisible" and non-top-level LVs) - udev is directed to
skip WATCH rule use.
- if the LV_TEMPORARY flag was not used, there would normally be
a WATCH event generated once the LV is closed after "zeroed"
stage. This will make problems with immediated deactivation that
follows.
2013-10-23 14:06:39 +02:00
struct lv_activate_opts laopts = { . exclusive = exclusive ,
. noscan = noscan ,
. temporary = temporary } ;
2011-06-17 14:14:19 +00:00
2013-03-17 21:29:58 +01:00
if ( ! _lv_activate ( cmd , lvid_s , & laopts , 0 , lv ) )
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 */
2013-10-08 13:27:21 +02:00
int lv_activate_with_filter ( struct cmd_context * cmd , const char * lvid_s , int exclusive ,
2014-09-22 15:50:07 +02:00
int noscan , int temporary , const struct logical_volume * lv )
2004-03-08 18:54:13 +00:00
{
activation: flag temporary LVs internally
Add LV_TEMPORARY flag for LVs with limited existence during command
execution. Such LVs are temporary in way that they need to be activated,
some action done and then removed immediately. Such LVs are just like
any normal LV - the only difference is that they are removed during
LVM command execution. This is also the case for LVs representing
future pool metadata spare LVs which we need to initialize by using
the usual LV before they are declared as pool metadata spare.
We can optimize some other parts like udev to do a better job if
it knows that the LV is temporary and any processing on it is just
useless.
This flag is orthogonal to LV_NOSCAN flag introduced recently
as LV_NOSCAN flag is primarily used to mark an LV for the scanning
to be avoided before the zeroing of the device happens. The LV_TEMPORARY
flag makes a difference between a full-fledged LV visible in the system
and the LV just used as a temporary overlay for some action that needs to
be done on underlying PVs.
For example: lvcreate --thinpool POOL --zero n -L 1G vg
- first, the usual LV is created to do a clean up for pool metadata
spare. The LV is activated, zeroed, deactivated.
- between "activated" and "zeroed" stage, the LV_NOSCAN flag is used
to avoid any scanning in udev
- betwen "zeroed" and "deactivated" stage, we need to avoid the WATCH
udev rule, but since the LV is just a usual LV, we can't make a
difference. The LV_TEMPORARY internal LV flag helps here. If we
create the LV with this flag, the DM_UDEV_DISABLE_DISK_RULES
and DM_UDEV_DISABLE_OTHER_RULES flag are set (just like as it is
with "invisible" and non-top-level LVs) - udev is directed to
skip WATCH rule use.
- if the LV_TEMPORARY flag was not used, there would normally be
a WATCH event generated once the LV is closed after "zeroed"
stage. This will make problems with immediated deactivation that
follows.
2013-10-23 14:06:39 +02:00
struct lv_activate_opts laopts = { . exclusive = exclusive ,
. noscan = noscan ,
. temporary = temporary } ;
2011-06-17 14:14:19 +00:00
2013-03-17 21:29:58 +01:00
if ( ! _lv_activate ( cmd , lvid_s , & laopts , 1 , lv ) )
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 )
{
2014-08-19 14:19:11 +02:00
int r ;
2003-11-12 19:16:48 +00:00
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 ;
}
2016-12-23 03:35:13 +01:00
/* Remove any existing, closed mapped device by @name */
static int _remove_dm_dev_by_name ( const char * name )
{
int r = 0 ;
struct dm_task * dmt ;
struct dm_info info ;
if ( ! ( dmt = dm_task_create ( DM_DEVICE_INFO ) ) )
return_0 ;
/* Check, if the device exists. */
if ( dm_task_set_name ( dmt , name ) & & dm_task_run ( dmt ) & & dm_task_get_info ( dmt , & info ) ) {
dm_task_destroy ( dmt ) ;
/* Ignore non-existing or open dm devices */
if ( ! info . exists | | info . open_count )
return 1 ;
if ( ! ( dmt = dm_task_create ( DM_DEVICE_REMOVE ) ) )
return_0 ;
if ( dm_task_set_name ( dmt , name ) )
r = dm_task_run ( dmt ) ;
}
dm_task_destroy ( dmt ) ;
return r ;
}
/* Work all segments of @lv removing any existing, closed "*-missing_N_0" sub devices. */
static int _lv_remove_any_missing_subdevs ( struct logical_volume * lv )
{
if ( lv ) {
uint32_t seg_no = 0 ;
char name [ 257 ] ;
struct lv_segment * seg ;
dm_list_iterate_items ( seg , & lv - > segments ) {
if ( dm_snprintf ( name , sizeof ( name ) , " %s-%s-missing_%u_0 " , seg - > lv - > vg - > name , seg - > lv - > name , seg_no ) < 0 )
2017-12-05 18:48:06 +01:00
return_0 ;
2016-12-23 03:35:13 +01:00
if ( ! _remove_dm_dev_by_name ( name ) )
return 0 ;
seg_no + + ;
}
}
return 1 ;
}
/* Remove any "*-missing_*" sub devices added by the activation layer for an rmate/rimage missing PV mapping */
int lv_deactivate_any_missing_subdevs ( const struct logical_volume * lv )
{
uint32_t s ;
struct lv_segment * seg = first_seg ( lv ) ;
for ( s = 0 ; s < seg - > area_count ; s + + ) {
if ( seg_type ( seg , s ) = = AREA_LV & &
! _lv_remove_any_missing_subdevs ( seg_lv ( seg , s ) ) )
return 0 ;
if ( seg - > meta_areas & & seg_metatype ( seg , s ) = = AREA_LV & &
! _lv_remove_any_missing_subdevs ( seg_metalv ( seg , s ) ) )
return 0 ;
}
return 1 ;
}
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 )
{
2014-03-20 10:31:21 +01:00
if ( critical_section ( ) )
/* May leak stacked operation */
log_error ( " Releasing activation in critical section. " ) ;
fs_unlock ( ) ; /* Implicit dev_manager_release(); */
2006-05-16 16:48:31 +00:00
}
2003-07-04 22:34:56 +00:00
void activation_exit ( void )
{
2014-03-20 10:31:21 +01:00
activation_release ( ) ;
2003-07-04 22:34:56 +00:00
dev_manager_exit ( ) ;
}
2003-01-08 22:44:07 +00:00
# endif
2018-02-27 14:13:00 +01:00
static int _component_cb ( struct logical_volume * lv , void * data )
{
struct logical_volume * * component_lv = ( struct logical_volume * * ) data ;
if ( lv_is_locked ( lv ) | | lv_is_pvmove ( lv ) | | /* ignoring */
/* thin-pool is special and it's using layered device */
( lv_is_thin_pool ( lv ) & & pool_is_active ( lv ) ) )
return - 1 ;
if ( lv_is_active ( lv ) ) {
if ( ! lv_is_component ( lv ) | | lv_is_visible ( lv ) )
return - 1 ; /* skip whole subtree */
log_debug_activation ( " Found active component LV %s. " , display_lvname ( lv ) ) ;
* component_lv = lv ;
return 0 ; /* break any further processing */
}
return 1 ;
}
/*
* Finds out for any LV if any of its component LVs are active .
* Function first checks if an existing LV is visible and active eventually
* it ' s lock holding LV is already active . In such case sub LV cannot be
* actived alone and no further checking is needed .
*
* Returns active component LV if there is such .
*/
const struct logical_volume * lv_component_is_active ( const struct logical_volume * lv )
{
const struct logical_volume * component_lv = NULL ;
const struct logical_volume * holder_lv = lv_lock_holder ( lv ) ;
if ( ( holder_lv ! = lv ) & & lv_is_active ( holder_lv ) )
return NULL ; /* Lock holding LV is active, do not check components */
if ( _component_cb ( ( struct logical_volume * ) lv , & holder_lv ) = = 1 )
( void ) for_each_sub_lv ( ( struct logical_volume * ) lv , _component_cb ,
( void * ) & component_lv ) ;
return component_lv ;
}
/*
* Finds out if any LV above is active , as stacked device tree can be composed of
* chained set of LVs .
*
* Returns active holder LV if there is such .
*/
const struct logical_volume * lv_holder_is_active ( const struct logical_volume * lv )
{
const struct logical_volume * holder ;
const struct seg_list * sl ;
if ( lv_is_locked ( lv ) | | lv_is_pvmove ( lv ) )
return NULL ; /* Skip pvmove/locked LV tracking */
dm_list_iterate_items ( sl , & lv - > segs_using_this_lv ) {
/* Recursive call for upper-stack holder */
if ( ( holder = lv_holder_is_active ( sl - > seg - > lv ) ) )
return holder ;
if ( lv_is_active ( sl - > seg - > lv ) ) {
log_debug_activation ( " Found active holder LV %s. " , display_lvname ( sl - > seg - > lv ) ) ;
return sl - > seg - > lv ;
}
}
return NULL ;
}
static int _deactivate_sub_lv_cb ( struct logical_volume * lv , void * data )
{
struct logical_volume * * slv = data ;
if ( lv_is_thin_pool ( lv ) | | lv_is_external_origin ( lv ) )
return - 1 ;
if ( ! deactivate_lv ( lv - > vg - > cmd , lv ) ) {
* slv = lv ;
return 0 ;
}
return 1 ;
}
/*
* Deactivates LV toghether with explicit deactivation call made also for all its component LVs .
*/
int deactivate_lv_with_sub_lv ( const struct logical_volume * lv )
{
struct logical_volume * flv ;
if ( ! deactivate_lv ( lv - > vg - > cmd , lv ) ) {
log_error ( " Cannot deactivate logical volume %s. " ,
display_lvname ( lv ) ) ;
return 0 ;
}
if ( ! for_each_sub_lv ( ( struct logical_volume * ) lv , _deactivate_sub_lv_cb , & flv ) ) {
log_error ( " Cannot deactivate subvolume %s of logical volume %s. " ,
display_lvname ( flv ) , display_lvname ( lv ) ) ;
return 0 ;
}
return 1 ;
}