2004-06-07 19:10:21 +00:00
/*
* Copyright ( C ) 1997 - 2004 Sistina Software , Inc . All rights reserved .
* Copyright ( C ) 2004 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 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
*/
# include "lib.h"
# include "label.h"
# include "metadata.h"
# include "lvmcache.h"
# include "filter.h"
# include "xlate.h"
# include "disk_rep.h"
/* FIXME: memcpy might not be portable */
# define CPIN_8(x, y, z) {memcpy((x), (y), (z));}
# define CPOUT_8(x, y, z) {memcpy((y), (x), (z));}
# define CPIN_16(x, y) {(x) = xlate16_be((y));}
# define CPOUT_16(x, y) {(y) = xlate16_be((x));}
# define CPIN_32(x, y) {(x) = xlate32_be((y));}
# define CPOUT_32(x, y) {(y) = xlate32_be((x));}
# define CPIN_64(x, y) {(x) = xlate64_be((y));}
# define CPOUT_64(x, y) {(y) = xlate64_be((x));}
static int __read_pool_disk ( const struct format_type * fmt , struct device * dev ,
2005-10-16 23:03:59 +00:00
struct dm_pool * mem , struct pool_list * pl ,
2004-06-07 19:10:21 +00:00
const char * vg_name )
{
2006-11-30 23:11:42 +00:00
char buf [ 512 ] __attribute ( ( aligned ( 8 ) ) ) ;
2004-06-07 19:10:21 +00:00
/* FIXME: Need to check the cache here first */
if ( ! dev_read ( dev , UINT64_C ( 0 ) , 512 , buf ) ) {
log_very_verbose ( " Failed to read PV data from %s " ,
dev_name ( dev ) ) ;
return 0 ;
}
if ( ! read_pool_label ( pl , fmt - > labeller , dev , buf , NULL ) ) {
stack ;
return 0 ;
}
return 1 ;
}
static void _add_pl_to_list ( struct list * head , struct pool_list * data )
{
struct pool_list * pl ;
2005-06-01 16:51:55 +00:00
list_iterate_items ( pl , head ) {
2004-06-07 19:10:21 +00:00
if ( id_equal ( & data - > pv_uuid , & pl - > pv_uuid ) ) {
2006-11-30 23:11:42 +00:00
char uuid [ ID_LEN + 7 ] __attribute ( ( aligned ( 8 ) ) ) ;
2004-06-07 19:10:21 +00:00
id_write_format ( & pl - > pv_uuid , uuid , ID_LEN + 7 ) ;
if ( MAJOR ( data - > dev - > dev ) ! = md_major ( ) ) {
log_very_verbose ( " Ignoring duplicate PV %s on "
" %s " , uuid ,
dev_name ( data - > dev ) ) ;
return ;
}
log_very_verbose ( " Duplicate PV %s - using md %s " ,
uuid , dev_name ( data - > dev ) ) ;
2005-06-01 16:51:55 +00:00
list_del ( & pl - > list ) ;
2004-06-07 19:10:21 +00:00
break ;
}
}
list_add ( head , & data - > list ) ;
}
int read_pool_label ( struct pool_list * pl , struct labeller * l ,
struct device * dev , char * buf , struct label * * label )
{
struct lvmcache_info * info ;
struct id pvid ;
struct id vgid ;
2006-11-30 23:11:42 +00:00
char uuid [ ID_LEN + 7 ] __attribute ( ( aligned ( 8 ) ) ) ;
2004-06-07 19:10:21 +00:00
struct pool_disk * pd = & pl - > pd ;
pool_label_in ( pd , buf ) ;
get_pool_pv_uuid ( & pvid , pd ) ;
id_write_format ( & pvid , uuid , ID_LEN + 7 ) ;
log_debug ( " Calculated uuid %s for %s " , uuid , dev_name ( dev ) ) ;
get_pool_vg_uuid ( & vgid , pd ) ;
id_write_format ( & vgid , uuid , ID_LEN + 7 ) ;
log_debug ( " Calculated uuid %s for %s " , uuid , pd - > pl_pool_name ) ;
if ( ! ( info = lvmcache_add ( l , ( char * ) & pvid , dev , pd - > pl_pool_name ,
2006-04-11 17:42:15 +00:00
( char * ) & vgid , 0 ) ) ) {
2004-06-07 19:10:21 +00:00
stack ;
return 0 ;
}
if ( label )
* label = info - > label ;
info - > device_size = xlate32_be ( pd - > pl_blocks ) < < SECTOR_SHIFT ;
list_init ( & info - > mdas ) ;
info - > status & = ~ CACHE_INVALID ;
pl - > dev = dev ;
pl - > pv = NULL ;
memcpy ( & pl - > pv_uuid , & pvid , sizeof ( pvid ) ) ;
return 1 ;
}
/**
* pool_label_out - copies a pool_label_t into a char buffer
* @ pl : ptr to a pool_label_t struct
* @ buf : ptr to raw space where label info will be copied
*
* This function is important because it takes care of all of
* the endian issues when copying to disk . This way , when
* machines of different architectures are used , they will
* be able to interpret ondisk labels correctly . Always use
* this function before writing to disk .
*/
2006-11-30 23:11:42 +00:00
void pool_label_out ( struct pool_disk * pl , void * buf )
2004-06-07 19:10:21 +00:00
{
struct pool_disk * bufpl = ( struct pool_disk * ) buf ;
CPOUT_64 ( pl - > pl_magic , bufpl - > pl_magic ) ;
CPOUT_64 ( pl - > pl_pool_id , bufpl - > pl_pool_id ) ;
CPOUT_8 ( pl - > pl_pool_name , bufpl - > pl_pool_name , POOL_NAME_SIZE ) ;
CPOUT_32 ( pl - > pl_version , bufpl - > pl_version ) ;
CPOUT_32 ( pl - > pl_subpools , bufpl - > pl_subpools ) ;
CPOUT_32 ( pl - > pl_sp_id , bufpl - > pl_sp_id ) ;
CPOUT_32 ( pl - > pl_sp_devs , bufpl - > pl_sp_devs ) ;
CPOUT_32 ( pl - > pl_sp_devid , bufpl - > pl_sp_devid ) ;
CPOUT_32 ( pl - > pl_sp_type , bufpl - > pl_sp_type ) ;
CPOUT_64 ( pl - > pl_blocks , bufpl - > pl_blocks ) ;
CPOUT_32 ( pl - > pl_striping , bufpl - > pl_striping ) ;
CPOUT_32 ( pl - > pl_sp_dmepdevs , bufpl - > pl_sp_dmepdevs ) ;
CPOUT_32 ( pl - > pl_sp_dmepid , bufpl - > pl_sp_dmepid ) ;
CPOUT_32 ( pl - > pl_sp_weight , bufpl - > pl_sp_weight ) ;
CPOUT_32 ( pl - > pl_minor , bufpl - > pl_minor ) ;
CPOUT_32 ( pl - > pl_padding , bufpl - > pl_padding ) ;
CPOUT_8 ( pl - > pl_reserve , bufpl - > pl_reserve , 184 ) ;
}
/**
* pool_label_in - copies a char buffer into a pool_label_t
* @ pl : ptr to a pool_label_t struct
* @ buf : ptr to raw space where label info is copied from
*
* This function is important because it takes care of all of
* the endian issues when information from disk is about to be
* used . This way , when machines of different architectures
* are used , they will be able to interpret ondisk labels
* correctly . Always use this function before using labels that
* were read from disk .
*/
2006-11-30 23:11:42 +00:00
void pool_label_in ( struct pool_disk * pl , void * buf )
2004-06-07 19:10:21 +00:00
{
struct pool_disk * bufpl = ( struct pool_disk * ) buf ;
CPIN_64 ( pl - > pl_magic , bufpl - > pl_magic ) ;
CPIN_64 ( pl - > pl_pool_id , bufpl - > pl_pool_id ) ;
CPIN_8 ( pl - > pl_pool_name , bufpl - > pl_pool_name , POOL_NAME_SIZE ) ;
CPIN_32 ( pl - > pl_version , bufpl - > pl_version ) ;
CPIN_32 ( pl - > pl_subpools , bufpl - > pl_subpools ) ;
CPIN_32 ( pl - > pl_sp_id , bufpl - > pl_sp_id ) ;
CPIN_32 ( pl - > pl_sp_devs , bufpl - > pl_sp_devs ) ;
CPIN_32 ( pl - > pl_sp_devid , bufpl - > pl_sp_devid ) ;
CPIN_32 ( pl - > pl_sp_type , bufpl - > pl_sp_type ) ;
CPIN_64 ( pl - > pl_blocks , bufpl - > pl_blocks ) ;
CPIN_32 ( pl - > pl_striping , bufpl - > pl_striping ) ;
CPIN_32 ( pl - > pl_sp_dmepdevs , bufpl - > pl_sp_dmepdevs ) ;
CPIN_32 ( pl - > pl_sp_dmepid , bufpl - > pl_sp_dmepid ) ;
CPIN_32 ( pl - > pl_sp_weight , bufpl - > pl_sp_weight ) ;
CPIN_32 ( pl - > pl_minor , bufpl - > pl_minor ) ;
CPIN_32 ( pl - > pl_padding , bufpl - > pl_padding ) ;
CPIN_8 ( pl - > pl_reserve , bufpl - > pl_reserve , 184 ) ;
}
static char _calc_char ( unsigned int id )
{
/*
* [ 0 - 9 A - Za - z ! # ] - 64 printable chars ( 6 - bits )
*/
if ( id < 10 )
return id + 48 ;
if ( id < 36 )
return ( id - 10 ) + 65 ;
if ( id < 62 )
return ( id - 36 ) + 97 ;
if ( id = = 62 )
return ' ! ' ;
if ( id = = 63 )
return ' # ' ;
return ' % ' ;
}
void get_pool_uuid ( char * uuid , uint64_t poolid , uint32_t spid , uint32_t devid )
{
int i ;
unsigned shifter = 0x003F ;
assert ( ID_LEN = = 32 ) ;
memset ( uuid , 0 , ID_LEN ) ;
strcat ( uuid , " POOL0000000000 " ) ;
/* We grab the entire 64 bits (+2 that get shifted in) */
for ( i = 13 ; i < 24 ; i + + ) {
uuid [ i ] = _calc_char ( ( ( unsigned ) poolid ) & shifter ) ;
poolid = poolid > > 6 ;
}
/* We grab the entire 32 bits (+4 that get shifted in) */
for ( i = 24 ; i < 30 ; i + + ) {
uuid [ i ] = _calc_char ( ( unsigned ) ( spid & shifter ) ) ;
spid = spid > > 6 ;
}
/*
* Since we can only have 128 devices , we only worry about the
* last 12 bits
*/
for ( i = 30 ; i < 32 ; i + + ) {
uuid [ i ] = _calc_char ( ( unsigned ) ( devid & shifter ) ) ;
devid = devid > > 6 ;
}
}
2005-10-16 23:03:59 +00:00
static int _read_vg_pds ( const struct format_type * fmt , struct dm_pool * mem ,
2004-06-07 19:10:21 +00:00
struct lvmcache_vginfo * vginfo , struct list * head ,
uint32_t * devcount )
{
2005-06-01 16:51:55 +00:00
struct lvmcache_info * info ;
2005-06-14 18:29:12 +00:00
struct pool_list * pl = NULL ;
2005-10-16 23:03:59 +00:00
struct dm_pool * tmpmem ;
2004-06-07 19:10:21 +00:00
uint32_t sp_count = 0 ;
uint32_t * sp_devs = NULL ;
2006-05-09 21:23:51 +00:00
uint32_t i ;
2004-06-07 19:10:21 +00:00
/* FIXME: maybe should return a different error in memory
* allocation failure */
2005-10-16 23:03:59 +00:00
if ( ! ( tmpmem = dm_pool_create ( " pool read_vg " , 512 ) ) ) {
2004-06-07 19:10:21 +00:00
stack ;
return 0 ;
}
2005-06-01 16:51:55 +00:00
list_iterate_items ( info , & vginfo - > infos ) {
if ( info - > dev & &
! ( pl = read_pool_disk ( fmt , info - > dev , mem , vginfo - > vgname ) ) )
2004-06-07 19:10:21 +00:00
break ;
/*
* We need to keep track of the total expected number
* of devices per subpool
*/
if ( ! sp_count ) {
2005-06-01 16:51:55 +00:00
/* FIXME pl left uninitialised if !info->dev */
2004-06-07 19:10:21 +00:00
sp_count = pl - > pd . pl_subpools ;
if ( ! ( sp_devs =
2005-10-16 23:03:59 +00:00
dm_pool_zalloc ( tmpmem ,
2004-06-07 19:10:21 +00:00
sizeof ( uint32_t ) * sp_count ) ) ) {
log_error ( " Unable to allocate %d 32-bit uints " ,
sp_count ) ;
2005-10-16 23:03:59 +00:00
dm_pool_destroy ( tmpmem ) ;
2004-06-07 19:10:21 +00:00
return 0 ;
}
}
/*
* watch out for a pool label with a different subpool
* count than the original - give up if it does
*/
if ( sp_count ! = pl - > pd . pl_subpools )
break ;
_add_pl_to_list ( head , pl ) ;
if ( sp_count > pl - > pd . pl_sp_id & & sp_devs [ pl - > pd . pl_sp_id ] = = 0 )
sp_devs [ pl - > pd . pl_sp_id ] = pl - > pd . pl_sp_devs ;
}
* devcount = 0 ;
2005-06-01 16:51:55 +00:00
for ( i = 0 ; i < sp_count ; i + + )
2004-06-07 19:10:21 +00:00
* devcount + = sp_devs [ i ] ;
2005-10-16 23:03:59 +00:00
dm_pool_destroy ( tmpmem ) ;
2004-06-07 19:10:21 +00:00
if ( pl & & * pl - > pd . pl_pool_name )
return 1 ;
return 0 ;
}
int read_pool_pds ( const struct format_type * fmt , const char * vg_name ,
2005-10-16 23:03:59 +00:00
struct dm_pool * mem , struct list * pdhead )
2004-06-07 19:10:21 +00:00
{
struct lvmcache_vginfo * vginfo ;
uint32_t totaldevs ;
int full_scan = - 1 ;
do {
/*
* If the cache scanning doesn ' t work , this will never work
*/
2006-04-12 17:54:11 +00:00
if ( vg_name & & ( vginfo = vginfo_from_vgname ( vg_name , NULL ) ) & &
2004-06-07 19:10:21 +00:00
vginfo - > infos . n ) {
if ( _read_vg_pds ( fmt , mem , vginfo , pdhead , & totaldevs ) ) {
/*
* If we found all the devices we were
* expecting , return success
*/
if ( list_size ( pdhead ) = = totaldevs )
return 1 ;
/*
* accept partial pool if we ' ve done a full
* rescan of the cache
*/
if ( full_scan > 0 )
return 1 ;
}
}
/* Failed */
list_init ( pdhead ) ;
full_scan + + ;
if ( full_scan > 1 ) {
log_debug ( " No devices for vg %s found in cache " ,
vg_name ) ;
return 0 ;
}
lvmcache_label_scan ( fmt - > cmd , full_scan ) ;
} while ( 1 ) ;
}
struct pool_list * read_pool_disk ( const struct format_type * fmt ,
2005-10-16 23:03:59 +00:00
struct device * dev , struct dm_pool * mem ,
2004-06-07 19:10:21 +00:00
const char * vg_name )
{
struct pool_list * pl ;
if ( ! dev_open ( dev ) ) {
stack ;
return NULL ;
}
2005-10-16 23:03:59 +00:00
if ( ! ( pl = dm_pool_zalloc ( mem , sizeof ( * pl ) ) ) ) {
2004-06-07 19:10:21 +00:00
log_error ( " Unable to allocate pool list structure " ) ;
return 0 ;
}
if ( ! __read_pool_disk ( fmt , dev , mem , pl , vg_name ) ) {
stack ;
return NULL ;
}
if ( ! dev_close ( dev ) )
stack ;
return pl ;
}