2017-10-18 21:29:32 +03:00
/*
* Copyright ( C ) 2001 - 2004 Sistina Software , Inc . All rights reserved .
* Copyright ( C ) 2004 - 2017 Red Hat , Inc . All rights reserved .
*
* 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 Lesser General Public License v .2 .1 .
*
* You should have received a copy of the GNU Lesser General Public License
* along with this program ; if not , write to the Free Software Foundation ,
* Inc . , 51 Franklin Street , Fifth Floor , Boston , MA 02110 - 1301 USA
*/
/*
* This file contains functions now used only by liblvm .
* Ideally this file should be empty as liblvm and toollib should be doing identical things .
* FIXME Merge all the code into different parts of the tree .
*/
# include "lib.h"
# include "toolcontext.h"
# include "lvm-string.h"
# include "metadata.h"
# include "label.h"
# include "lvm-signal.h"
# include "lvmcache.h"
# include "lvmetad.h"
int vg_reduce ( struct volume_group * vg , const char * pv_name )
{
struct physical_volume * pv ;
struct pv_list * pvl ;
if ( ! ( pvl = find_pv_in_vg ( vg , pv_name ) ) ) {
log_error ( " Physical volume %s not in volume group %s. " ,
pv_name , vg - > name ) ;
return 0 ;
}
pv = pvl - > pv ;
if ( vgreduce_single ( vg - > cmd , vg , pv , 0 ) ) {
dm_list_add ( & vg - > removed_pvs , & pvl - > list ) ;
return 1 ;
}
log_error ( " Unable to remove physical volume '%s' from "
" volume group '%s'. " , pv_name , vg - > name ) ;
return 0 ;
}
static int _pvcreate_write ( struct cmd_context * cmd , struct pv_to_write * pvw )
{
struct physical_volume * pv = pvw - > pv ;
struct device * dev = pv - > dev ;
const char * pv_name = dev_name ( dev ) ;
if ( pvw - > new_pv ) {
/* Wipe existing label first */
if ( ! label_remove ( pv_dev ( pv ) ) ) {
log_error ( " Failed to wipe existing label on %s " , pv_name ) ;
return 0 ;
}
if ( pvw - > pp - > zero ) {
log_verbose ( " Zeroing start of device %s " , pv_name ) ;
if ( ! dev_open_quiet ( dev ) ) {
log_error ( " %s not opened: device not zeroed " , pv_name ) ;
return 0 ;
}
2017-12-05 02:18:56 +03:00
if ( ! dev_set ( dev , UINT64_C ( 0 ) , ( size_t ) 2048 , DEV_IO_LABEL , 0 ) ) {
2017-10-18 21:29:32 +03:00
log_error ( " %s not wiped: aborting " , pv_name ) ;
if ( ! dev_close ( dev ) )
stack ;
return 0 ;
}
if ( ! dev_close ( dev ) )
stack ;
}
}
log_verbose ( " Writing physical volume data to disk \" %s \" " ,
pv_name ) ;
if ( ! ( pv_write ( cmd , pv , 1 ) ) ) {
log_error ( " Failed to write physical volume \" %s \" " , pv_name ) ;
return 0 ;
}
if ( pvw - > new_pv )
log_print_unless_silent ( " Physical volume \" %s \" successfully created " , pv_name ) ;
else
log_verbose ( " Physical volume \" %s \" successfully written " , pv_name ) ;
return 1 ;
}
static int _verify_pv_create_params ( struct pvcreate_params * pp )
{
/*
* FIXME : Some of these checks are duplicates in pvcreate_params_validate .
*/
if ( pp - > pva . pvmetadatacopies > 2 ) {
log_error ( " Metadatacopies may only be 0, 1 or 2 " ) ;
return 0 ;
}
if ( pp - > pva . data_alignment > UINT32_MAX ) {
log_error ( " Physical volume data alignment is too big. " ) ;
return 0 ;
}
if ( pp - > pva . data_alignment_offset > UINT32_MAX ) {
log_error ( " Physical volume data alignment offset is too big. " ) ;
return 0 ;
}
return 1 ;
}
/*
* See if we may pvcreate on this device .
* 0 indicates we may not .
*/
static int _pvcreate_check ( struct cmd_context * cmd , const char * name ,
struct pvcreate_params * pp , int * wiped )
{
static const char really_init_msg [ ] = " Really INITIALIZE physical volume " ;
static const char not_init_msg [ ] = " physical volume not initialized " ;
struct physical_volume * pv ;
struct device * dev ;
int r = 0 ;
int scan_needed = 0 ;
int filter_refresh_needed = 0 ;
int used ;
/* FIXME Check partition type is LVM unless --force is given */
* wiped = 0 ;
/* Is there a pv here already? */
pv = find_pv_by_name ( cmd , name , 1 , 1 ) ;
/* Allow partial & exported VGs to be destroyed. */
/* We must have -ff to overwrite a non orphan */
if ( pv ) {
if ( ! is_orphan ( pv ) & & pp - > force ! = DONT_PROMPT_OVERRIDE ) {
log_error ( " Can't initialize physical volume \" %s \" of "
" volume group \" %s \" without -ff. " , name , pv_vg_name ( pv ) ) ;
goto out ;
}
if ( ( used = is_used_pv ( pv ) ) < 0 )
goto_out ;
if ( used & & pp - > force ! = DONT_PROMPT_OVERRIDE ) {
log_error ( " PV %s is used by a VG but its metadata is missing. " , name ) ;
log_error ( " Can't initialize PV '%s' without -ff. " , name ) ;
goto out ;
}
}
/* prompt */
if ( pv & & ! pp - > yes ) {
if ( is_orphan ( pv ) ) {
if ( used ) {
if ( yes_no_prompt ( " %s \" %s \" that is marked as belonging to a VG [y/n]? " ,
really_init_msg , name ) = = ' n ' ) {
log_error ( " %s: %s " , name , not_init_msg ) ;
goto out ;
}
}
} else {
if ( yes_no_prompt ( " %s \" %s \" of volume group \" %s \" [y/n]? " ,
really_init_msg , name , pv_vg_name ( pv ) ) = = ' n ' ) {
log_error ( " %s: %s " , name , not_init_msg ) ;
goto out ;
}
}
}
if ( sigint_caught ( ) )
goto_out ;
dev = dev_cache_get ( name , cmd - > full_filter ) ;
/*
* Refresh + rescan at the end is needed if :
* - we don ' t obtain device list from udev ,
* hence persistent cache file is used
* and we need to trash it and reevaluate
* for any changes done outside - adding
* any new foreign signature which may affect
* filtering - before we do pvcreate , we
* need to be sure that we have up - to - date
* view for filters
*
* - we have wiped existing foreign signatures
* from dev as this may affect what ' s filtered
* as well
*
*
* Only rescan at the end is needed if :
* - we ' ve just checked whether dev is fileterd
* by MD filter . We do the refresh in - situ ,
* so no need to require the refresh at the
* end of this fn . This is to allow for
* wiping MD signature during pvcreate for
* the dev - the dev would normally be
* filtered because of MD filter .
* This is an exception .
*/
/* Is there an md superblock here? */
if ( ! dev & & md_filtering ( ) ) {
if ( ! refresh_filters ( cmd ) )
goto_out ;
init_md_filtering ( 0 ) ;
dev = dev_cache_get ( name , cmd - > full_filter ) ;
init_md_filtering ( 1 ) ;
scan_needed = 1 ;
} else if ( ! obtain_device_list_from_udev ( ) )
filter_refresh_needed = scan_needed = 1 ;
if ( ! dev ) {
log_error ( " Device %s not found (or ignored by filtering). " , name ) ;
goto out ;
}
/*
* This test will fail if the device belongs to an MD array .
*/
if ( ! dev_test_excl ( dev ) ) {
/* FIXME Detect whether device-mapper itself is still using it */
log_error ( " Can't open %s exclusively. Mounted filesystem? " ,
name ) ;
goto out ;
}
2018-02-13 17:58:35 +03:00
dev_close ( dev ) ;
2017-10-18 21:29:32 +03:00
if ( ! wipe_known_signatures ( cmd , dev , name ,
TYPE_LVM1_MEMBER | TYPE_LVM2_MEMBER ,
0 , pp - > yes , pp - > force , wiped ) ) {
log_error ( " Aborting pvcreate on %s. " , name ) ;
goto out ;
}
if ( * wiped )
filter_refresh_needed = scan_needed = 1 ;
if ( sigint_caught ( ) )
goto_out ;
if ( pv & & ! is_orphan ( pv ) & & pp - > force )
log_warn ( " WARNING: Forcing physical volume creation on "
" %s%s%s%s " , name ,
! is_orphan ( pv ) ? " of volume group \" " : " " ,
pv_vg_name ( pv ) ,
! is_orphan ( pv ) ? " \" " : " " ) ;
r = 1 ;
out :
if ( filter_refresh_needed )
if ( ! refresh_filters ( cmd ) ) {
stack ;
r = 0 ;
}
if ( scan_needed ) {
if ( ! lvmcache_label_scan ( cmd ) ) {
stack ;
r = 0 ;
}
}
free_pv_fid ( pv ) ;
return r ;
}
/*
* pvcreate_vol ( ) - initialize a device with PV label and metadata area
*
* Parameters :
* - pv_name : device path to initialize
* - pp : parameters to pass to pv_create ; if NULL , use default values
*
* Returns :
* NULL : error
* struct physical_volume * ( non - NULL ) : handle to physical volume created
*/
struct physical_volume * pvcreate_vol ( struct cmd_context * cmd , const char * pv_name ,
struct pvcreate_params * pp , int write_now )
{
struct physical_volume * pv = NULL ;
struct device * dev ;
int wiped = 0 ;
struct dm_list mdas ;
struct pvcreate_params default_pp ;
char buffer [ 64 ] __attribute__ ( ( aligned ( 8 ) ) ) ;
dev_ext_t dev_ext_src ;
pvcreate_params_set_defaults ( & default_pp ) ;
if ( ! pp )
pp = & default_pp ;
if ( ! _verify_pv_create_params ( pp ) ) {
goto bad ;
}
if ( pp - > pva . idp ) {
2018-02-07 22:14:08 +03:00
if ( ( dev = lvmcache_device_from_pvid ( cmd , pp - > pva . idp , NULL ) ) & &
2017-10-18 21:29:32 +03:00
( dev ! = dev_cache_get ( pv_name , cmd - > full_filter ) ) ) {
if ( ! id_write_format ( ( const struct id * ) & pp - > pva . idp - > uuid ,
buffer , sizeof ( buffer ) ) )
goto_bad ;
log_error ( " uuid %s already in use on \" %s \" " , buffer ,
dev_name ( dev ) ) ;
goto bad ;
}
}
if ( ! _pvcreate_check ( cmd , pv_name , pp , & wiped ) )
goto_bad ;
if ( sigint_caught ( ) )
goto_bad ;
/*
* wipe_known_signatures called in _pvcreate_check fires
* WATCH event to update udev database . But at the moment ,
* we have no way to synchronize with such event - we may
* end up still seeing the old info in udev db and pvcreate
* can fail to proceed because of the device still being
* filtered ( because of the stale info in udev db ) .
* Disable udev dev - ext source temporarily here for
* this reason and rescan with DEV_EXT_NONE dev - ext
* source ( so filters use DEV_EXT_NONE source ) .
*/
dev_ext_src = external_device_info_source ( ) ;
if ( wiped & & ( dev_ext_src = = DEV_EXT_UDEV ) )
init_external_device_info_source ( DEV_EXT_NONE ) ;
dev = dev_cache_get ( pv_name , cmd - > full_filter ) ;
init_external_device_info_source ( dev_ext_src ) ;
if ( ! dev ) {
log_error ( " %s: Couldn't find device. Check your filters? " ,
pv_name ) ;
goto bad ;
}
dm_list_init ( & mdas ) ;
if ( ! ( pv = pv_create ( cmd , dev , & pp - > pva ) ) ) {
log_error ( " Failed to setup physical volume \" %s \" " , pv_name ) ;
goto bad ;
}
log_verbose ( " Set up physical volume for \" %s \" with % " PRIu64
" available sectors " , pv_name , pv_size ( pv ) ) ;
pv - > status | = UNLABELLED_PV ;
if ( write_now ) {
struct pv_to_write pvw ;
pvw . pp = pp ;
pvw . pv = pv ;
pvw . new_pv = 1 ;
if ( ! _pvcreate_write ( cmd , & pvw ) )
goto bad ;
}
return pv ;
bad :
return NULL ;
}
/*
* Extend a VG by a single PV / device path
*
* Parameters :
* - vg : handle of volume group to extend by ' pv_name '
* - pv_name : device path of PV to add to VG
* - pp : parameters to pass to implicit pvcreate ; if NULL , do not pvcreate
* - max_phys_block_size : largest physical block size found amongst PVs in a VG
*
*/
static int _vg_extend_single_pv ( struct volume_group * vg , char * pv_name ,
struct pvcreate_params * pp ,
unsigned int * max_phys_block_size )
{
struct physical_volume * pv ;
struct pv_to_write * pvw ;
int new_pv = 0 ;
pv = find_pv_by_name ( vg - > cmd , pv_name , 1 , 1 ) ;
if ( ! pv & & ! pp ) {
log_error ( " %s not identified as an existing "
" physical volume " , pv_name ) ;
return 0 ;
}
if ( ! pv & & pp ) {
if ( ! ( pv = pvcreate_vol ( vg - > cmd , pv_name , pp , 0 ) ) )
return_0 ;
new_pv = 1 ;
}
if ( ! ( check_dev_block_size_for_vg ( pv - > dev , ( const struct volume_group * ) vg ,
max_phys_block_size ) ) )
goto_bad ;
if ( ! add_pv_to_vg ( vg , pv_name , pv , new_pv ) )
goto_bad ;
if ( ( pv - > fmt - > features & FMT_PV_FLAGS ) | |
( pv - > status & UNLABELLED_PV ) ) {
if ( ! ( pvw = dm_pool_zalloc ( vg - > vgmem , sizeof ( * pvw ) ) ) ) {
log_error ( " pv_to_write allocation for '%s' failed " , pv_name ) ;
return 0 ;
}
pvw - > pv = pv ;
pvw - > pp = new_pv ? pp : NULL ;
pvw - > new_pv = new_pv ;
dm_list_add ( & vg - > pvs_to_write , & pvw - > list ) ;
}
return 1 ;
bad :
free_pv_fid ( pv ) ;
return 0 ;
}
/*
* Extend a VG by a single PV / device path
*
* Parameters :
* - vg : handle of volume group to extend by ' pv_name '
* - pv_count : count of device paths of PVs
* - pv_names : device paths of PVs to add to VG
* - pp : parameters to pass to implicit pvcreate ; if NULL , do not pvcreate
*
*/
int vg_extend ( struct volume_group * vg , int pv_count , const char * const * pv_names ,
struct pvcreate_params * pp )
{
int i ;
char * pv_name ;
unsigned int max_phys_block_size = 0 ;
if ( vg_bad_status_bits ( vg , RESIZEABLE_VG ) )
return_0 ;
/* attach each pv */
for ( i = 0 ; i < pv_count ; i + + ) {
if ( ! ( pv_name = dm_strdup ( pv_names [ i ] ) ) ) {
log_error ( " Failed to duplicate pv name %s. " , pv_names [ i ] ) ;
return 0 ;
}
dm_unescape_colons_and_at_signs ( pv_name , NULL , NULL ) ;
if ( ! _vg_extend_single_pv ( vg , pv_name , pp , & max_phys_block_size ) ) {
log_error ( " Unable to add physical volume '%s' to "
" volume group '%s'. " , pv_name , vg - > name ) ;
dm_free ( pv_name ) ;
return 0 ;
}
dm_free ( pv_name ) ;
}
( void ) check_pv_dev_sizes ( vg ) ;
/* FIXME Decide whether to initialise and add new mdahs to format instance */
return 1 ;
}
/*
* Decide whether it is " safe " to wipe the labels on this device .
* 0 indicates we may not .
*/
static int _pvremove_check ( struct cmd_context * cmd , const char * name ,
unsigned force_count , unsigned prompt , struct dm_list * pvslist )
{
static const char really_wipe_msg [ ] = " Really WIPE LABELS from physical volume " ;
struct device * dev ;
2018-04-20 18:43:50 +03:00
struct label * label ;
2017-10-18 21:29:32 +03:00
struct pv_list * pvl ;
struct physical_volume * pv = NULL ;
int used ;
int r = 0 ;
/* FIXME Check partition type is LVM unless --force is given */
if ( ! ( dev = dev_cache_get ( name , cmd - > filter ) ) ) {
log_error ( " Device %s not found. " , name ) ;
return 0 ;
}
/* Is there a pv here already? */
/* If not, this is an error unless you used -f. */
2018-04-20 18:43:50 +03:00
if ( ! label_read ( dev , & label , 0 ) ) {
2017-10-18 21:29:32 +03:00
if ( force_count )
return 1 ;
log_error ( " No PV label found on %s. " , name ) ;
return 0 ;
}
dm_list_iterate_items ( pvl , pvslist )
if ( pvl - > pv - > dev = = dev )
pv = pvl - > pv ;
if ( ! pv ) {
log_error ( INTERNAL_ERROR " Physical Volume %s has a label, "
" but is neither in a VG nor orphan. " , name ) ;
goto out ; /* better safe than sorry */
}
if ( is_orphan ( pv ) ) {
if ( ( used = is_used_pv ( pv ) ) < 0 )
goto_out ;
if ( used ) {
log_warn ( " WARNING: PV %s is used by a VG but its metadata is missing. " , name ) ;
if ( force_count < 2 )
goto_bad ;
if ( ! prompt & &
yes_no_prompt ( " %s \" %s \" that is marked as belonging to a VG [y/n]? " ,
really_wipe_msg , name ) = = ' n ' )
goto_bad ;
}
} else {
log_warn ( " WARNING: PV %s is used by VG %s (consider using vgreduce). " , name , pv_vg_name ( pv ) ) ;
if ( force_count < 2 )
goto_bad ;
if ( ! prompt & &
yes_no_prompt ( " %s \" %s \" of volume group \" %s \" [y/n]? " ,
really_wipe_msg , name , pv_vg_name ( pv ) ) = = ' n ' )
goto_bad ;
}
if ( force_count )
log_warn ( " WARNING: Wiping physical volume label from "
" %s%s%s%s " , name ,
! is_orphan ( pv ) ? " of volume group \" " : " " ,
pv_vg_name ( pv ) ,
! is_orphan ( pv ) ? " \" " : " " ) ;
r = 1 ;
bad :
if ( ! r ) {
log_error ( " %s: physical volume label not removed. " , name ) ;
if ( force_count < 2 ) /* Show hint as log_error() */
log_error ( " (If you are certain you need pvremove, "
" then confirm by using --force twice.) " ) ;
}
out :
return r ;
}
static int _pvremove_single ( struct cmd_context * cmd , const char * pv_name ,
void * handle __attribute__ ( ( unused ) ) , unsigned force_count ,
unsigned prompt , struct dm_list * pvslist )
{
struct device * dev ;
struct lvmcache_info * info ;
int r = 0 ;
if ( ! _pvremove_check ( cmd , pv_name , force_count , prompt , pvslist ) )
goto out ;
if ( ! ( dev = dev_cache_get ( pv_name , cmd - > filter ) ) ) {
log_error ( " %s: Couldn't find device. Check your filters? " ,
pv_name ) ;
goto out ;
}
info = lvmcache_info_from_pvid ( dev - > pvid , dev , 0 ) ;
if ( ! dev_test_excl ( dev ) ) {
/* FIXME Detect whether device-mapper is still using the device */
log_error ( " Can't open %s exclusively - not removing. "
" Mounted filesystem? " , dev_name ( dev ) ) ;
goto out ;
}
/* Wipe existing label(s) */
if ( ! label_remove ( dev ) ) {
log_error ( " Failed to wipe existing label(s) on %s " , pv_name ) ;
goto out ;
}
if ( ! lvmetad_pv_gone_by_dev ( dev ) )
goto_out ;
log_print_unless_silent ( " Labels on physical volume \" %s \" successfully wiped " ,
pv_name ) ;
r = 1 ;
out :
return r ;
}
int pvremove_many ( struct cmd_context * cmd , struct dm_list * pv_names ,
unsigned force_count , unsigned prompt )
{
int ret = 1 ;
struct dm_list * pvslist = NULL ;
struct pv_list * pvl ;
const struct dm_str_list * pv_name ;
if ( ! lock_vol ( cmd , VG_ORPHANS , LCK_VG_WRITE , NULL ) ) {
log_error ( " Can't get lock for orphan PVs " ) ;
return 0 ;
}
lvmcache_seed_infos_from_lvmetad ( cmd ) ;
if ( ! ( pvslist = get_pvs ( cmd ) ) ) {
ret = 0 ;
goto_out ;
}
dm_list_iterate_items ( pv_name , pv_names ) {
if ( ! _pvremove_single ( cmd , pv_name - > str , NULL , force_count , prompt , pvslist ) ) {
stack ;
ret = 0 ;
}
if ( sigint_caught ( ) ) {
ret = 0 ;
goto_out ;
}
}
out :
unlock_vg ( cmd , NULL , VG_ORPHANS ) ;
if ( pvslist )
dm_list_iterate_items ( pvl , pvslist )
free_pv_fid ( pvl - > pv ) ;
return ret ;
}
/* FIXME: liblvm todo - make into function that returns handle */
struct physical_volume * find_pv_by_name ( struct cmd_context * cmd ,
const char * pv_name ,
int allow_orphan , int allow_unformatted )
{
struct device * dev ;
struct pv_list * pvl ;
struct dm_list * pvslist ;
struct physical_volume * pv = NULL ;
lvmcache_seed_infos_from_lvmetad ( cmd ) ;
if ( ! ( dev = dev_cache_get ( pv_name , cmd - > filter ) ) ) {
if ( ! allow_unformatted )
log_error ( " Physical volume %s not found " , pv_name ) ;
return_NULL ;
}
if ( ! ( pvslist = get_pvs ( cmd ) ) )
return_NULL ;
dm_list_iterate_items ( pvl , pvslist )
if ( pvl - > pv - > dev = = dev )
pv = pvl - > pv ;
else
free_pv_fid ( pvl - > pv ) ;
if ( ! pv & & ! allow_unformatted )
log_error ( " Physical volume %s not found " , pv_name ) ;
if ( pv & & ! allow_orphan & & is_orphan_vg ( pv - > vg_name ) ) {
log_error ( " Physical volume %s not in a volume group " , pv_name ) ;
goto bad ;
}
return pv ;
bad :
free_pv_fid ( pv ) ;
return NULL ;
}