2001-10-09 20:05:34 +04:00
/*
2004-03-30 23:35:44 +04:00
* Copyright ( C ) 2001 - 2004 Sistina Software , Inc . All rights reserved .
2005-11-09 01:52:26 +03:00
* Copyright ( C ) 2004 - 2005 Red Hat , Inc . All rights reserved .
2001-10-09 20:05:34 +04:00
*
2004-03-30 23:35:44 +04: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
* of the GNU General Public License v .2 .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software Foundation ,
* Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
2001-10-09 20:05:34 +04:00
*/
2002-11-18 17:01:16 +03:00
# include "lib.h"
2001-10-16 20:25:28 +04:00
# include "metadata.h"
2001-10-09 20:05:34 +04:00
# include "activate.h"
2003-07-05 02:34:56 +04:00
# include "memlock.h"
2001-11-15 17:27:51 +03:00
# include "display.h"
2001-11-12 15:20:58 +03:00
# include "fs.h"
2005-10-17 22:00:02 +04:00
# include "lvm-exec.h"
2004-04-08 19:23:23 +04:00
# include "lvm-file.h"
2001-12-31 22:09:51 +03:00
# include "lvm-string.h"
2002-02-25 15:56:16 +03:00
# include "toolcontext.h"
2002-02-26 14:49:17 +03:00
# include "dev_manager.h"
2004-03-08 21:13:22 +03:00
# include "str_list.h"
2004-05-05 01:25:57 +04:00
# include "config.h"
2005-10-25 23:08:21 +04:00
# include "filter.h"
2005-12-02 23:35:07 +03:00
# include "segtype.h"
2002-01-21 14:06:32 +03:00
# include <limits.h>
2002-02-18 18:52:48 +03:00
# include <fcntl.h>
2003-07-05 02:34:56 +04:00
# include <unistd.h>
2002-01-21 14:06:32 +03:00
2002-02-11 18:48:34 +03:00
# define _skip(fmt, args...) log_very_verbose("Skipping: " fmt , ## args)
2001-10-09 20:05:34 +04:00
2004-04-08 19:23:23 +04:00
int lvm1_present ( struct cmd_context * cmd )
{
char path [ PATH_MAX ] ;
if ( lvm_snprintf ( path , sizeof ( path ) , " %s/lvm/global " , cmd - > proc_dir )
< 0 ) {
log_error ( " LVM1 proc global snprintf failed " ) ;
return 0 ;
}
if ( path_exists ( path ) )
return 1 ;
else
return 0 ;
}
2003-01-09 01:44:07 +03:00
# ifndef DEVMAPPER_SUPPORT
void set_activation ( int act )
{
2004-03-19 19:26:46 +03:00
static int warned = 0 ;
if ( warned | | ! act )
return ;
log_error ( " Compiled without libdevmapper support. "
" Can't enable activation. " ) ;
warned = 1 ;
2003-01-09 01:44:07 +03: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-20 00:01:39 +03:00
int target_version ( const char * target_name , uint32_t * maj ,
uint32_t * min , uint32_t * patchlevel )
{
return 0 ;
}
2004-03-26 22:52:09 +03:00
int target_present ( const char * target_name )
{
return 0 ;
}
2005-10-17 22:00:02 +04:00
int lv_info ( struct cmd_context * cmd , const struct logical_volume * lv , struct lvinfo * info ,
2005-01-13 01:58:21 +03:00
int with_open_count )
2003-01-09 01:44:07 +03:00
{
return 0 ;
}
2004-03-26 18:35:01 +03:00
int lv_info_by_lvid ( struct cmd_context * cmd , const char * lvid_s ,
2005-01-13 01:58:21 +03:00
struct lvinfo * info , int with_open_count )
2004-03-26 18:35:01 +03:00
{
return 0 ;
}
2003-01-09 01:44:07 +03:00
int lv_snapshot_percent ( struct logical_volume * lv , float * percent )
{
return 0 ;
}
2003-07-05 02:34:56 +04:00
int lv_mirror_percent ( struct logical_volume * lv , int wait , float * percent ,
uint32_t * event_nr )
2003-04-30 19:26:25 +04:00
{
return 0 ;
}
2003-01-09 01:44:07 +03:00
int lvs_in_vg_activated ( struct volume_group * vg )
{
return 0 ;
}
int lvs_in_vg_opened ( struct volume_group * vg )
{
return 0 ;
}
2004-03-08 21:54:13 +03:00
int lv_suspend ( struct cmd_context * cmd , const char * lvid_s )
{
return 1 ;
}
2003-01-09 01:44:07 +03:00
int lv_suspend_if_active ( struct cmd_context * cmd , const char * lvid_s )
{
return 1 ;
}
2004-03-08 21:54:13 +03:00
int lv_resume ( struct cmd_context * cmd , const char * lvid_s )
{
return 1 ;
}
2003-01-09 01:44:07 +03:00
int lv_resume_if_active ( struct cmd_context * cmd , const char * lvid_s )
{
return 1 ;
}
int lv_deactivate ( struct cmd_context * cmd , const char * lvid_s )
{
return 1 ;
}
2004-03-08 21:54:13 +03:00
int lv_activation_filter ( struct cmd_context * cmd , const char * lvid_s ,
int * activate_lv )
{
return 1 ;
}
2005-08-15 03:18:28 +04:00
int lv_activate ( struct cmd_context * cmd , const char * lvid_s , int exclusive )
2003-01-09 01:44:07 +03:00
{
return 1 ;
}
2005-08-15 03:18:28 +04:00
int lv_activate_with_filter ( struct cmd_context * cmd , const char * lvid_s , int exclusive )
2004-03-08 21:54:13 +03:00
{
return 1 ;
}
2003-11-12 22:16:48 +03:00
int lv_mknodes ( struct cmd_context * cmd , const struct logical_volume * lv )
{
return 1 ;
}
2005-10-25 23:08:21 +04:00
int pv_uses_vg ( struct cmd_context * cmd , struct physical_volume * pv ,
struct volume_group * vg )
{
return 0 ;
}
2003-07-05 02:34:56 +04:00
void activation_exit ( void )
{
return ;
}
2003-01-09 01:44:07 +03:00
# else /* DEVMAPPER_SUPPORT */
2002-11-18 17:01:16 +03:00
static int _activation = 1 ;
2002-12-20 02:25:55 +03:00
void set_activation ( int act )
2002-11-18 17:01:16 +03:00
{
2002-12-20 02:25:55 +03:00
if ( act = = _activation )
2002-11-18 17:01:16 +03:00
return ;
2002-12-20 02:25:55 +03:00
_activation = act ;
2002-11-18 17:01:16 +03:00
if ( _activation )
log_verbose ( " Activation enabled. Device-mapper kernel "
" driver will be used. " ) ;
else
2005-12-22 19:13:38 +03:00
log_print ( " WARNING: Activation disabled. No device-mapper "
" interaction will be attempted. " ) ;
2002-11-18 17:01:16 +03:00
}
2002-12-20 02:25:55 +03:00
int activation ( void )
2002-11-18 17:01:16 +03:00
{
return _activation ;
}
2004-03-08 21:13:22 +03:00
static int _passes_activation_filter ( struct cmd_context * cmd ,
struct logical_volume * lv )
{
2004-05-04 22:28:15 +04:00
const struct config_node * cn ;
2004-03-08 21:13:22 +03:00
struct config_value * cv ;
char * str ;
char path [ PATH_MAX ] ;
2004-03-08 21:28:45 +03:00
if ( ! ( cn = find_config_node ( cmd - > cft - > root , " activation/volume_list " ) ) ) {
2004-05-04 22:28:15 +04:00
/* If no host tags defined, activate */
2004-03-08 21:13:22 +03:00
if ( 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 ) | |
str_list_match_list ( & cmd - > tags , & lv - > vg - > tags ) )
return 1 ;
/* Don't activate */
return 0 ;
}
for ( cv = cn - > v ; cv ; cv = cv - > next ) {
if ( cv - > type ! = CFG_STRING ) {
log_error ( " Ignoring invalid string in config file "
" activation/volume_list " ) ;
continue ;
}
str = cv - > v . str ;
if ( ! * str ) {
log_error ( " Ignoring empty string in config file "
" activation/volume_list " ) ;
continue ;
}
/* Tag? */
if ( * str = = ' @ ' ) {
str + + ;
if ( ! * str ) {
log_error ( " Ignoring empty tag in config file "
" activation/volume_list " ) ;
continue ;
}
/* If any host tag matches any LV or VG tag, activate */
if ( ! strcmp ( str , " * " ) ) {
if ( str_list_match_list ( & cmd - > tags , & lv - > tags )
| | str_list_match_list ( & cmd - > tags ,
& lv - > vg - > tags ) )
return 1 ;
else
continue ;
}
/* If supplied tag matches LV or VG tag, activate */
if ( str_list_match_item ( & lv - > tags , str ) | |
str_list_match_item ( & lv - > vg - > tags , str ) )
return 1 ;
else
continue ;
}
if ( ! index ( str , ' / ' ) ) {
/* vgname supplied */
if ( ! strcmp ( str , lv - > vg - > name ) )
return 1 ;
else
continue ;
}
/* vgname/lvname */
if ( lvm_snprintf ( path , sizeof ( path ) , " %s/%s " , lv - > vg - > name ,
lv - > name ) < 0 ) {
log_error ( " lvm_snprintf error from %s/%s " , lv - > vg - > name ,
lv - > name ) ;
continue ;
}
if ( ! strcmp ( path , str ) )
return 1 ;
}
return 0 ;
}
2002-01-17 19:39:24 +03:00
int library_version ( char * version , size_t size )
{
2002-11-18 17:01:16 +03:00
if ( ! activation ( ) )
return 0 ;
2005-10-17 22:00:02 +04:00
return dm_get_library_version ( version , size ) ;
2002-01-17 19:39:24 +03:00
}
int driver_version ( char * version , size_t size )
{
2002-11-18 17:01:16 +03:00
if ( ! activation ( ) )
return 0 ;
2002-01-17 19:39:24 +03:00
log_very_verbose ( " Getting driver version " ) ;
2005-10-17 22:00:02 +04:00
return dm_driver_version ( version , size ) ;
2002-01-17 19:39:24 +03:00
}
2005-12-20 00:01:39 +03:00
int target_version ( const char * target_name , uint32_t * maj ,
uint32_t * min , uint32_t * patchlevel )
2004-03-26 22:52:09 +03: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-09 01:52:26 +03:00
if ( ! ( dmt = dm_task_create ( DM_DEVICE_LIST_VERSIONS ) ) )
return_0 ;
2004-03-26 22:52:09 +03:00
if ( ! dm_task_run ( dmt ) ) {
log_debug ( " Failed to get %s target version " , target_name ) ;
/* Assume this was because LIST_VERSIONS isn't supported */
return 1 ;
}
target = dm_task_get_versions ( dmt ) ;
do {
last_target = target ;
if ( ! strcmp ( target_name , target - > name ) ) {
r = 1 ;
2005-12-20 00:01:39 +03:00
* maj = target - > version [ 0 ] ;
* min = target - > version [ 1 ] ;
* patchlevel = target - > version [ 2 ] ;
2004-03-26 22:52:09 +03:00
goto out ;
}
target = ( void * ) target + target - > next ;
} while ( last_target ! = target ) ;
out :
dm_task_destroy ( dmt ) ;
return r ;
}
2005-11-09 01:52:26 +03:00
int target_present ( const char * target_name , int use_modprobe )
2005-10-17 22:00:02 +04:00
{
2005-12-20 00:01:39 +03:00
uint32_t maj , min , patchlevel ;
2005-10-19 17:59:18 +04:00
# ifdef MODPROBE_CMD
2005-10-17 22:00:02 +04:00
char module [ 128 ] ;
2005-10-19 17:59:18 +04:00
# endif
2005-10-17 22:00:02 +04:00
if ( ! activation ( ) )
return 0 ;
# ifdef MODPROBE_CMD
2005-11-09 01:52:26 +03:00
if ( use_modprobe ) {
2005-12-20 00:01:39 +03:00
if ( target_version ( target_name , & maj , & min , & patchlevel ) )
2005-11-09 01:52:26 +03:00
return 1 ;
2005-10-17 22:00:02 +04:00
2005-11-09 01:52:26 +03:00
if ( lvm_snprintf ( module , sizeof ( module ) , " dm-%s " , target_name )
< 0 ) {
log_error ( " target_present module name too long: %s " ,
target_name ) ;
return 0 ;
}
2005-10-17 22:00:02 +04:00
2005-11-09 01:52:26 +03:00
if ( ! exec_cmd ( MODPROBE_CMD , module , " " , " " ) )
return_0 ;
2005-10-17 22:00:02 +04:00
}
# endif
2005-12-20 00:01:39 +03:00
return target_version ( target_name , & maj , & min , & patchlevel ) ;
2005-10-17 22:00:02 +04:00
}
2002-02-26 14:49:17 +03:00
/*
* Returns 1 if info structure populated , else 0 on failure .
*/
2005-10-17 22:00:02 +04:00
static int _lv_info ( struct cmd_context * cmd , const struct logical_volume * lv , int with_mknodes ,
2005-01-13 01:58:21 +03:00
struct lvinfo * info , int with_open_count )
2001-11-07 14:51:42 +03:00
{
2003-01-09 01:44:07 +03:00
struct dm_info dminfo ;
2005-10-17 22:00:02 +04:00
char * name ;
2002-01-11 02:21:07 +03:00
2002-11-18 17:01:16 +03:00
if ( ! activation ( ) )
return 0 ;
2005-11-09 01:52:26 +03:00
if ( ! ( name = build_dm_name ( cmd - > mem , lv - > vg - > name , lv - > name , NULL ) ) )
return_0 ;
2002-01-11 02:21:07 +03:00
2005-10-17 22:00:02 +04:00
log_debug ( " Getting device info for %s " , name ) ;
2005-10-19 17:59:18 +04:00
if ( ! dev_manager_info ( lv - > vg - > cmd - > mem , name , lv , with_mknodes ,
with_open_count , & dminfo ) ) {
2005-10-17 22:00:02 +04:00
dm_pool_free ( cmd - > mem , name ) ;
2005-11-09 01:52:26 +03:00
return_0 ;
2005-10-17 22:00:02 +04:00
}
2002-01-11 02:21:07 +03:00
2003-01-09 01:44:07 +03: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-09 01:52:26 +03:00
info - > live_table = dminfo . live_table ;
info - > inactive_table = dminfo . inactive_table ;
2003-01-09 01:44:07 +03:00
2005-10-17 22:00:02 +04:00
dm_pool_free ( cmd - > mem , name ) ;
return 1 ;
2002-02-26 14:49:17 +03:00
}
2002-02-11 20:42:02 +03:00
2005-10-17 22:00:02 +04:00
int lv_info ( struct cmd_context * cmd , const struct logical_volume * lv , struct lvinfo * info ,
2005-01-13 01:58:21 +03:00
int with_open_count )
2003-11-13 17:11:41 +03:00
{
2005-10-17 22:00:02 +04:00
return _lv_info ( cmd , lv , 0 , info , with_open_count ) ;
2003-11-13 17:11:41 +03:00
}
2004-03-26 18:35:01 +03:00
int lv_info_by_lvid ( struct cmd_context * cmd , const char * lvid_s ,
2005-01-13 01:58:21 +03:00
struct lvinfo * info , int with_open_count )
2004-03-26 18:35:01 +03:00
{
struct logical_volume * lv ;
2005-10-31 23:15:28 +03:00
if ( ! ( lv = lv_from_lvid ( cmd , lvid_s , 0 ) ) )
2004-03-26 18:35:01 +03:00
return 0 ;
2005-10-17 22:00:02 +04:00
return _lv_info ( cmd , lv , 0 , info , with_open_count ) ;
2004-03-26 18:35:01 +03:00
}
2002-05-10 01:17:57 +04:00
/*
* Returns 1 if percent set , else 0 on failure .
*/
2002-05-22 18:03:45 +04:00
int lv_snapshot_percent ( struct logical_volume * lv , float * percent )
2002-05-10 01:17:57 +04:00
{
int r ;
struct dev_manager * dm ;
2002-11-18 17:01:16 +03:00
if ( ! activation ( ) )
return 0 ;
2005-11-09 01:52:26 +03:00
if ( ! ( dm = dev_manager_create ( lv - > vg - > cmd , lv - > vg - > name ) ) )
return_0 ;
2002-05-10 01:17:57 +04:00
2002-05-22 18:03:45 +04:00
if ( ! ( r = dev_manager_snapshot_percent ( dm , lv , percent ) ) )
2002-05-10 01:17:57 +04:00
stack ;
2002-05-22 18:03:45 +04:00
2002-05-10 01:17:57 +04:00
dev_manager_destroy ( dm ) ;
return r ;
}
2003-04-30 19:26:25 +04:00
/* FIXME Merge with snapshot_percent */
2005-10-17 22:00:02 +04:00
int lv_mirror_percent ( struct cmd_context * cmd , struct logical_volume * lv , int wait , float * percent ,
2003-04-30 19:26:25 +04:00
uint32_t * event_nr )
{
int r ;
struct dev_manager * dm ;
2004-03-26 22:10:48 +03:00
struct lvinfo info ;
2003-04-30 19:26:25 +04:00
if ( ! activation ( ) )
return 0 ;
2005-11-09 01:52:26 +03:00
if ( ! lv_info ( cmd , lv , & info , 0 ) )
return_0 ;
2004-03-26 22:10:48 +03:00
if ( ! info . exists )
return 0 ;
2005-11-09 01:52:26 +03:00
if ( ! ( dm = dev_manager_create ( lv - > vg - > cmd , lv - > vg - > name ) ) )
return_0 ;
2003-04-30 19:26:25 +04:00
if ( ! ( r = dev_manager_mirror_percent ( dm , lv , wait , percent , event_nr ) ) )
stack ;
dev_manager_destroy ( dm ) ;
return r ;
}
2005-10-17 22:00:02 +04:00
static int _lv_active ( struct cmd_context * cmd , struct logical_volume * lv )
2001-11-07 18:02:07 +03:00
{
2003-01-09 01:44:07 +03:00
struct lvinfo info ;
2001-11-07 18:02:07 +03:00
2005-10-17 22:00:02 +04:00
if ( ! lv_info ( cmd , lv , & info , 0 ) ) {
2001-11-07 18:02:07 +03:00
stack ;
2002-02-26 14:49:17 +03:00
return - 1 ;
2001-11-07 18:02:07 +03:00
}
2002-02-26 14:49:17 +03:00
return info . exists ;
2001-11-07 18:02:07 +03:00
}
2005-10-17 22:00:02 +04:00
static int _lv_open_count ( struct cmd_context * cmd , struct logical_volume * lv )
2002-02-18 18:52:48 +03:00
{
2003-01-09 01:44:07 +03:00
struct lvinfo info ;
2002-02-18 18:52:48 +03:00
2005-10-17 22:00:02 +04:00
if ( ! lv_info ( cmd , lv , & info , 1 ) ) {
2002-02-18 18:52:48 +03:00
stack ;
2002-02-26 14:49:17 +03:00
return - 1 ;
2002-02-18 18:52:48 +03:00
}
2002-02-26 14:49:17 +03:00
return info . open_count ;
2002-02-18 18:52:48 +03:00
}
2004-03-08 21:54:13 +03:00
static int _lv_activate_lv ( struct logical_volume * lv )
2001-10-09 20:05:34 +04:00
{
2002-02-11 18:48:34 +03:00
int r ;
2002-02-26 14:49:17 +03:00
struct dev_manager * dm ;
2001-10-09 20:05:34 +04:00
2005-11-09 01:52:26 +03:00
if ( ! ( dm = dev_manager_create ( lv - > vg - > cmd , lv - > vg - > name ) ) )
return_0 ;
2001-11-02 16:45:05 +03:00
2002-03-07 19:48:46 +03:00
if ( ! ( r = dev_manager_activate ( dm , lv ) ) )
2002-02-11 18:48:34 +03:00
stack ;
2001-11-02 16:45:05 +03:00
2002-02-26 14:49:17 +03:00
dev_manager_destroy ( dm ) ;
2001-11-02 16:45:05 +03:00
return r ;
2001-10-09 20:05:34 +04:00
}
2001-10-16 20:25:28 +04:00
2005-11-09 01:52:26 +03:00
static int _lv_preload ( struct logical_volume * lv )
2001-10-31 20:59:52 +03:00
{
2002-02-26 14:49:17 +03:00
int r ;
struct dev_manager * dm ;
2001-11-07 14:51:42 +03:00
2005-11-09 01:52:26 +03:00
if ( ! ( dm = dev_manager_create ( lv - > vg - > cmd , lv - > vg - > name ) ) )
return_0 ;
if ( ! ( r = dev_manager_preload ( dm , lv ) ) )
2002-02-11 18:48:34 +03:00
stack ;
2005-11-09 01:52:26 +03:00
dev_manager_destroy ( dm ) ;
return r ;
}
static int _lv_deactivate ( struct logical_volume * lv )
{
int r ;
struct dev_manager * dm ;
if ( ! ( dm = dev_manager_create ( lv - > vg - > cmd , lv - > vg - > name ) ) )
return_0 ;
2001-11-07 14:51:42 +03:00
2002-02-26 14:49:17 +03:00
if ( ! ( r = dev_manager_deactivate ( dm , lv ) ) )
2001-11-07 14:51:42 +03:00
stack ;
2002-02-26 14:49:17 +03:00
dev_manager_destroy ( dm ) ;
return r ;
2001-11-07 14:51:42 +03:00
}
2004-03-08 21:54:13 +03:00
static int _lv_suspend_lv ( struct logical_volume * lv )
2002-01-11 02:21:07 +03:00
{
2002-03-14 18:36:07 +03:00
int r ;
struct dev_manager * dm ;
2001-11-28 21:03:11 +03:00
2005-11-09 01:52:26 +03:00
if ( ! ( dm = dev_manager_create ( lv - > vg - > cmd , lv - > vg - > name ) ) )
return_0 ;
2001-10-31 20:59:52 +03:00
2002-03-14 18:36:07 +03:00
if ( ! ( r = dev_manager_suspend ( dm , lv ) ) )
2001-11-07 14:51:42 +03:00
stack ;
2001-10-31 20:59:52 +03:00
2002-03-14 18:36:07 +03:00
dev_manager_destroy ( dm ) ;
return r ;
2002-02-11 18:48:34 +03:00
}
2002-01-11 02:21:07 +03:00
2002-03-01 22:08:11 +03:00
/*
2004-05-11 22:18:14 +04:00
* These two functions return the number of visible LVs in the state ,
2002-03-01 22:08:11 +03:00
* or - 1 on error .
*/
2001-11-02 19:28:04 +03:00
int lvs_in_vg_activated ( struct volume_group * vg )
{
2005-06-01 20:51:55 +04:00
struct lv_list * lvl ;
2001-11-08 19:15:58 +03:00
int count = 0 ;
2001-11-07 14:51:42 +03:00
2002-11-18 17:01:16 +03:00
if ( ! activation ( ) )
return 0 ;
2005-06-01 20:51:55 +04:00
list_iterate_items ( lvl , & vg - > lvs ) {
if ( lvl - > lv - > status & VISIBLE_LV )
2005-10-17 22:00:02 +04:00
count + = ( _lv_active ( vg - > cmd , lvl - > lv ) = = 1 ) ;
2001-11-07 14:51:42 +03:00
}
return count ;
2001-11-02 19:28:04 +03:00
}
2001-11-07 18:02:07 +03:00
int lvs_in_vg_opened ( struct volume_group * vg )
{
2005-06-01 20:51:55 +04:00
struct lv_list * lvl ;
2001-11-08 19:15:58 +03:00
int count = 0 ;
2001-11-07 18:02:07 +03:00
2002-11-18 17:01:16 +03:00
if ( ! activation ( ) )
return 0 ;
2005-06-01 20:51:55 +04:00
list_iterate_items ( lvl , & vg - > lvs ) {
if ( lvl - > lv - > status & VISIBLE_LV )
2005-10-17 22:00:02 +04:00
count + = ( _lv_open_count ( vg - > cmd , lvl - > lv ) > 0 ) ;
2001-11-07 18:02:07 +03:00
}
return count ;
}
2002-02-25 15:56:16 +03:00
2006-01-27 21:38:14 +03:00
static int _register_dev_for_events ( struct cmd_context * cmd ,
struct logical_volume * lv , int do_reg )
2005-12-02 23:35:07 +03:00
{
2006-01-27 21:38:14 +03:00
# ifdef DMEVENTD
2005-12-02 23:35:07 +03:00
struct list * tmp ;
struct lv_segment * seg ;
int ( * reg ) ( struct dm_pool * mem , struct lv_segment * ,
struct config_tree * cft , int events ) ;
list_iterate ( tmp , & lv - > segments ) {
seg = list_item ( tmp , struct lv_segment ) ;
2006-01-27 21:38:14 +03:00
reg = NULL ;
2005-12-08 20:49:34 +03:00
if ( do_reg ) {
if ( seg - > segtype - > ops - > target_register_events )
reg = seg - > segtype - > ops - > target_register_events ;
2005-12-20 00:01:39 +03:00
} else if ( seg - > segtype - > ops - > target_unregister_events )
2005-12-08 20:49:34 +03:00
reg = seg - > segtype - > ops - > target_unregister_events ;
2005-12-02 23:35:07 +03:00
if ( reg )
/* FIXME specify events */
if ( ! reg ( cmd - > mem , seg , cmd - > cft , 0 ) ) {
stack ;
return 0 ;
}
}
2005-12-08 20:49:34 +03:00
2006-01-27 21:38:14 +03:00
# endif
2005-12-02 23:35:07 +03:00
return 1 ;
}
2004-03-08 21:54:13 +03:00
static int _lv_suspend ( struct cmd_context * cmd , const char * lvid_s ,
int error_if_not_suspended )
2002-02-25 15:56:16 +03:00
{
2006-01-27 23:39:37 +03:00
struct logical_volume * lv , * lv_pre ;
2003-01-09 01:44:07 +03:00
struct lvinfo info ;
2002-02-25 15:56:16 +03:00
2002-11-18 17:01:16 +03:00
if ( ! activation ( ) )
return 1 ;
2006-01-27 23:39:37 +03:00
if ( ! ( lv = lv_from_lvid ( cmd , lvid_s , 0 ) ) )
return 0 ;
2005-10-31 23:15:28 +03:00
/* Use precommitted metadata if present */
2006-01-27 23:39:37 +03:00
if ( ! ( lv_pre = lv_from_lvid ( cmd , lvid_s , 1 ) ) )
2002-02-25 15:56:16 +03:00
return 0 ;
2002-03-14 18:36:07 +03:00
if ( test_mode ( ) ) {
_skip ( " Suspending '%s'. " , lv - > name ) ;
2003-01-17 23:16:23 +03:00
return 1 ;
2002-03-14 18:36:07 +03:00
}
2005-11-09 01:52:26 +03:00
if ( ! lv_info ( cmd , lv , & info , 0 ) )
return_0 ;
2002-03-11 22:02:28 +03:00
2003-07-05 02:34:56 +04:00
if ( ! info . exists | | info . suspended )
2004-03-08 21:54:13 +03:00
return error_if_not_suspended ? 0 : 1 ;
2003-07-05 02:34:56 +04:00
2005-11-09 01:52:26 +03:00
/* If VG was precommitted, preload devices for the LV */
2006-01-27 23:39:37 +03:00
if ( ( lv_pre - > vg - > status & PRECOMMITTED ) ) {
if ( ! _lv_preload ( lv_pre ) ) {
2005-11-09 01:52:26 +03:00
/* FIXME Revert preloading */
return_0 ;
}
}
2006-01-27 21:38:14 +03:00
if ( ! _register_dev_for_events ( cmd , lv , 0 ) )
stack ;
2003-07-05 02:34:56 +04:00
memlock_inc ( ) ;
2006-01-27 23:39:37 +03:00
if ( ! _lv_suspend_lv ( lv_pre ) ) {
2003-07-05 02:34:56 +04:00
memlock_dec ( ) ;
fs_unlock ( ) ;
return 0 ;
}
2002-03-01 22:08:11 +03:00
2002-02-25 15:56:16 +03:00
return 1 ;
}
2004-03-08 21:54:13 +03:00
/* Returns success if the device is not active */
int lv_suspend_if_active ( struct cmd_context * cmd , const char * lvid_s )
{
return _lv_suspend ( cmd , lvid_s , 0 ) ;
}
int lv_suspend ( struct cmd_context * cmd , const char * lvid_s )
{
return _lv_suspend ( cmd , lvid_s , 1 ) ;
}
static int _lv_resume ( struct cmd_context * cmd , const char * lvid_s ,
int error_if_not_active )
2002-02-25 15:56:16 +03:00
{
struct logical_volume * lv ;
2003-01-09 01:44:07 +03:00
struct lvinfo info ;
2002-02-25 15:56:16 +03:00
2002-11-18 17:01:16 +03:00
if ( ! activation ( ) )
return 1 ;
2005-10-31 23:15:28 +03:00
if ( ! ( lv = lv_from_lvid ( cmd , lvid_s , 0 ) ) )
2002-02-25 15:56:16 +03:00
return 0 ;
2002-03-14 18:36:07 +03:00
if ( test_mode ( ) ) {
_skip ( " Resuming '%s'. " , lv - > name ) ;
2003-01-17 23:16:23 +03:00
return 1 ;
2002-03-14 18:36:07 +03:00
}
2005-11-09 01:52:26 +03:00
if ( ! lv_info ( cmd , lv , & info , 0 ) )
return_0 ;
2002-03-11 22:02:28 +03:00
2003-07-05 02:34:56 +04:00
if ( ! info . exists | | ! info . suspended )
2004-03-08 21:54:13 +03:00
return error_if_not_active ? 0 : 1 ;
2003-07-05 02:34:56 +04:00
2004-03-08 21:54:13 +03:00
if ( ! _lv_activate_lv ( lv ) )
2003-07-05 02:34:56 +04:00
return 0 ;
memlock_dec ( ) ;
fs_unlock ( ) ;
2002-02-25 15:56:16 +03:00
2006-01-27 21:38:14 +03:00
if ( ! _register_dev_for_events ( cmd , lv , 1 ) )
stack ;
2005-12-02 23:35:07 +03:00
2002-02-25 15:56:16 +03:00
return 1 ;
}
2004-03-08 21:54:13 +03:00
/* Returns success if the device is not active */
int lv_resume_if_active ( struct cmd_context * cmd , const char * lvid_s )
{
return _lv_resume ( cmd , lvid_s , 0 ) ;
}
int lv_resume ( struct cmd_context * cmd , const char * lvid_s )
{
return _lv_resume ( cmd , lvid_s , 1 ) ;
}
2002-03-11 18:08:39 +03:00
int lv_deactivate ( struct cmd_context * cmd , const char * lvid_s )
2002-02-27 15:26:41 +03:00
{
struct logical_volume * lv ;
2003-01-09 01:44:07 +03:00
struct lvinfo info ;
2003-07-05 02:34:56 +04:00
int r ;
2002-02-27 15:26:41 +03:00
2002-11-18 17:01:16 +03:00
if ( ! activation ( ) )
return 1 ;
2005-10-31 23:15:28 +03:00
if ( ! ( lv = lv_from_lvid ( cmd , lvid_s , 0 ) ) )
2002-02-27 15:26:41 +03:00
return 0 ;
2002-03-14 18:36:07 +03:00
if ( test_mode ( ) ) {
_skip ( " Deactivating '%s'. " , lv - > name ) ;
2003-01-17 23:16:23 +03:00
return 1 ;
2002-03-14 18:36:07 +03:00
}
2005-11-09 01:52:26 +03:00
if ( ! lv_info ( cmd , lv , & info , 1 ) )
return_0 ;
2002-03-11 22:02:28 +03:00
2003-07-05 02:34:56 +04:00
if ( ! info . exists )
return 1 ;
2002-02-27 15:26:41 +03:00
2004-03-26 22:13:39 +03:00
if ( info . open_count & & ( lv - > status & VISIBLE_LV ) ) {
2005-11-09 01:52:26 +03:00
log_error ( " LV %s/%s in use: not deactivating " , lv - > vg - > name ,
2003-10-22 02:00:36 +04:00
lv - > name ) ;
return 0 ;
}
2006-01-27 21:38:14 +03:00
if ( ! _register_dev_for_events ( cmd , lv , 0 ) )
stack ;
2005-12-02 23:35:07 +03:00
2003-07-05 02:34:56 +04:00
memlock_inc ( ) ;
r = _lv_deactivate ( lv ) ;
memlock_dec ( ) ;
fs_unlock ( ) ;
return r ;
2002-02-27 15:26:41 +03:00
}
2004-03-08 21:54:13 +03:00
/* Test if LV passes filter */
int lv_activation_filter ( struct cmd_context * cmd , const char * lvid_s ,
int * activate_lv )
{
struct logical_volume * lv ;
if ( ! activation ( ) )
goto activate ;
2005-10-31 23:15:28 +03:00
if ( ! ( lv = lv_from_lvid ( cmd , lvid_s , 0 ) ) )
2004-03-08 21:54:13 +03:00
return 0 ;
if ( ! _passes_activation_filter ( cmd , lv ) ) {
log_verbose ( " Not activating %s/%s due to config file settings " ,
lv - > vg - > name , lv - > name ) ;
* activate_lv = 0 ;
return 1 ;
}
activate :
* activate_lv = 1 ;
return 1 ;
}
2005-08-15 03:18:28 +04:00
static int _lv_activate ( struct cmd_context * cmd , const char * lvid_s ,
int exclusive , int filter )
2002-02-27 15:26:41 +03:00
{
struct logical_volume * lv ;
2003-01-09 01:44:07 +03:00
struct lvinfo info ;
2003-07-05 02:34:56 +04:00
int r ;
2002-02-27 15:26:41 +03:00
2002-11-18 17:01:16 +03:00
if ( ! activation ( ) )
return 1 ;
2005-10-31 23:15:28 +03:00
if ( ! ( lv = lv_from_lvid ( cmd , lvid_s , 0 ) ) )
2002-02-27 15:26:41 +03:00
return 0 ;
2004-03-08 21:54:13 +03:00
if ( filter & & ! _passes_activation_filter ( cmd , lv ) ) {
2004-03-08 21:13:22 +03:00
log_verbose ( " Not activating %s/%s due to config file settings " ,
lv - > vg - > name , lv - > name ) ;
return 0 ;
}
2002-03-14 18:36:07 +03:00
if ( test_mode ( ) ) {
_skip ( " Activating '%s'. " , lv - > name ) ;
2003-01-17 23:16:23 +03:00
return 1 ;
2002-03-14 18:36:07 +03:00
}
2005-11-09 01:52:26 +03:00
if ( ! lv_info ( cmd , lv , & info , 0 ) )
return_0 ;
2002-03-01 22:08:11 +03:00
2005-11-09 01:52:26 +03:00
if ( info . exists & & ! info . suspended & & info . live_table )
2003-07-05 02:34:56 +04:00
return 1 ;
2002-02-27 15:26:41 +03:00
2005-08-15 03:18:28 +04:00
if ( exclusive )
lv - > status | = ACTIVATE_EXCL ;
2003-07-05 02:34:56 +04:00
memlock_inc ( ) ;
2004-03-08 21:54:13 +03:00
r = _lv_activate_lv ( lv ) ;
2003-07-05 02:34:56 +04:00
memlock_dec ( ) ;
fs_unlock ( ) ;
2006-01-27 21:38:14 +03:00
if ( ! _register_dev_for_events ( cmd , lv , 1 ) )
stack ;
2005-12-02 23:35:07 +03:00
2003-07-05 02:34:56 +04:00
return r ;
2002-02-27 15:26:41 +03:00
}
2003-01-09 01:44:07 +03:00
2004-03-08 21:54:13 +03:00
/* Activate LV */
2005-08-15 03:18:28 +04:00
int lv_activate ( struct cmd_context * cmd , const char * lvid_s , int exclusive )
2004-03-08 21:54:13 +03:00
{
2005-08-15 03:18:28 +04:00
return _lv_activate ( cmd , lvid_s , exclusive , 0 ) ;
2004-03-08 21:54:13 +03:00
}
/* Activate LV only if it passes filter */
2005-08-15 03:18:28 +04:00
int lv_activate_with_filter ( struct cmd_context * cmd , const char * lvid_s , int exclusive )
2004-03-08 21:54:13 +03:00
{
2005-08-15 03:18:28 +04:00
return _lv_activate ( cmd , lvid_s , exclusive , 1 ) ;
2004-03-08 21:54:13 +03:00
}
2003-11-12 22:16:48 +03:00
int lv_mknodes ( struct cmd_context * cmd , const struct logical_volume * lv )
{
struct lvinfo info ;
int r = 1 ;
2004-03-30 18:40:03 +04:00
if ( ! lv ) {
2005-10-17 03:03:59 +04:00
r = dm_mknodes ( NULL ) ;
2004-03-30 18:40:03 +04:00
fs_unlock ( ) ;
return r ;
}
2005-11-09 01:52:26 +03:00
if ( ! _lv_info ( cmd , lv , 1 , & info , 0 ) )
return_0 ;
2003-11-12 22:16:48 +03:00
if ( info . exists )
2004-03-30 18:38:57 +04:00
r = dev_manager_lv_mknodes ( lv ) ;
2003-11-12 22:16:48 +03:00
else
2004-03-30 18:38:57 +04:00
r = dev_manager_lv_rmnodes ( lv ) ;
2003-11-12 22:16:48 +03:00
fs_unlock ( ) ;
return r ;
}
2005-10-25 23:08:21 +04:00
/*
* Does PV use VG somewhere in its construction ?
* Returns 1 on failure .
*/
int pv_uses_vg ( struct cmd_context * cmd , struct physical_volume * pv ,
struct volume_group * vg )
{
struct dev_manager * dm ;
int r ;
if ( ! activation ( ) )
return 0 ;
if ( ! dm_is_dm_major ( MAJOR ( pv - > dev - > dev ) ) )
return 0 ;
if ( ! ( dm = dev_manager_create ( cmd , vg - > name ) ) ) {
stack ;
return 1 ;
}
r = dev_manager_device_uses_vg ( dm , pv - > dev , vg ) ;
dev_manager_destroy ( dm ) ;
return r ;
}
2003-07-05 02:34:56 +04:00
void activation_exit ( void )
{
dev_manager_exit ( ) ;
}
2003-01-09 01:44:07 +03:00
# endif