2002-11-18 14:04:08 +00:00
/*
2004-03-30 19:35:44 +00:00
* Copyright ( C ) 2001 - 2004 Sistina Software , Inc . All rights reserved .
2007-08-20 20:55:30 +00:00
* Copyright ( C ) 2004 - 2007 Red Hat , Inc . All rights reserved .
2002-11-18 14:04:08 +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 ,
* Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
2002-11-18 14:04:08 +00:00
*/
# include "lib.h"
# include "metadata.h"
# include "import-export.h"
# include "display.h"
# include "toolcontext.h"
2003-07-04 22:34:56 +00:00
# include "lvmcache.h"
2003-09-17 20:35:57 +00:00
# include "lv_alloc.h"
2005-04-19 20:58:25 +00:00
# include "pv_alloc.h"
2004-09-16 18:40:56 +00:00
# include "segtype.h"
2004-05-04 21:25:57 +00:00
# include "text_import.h"
2002-11-18 14:04:08 +00:00
2005-10-16 23:03:59 +00:00
typedef int ( * section_fn ) ( struct format_instance * fid , struct dm_pool * mem ,
2002-11-18 14:04:08 +00:00
struct volume_group * vg , struct config_node * pvn ,
struct config_node * vgn ,
2005-10-16 23:03:59 +00:00
struct dm_hash_table * pv_hash ) ;
2002-11-18 14:04:08 +00:00
# define _read_int32(root, path, result) \
2005-03-10 22:34:17 +00:00
get_config_uint32 ( root , path , ( uint32_t * ) result )
2002-11-18 14:04:08 +00:00
# define _read_uint32(root, path, result) \
2004-03-08 18:28:45 +00:00
get_config_uint32 ( root , path , result )
2002-11-18 14:04:08 +00:00
# define _read_int64(root, path, result) \
2004-03-08 18:28:45 +00:00
get_config_uint64 ( root , path , result )
2002-11-18 14:04:08 +00:00
/*
* Logs an attempt to read an invalid format file .
*/
static void _invalid_format ( const char * str )
{
log_error ( " Can't process text format file - %s. " , str ) ;
}
/*
* Checks that the config file contains vg metadata , and that it
* we recognise the version number ,
*/
2004-03-08 18:28:45 +00:00
static int _check_version ( struct config_tree * cft )
2002-11-18 14:04:08 +00:00
{
struct config_node * cn ;
struct config_value * cv ;
/*
* Check the contents field .
*/
2004-03-08 18:28:45 +00:00
if ( ! ( cn = find_config_node ( cft - > root , CONTENTS_FIELD ) ) ) {
2002-11-18 14:04:08 +00:00
_invalid_format ( " missing contents field " ) ;
return 0 ;
}
cv = cn - > v ;
if ( ! cv | | cv - > type ! = CFG_STRING | | strcmp ( cv - > v . str , CONTENTS_VALUE ) ) {
_invalid_format ( " unrecognised contents field " ) ;
return 0 ;
}
/*
* Check the version number .
*/
2004-03-08 18:28:45 +00:00
if ( ! ( cn = find_config_node ( cft - > root , FORMAT_VERSION_FIELD ) ) ) {
2002-11-18 14:04:08 +00:00
_invalid_format ( " missing version number " ) ;
return 0 ;
}
cv = cn - > v ;
if ( ! cv | | cv - > type ! = CFG_INT | | cv - > v . i ! = FORMAT_VERSION_VALUE ) {
_invalid_format ( " unrecognised version number " ) ;
return 0 ;
}
return 1 ;
}
static int _read_id ( struct id * id , struct config_node * cn , const char * path )
{
struct config_value * cv ;
2004-03-08 18:28:45 +00:00
if ( ! ( cn = find_config_node ( cn , path ) ) ) {
2002-11-18 14:04:08 +00:00
log_error ( " Couldn't find uuid. " ) ;
return 0 ;
}
cv = cn - > v ;
if ( ! cv | | ! cv - > v . str ) {
log_error ( " uuid must be a string. " ) ;
return 0 ;
}
if ( ! id_read_format ( id , cv - > v . str ) ) {
log_error ( " Invalid uuid. " ) ;
return 0 ;
}
return 1 ;
}
2005-10-16 23:03:59 +00:00
static int _read_pv ( struct format_instance * fid , struct dm_pool * mem ,
2002-11-18 14:04:08 +00:00
struct volume_group * vg , struct config_node * pvn ,
2007-08-22 14:38:18 +00:00
struct config_node * vgn __attribute ( ( unused ) ) ,
struct dm_hash_table * pv_hash )
2002-11-18 14:04:08 +00:00
{
struct physical_volume * pv ;
struct pv_list * pvl ;
struct config_node * cn ;
2006-10-19 12:53:47 +00:00
uint64_t size ;
2002-11-18 14:04:08 +00:00
2005-10-16 23:03:59 +00:00
if ( ! ( pvl = dm_pool_zalloc ( mem , sizeof ( * pvl ) ) ) | |
! ( pvl - > pv = dm_pool_zalloc ( mem , sizeof ( * pvl - > pv ) ) ) ) {
2002-11-18 14:04:08 +00:00
stack ;
return 0 ;
}
pv = pvl - > pv ;
/*
* Add the pv to the pv hash for quick lookup when we read
* the lv segments .
*/
2005-10-16 23:03:59 +00:00
if ( ! dm_hash_insert ( pv_hash , pvn - > key , pv ) ) {
2002-11-18 14:04:08 +00:00
stack ;
return 0 ;
}
if ( ! ( pvn = pvn - > child ) ) {
log_error ( " Empty pv section. " ) ;
return 0 ;
}
if ( ! _read_id ( & pv - > id , pvn , " id " ) ) {
log_error ( " Couldn't read uuid for volume group. " ) ;
return 0 ;
}
/*
* Convert the uuid into a device .
*/
if ( ! ( pv - > dev = device_from_pvid ( fid - > fmt - > cmd , & pv - > id ) ) ) {
2006-11-30 23:11:42 +00:00
char buffer [ 64 ] __attribute ( ( aligned ( 8 ) ) ) ;
2002-11-18 14:04:08 +00:00
if ( ! id_write_format ( & pv - > id , buffer , sizeof ( buffer ) ) )
log_error ( " Couldn't find device. " ) ;
else
log_error ( " Couldn't find device with uuid '%s'. " ,
buffer ) ;
if ( partial_mode ( ) )
vg - > status | = PARTIAL_VG ;
else
return 0 ;
}
2005-10-16 23:03:59 +00:00
if ( ! ( pv - > vg_name = dm_pool_strdup ( mem , vg - > name ) ) ) {
2002-11-18 14:04:08 +00:00
stack ;
return 0 ;
}
2006-04-12 21:23:04 +00:00
memcpy ( & pv - > vgid , & vg - > id , sizeof ( vg - > id ) ) ;
2004-03-08 18:28:45 +00:00
if ( ! ( cn = find_config_node ( pvn , " status " ) ) ) {
2002-11-18 14:04:08 +00:00
log_error ( " Couldn't find status flags for physical volume. " ) ;
return 0 ;
}
if ( ! ( read_flags ( & pv - > status , PV_FLAGS , cn - > v ) ) ) {
log_error ( " Couldn't read status flags for physical volume. " ) ;
return 0 ;
}
2006-10-07 23:17:17 +00:00
/* Late addition */
_read_int64 ( pvn , " dev_size " , & pv - > size ) ;
2002-11-18 14:04:08 +00:00
if ( ! _read_int64 ( pvn , " pe_start " , & pv - > pe_start ) ) {
log_error ( " Couldn't read extent size for volume group. " ) ;
return 0 ;
}
if ( ! _read_int32 ( pvn , " pe_count " , & pv - > pe_count ) ) {
log_error ( " Couldn't find extent count (pe_count) for "
" physical volume. " ) ;
return 0 ;
}
2004-03-08 17:19:15 +00:00
list_init ( & pv - > tags ) ;
2005-04-19 20:58:25 +00:00
list_init ( & pv - > segments ) ;
2004-03-08 17:19:15 +00:00
/* Optional tags */
2004-03-08 18:28:45 +00:00
if ( ( cn = find_config_node ( pvn , " tags " ) ) & &
2004-03-08 17:19:15 +00:00
! ( read_tags ( mem , & pv - > tags , cn - > v ) ) ) {
log_error ( " Couldn't read tags for physical volume %s in %s. " ,
dev_name ( pv - > dev ) , vg - > name ) ;
return 0 ;
}
2002-11-18 14:04:08 +00:00
/* adjust the volume group. */
vg - > extent_count + = pv - > pe_count ;
vg - > free_count + = pv - > pe_count ;
pv - > pe_size = vg - > extent_size ;
2006-10-07 23:17:17 +00:00
2002-11-18 14:04:08 +00:00
pv - > pe_alloc_count = 0 ;
pv - > fmt = fid - > fmt ;
2006-10-19 12:53:47 +00:00
/* Fix up pv size if missing */
if ( ! pv - > size & & pv - > dev ) {
if ( ! dev_get_size ( pv - > dev , & pv - > size ) ) {
log_error ( " %s: Couldn't get size. " , dev_name ( pv - > dev ) ) ;
return 0 ;
}
log_verbose ( " Fixing up missing format1 size (%s) "
" for PV %s " , display_size ( fid - > fmt - > cmd , pv - > size ) ,
dev_name ( pv - > dev ) ) ;
if ( vg ) {
size = pv - > pe_count * ( uint64_t ) vg - > extent_size +
pv - > pe_start ;
if ( size > pv - > size )
log_error ( " WARNING: Physical Volume %s is too "
" large for underlying device " ,
dev_name ( pv - > dev ) ) ;
}
}
2005-04-19 20:58:25 +00:00
if ( ! alloc_pv_segment_whole_pv ( mem , pv ) ) {
stack ;
return 0 ;
}
2002-11-18 14:04:08 +00:00
vg - > pv_count + + ;
list_add ( & vg - > pvs , & pvl - > list ) ;
return 1 ;
}
static void _insert_segment ( struct logical_volume * lv , struct lv_segment * seg )
{
struct lv_segment * comp ;
2005-06-01 16:51:55 +00:00
list_iterate_items ( comp , & lv - > segments ) {
2002-11-18 14:04:08 +00:00
if ( comp - > le > seg - > le ) {
list_add ( & comp - > list , & seg - > list ) ;
return ;
}
}
lv - > le_count + = seg - > len ;
list_add ( & lv - > segments , & seg - > list ) ;
}
2005-10-16 23:03:59 +00:00
static int _read_segment ( struct dm_pool * mem , struct volume_group * vg ,
2002-11-18 14:04:08 +00:00
struct logical_volume * lv , struct config_node * sn ,
2005-10-16 23:03:59 +00:00
struct dm_hash_table * pv_hash )
2002-11-18 14:04:08 +00:00
{
2004-05-04 21:25:57 +00:00
uint32_t area_count = 0u ;
2002-11-18 14:04:08 +00:00
struct lv_segment * seg ;
struct config_node * cn ;
struct config_value * cv ;
uint32_t start_extent , extent_count ;
2004-05-04 21:25:57 +00:00
struct segment_type * segtype ;
const char * segtype_str ;
2002-11-18 14:04:08 +00:00
if ( ! ( sn = sn - > child ) ) {
log_error ( " Empty segment section. " ) ;
return 0 ;
}
if ( ! _read_int32 ( sn , " start_extent " , & start_extent ) ) {
log_error ( " Couldn't read 'start_extent' for segment '%s'. " ,
sn - > key ) ;
return 0 ;
}
if ( ! _read_int32 ( sn , " extent_count " , & extent_count ) ) {
log_error ( " Couldn't read 'extent_count' for segment '%s'. " ,
sn - > key ) ;
return 0 ;
}
2004-05-04 21:25:57 +00:00
segtype_str = " striped " ;
2004-03-08 18:28:45 +00:00
if ( ( cn = find_config_node ( sn , " type " ) ) ) {
2002-11-18 14:04:08 +00:00
cv = cn - > v ;
if ( ! cv | | ! cv - > v . str ) {
log_error ( " Segment type must be a string. " ) ;
return 0 ;
}
2004-05-04 21:25:57 +00:00
segtype_str = cv - > v . str ;
2002-11-18 14:04:08 +00:00
}
2004-05-04 21:25:57 +00:00
if ( ! ( segtype = get_segtype_from_string ( vg - > cmd , segtype_str ) ) ) {
stack ;
return 0 ;
2002-11-18 14:04:08 +00:00
}
2004-05-04 21:25:57 +00:00
if ( segtype - > ops - > text_import_area_count & &
! segtype - > ops - > text_import_area_count ( sn , & area_count ) ) {
stack ;
return 0 ;
2003-04-30 15:22:36 +00:00
}
2005-04-22 15:44:00 +00:00
if ( ! ( seg = alloc_lv_segment ( mem , segtype , lv , start_extent ,
2005-06-01 16:51:55 +00:00
extent_count , 0 , 0 , NULL , area_count ,
extent_count , 0 , 0 , 0 ) ) ) {
2003-09-17 20:35:57 +00:00
log_error ( " Segment allocation failed " ) ;
2002-11-18 14:04:08 +00:00
return 0 ;
}
2004-05-04 21:25:57 +00:00
if ( seg - > segtype - > ops - > text_import & &
! seg - > segtype - > ops - > text_import ( seg , sn , pv_hash ) ) {
stack ;
return 0 ;
}
2002-11-18 14:04:08 +00:00
2004-03-08 17:19:15 +00:00
/* Optional tags */
2004-03-08 18:28:45 +00:00
if ( ( cn = find_config_node ( sn , " tags " ) ) & &
2004-03-08 17:19:15 +00:00
! ( read_tags ( mem , & seg - > tags , cn - > v ) ) ) {
log_error ( " Couldn't read tags for a segment of %s/%s. " ,
vg - > name , lv - > name ) ;
return 0 ;
}
2004-05-04 21:25:57 +00:00
/*
* Insert into correct part of segment list .
*/
_insert_segment ( lv , seg ) ;
2002-11-18 14:04:08 +00:00
2005-05-09 16:59:01 +00:00
if ( seg_is_mirrored ( seg ) )
2004-05-04 21:25:57 +00:00
lv - > status | = MIRRORED ;
2002-11-18 14:04:08 +00:00
2005-05-09 16:59:01 +00:00
if ( seg_is_virtual ( seg ) )
2004-05-11 16:01:58 +00:00
lv - > status | = VIRTUAL ;
2004-05-04 21:25:57 +00:00
return 1 ;
}
2002-11-18 14:04:08 +00:00
2004-05-04 21:25:57 +00:00
int text_import_areas ( struct lv_segment * seg , const struct config_node * sn ,
2005-10-16 23:03:59 +00:00
const struct config_node * cn , struct dm_hash_table * pv_hash ,
2005-06-03 14:49:51 +00:00
uint32_t flags )
2004-05-04 21:25:57 +00:00
{
unsigned int s ;
struct config_value * cv ;
struct logical_volume * lv1 ;
const char * seg_name = sn - > key ;
2002-11-18 14:04:08 +00:00
2004-05-04 21:25:57 +00:00
if ( ! seg - > area_count ) {
log_error ( " Zero areas not allowed for segment '%s' " , sn - > key ) ;
return 0 ;
}
2002-11-18 14:04:08 +00:00
2004-05-04 21:25:57 +00:00
for ( cv = cn - > v , s = 0 ; cv & & s < seg - > area_count ; s + + , cv = cv - > next ) {
2002-11-18 14:04:08 +00:00
2004-05-04 21:25:57 +00:00
/* first we read the pv */
const char * bad = " Badly formed areas array for "
" segment '%s'. " ;
struct physical_volume * pv ;
2002-11-18 14:04:08 +00:00
2004-05-04 21:25:57 +00:00
if ( cv - > type ! = CFG_STRING ) {
log_error ( bad , sn - > key ) ;
2002-11-18 14:04:08 +00:00
return 0 ;
}
2004-05-04 21:25:57 +00:00
if ( ! cv - > next ) {
log_error ( bad , sn - > key ) ;
2003-04-24 22:23:24 +00:00
return 0 ;
}
2004-05-04 21:25:57 +00:00
if ( cv - > next - > type ! = CFG_INT ) {
log_error ( bad , sn - > key ) ;
2003-04-30 15:22:36 +00:00
return 0 ;
}
2004-05-04 21:25:57 +00:00
/* FIXME Cope if LV not yet read in */
2005-10-16 23:03:59 +00:00
if ( ( pv = dm_hash_lookup ( pv_hash , cv - > v . str ) ) ) {
2007-08-22 14:38:18 +00:00
if ( ! set_lv_segment_area_pv ( seg , s , pv , ( uint32_t ) cv - > next - > v . i ) ) {
2005-05-03 17:28:23 +00:00
stack ;
return 0 ;
}
2004-05-04 21:25:57 +00:00
} else if ( ( lv1 = find_lv ( seg - > lv - > vg , cv - > v . str ) ) ) {
2007-08-22 14:38:18 +00:00
set_lv_segment_area_lv ( seg , s , lv1 , ( uint32_t ) cv - > next - > v . i ,
2005-06-03 14:49:51 +00:00
flags ) ;
2004-05-04 21:25:57 +00:00
} else {
log_error ( " Couldn't find volume '%s' "
" for segment '%s'. " ,
2006-05-09 21:23:51 +00:00
cv - > v . str ? : " NULL " , seg_name ) ;
2002-11-18 14:04:08 +00:00
return 0 ;
}
2004-05-04 21:25:57 +00:00
cv = cv - > next ;
2002-11-18 14:04:08 +00:00
}
/*
2004-05-04 21:25:57 +00:00
* Check we read the correct number of stripes .
2002-11-18 14:04:08 +00:00
*/
2004-05-04 21:25:57 +00:00
if ( cv | | ( s < seg - > area_count ) ) {
log_error ( " Incorrect number of areas in area array "
" for segment '%s'. " , seg_name ) ;
return 0 ;
}
2002-11-18 14:04:08 +00:00
return 1 ;
}
2005-10-16 23:03:59 +00:00
static int _read_segments ( struct dm_pool * mem , struct volume_group * vg ,
2002-11-18 14:04:08 +00:00
struct logical_volume * lv , struct config_node * lvn ,
2005-10-16 23:03:59 +00:00
struct dm_hash_table * pv_hash )
2002-11-18 14:04:08 +00:00
{
struct config_node * sn ;
int count = 0 , seg_count ;
for ( sn = lvn ; sn ; sn = sn - > sib ) {
/*
* All sub - sections are assumed to be segments .
*/
if ( ! sn - > v ) {
if ( ! _read_segment ( mem , vg , lv , sn , pv_hash ) ) {
stack ;
return 0 ;
}
count + + ;
}
/* FIXME Remove this restriction */
if ( ( lv - > status & SNAPSHOT ) & & count > 1 ) {
log_error ( " Only one segment permitted for snapshot " ) ;
return 0 ;
}
}
if ( ! _read_int32 ( lvn , " segment_count " , & seg_count ) ) {
log_error ( " Couldn't read segment count for logical volume. " ) ;
return 0 ;
}
if ( seg_count ! = count ) {
log_error ( " segment_count and actual number of segments "
" disagree. " ) ;
return 0 ;
}
/*
* Check there are no gaps or overlaps in the lv .
*/
2005-10-27 21:51:28 +00:00
if ( ! check_lv_segments ( lv , 0 ) ) {
2002-11-18 14:04:08 +00:00
stack ;
return 0 ;
}
/*
* Merge segments in case someones been editing things by hand .
*/
if ( ! lv_merge_segments ( lv ) ) {
stack ;
return 0 ;
}
return 1 ;
}
2007-08-22 14:38:18 +00:00
static int _read_lvnames ( struct format_instance * fid __attribute ( ( unused ) ) ,
struct dm_pool * mem ,
2003-04-24 22:23:24 +00:00
struct volume_group * vg , struct config_node * lvn ,
2007-08-22 14:38:18 +00:00
struct config_node * vgn __attribute ( ( unused ) ) ,
struct dm_hash_table * pv_hash __attribute ( ( unused ) ) )
2002-11-18 14:04:08 +00:00
{
struct logical_volume * lv ;
struct lv_list * lvl ;
struct config_node * cn ;
2005-10-16 23:03:59 +00:00
if ( ! ( lvl = dm_pool_zalloc ( mem , sizeof ( * lvl ) ) ) | |
! ( lvl - > lv = dm_pool_zalloc ( mem , sizeof ( * lvl - > lv ) ) ) ) {
2002-11-18 14:04:08 +00:00
stack ;
return 0 ;
}
lv = lvl - > lv ;
2005-10-16 23:03:59 +00:00
if ( ! ( lv - > name = dm_pool_strdup ( mem , lvn - > key ) ) ) {
2002-11-18 14:04:08 +00:00
stack ;
return 0 ;
}
if ( ! ( lvn = lvn - > child ) ) {
log_error ( " Empty logical volume section. " ) ;
return 0 ;
}
2004-03-08 18:28:45 +00:00
if ( ! ( cn = find_config_node ( lvn , " status " ) ) ) {
2002-11-18 14:04:08 +00:00
log_error ( " Couldn't find status flags for logical volume. " ) ;
return 0 ;
}
if ( ! ( read_flags ( & lv - > status , LV_FLAGS , cn - > v ) ) ) {
log_error ( " Couldn't read status flags for logical volume. " ) ;
return 0 ;
}
2004-05-18 22:12:53 +00:00
lv - > alloc = ALLOC_INHERIT ;
2004-03-08 18:28:45 +00:00
if ( ( cn = find_config_node ( lvn , " allocation_policy " ) ) ) {
2002-11-18 14:04:08 +00:00
struct config_value * cv = cn - > v ;
if ( ! cv | | ! cv - > v . str ) {
log_error ( " allocation_policy must be a string. " ) ;
return 0 ;
}
lv - > alloc = get_alloc_from_string ( cv - > v . str ) ;
2004-05-18 22:12:53 +00:00
if ( lv - > alloc = = ALLOC_INVALID ) {
stack ;
return 0 ;
}
2002-11-18 14:04:08 +00:00
}
/* read_ahead defaults to 0 */
if ( ! _read_int32 ( lvn , " read_ahead " , & lv - > read_ahead ) )
lv - > read_ahead = 0 ;
2005-04-07 12:39:44 +00:00
lv - > snapshot = NULL ;
list_init ( & lv - > snapshot_segs ) ;
2003-04-24 22:23:24 +00:00
list_init ( & lv - > segments ) ;
2004-03-08 17:19:15 +00:00
list_init ( & lv - > tags ) ;
/* Optional tags */
2004-03-08 18:28:45 +00:00
if ( ( cn = find_config_node ( lvn , " tags " ) ) & &
2004-03-08 17:19:15 +00:00
! ( read_tags ( mem , & lv - > tags , cn - > v ) ) ) {
log_error ( " Couldn't read tags for logical volume %s/%s. " ,
vg - > name , lv - > name ) ;
return 0 ;
}
2003-04-24 22:23:24 +00:00
lv - > vg = vg ;
vg - > lv_count + + ;
list_add ( & vg - > lvs , & lvl - > list ) ;
return 1 ;
}
2007-08-22 14:38:18 +00:00
static int _read_lvsegs ( struct format_instance * fid __attribute ( ( unused ) ) ,
struct dm_pool * mem ,
2003-04-24 22:23:24 +00:00
struct volume_group * vg , struct config_node * lvn ,
2007-08-22 14:38:18 +00:00
struct config_node * vgn __attribute ( ( unused ) ) ,
struct dm_hash_table * pv_hash )
2003-04-24 22:23:24 +00:00
{
struct logical_volume * lv ;
struct lv_list * lvl ;
if ( ! ( lvl = find_lv_in_vg ( vg , lvn - > key ) ) ) {
log_error ( " Lost logical volume reference %s " , lvn - > key ) ;
return 0 ;
}
lv = lvl - > lv ;
if ( ! ( lvn = lvn - > child ) ) {
log_error ( " Empty logical volume section. " ) ;
return 0 ;
}
/* FIXME: read full lvid */
if ( ! _read_id ( & lv - > lvid . id [ 1 ] , lvn , " id " ) ) {
log_error ( " Couldn't read uuid for logical volume %s. " ,
lv - > name ) ;
return 0 ;
}
memcpy ( & lv - > lvid . id [ 0 ] , & lv - > vg - > id , sizeof ( lv - > lvid . id [ 0 ] ) ) ;
if ( ! _read_segments ( mem , vg , lv , lvn , pv_hash ) ) {
stack ;
return 0 ;
}
2002-11-18 14:04:08 +00:00
lv - > size = ( uint64_t ) lv - > le_count * ( uint64_t ) vg - > extent_size ;
2005-04-07 12:39:44 +00:00
/*
* FIXME We now have 2 LVs for each snapshot . The real one was
* created by vg_add_snapshot from the segment text_import .
*/
if ( lv - > status & SNAPSHOT ) {
2003-04-24 22:23:24 +00:00
vg - > lv_count - - ;
list_del ( & lvl - > list ) ;
2005-04-07 12:39:44 +00:00
return 1 ;
}
lv - > minor = - 1 ;
if ( ( lv - > status & FIXED_MINOR ) & &
! _read_int32 ( lvn , " minor " , & lv - > minor ) ) {
log_error ( " Couldn't read minor number for logical "
" volume %s. " , lv - > name ) ;
return 0 ;
}
lv - > major = - 1 ;
if ( ( lv - > status & FIXED_MINOR ) & &
! _read_int32 ( lvn , " major " , & lv - > major ) ) {
log_error ( " Couldn't read major number for logical "
" volume %s. " , lv - > name ) ;
2002-11-18 14:04:08 +00:00
}
return 1 ;
}
static int _read_sections ( struct format_instance * fid ,
const char * section , section_fn fn ,
2005-10-16 23:03:59 +00:00
struct dm_pool * mem ,
2002-11-18 14:04:08 +00:00
struct volume_group * vg , struct config_node * vgn ,
2005-10-16 23:03:59 +00:00
struct dm_hash_table * pv_hash , int optional )
2002-11-18 14:04:08 +00:00
{
struct config_node * n ;
2004-03-08 18:28:45 +00:00
if ( ! ( n = find_config_node ( vgn , section ) ) ) {
2002-11-18 14:04:08 +00:00
if ( ! optional ) {
log_error ( " Couldn't find section '%s'. " , section ) ;
return 0 ;
}
return 1 ;
}
for ( n = n - > child ; n ; n = n - > sib ) {
if ( ! fn ( fid , mem , vg , n , vgn , pv_hash ) ) {
stack ;
return 0 ;
}
}
return 1 ;
}
static struct volume_group * _read_vg ( struct format_instance * fid ,
2004-03-08 18:28:45 +00:00
struct config_tree * cft )
2002-11-18 14:04:08 +00:00
{
struct config_node * vgn , * cn ;
struct volume_group * vg ;
2005-10-16 23:03:59 +00:00
struct dm_hash_table * pv_hash = NULL ;
struct dm_pool * mem = fid - > fmt - > cmd - > mem ;
2002-11-18 14:04:08 +00:00
/* skip any top-level values */
2004-03-08 18:28:45 +00:00
for ( vgn = cft - > root ; ( vgn & & vgn - > v ) ; vgn = vgn - > sib ) ;
2002-11-18 14:04:08 +00:00
if ( ! vgn ) {
log_error ( " Couldn't find volume group in file. " ) ;
return NULL ;
}
2005-10-16 23:03:59 +00:00
if ( ! ( vg = dm_pool_zalloc ( mem , sizeof ( * vg ) ) ) ) {
2002-11-18 14:04:08 +00:00
stack ;
return NULL ;
}
vg - > cmd = fid - > fmt - > cmd ;
/* FIXME Determine format type from file contents */
/* eg Set to instance of fmt1 here if reading a format1 backup? */
vg - > fid = fid ;
2007-04-26 16:44:59 +00:00
if ( ! ( vg - > name = dm_pool_strdup ( mem , vgn - > key ) ) )
goto_bad ;
2002-11-18 14:04:08 +00:00
2007-04-26 16:44:59 +00:00
if ( ! ( vg - > system_id = dm_pool_zalloc ( mem , NAME_LEN ) ) )
goto_bad ;
2002-11-18 14:04:08 +00:00
vgn = vgn - > child ;
2004-03-08 18:28:45 +00:00
if ( ( cn = find_config_node ( vgn , " system_id " ) ) & & cn - > v ) {
2002-11-18 14:04:08 +00:00
if ( ! cn - > v - > v . str ) {
log_error ( " system_id must be a string " ) ;
goto bad ;
}
strncpy ( vg - > system_id , cn - > v - > v . str , NAME_LEN ) ;
}
if ( ! _read_id ( & vg - > id , vgn , " id " ) ) {
log_error ( " Couldn't read uuid for volume group %s. " , vg - > name ) ;
goto bad ;
}
if ( ! _read_int32 ( vgn , " seqno " , & vg - > seqno ) ) {
log_error ( " Couldn't read 'seqno' for volume group %s. " ,
vg - > name ) ;
goto bad ;
}
2004-03-08 18:28:45 +00:00
if ( ! ( cn = find_config_node ( vgn , " status " ) ) ) {
2002-11-18 14:04:08 +00:00
log_error ( " Couldn't find status flags for volume group %s. " ,
vg - > name ) ;
goto bad ;
}
if ( ! ( read_flags ( & vg - > status , VG_FLAGS , cn - > v ) ) ) {
log_error ( " Couldn't read status flags for volume group %s. " ,
vg - > name ) ;
goto bad ;
}
if ( ! _read_int32 ( vgn , " extent_size " , & vg - > extent_size ) ) {
log_error ( " Couldn't read extent size for volume group %s. " ,
vg - > name ) ;
goto bad ;
}
/*
* ' extent_count ' and ' free_count ' get filled in
* implicitly when reading in the pv ' s and lv ' s .
*/
if ( ! _read_int32 ( vgn , " max_lv " , & vg - > max_lv ) ) {
log_error ( " Couldn't read 'max_lv' for volume group %s. " ,
vg - > name ) ;
goto bad ;
}
if ( ! _read_int32 ( vgn , " max_pv " , & vg - > max_pv ) ) {
log_error ( " Couldn't read 'max_pv' for volume group %s. " ,
vg - > name ) ;
goto bad ;
}
2004-05-18 22:12:53 +00:00
vg - > alloc = ALLOC_NORMAL ;
if ( ( cn = find_config_node ( vgn , " allocation_policy " ) ) ) {
struct config_value * cv = cn - > v ;
if ( ! cv | | ! cv - > v . str ) {
log_error ( " allocation_policy must be a string. " ) ;
return 0 ;
}
vg - > alloc = get_alloc_from_string ( cv - > v . str ) ;
if ( vg - > alloc = = ALLOC_INVALID ) {
stack ;
return 0 ;
}
}
2002-11-18 14:04:08 +00:00
/*
* The pv hash memoises the pv section names - > pv
* structures .
*/
2005-10-16 23:03:59 +00:00
if ( ! ( pv_hash = dm_hash_create ( 32 ) ) ) {
2002-11-18 14:04:08 +00:00
log_error ( " Couldn't create hash table. " ) ;
goto bad ;
}
list_init ( & vg - > pvs ) ;
if ( ! _read_sections ( fid , " physical_volumes " , _read_pv , mem , vg ,
vgn , pv_hash , 0 ) ) {
log_error ( " Couldn't find all physical volumes for volume "
" group %s. " , vg - > name ) ;
goto bad ;
}
list_init ( & vg - > lvs ) ;
2004-03-08 17:19:15 +00:00
list_init ( & vg - > tags ) ;
/* Optional tags */
2004-03-08 18:28:45 +00:00
if ( ( cn = find_config_node ( vgn , " tags " ) ) & &
2004-03-08 17:19:15 +00:00
! ( read_tags ( mem , & vg - > tags , cn - > v ) ) ) {
log_error ( " Couldn't read tags for volume group %s. " , vg - > name ) ;
goto bad ;
}
2003-04-24 22:23:24 +00:00
if ( ! _read_sections ( fid , " logical_volumes " , _read_lvnames , mem , vg ,
2002-11-18 14:04:08 +00:00
vgn , pv_hash , 1 ) ) {
2003-04-24 22:23:24 +00:00
log_error ( " Couldn't read all logical volume names for volume "
2002-11-18 14:04:08 +00:00
" group %s. " , vg - > name ) ;
goto bad ;
}
2003-04-24 22:23:24 +00:00
if ( ! _read_sections ( fid , " logical_volumes " , _read_lvsegs , mem , vg ,
vgn , pv_hash , 1 ) ) {
log_error ( " Couldn't read all logical volumes for "
" volume group %s. " , vg - > name ) ;
goto bad ;
}
2005-10-27 21:51:28 +00:00
if ( ! fixup_imported_mirrors ( vg ) ) {
log_error ( " Failed to fixup mirror pointers after import for "
" volume group %s. " , vg - > name ) ;
goto bad ;
}
2005-10-16 23:03:59 +00:00
dm_hash_destroy ( pv_hash ) ;
2002-11-18 14:04:08 +00:00
if ( vg - > status & PARTIAL_VG ) {
vg - > status & = ~ LVM_WRITE ;
vg - > status | = LVM_READ ;
}
/*
* Finished .
*/
return vg ;
bad :
if ( pv_hash )
2005-10-16 23:03:59 +00:00
dm_hash_destroy ( pv_hash ) ;
2002-11-18 14:04:08 +00:00
2005-10-16 23:03:59 +00:00
dm_pool_free ( mem , vg ) ;
2002-11-18 14:04:08 +00:00
return NULL ;
}
2005-10-16 23:03:59 +00:00
static void _read_desc ( struct dm_pool * mem ,
2004-03-08 18:28:45 +00:00
struct config_tree * cft , time_t * when , char * * desc )
2002-11-18 14:04:08 +00:00
{
const char * d ;
unsigned int u = 0u ;
log_suppress ( 1 ) ;
2004-03-08 18:28:45 +00:00
d = find_config_str ( cft - > root , " description " , " " ) ;
2002-11-18 14:04:08 +00:00
log_suppress ( 0 ) ;
2005-10-16 23:03:59 +00:00
* desc = dm_pool_strdup ( mem , d ) ;
2002-11-18 14:04:08 +00:00
2004-03-08 18:28:45 +00:00
get_config_uint32 ( cft - > root , " creation_time " , & u ) ;
2002-11-18 14:04:08 +00:00
* when = u ;
}
2006-04-11 13:55:59 +00:00
static const char * _read_vgname ( const struct format_type * fmt ,
2006-04-11 17:42:15 +00:00
struct config_tree * cft , struct id * vgid ,
2006-04-13 17:32:24 +00:00
uint32_t * vgstatus , char * * creation_host )
2006-04-11 13:55:59 +00:00
{
2006-04-11 17:42:15 +00:00
struct config_node * vgn , * cn ;
2006-04-11 13:55:59 +00:00
struct dm_pool * mem = fmt - > cmd - > mem ;
char * vgname ;
2006-04-13 17:32:24 +00:00
int old_suppress ;
old_suppress = log_suppress ( 2 ) ;
* creation_host = dm_pool_strdup ( mem ,
find_config_str ( cft - > root ,
" creation_host " , " " ) ) ;
log_suppress ( old_suppress ) ;
2006-04-11 13:55:59 +00:00
/* skip any top-level values */
for ( vgn = cft - > root ; ( vgn & & vgn - > v ) ; vgn = vgn - > sib ) ;
if ( ! vgn ) {
log_error ( " Couldn't find volume group in file. " ) ;
return 0 ;
}
if ( ! ( vgname = dm_pool_strdup ( mem , vgn - > key ) ) )
return_0 ;
vgn = vgn - > child ;
if ( ! _read_id ( vgid , vgn , " id " ) ) {
log_error ( " Couldn't read uuid for volume group %s. " , vgname ) ;
return 0 ;
}
2006-04-11 17:42:15 +00:00
if ( ! ( cn = find_config_node ( vgn , " status " ) ) ) {
log_error ( " Couldn't find status flags for volume group %s. " ,
vgname ) ;
return 0 ;
}
if ( ! ( read_flags ( vgstatus , VG_FLAGS , cn - > v ) ) ) {
log_error ( " Couldn't read status flags for volume group %s. " ,
vgname ) ;
return 0 ;
}
2006-04-11 13:55:59 +00:00
return vgname ;
}
2002-11-18 14:04:08 +00:00
static struct text_vg_version_ops _vsn1_ops = {
2006-05-09 21:23:51 +00:00
. check_version = _check_version ,
. read_vg = _read_vg ,
. read_desc = _read_desc ,
. read_vgname = _read_vgname ,
2002-11-18 14:04:08 +00:00
} ;
struct text_vg_version_ops * text_vg_vsn1_init ( void )
{
return & _vsn1_ops ;
2006-05-09 21:23:51 +00:00
}