2010-09-30 17:15:42 +04:00
/*
2010-10-01 00:47:18 +04:00
* Copyright ( C ) 2001 - 2004 Sistina Software , Inc . All rights reserved .
* Copyright ( C ) 2004 - 2010 Red Hat , Inc . All rights reserved .
2010-09-30 17:15:42 +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 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 ,
2016-01-21 13:49:46 +03:00
* Inc . , 51 Franklin Street , Fifth Floor , Boston , MA 02110 - 1301 USA
2010-09-30 17:15:42 +04:00
*/
# include "lib.h"
# include "metadata.h"
# include "lvmcache.h"
/*
* FIXME : Check for valid handle before dereferencing field or log error ?
*/
# define pv_field(handle, field) ((handle)->field)
2010-09-30 18:09:22 +04:00
char * pv_fmt_dup ( const struct physical_volume * pv )
{
if ( ! pv - > fmt )
return NULL ;
return dm_pool_strdup ( pv - > vg - > vgmem , pv - > fmt - > name ) ;
}
2015-09-21 14:38:53 +03:00
char * pv_name_dup ( struct dm_pool * mem , const struct physical_volume * pv )
2010-09-30 18:09:22 +04:00
{
2015-09-21 14:38:53 +03:00
return dm_pool_strdup ( mem ? mem : pv - > vg - > vgmem , dev_name ( pv - > dev ) ) ;
2010-09-30 18:09:22 +04:00
}
2010-09-30 17:15:42 +04:00
/*
* Gets / Sets for external LVM library
*/
struct id pv_id ( const struct physical_volume * pv )
{
return pv_field ( pv , id ) ;
}
2015-09-21 14:38:53 +03:00
char * pv_uuid_dup ( struct dm_pool * mem , const struct physical_volume * pv )
2010-09-30 18:07:47 +04:00
{
2015-09-21 14:38:53 +03:00
return id_format_and_copy ( mem ? mem : pv - > vg - > vgmem , & pv - > id ) ;
2010-09-30 18:07:47 +04:00
}
2010-09-30 18:08:19 +04:00
char * pv_tags_dup ( const struct physical_volume * pv )
{
return tags_format_and_copy ( pv - > vg - > vgmem , & pv - > tags ) ;
}
2010-09-30 17:15:42 +04:00
const struct format_type * pv_format_type ( const struct physical_volume * pv )
{
return pv_field ( pv , fmt ) ;
}
struct id pv_vgid ( const struct physical_volume * pv )
{
return pv_field ( pv , vgid ) ;
}
struct device * pv_dev ( const struct physical_volume * pv )
{
return pv_field ( pv , dev ) ;
}
const char * pv_vg_name ( const struct physical_volume * pv )
{
2014-03-19 02:54:46 +04:00
/* Avoid exposing internal orphan names to users */
return ( ! is_orphan ( pv ) ) ? pv_field ( pv , vg_name ) : " " ;
2010-09-30 17:15:42 +04:00
}
const char * pv_dev_name ( const struct physical_volume * pv )
{
return dev_name ( pv_dev ( pv ) ) ;
}
uint64_t pv_size ( const struct physical_volume * pv )
{
return pv_field ( pv , size ) ;
}
uint64_t pv_dev_size ( const struct physical_volume * pv )
{
uint64_t size ;
if ( ! dev_get_size ( pv - > dev , & size ) )
size = 0 ;
2014-01-20 23:22:56 +04:00
2010-09-30 17:15:42 +04:00
return size ;
}
uint64_t pv_size_field ( const struct physical_volume * pv )
{
uint64_t size ;
if ( ! pv - > pe_count )
size = pv - > size ;
else
size = ( uint64_t ) pv - > pe_count * pv - > pe_size ;
2014-01-20 23:22:56 +04:00
2010-09-30 17:15:42 +04:00
return size ;
}
uint64_t pv_free ( const struct physical_volume * pv )
{
uint64_t freespace ;
2013-02-21 16:28:07 +04:00
if ( ! pv - > vg | | is_orphan_vg ( pv - > vg - > name ) )
2010-09-30 17:15:42 +04:00
freespace = pv - > size ;
else
freespace = ( uint64_t )
( pv - > pe_count - pv - > pe_alloc_count ) * pv - > pe_size ;
2014-01-20 23:22:56 +04:00
2010-09-30 17:15:42 +04:00
return freespace ;
}
uint64_t pv_status ( const struct physical_volume * pv )
{
return pv_field ( pv , status ) ;
}
uint32_t pv_pe_size ( const struct physical_volume * pv )
{
return pv_field ( pv , pe_size ) ;
}
2013-05-28 14:37:22 +04:00
uint64_t pv_ba_start ( const struct physical_volume * pv )
2013-02-14 19:04:35 +04:00
{
2013-05-28 14:37:22 +04:00
return pv_field ( pv , ba_start ) ;
2013-02-14 19:04:35 +04:00
}
2013-05-28 14:37:22 +04:00
uint64_t pv_ba_size ( const struct physical_volume * pv )
2013-02-14 19:04:35 +04:00
{
2013-05-28 14:37:22 +04:00
return pv_field ( pv , ba_size ) ;
2013-02-14 19:04:35 +04:00
}
2010-09-30 17:15:42 +04:00
uint64_t pv_pe_start ( const struct physical_volume * pv )
{
return pv_field ( pv , pe_start ) ;
}
uint32_t pv_pe_count ( const struct physical_volume * pv )
{
return pv_field ( pv , pe_count ) ;
}
uint32_t pv_pe_alloc_count ( const struct physical_volume * pv )
{
return pv_field ( pv , pe_alloc_count ) ;
}
uint32_t pv_mda_count ( const struct physical_volume * pv )
{
struct lvmcache_info * info ;
2012-02-10 05:28:27 +04:00
info = lvmcache_info_from_pvid ( ( const char * ) & pv - > id . uuid , 0 ) ;
2014-01-20 23:22:56 +04:00
2012-02-10 05:28:27 +04:00
return info ? lvmcache_mda_count ( info ) : UINT64_C ( 0 ) ;
}
static int _count_unignored ( struct metadata_area * mda , void * baton )
{
uint32_t * count = baton ;
2014-01-20 23:22:56 +04:00
2012-02-10 05:28:27 +04:00
if ( ! mda_is_ignored ( mda ) )
( * count ) + + ;
2014-01-20 23:22:56 +04:00
2012-02-10 05:28:27 +04:00
return 1 ;
2010-09-30 17:15:42 +04:00
}
uint32_t pv_mda_used_count ( const struct physical_volume * pv )
{
struct lvmcache_info * info ;
uint32_t used_count = 0 ;
2012-02-10 05:28:27 +04:00
info = lvmcache_info_from_pvid ( ( const char * ) & pv - > id . uuid , 0 ) ;
2010-09-30 17:15:42 +04:00
if ( ! info )
return 0 ;
2012-02-10 05:28:27 +04:00
lvmcache_foreach_mda ( info , _count_unignored , & used_count ) ;
2014-01-20 23:22:56 +04:00
2010-09-30 17:15:42 +04:00
return used_count ;
}
/**
* is_orphan - Determine whether a pv is an orphan based on its vg_name
* @ pv : handle to the physical volume
*/
int is_orphan ( const struct physical_volume * pv )
{
return is_orphan_vg ( pv_field ( pv , vg_name ) ) ;
}
/**
* is_pv - Determine whether a pv is a real pv or dummy one
* @ pv : handle to device
*/
int is_pv ( const struct physical_volume * pv )
{
return ( pv_field ( pv , vg_name ) ? 1 : 0 ) ;
}
int is_missing_pv ( const struct physical_volume * pv )
{
return pv_field ( pv , status ) & MISSING_PV ? 1 : 0 ;
}
2016-02-12 12:17:18 +03:00
int is_used_pv ( const struct physical_volume * pv )
{
struct lvmcache_info * info ;
uint32_t ext_flags ;
2016-02-15 17:50:11 +03:00
if ( ! pv - > fmt )
2016-02-15 16:46:31 +03:00
return 0 ;
2016-02-12 12:17:18 +03:00
if ( ! is_orphan ( pv ) )
return 1 ;
if ( ! ( pv - > fmt - > features & FMT_PV_FLAGS ) )
return 0 ;
if ( ! ( info = lvmcache_info_from_pvid ( ( const char * ) & pv - > id , 0 ) ) ) {
log_error ( " Failed to find cached info for PV %s. " , pv_dev_name ( pv ) ) ;
return - 1 ;
}
ext_flags = lvmcache_ext_flags ( info ) ;
return ext_flags & PV_EXT_USED ? 1 : 0 ;
}
2010-09-30 17:52:55 +04:00
char * pv_attr_dup ( struct dm_pool * mem , const struct physical_volume * pv )
{
char * repstr ;
2016-02-11 17:34:05 +03:00
int used = is_used_pv ( pv ) ;
2016-04-29 22:42:14 +03:00
int duplicate = lvmcache_dev_is_unchosen_duplicate ( pv - > dev ) ;
2010-09-30 17:52:55 +04:00
2016-02-26 18:46:37 +03:00
if ( ! ( repstr = dm_pool_zalloc ( mem , 4 ) ) ) {
2010-09-30 17:52:55 +04:00
log_error ( " dm_pool_alloc failed " ) ;
return NULL ;
}
2016-02-26 18:46:37 +03:00
/*
* An allocatable PV is always used , so we don ' t need to show ' u ' .
*/
2016-04-29 22:42:14 +03:00
if ( duplicate )
repstr [ 0 ] = ' d ' ;
else if ( pv - > status & ALLOCATABLE_PV )
repstr [ 0 ] = ' a ' ;
else if ( used > 0 )
repstr [ 0 ] = ' u ' ;
else
repstr [ 0 ] = ' - ' ;
2010-09-30 18:07:19 +04:00
repstr [ 1 ] = ( pv - > status & EXPORTED_VG ) ? ' x ' : ' - ' ;
2011-06-29 18:56:33 +04:00
repstr [ 2 ] = ( pv - > status & MISSING_PV ) ? ' m ' : ' - ' ;
2011-09-07 17:42:00 +04:00
2010-09-30 17:52:55 +04:00
return repstr ;
}
2010-09-30 18:09:10 +04:00
uint64_t pv_mda_size ( const struct physical_volume * pv )
{
struct lvmcache_info * info ;
uint64_t min_mda_size = 0 ;
const char * pvid = ( const char * ) ( & pv - > id . uuid ) ;
/* PVs could have 2 mdas of different sizes (rounding effect) */
2012-02-10 05:28:27 +04:00
if ( ( info = lvmcache_info_from_pvid ( pvid , 0 ) ) )
min_mda_size = lvmcache_smallest_mda_size ( info ) ;
2010-09-30 18:09:10 +04:00
return min_mda_size ;
}
2014-01-20 23:22:56 +04:00
static int _pv_mda_free ( struct metadata_area * mda , void * baton )
{
2012-02-10 05:28:27 +04:00
uint64_t mda_free ;
uint64_t * freespace = baton ;
if ( ! mda - > ops - > mda_free_sectors )
return 1 ;
mda_free = mda - > ops - > mda_free_sectors ( mda ) ;
if ( mda_free < * freespace )
* freespace = mda_free ;
2014-01-20 23:22:56 +04:00
2012-02-10 05:28:27 +04:00
return 1 ;
}
2013-07-29 20:52:28 +04:00
uint64_t lvmcache_info_mda_free ( struct lvmcache_info * info )
2010-09-30 18:09:10 +04:00
{
2012-02-10 05:28:27 +04:00
uint64_t freespace = UINT64_MAX ;
2010-09-30 18:09:10 +04:00
2014-01-14 07:17:27 +04:00
if ( info )
lvmcache_foreach_mda ( info , _pv_mda_free , & freespace ) ;
2010-09-30 18:09:10 +04:00
if ( freespace = = UINT64_MAX )
freespace = UINT64_C ( 0 ) ;
2012-02-10 05:28:27 +04:00
2010-09-30 18:09:10 +04:00
return freespace ;
}
2013-07-29 20:52:28 +04:00
uint64_t pv_mda_free ( const struct physical_volume * pv )
{
const char * pvid = ( const char * ) & pv - > id . uuid ;
struct lvmcache_info * info ;
2014-01-20 23:22:56 +04:00
2013-07-29 20:52:28 +04:00
if ( ( info = lvmcache_info_from_pvid ( pvid , 0 ) ) )
return lvmcache_info_mda_free ( info ) ;
2014-01-20 23:22:56 +04:00
2013-07-29 20:52:28 +04:00
return 0 ;
}
2010-09-30 18:09:10 +04:00
uint64_t pv_used ( const struct physical_volume * pv )
{
uint64_t used ;
if ( ! pv - > pe_count )
used = 0LL ;
else
used = ( uint64_t ) pv - > pe_alloc_count * pv - > pe_size ;
2014-01-20 23:22:56 +04:00
2010-09-30 18:09:10 +04:00
return used ;
}
2012-02-10 05:28:27 +04:00
struct _pv_mda_set_ignored_baton {
unsigned mda_ignored ;
struct dm_list * mdas_in_use , * mdas_ignored , * mdas_to_change ;
} ;
static int _pv_mda_set_ignored_one ( struct metadata_area * mda , void * baton )
{
struct _pv_mda_set_ignored_baton * b = baton ;
struct metadata_area * vg_mda , * tmda ;
if ( mda_is_ignored ( mda ) & & ! b - > mda_ignored ) {
/* Changing an ignored mda to one in_use requires moving it */
dm_list_iterate_items_safe ( vg_mda , tmda , b - > mdas_ignored )
if ( mda_locns_match ( mda , vg_mda ) ) {
mda_set_ignored ( vg_mda , b - > mda_ignored ) ;
dm_list_move ( b - > mdas_in_use , & vg_mda - > list ) ;
}
}
dm_list_iterate_items_safe ( vg_mda , tmda , b - > mdas_in_use )
if ( mda_locns_match ( mda , vg_mda ) )
/* Don't move mda: needs writing to disk. */
mda_set_ignored ( vg_mda , b - > mda_ignored ) ;
mda_set_ignored ( mda , b - > mda_ignored ) ;
2014-01-20 23:22:56 +04:00
2012-02-10 05:28:27 +04:00
return 1 ;
}
2010-09-30 17:15:42 +04:00
unsigned pv_mda_set_ignored ( const struct physical_volume * pv , unsigned mda_ignored )
{
struct lvmcache_info * info ;
2012-02-10 05:28:27 +04:00
struct _pv_mda_set_ignored_baton baton ;
struct metadata_area * mda ;
2010-09-30 17:15:42 +04:00
2012-02-10 05:28:27 +04:00
if ( ! ( info = lvmcache_info_from_pvid ( ( const char * ) & pv - > id . uuid , 0 ) ) )
2010-09-30 17:15:42 +04:00
return_0 ;
2012-02-10 05:28:27 +04:00
baton . mda_ignored = mda_ignored ;
baton . mdas_in_use = & pv - > fid - > metadata_areas_in_use ;
baton . mdas_ignored = & pv - > fid - > metadata_areas_ignored ;
baton . mdas_to_change = baton . mda_ignored ? baton . mdas_in_use : baton . mdas_ignored ;
2011-02-21 15:33:16 +03:00
2010-09-30 17:15:42 +04:00
if ( is_orphan ( pv ) ) {
2012-02-10 05:28:27 +04:00
dm_list_iterate_items ( mda , baton . mdas_to_change )
mda_set_ignored ( mda , baton . mda_ignored ) ;
2010-09-30 17:15:42 +04:00
return 1 ;
}
/*
* Do not allow disabling of the the last PV in a VG .
*/
if ( pv_mda_used_count ( pv ) = = vg_mda_used_count ( pv - > vg ) ) {
log_error ( " Cannot disable all metadata areas in volume group %s. " ,
pv - > vg - > name ) ;
return 0 ;
}
/*
* Non - orphan case is more complex .
* If the PV ' s mdas are ignored , and we wish to un - ignore ,
* we clear the bit and move them from the ignored mda list to the
* in_use list , ensuring the new state will get written to disk
* in the vg_write ( ) path .
* If the PV ' s mdas are not ignored , and we are setting
* them to ignored , we set the bit but leave them on the in_use
* list , ensuring the new state will get written to disk in the
* vg_write ( ) path .
*/
2011-02-21 15:33:16 +03:00
/* FIXME: Try not to update the cache here! Also, try to iterate over
* PV mdas only using the format instance ' s index somehow
* ( i . e . try to avoid using mda_locn_match call ) . */
2012-02-10 05:28:27 +04:00
lvmcache_foreach_mda ( info , _pv_mda_set_ignored_one , & baton ) ;
2010-09-30 17:15:42 +04:00
return 1 ;
}
2013-07-29 20:54:59 +04:00
struct label * pv_label ( const struct physical_volume * pv )
{
struct lvmcache_info * info =
lvmcache_info_from_pvid ( ( const char * ) & pv - > id . uuid , 0 ) ;
2014-01-20 23:22:56 +04:00
if ( info )
return lvmcache_get_label ( info ) ;
/* process_each_pv() may create dummy PVs that have no label */
if ( pv - > vg & & pv - > dev )
log_error ( INTERNAL_ERROR " PV %s unexpectedly not in cache. " ,
dev_name ( pv - > dev ) ) ;
return NULL ;
2013-07-29 20:54:59 +04:00
}