2001-10-04 17:48:55 +00:00
/*
* Copyright ( C ) 2001 Sistina Software ( UK ) Limited .
*
2001-10-31 12:47:01 +00:00
* This file is released under the LGPL .
2001-10-04 17:48:55 +00:00
*/
# include "disk-rep.h"
# include "dbg_malloc.h"
# include "pool.h"
# include "hash.h"
2002-02-13 21:28:56 +00:00
# include "limits.h"
2001-10-04 17:48:55 +00:00
# include "list.h"
2001-10-08 09:45:16 +00:00
# include "log.h"
2001-10-15 20:29:15 +00:00
# include "display.h"
2002-02-11 20:50:53 +00:00
# include "toolcontext.h"
2001-10-04 17:48:55 +00:00
2001-10-16 16:25:28 +00:00
/* VG consistency checks */
2002-01-29 17:23:33 +00:00
static int _check_vgs ( struct list * pvs , int * partial )
2001-10-04 17:48:55 +00:00
{
2002-01-29 17:23:33 +00:00
struct list * pvh , * t ;
2001-10-25 13:08:29 +00:00
struct disk_list * dl = NULL ;
2001-10-16 16:25:28 +00:00
struct disk_list * first = NULL ;
int pv_count = 0 ;
2002-01-29 17:23:33 +00:00
int exported = - 1 ;
* partial = 0 ;
2001-10-04 17:48:55 +00:00
2002-01-29 17:23:33 +00:00
/*
* If there are exported and unexported PVs , ignore exported ones .
* This means an active VG won ' t be affected if disks are inserted
* bearing an exported VG with the same name .
2002-02-12 16:31:31 +00:00
*/
2001-10-31 12:47:01 +00:00
list_iterate ( pvh , pvs ) {
dl = list_item ( pvh , struct disk_list ) ;
2001-10-04 17:48:55 +00:00
2002-01-29 17:23:33 +00:00
if ( exported < 0 ) {
exported = dl - > pvd . pv_status & VG_EXPORTED ;
continue ;
}
if ( exported ! = ( dl - > pvd . pv_status & VG_EXPORTED ) ) {
/* Remove exported PVs */
list_iterate_safe ( pvh , t , pvs ) {
dl = list_item ( pvh , struct disk_list ) ;
if ( dl - > pvd . pv_status & VG_EXPORTED )
list_del ( pvh ) ;
}
break ;
}
}
/* Remove any PVs with VG structs that differ from the first */
list_iterate_safe ( pvh , t , pvs ) {
dl = list_item ( pvh , struct disk_list ) ;
2001-10-09 14:26:45 +00:00
if ( ! first )
2001-10-16 16:25:28 +00:00
first = dl ;
2001-11-12 12:16:57 +00:00
2001-10-31 17:59:52 +00:00
else if ( memcmp ( & first - > vgd , & dl - > vgd , sizeof ( first - > vgd ) ) ) {
2002-01-27 21:30:47 +00:00
log_error ( " VG data differs between PVs %s and %s " ,
dev_name ( first - > dev ) , dev_name ( dl - > dev ) ) ;
2002-01-29 17:23:33 +00:00
list_del ( pvh ) ;
if ( partial_mode ( ) ) {
* partial = 1 ;
continue ;
}
2001-10-04 17:48:55 +00:00
return 0 ;
}
2001-10-16 16:25:28 +00:00
pv_count + + ;
}
/* On entry to fn, list known to be non-empty */
2002-01-29 17:23:33 +00:00
if ( pv_count ! = dl - > vgd . pv_cur ) {
log_error ( " %d PV(s) found for VG %s: expected %d " ,
pv_count , dl - > pvd . vg_name , dl - > vgd . pv_cur ) ;
if ( ! partial_mode ( ) )
return 0 ;
* partial = 1 ;
2001-10-04 17:48:55 +00:00
}
return 1 ;
}
2002-04-24 18:20:51 +00:00
static struct volume_group * _build_vg ( struct format_instance * fid ,
2002-02-13 11:43:29 +00:00
struct list * pvs )
2001-10-04 17:48:55 +00:00
{
2002-04-24 18:20:51 +00:00
struct pool * mem = fid - > fmt - > cmd - > mem ;
2001-10-04 17:48:55 +00:00
struct volume_group * vg = pool_alloc ( mem , sizeof ( * vg ) ) ;
2001-10-09 17:09:46 +00:00
struct disk_list * dl ;
2002-01-29 17:23:33 +00:00
int partial ;
2001-10-09 17:09:46 +00:00
2001-10-15 18:39:40 +00:00
if ( ! vg )
goto bad ;
2001-10-04 17:48:55 +00:00
2001-10-15 18:39:40 +00:00
if ( list_empty ( pvs ) )
goto bad ;
2001-10-09 17:09:46 +00:00
2001-10-04 17:48:55 +00:00
memset ( vg , 0 , sizeof ( * vg ) ) ;
2002-04-24 18:20:51 +00:00
vg - > cmd = fid - > fmt - > cmd ;
vg - > fid = fid ;
vg - > seqno = 0 ;
2001-10-31 12:47:01 +00:00
list_init ( & vg - > pvs ) ;
list_init ( & vg - > lvs ) ;
2002-02-13 11:43:29 +00:00
list_init ( & vg - > snapshots ) ;
2001-10-08 16:08:16 +00:00
2002-01-29 17:23:33 +00:00
if ( ! _check_vgs ( pvs , & partial ) )
2001-10-15 18:39:40 +00:00
goto bad ;
2001-10-09 14:26:45 +00:00
2002-01-29 17:23:33 +00:00
dl = list_item ( pvs - > n , struct disk_list ) ;
if ( ! import_vg ( mem , vg , dl , partial ) )
2001-10-04 17:48:55 +00:00
goto bad ;
2002-04-24 18:20:51 +00:00
if ( ! import_pvs ( fid , mem , vg , pvs , & vg - > pvs , & vg - > pv_count ) )
2001-10-04 17:48:55 +00:00
goto bad ;
2001-10-09 14:26:45 +00:00
if ( ! import_lvs ( mem , vg , pvs ) )
2001-10-04 17:48:55 +00:00
goto bad ;
2001-10-09 14:26:45 +00:00
if ( ! import_extents ( mem , vg , pvs ) )
2001-10-08 09:45:16 +00:00
goto bad ;
2002-02-13 11:43:29 +00:00
if ( ! import_snapshots ( mem , vg , pvs ) )
goto bad ;
2001-10-04 17:48:55 +00:00
return vg ;
2002-04-24 18:20:51 +00:00
bad :
2001-10-04 17:48:55 +00:00
stack ;
pool_free ( mem , vg ) ;
return NULL ;
}
2002-04-24 18:20:51 +00:00
static struct volume_group * _vg_read ( struct format_instance * fid ,
const char * vg_name , void * mda )
2001-10-04 17:48:55 +00:00
{
struct pool * mem = pool_create ( 1024 * 10 ) ;
2001-10-31 12:47:01 +00:00
struct list pvs ;
2001-11-12 15:10:01 +00:00
struct volume_group * vg = NULL ;
2001-10-31 12:47:01 +00:00
list_init ( & pvs ) ;
2001-10-04 17:48:55 +00:00
if ( ! mem ) {
stack ;
return NULL ;
}
2002-04-24 18:20:51 +00:00
/* Strip dev_dir if present */
vg_name = strip_dir ( vg_name , fid - > fmt - > cmd - > dev_dir ) ;
2001-10-11 21:35:55 +00:00
2002-04-24 18:20:51 +00:00
if ( ! read_pvs_in_vg
( fid - > fmt , vg_name , fid - > fmt - > cmd - > filter , mem , & pvs ) ) {
2001-10-04 17:48:55 +00:00
stack ;
2001-11-12 15:10:01 +00:00
goto bad ;
2001-10-04 17:48:55 +00:00
}
2002-04-24 18:20:51 +00:00
if ( ! ( vg = _build_vg ( fid , & pvs ) ) ) {
2001-10-04 17:48:55 +00:00
stack ;
2001-11-12 15:10:01 +00:00
goto bad ;
}
2002-04-24 18:20:51 +00:00
bad :
2001-10-04 17:48:55 +00:00
pool_destroy ( mem ) ;
return vg ;
}
2001-10-08 09:45:16 +00:00
static struct disk_list * _flatten_pv ( struct pool * mem , struct volume_group * vg ,
2001-10-09 14:26:45 +00:00
struct physical_volume * pv ,
2001-11-14 13:52:38 +00:00
const char * dev_dir )
2001-10-05 16:36:53 +00:00
{
2001-10-09 10:47:52 +00:00
struct disk_list * dl = pool_alloc ( mem , sizeof ( * dl ) ) ;
2001-10-08 12:11:33 +00:00
2001-10-09 10:47:52 +00:00
if ( ! dl ) {
stack ;
return NULL ;
}
dl - > mem = mem ;
dl - > dev = pv - > dev ;
2001-10-31 12:47:01 +00:00
list_init ( & dl - > uuids ) ;
2001-10-31 17:59:52 +00:00
list_init ( & dl - > lvds ) ;
2001-10-09 10:47:52 +00:00
2002-01-29 17:23:33 +00:00
if ( ! export_pv ( mem , vg , & dl - > pvd , pv ) | |
2001-10-31 17:59:52 +00:00
! export_vg ( & dl - > vgd , vg ) | |
2001-10-09 14:26:45 +00:00
! export_uuids ( dl , vg ) | |
2001-11-14 13:52:38 +00:00
! export_lvs ( dl , vg , pv , dev_dir ) | |
2001-10-10 09:25:04 +00:00
! calculate_layout ( dl ) ) {
2001-10-09 10:47:52 +00:00
stack ;
2001-10-15 18:39:40 +00:00
pool_free ( mem , dl ) ;
2001-10-09 10:47:52 +00:00
return NULL ;
}
return dl ;
2001-10-05 16:36:53 +00:00
}
2002-04-24 18:20:51 +00:00
static int _flatten_vg ( struct format_instance * fid , struct pool * mem ,
struct volume_group * vg ,
2001-11-14 13:52:38 +00:00
struct list * pvds , const char * dev_dir ,
2001-10-11 14:10:18 +00:00
struct dev_filter * filter )
2001-10-05 16:36:53 +00:00
{
2001-10-31 12:47:01 +00:00
struct list * pvh ;
2001-10-09 10:47:52 +00:00
struct pv_list * pvl ;
2001-10-05 16:36:53 +00:00
struct disk_list * data ;
2001-10-31 12:47:01 +00:00
list_iterate ( pvh , & vg - > pvs ) {
pvl = list_item ( pvh , struct pv_list ) ;
2001-10-05 16:36:53 +00:00
2002-01-21 16:05:23 +00:00
if ( ! ( data = _flatten_pv ( mem , vg , pvl - > pv , dev_dir ) ) ) {
2001-10-05 16:36:53 +00:00
stack ;
return 0 ;
}
2001-11-14 13:52:38 +00:00
list_add ( pvds , & data - > list ) ;
2001-10-05 16:36:53 +00:00
}
2001-10-11 10:55:19 +00:00
2001-11-14 13:52:38 +00:00
export_numbers ( pvds , vg ) ;
export_pv_act ( pvds ) ;
2001-10-11 13:05:55 +00:00
2002-04-24 18:20:51 +00:00
if ( ! export_vg_number ( fid , pvds , vg - > name , filter ) ) {
2001-10-11 14:10:18 +00:00
stack ;
return 0 ;
}
2001-10-05 16:36:53 +00:00
return 1 ;
}
2002-04-24 18:20:51 +00:00
static int _vg_write ( struct format_instance * fid , struct volume_group * vg ,
void * mdl )
2001-10-05 16:36:53 +00:00
{
struct pool * mem = pool_create ( 1024 * 10 ) ;
2001-11-14 13:52:38 +00:00
struct list pvds ;
2001-10-05 16:36:53 +00:00
int r = 0 ;
if ( ! mem ) {
stack ;
return 0 ;
}
2001-11-14 13:52:38 +00:00
list_init ( & pvds ) ;
2001-10-10 13:09:40 +00:00
2002-04-24 18:20:51 +00:00
r = ( _flatten_vg ( fid , mem , vg , & pvds , fid - > fmt - > cmd - > dev_dir ,
fid - > fmt - > cmd - > filter ) & &
write_disks ( fid - > fmt , & pvds ) ) ;
2001-10-05 16:36:53 +00:00
pool_destroy ( mem ) ;
return r ;
}
2001-10-04 17:48:55 +00:00
2002-04-24 18:20:51 +00:00
int _pv_read ( struct format_type * fmt , const char * name ,
struct physical_volume * pv )
2001-10-09 08:58:52 +00:00
{
struct pool * mem = pool_create ( 1024 ) ;
struct disk_list * dl ;
2001-10-09 16:05:34 +00:00
struct device * dev ;
2002-04-24 18:20:51 +00:00
int r = 0 ;
2001-10-09 08:58:52 +00:00
2002-04-24 18:20:51 +00:00
log_very_verbose ( " Reading physical volume data %s from disk " , name ) ;
2001-10-17 15:29:31 +00:00
2001-10-09 08:58:52 +00:00
if ( ! mem ) {
stack ;
2002-04-24 18:20:51 +00:00
return 0 ;
2001-10-09 08:58:52 +00:00
}
2002-04-24 18:20:51 +00:00
if ( ! ( dev = dev_cache_get ( name , fmt - > cmd - > filter ) ) ) {
2001-10-09 16:05:34 +00:00
stack ;
2001-11-12 12:16:57 +00:00
goto out ;
2001-10-09 16:05:34 +00:00
}
2002-04-24 18:20:51 +00:00
if ( ! ( dl = read_disk ( fmt , dev , mem , NULL ) ) ) {
2001-10-09 08:58:52 +00:00
stack ;
2001-11-12 12:16:57 +00:00
goto out ;
2001-10-09 08:58:52 +00:00
}
2002-04-24 18:20:51 +00:00
if ( ! import_pv ( fmt - > cmd - > mem , dl - > dev , NULL , pv , & dl - > pvd ) ) {
2001-10-09 08:58:52 +00:00
stack ;
2001-11-12 12:16:57 +00:00
goto out ;
2001-10-09 08:58:52 +00:00
}
2002-04-24 18:20:51 +00:00
pv - > fid = fmt - > ops - > create_instance ( fmt , NULL , NULL ) ;
r = 1 ;
2001-10-09 08:58:52 +00:00
2002-04-24 18:20:51 +00:00
out :
2001-10-09 08:58:52 +00:00
pool_destroy ( mem ) ;
2002-04-24 18:20:51 +00:00
return r ;
2001-10-09 08:58:52 +00:00
}
2002-04-24 18:20:51 +00:00
static struct list * _get_pvs ( struct format_type * fmt , struct list * results )
2001-10-08 17:53:43 +00:00
{
struct pool * mem = pool_create ( 1024 * 10 ) ;
2002-04-24 18:20:51 +00:00
struct list pvs ;
2001-10-08 17:53:43 +00:00
uint32_t count ;
if ( ! mem ) {
stack ;
return NULL ;
}
2001-10-31 12:47:01 +00:00
list_init ( & pvs ) ;
2001-10-08 17:53:43 +00:00
2002-04-24 18:20:51 +00:00
if ( ! read_pvs_in_vg ( fmt , NULL , fmt - > cmd - > filter , mem , & pvs ) ) {
2001-10-08 17:53:43 +00:00
stack ;
goto bad ;
}
2002-04-24 18:20:51 +00:00
if ( ! import_pvs ( NULL , fmt - > cmd - > mem , NULL , & pvs , results , & count ) ) {
2001-10-08 17:53:43 +00:00
stack ;
goto bad ;
}
pool_destroy ( mem ) ;
return results ;
2002-04-24 18:20:51 +00:00
bad :
2001-10-15 18:39:40 +00:00
pool_destroy ( mem ) ;
2001-10-08 17:53:43 +00:00
return NULL ;
}
2001-10-31 12:47:01 +00:00
static int _find_vg_name ( struct list * names , const char * vg )
2001-10-09 09:22:50 +00:00
{
2001-10-31 12:47:01 +00:00
struct list * nh ;
2001-10-09 09:22:50 +00:00
struct name_list * nl ;
2001-10-31 12:47:01 +00:00
list_iterate ( nh , names ) {
nl = list_item ( nh , struct name_list ) ;
2001-10-09 09:22:50 +00:00
if ( ! strcmp ( nl - > name , vg ) )
return 1 ;
}
return 0 ;
}
2002-04-24 18:20:51 +00:00
static struct list * _get_vgs ( struct format_type * fmt , struct list * names )
2001-10-09 09:22:50 +00:00
{
2001-10-31 12:47:01 +00:00
struct list * pvh ;
2002-04-24 18:20:51 +00:00
struct list * pvs ;
2001-10-09 09:22:50 +00:00
struct name_list * nl ;
2002-04-24 18:20:51 +00:00
if ( ! ( pvs = pool_alloc ( fmt - > cmd - > mem , sizeof ( * pvs ) ) ) ) {
log_error ( " PV list allocation failed " ) ;
goto err ;
2001-10-09 09:22:50 +00:00
}
2002-04-24 18:20:51 +00:00
list_init ( pvs ) ;
2001-10-09 09:22:50 +00:00
2002-04-24 18:20:51 +00:00
if ( ! _get_pvs ( fmt , pvs ) ) {
2001-10-09 09:22:50 +00:00
stack ;
2002-04-24 18:20:51 +00:00
goto err ;
2001-10-09 09:22:50 +00:00
}
2001-10-31 12:47:01 +00:00
list_iterate ( pvh , pvs ) {
struct pv_list * pvl = list_item ( pvh , struct pv_list ) ;
2001-10-09 09:22:50 +00:00
2002-01-21 16:05:23 +00:00
if ( ! ( * pvl - > pv - > vg_name ) | |
2002-04-24 18:20:51 +00:00
_find_vg_name ( names , pvl - > pv - > vg_name ) )
2001-10-09 09:22:50 +00:00
continue ;
2002-04-24 18:20:51 +00:00
if ( ! ( nl = pool_alloc ( fmt - > cmd - > mem , sizeof ( * nl ) ) ) ) {
2001-10-09 09:22:50 +00:00
stack ;
2002-04-24 18:20:51 +00:00
goto err ;
2001-10-09 09:22:50 +00:00
}
2002-04-24 18:20:51 +00:00
if ( ! ( nl - > name = pool_strdup ( fmt - > cmd - > mem , pvl - > pv - > vg_name ) ) ) {
2001-10-09 09:22:50 +00:00
stack ;
2002-04-24 18:20:51 +00:00
goto err ;
2001-10-09 09:22:50 +00:00
}
2001-10-31 12:47:01 +00:00
list_add ( names , & nl - > list ) ;
2001-10-09 09:22:50 +00:00
}
2001-10-16 16:25:28 +00:00
if ( list_empty ( names ) )
2002-04-30 12:27:13 +00:00
pool_free ( fmt - > cmd - > mem , pvs ) ;
2001-10-16 16:25:28 +00:00
2001-10-09 09:22:50 +00:00
return names ;
2002-04-24 18:20:51 +00:00
err :
pool_free ( fmt - > cmd - > mem , pvs ) ;
2001-10-09 09:22:50 +00:00
return NULL ;
}
2002-04-24 18:20:51 +00:00
static int _pv_setup ( struct format_instance * fid , struct physical_volume * pv ,
2001-10-10 09:25:04 +00:00
struct volume_group * vg )
{
2002-02-15 14:33:59 +00:00
/* setup operations for the PV structure */
2002-02-20 18:29:30 +00:00
if ( pv - > size > MAX_PV_SIZE )
pv - > size - - ;
2002-02-15 01:26:16 +00:00
if ( pv - > size > MAX_PV_SIZE ) {
2002-02-20 18:29:30 +00:00
/* FIXME Limit hardcoded */
log_error ( " Physical volumes cannot be bigger than 2TB " ) ;
2002-02-15 01:26:16 +00:00
return 0 ;
}
2002-02-20 18:29:30 +00:00
/* Nothing more to do if pe_size isn't known */
2002-04-24 18:20:51 +00:00
if ( ! vg )
2002-02-20 18:29:30 +00:00
return 1 ;
2002-02-15 14:33:59 +00:00
2001-10-10 10:55:55 +00:00
/*
* This works out pe_start and pe_count .
*/
if ( ! calculate_extent_count ( pv ) ) {
stack ;
return 0 ;
}
2001-10-10 09:25:04 +00:00
return 1 ;
}
2002-02-21 19:04:37 +00:00
static int _find_free_lvnum ( struct logical_volume * lv )
{
int lvnum_used [ MAX_LV ] ;
int i = 0 ;
struct list * lvh ;
struct lv_list * lvl ;
memset ( & lvnum_used , 0 , sizeof ( lvnum_used ) ) ;
list_iterate ( lvh , & lv - > vg - > lvs ) {
lvl = list_item ( lvh , struct lv_list ) ;
2002-03-05 20:03:09 +00:00
lvnum_used [ lvnum_from_lvid ( & lvl - > lv - > lvid ) ] = 1 ;
2002-02-21 19:04:37 +00:00
}
while ( lvnum_used [ i ] )
i + + ;
return i ;
}
2002-04-24 18:20:51 +00:00
static int _lv_setup ( struct format_instance * fid , struct logical_volume * lv )
2002-01-24 17:16:36 +00:00
{
2002-02-13 21:28:56 +00:00
uint64_t max_size = UINT_MAX ;
2002-04-05 14:32:22 +00:00
if ( ! * lv - > lvid . s )
lvid_from_lvnum ( & lv - > lvid , & lv - > vg - > id , _find_free_lvnum ( lv ) ) ;
2002-02-21 19:04:37 +00:00
2002-01-24 19:20:35 +00:00
if ( lv - > le_count > MAX_LE_TOTAL ) {
2002-01-27 21:30:47 +00:00
log_error ( " logical volumes cannot contain more than "
" %d extents. " , MAX_LE_TOTAL ) ;
2002-01-24 17:16:36 +00:00
return 0 ;
}
2002-02-13 21:28:56 +00:00
if ( lv - > size > max_size ) {
2002-04-24 18:20:51 +00:00
char * dummy = display_size ( max_size , SIZE_SHORT ) ;
2002-02-13 21:28:56 +00:00
log_error ( " logical volumes cannot be larger than %s " , dummy ) ;
dbg_free ( dummy ) ;
return 0 ;
}
2002-01-24 17:16:36 +00:00
return 1 ;
}
2002-04-24 18:20:51 +00:00
static int _pv_write ( struct format_instance * fid , struct physical_volume * pv ,
void * mdl )
2001-10-09 17:44:58 +00:00
{
2001-10-10 10:05:29 +00:00
struct pool * mem ;
struct disk_list * dl ;
2001-10-31 12:47:01 +00:00
struct list pvs ;
2001-10-10 10:05:29 +00:00
2001-10-31 12:47:01 +00:00
list_init ( & pvs ) ;
2001-10-10 10:05:29 +00:00
2002-04-24 18:20:51 +00:00
if ( * pv - > vg_name | | pv - > pe_alloc_count ) {
2001-10-31 12:47:01 +00:00
log_error ( " Assertion failed: can't _pv_write non-orphan PV "
2001-10-11 21:35:55 +00:00
" (in VG %s) " , pv - > vg_name ) ;
2001-10-10 10:05:29 +00:00
return 0 ;
}
2001-10-18 16:55:19 +00:00
/* Ensure any residual PE structure is gone */
2002-04-24 18:20:51 +00:00
pv - > pe_size = pv - > pe_count = 0 ;
pv - > pe_start = PE_ALIGN ;
2001-10-18 16:55:19 +00:00
2001-10-10 10:05:29 +00:00
if ( ! ( mem = pool_create ( 1024 ) ) ) {
stack ;
return 0 ;
}
if ( ! ( dl = pool_alloc ( mem , sizeof ( * dl ) ) ) ) {
stack ;
2001-10-15 18:39:40 +00:00
goto bad ;
2001-10-10 10:05:29 +00:00
}
dl - > mem = mem ;
dl - > dev = pv - > dev ;
2002-01-29 17:23:33 +00:00
if ( ! export_pv ( mem , NULL , & dl - > pvd , pv ) ) {
2001-10-10 10:05:29 +00:00
stack ;
goto bad ;
}
2002-01-29 15:52:11 +00:00
/* must be set to be able to zero gap after PV structure in
dev_write in order to make other disk tools happy */
dl - > pvd . pv_on_disk . base = METADATA_BASE ;
dl - > pvd . pv_on_disk . size = PV_SIZE ;
2002-04-24 18:20:51 +00:00
dl - > pvd . pe_on_disk . base = PE_ALIGN * SECTOR_SIZE ;
2002-01-29 15:52:11 +00:00
2001-10-31 12:47:01 +00:00
list_add ( & pvs , & dl - > list ) ;
2002-04-24 18:20:51 +00:00
if ( ! write_disks ( fid - > fmt , & pvs ) ) {
2001-10-10 10:05:29 +00:00
stack ;
goto bad ;
}
pool_destroy ( mem ) ;
2001-10-09 17:44:58 +00:00
return 1 ;
2001-10-10 10:05:29 +00:00
2002-04-24 18:20:51 +00:00
bad :
2001-10-10 10:05:29 +00:00
pool_destroy ( mem ) ;
return 0 ;
2001-10-09 17:44:58 +00:00
}
2002-04-24 18:20:51 +00:00
int _vg_setup ( struct format_instance * fid , struct volume_group * vg )
2001-10-12 14:25:53 +00:00
{
/* just check max_pv and max_lv */
2001-10-15 18:39:40 +00:00
if ( vg - > max_lv > = MAX_LV )
vg - > max_lv = MAX_LV - 1 ;
2001-10-12 14:25:53 +00:00
2002-04-24 18:20:51 +00:00
if ( vg - > max_pv > = MAX_PV )
2001-10-15 18:39:40 +00:00
vg - > max_pv = MAX_PV - 1 ;
2001-10-12 14:25:53 +00:00
2001-10-15 20:29:15 +00:00
if ( vg - > extent_size > MAX_PE_SIZE | | vg - > extent_size < MIN_PE_SIZE ) {
char * dummy , * dummy2 ;
log_error ( " Extent size must be between %s and %s " ,
2002-04-24 18:20:51 +00:00
( dummy = display_size ( MIN_PE_SIZE / 2 , SIZE_SHORT ) ) ,
( dummy2 = display_size ( MAX_PE_SIZE / 2 , SIZE_SHORT ) ) ) ;
2001-10-15 20:29:15 +00:00
dbg_free ( dummy ) ;
dbg_free ( dummy2 ) ;
return 0 ;
}
if ( vg - > extent_size % MIN_PE_SIZE ) {
char * dummy ;
log_error ( " Extent size must be multiple of %s " ,
2002-04-24 18:20:51 +00:00
( dummy = display_size ( MIN_PE_SIZE / 2 , SIZE_SHORT ) ) ) ;
2001-10-15 20:29:15 +00:00
dbg_free ( dummy ) ;
return 0 ;
}
/* Redundant? */
if ( vg - > extent_size & ( vg - > extent_size - 1 ) ) {
log_error ( " Extent size must be power of 2 " ) ;
return 0 ;
}
2001-10-12 14:25:53 +00:00
return 1 ;
}
2001-10-09 17:44:58 +00:00
2002-04-24 18:20:51 +00:00
struct format_instance * _create_instance ( struct format_type * fmt ,
const char * vgname , void * private )
{
struct format_instance * fid ;
struct metadata_area * mda ;
if ( ! ( fid = pool_alloc ( fmt - > cmd - > mem , sizeof ( * fid ) ) ) ) {
stack ;
return NULL ;
}
fid - > fmt = fmt ;
list_init ( & fid - > metadata_areas ) ;
/* Define a NULL metadata area */
if ( ! ( mda = pool_alloc ( fmt - > cmd - > mem , sizeof ( * mda ) ) ) ) {
stack ;
pool_free ( fmt - > cmd - > mem , fid ) ;
return NULL ;
}
mda - > metadata_locn = NULL ;
list_add ( & fid - > metadata_areas , & mda - > list ) ;
return fid ;
}
void _destroy_instance ( struct format_instance * fid )
2001-10-04 17:48:55 +00:00
{
2002-04-24 18:20:51 +00:00
return ;
2001-10-08 12:11:33 +00:00
}
2002-04-24 18:20:51 +00:00
void _destroy ( struct format_type * fmt )
{
dbg_free ( fmt ) ;
}
2001-10-08 12:11:33 +00:00
2001-11-12 12:16:57 +00:00
static struct format_handler _format1_ops = {
2002-04-24 18:20:51 +00:00
get_vgs : _get_vgs ,
get_pvs : _get_pvs ,
pv_read : _pv_read ,
pv_setup : _pv_setup ,
pv_write : _pv_write ,
lv_setup : _lv_setup ,
vg_read : _vg_read ,
vg_setup : _vg_setup ,
vg_write : _vg_write ,
create_instance : _create_instance ,
destroy_instance : _destroy_instance ,
destroy : _destroy ,
2001-11-12 12:16:57 +00:00
} ;
2001-10-30 17:53:21 +00:00
2002-04-24 18:20:51 +00:00
struct format_type * create_lvm1_format ( struct cmd_context * cmd )
2001-11-12 12:16:57 +00:00
{
2002-04-24 18:20:51 +00:00
struct format_type * fmt = dbg_malloc ( sizeof ( * fmt ) ) ;
2001-10-08 12:11:33 +00:00
2002-04-24 18:20:51 +00:00
if ( ! fmt ) {
2001-10-08 12:11:33 +00:00
stack ;
2001-11-12 12:16:57 +00:00
return NULL ;
2001-10-08 12:11:33 +00:00
}
2002-04-24 18:20:51 +00:00
fmt - > cmd = cmd ;
fmt - > ops = & _format1_ops ;
fmt - > name = FMT_LVM1_NAME ;
fmt - > features = 0 ;
fmt - > private = NULL ;
2001-10-08 12:11:33 +00:00
2002-04-24 18:20:51 +00:00
return fmt ;
2001-10-04 17:48:55 +00:00
}