2001-10-04 17:48:55 +00:00
/*
2008-01-30 14:00:02 +00:00
* Copyright ( C ) 2001 - 2004 Sistina Software , Inc . All rights reserved .
2012-02-08 10:49:36 +00:00
* Copyright ( C ) 2004 - 2012 Red Hat , Inc . All rights reserved .
2001-10-04 17:48:55 +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
2001-10-04 17:48:55 +00:00
*/
2002-11-18 14:01:16 +00:00
# include "lib.h"
2001-10-04 17:48:55 +00:00
# include "disk-rep.h"
2002-02-13 21:28:56 +00:00
# include "limits.h"
2001-10-15 20:29:15 +00:00
# include "display.h"
2002-02-11 20:50:53 +00:00
# include "toolcontext.h"
2002-11-18 14:01:16 +00:00
# include "lvm1-label.h"
# include "format1.h"
2004-09-16 18:40:56 +00:00
# include "segtype.h"
2010-09-22 13:45:21 +00:00
# include "pv_alloc.h"
2001-10-04 17:48:55 +00:00
2001-10-16 16:25:28 +00:00
/* VG consistency checks */
2010-09-22 13:45:21 +00:00
static int _check_vgs ( struct dm_list * pvs , struct volume_group * vg )
2001-10-04 17:48:55 +00:00
{
2008-11-03 22:14:30 +00:00
struct dm_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 ;
2002-12-19 23:25:55 +00:00
uint32_t pv_count = 0 ;
uint32_t exported = 0 ;
int first_time = 1 ;
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
*/
2008-11-03 22:14:30 +00:00
dm_list_iterate_items ( dl , pvs ) {
2002-12-19 23:25:55 +00:00
if ( first_time ) {
2002-01-29 17:23:33 +00:00
exported = dl - > pvd . pv_status & VG_EXPORTED ;
2002-12-19 23:25:55 +00:00
first_time = 0 ;
2002-01-29 17:23:33 +00:00
continue ;
}
if ( exported ! = ( dl - > pvd . pv_status & VG_EXPORTED ) ) {
/* Remove exported PVs */
2008-11-03 22:14:30 +00:00
dm_list_iterate_safe ( pvh , t , pvs ) {
dl = dm_list_item ( pvh , struct disk_list ) ;
2002-01-29 17:23:33 +00:00
if ( dl - > pvd . pv_status & VG_EXPORTED )
2008-11-03 22:14:30 +00:00
dm_list_del ( pvh ) ;
2002-01-29 17:23:33 +00:00
}
break ;
}
}
/* Remove any PVs with VG structs that differ from the first */
2008-11-03 22:14:30 +00:00
dm_list_iterate_safe ( pvh , t , pvs ) {
dl = dm_list_item ( pvh , struct disk_list ) ;
2002-01-29 17:23:33 +00:00
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 ) ) ;
2013-01-07 22:30:29 +00:00
log_debug_metadata ( " VG data on %s: %s %s % " PRIu32 " % " PRIu32
" % " PRIu32 " % " PRIu32 " % " PRIu32 " % "
PRIu32 " % " PRIu32 " % " PRIu32 " % " PRIu32
" % " PRIu32 " % " PRIu32 " % " PRIu32 " % "
PRIu32 " % " PRIu32 " % " PRIu32 ,
dev_name ( first - > dev ) , first - > vgd . vg_uuid ,
first - > vgd . vg_name_dummy ,
first - > vgd . vg_number , first - > vgd . vg_access ,
first - > vgd . vg_status , first - > vgd . lv_max ,
first - > vgd . lv_cur , first - > vgd . lv_open ,
first - > vgd . pv_max , first - > vgd . pv_cur ,
first - > vgd . pv_act , first - > vgd . dummy ,
first - > vgd . vgda , first - > vgd . pe_size ,
first - > vgd . pe_total , first - > vgd . pe_allocated ,
first - > vgd . pvg_total ) ;
log_debug_metadata ( " VG data on %s: %s %s % " PRIu32 " % " PRIu32
" % " PRIu32 " % " PRIu32 " % " PRIu32 " % "
PRIu32 " % " PRIu32 " % " PRIu32 " % " PRIu32
" % " PRIu32 " % " PRIu32 " % " PRIu32 " % "
PRIu32 " % " PRIu32 " % " PRIu32 ,
dev_name ( dl - > dev ) , dl - > vgd . vg_uuid ,
dl - > vgd . vg_name_dummy , dl - > vgd . vg_number ,
dl - > vgd . vg_access , dl - > vgd . vg_status ,
dl - > vgd . lv_max , dl - > vgd . lv_cur ,
dl - > vgd . lv_open , dl - > vgd . pv_max ,
dl - > vgd . pv_cur , dl - > vgd . pv_act , dl - > vgd . dummy ,
dl - > vgd . vgda , dl - > vgd . pe_size ,
dl - > vgd . pe_total , dl - > vgd . pe_allocated ,
dl - > vgd . pvg_total ) ;
2008-11-03 22:14:30 +00:00
dm_list_del ( pvh ) ;
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 */
2003-08-20 15:48:27 +00:00
if ( pv_count ! = first - > vgd . pv_cur ) {
2002-01-29 17:23:33 +00:00
log_error ( " %d PV(s) found for VG %s: expected %d " ,
2003-08-20 15:48:27 +00:00
pv_count , first - > pvd . vg_name , first - > vgd . pv_cur ) ;
2010-09-22 13:45:21 +00:00
vg - > status | = PARTIAL_VG ;
2001-10-04 17:48:55 +00:00
}
return 1 ;
}
2010-09-22 13:45:21 +00:00
static int _fix_partial_vg ( struct volume_group * vg , struct dm_list * pvs )
{
uint32_t extent_count = 0 ;
struct disk_list * dl ;
struct dm_list * pvh ;
struct pv_list * pvl ;
2010-10-04 18:59:01 +00:00
struct lv_list * ll ;
struct lv_segment * seg ;
/*
* FIXME : code should remap missing segments to error segment .
* Also current mapping code allocates 1 segment per missing extent .
* For now bail out completely - allocated structures are not complete
*/
dm_list_iterate_items ( ll , & vg - > lvs )
dm_list_iterate_items ( seg , & ll - > lv - > segments ) {
/* area_count is always 1 here, s == 0 */
if ( seg_type ( seg , 0 ) ! = AREA_PV )
continue ;
if ( seg_pv ( seg , 0 ) )
continue ;
log_error ( " Partial mode support for missing lvm1 PVs and "
" partially available LVs is currently not implemented. " ) ;
return 0 ;
}
2010-09-22 13:45:21 +00:00
dm_list_iterate ( pvh , pvs ) {
dl = dm_list_item ( pvh , struct disk_list ) ;
extent_count + = dl - > pvd . pe_total ;
}
/* FIXME: move this to one place to pv_manip */
if ( ! ( pvl = dm_pool_zalloc ( vg - > vgmem , sizeof ( * pvl ) ) ) | |
! ( pvl - > pv = dm_pool_zalloc ( vg - > vgmem , sizeof ( * pvl - > pv ) ) ) )
return_0 ;
2010-10-04 18:59:01 +00:00
/* Use vg uuid with replaced first chars to "missing" as missing PV UUID */
memcpy ( & pvl - > pv - > id . uuid , vg - > id . uuid , sizeof ( pvl - > pv - > id . uuid ) ) ;
memcpy ( & pvl - > pv - > id . uuid , " missing " , 7 ) ;
2010-09-22 13:45:21 +00:00
if ( ! ( pvl - > pv - > vg_name = dm_pool_strdup ( vg - > vgmem , vg - > name ) ) )
goto_out ;
memcpy ( & pvl - > pv - > vgid , & vg - > id , sizeof ( vg - > id ) ) ;
pvl - > pv - > status | = MISSING_PV ;
dm_list_init ( & pvl - > pv - > tags ) ;
dm_list_init ( & pvl - > pv - > segments ) ;
pvl - > pv - > pe_size = vg - > extent_size ;
pvl - > pv - > pe_count = vg - > extent_count - extent_count ;
if ( ! alloc_pv_segment_whole_pv ( vg - > vgmem , pvl - > pv ) )
goto_out ;
add_pvl_to_vgs ( vg , pvl ) ;
2013-01-07 22:30:29 +00:00
log_debug_metadata ( " %s: partial VG, allocated missing PV using %d extents. " ,
vg - > name , pvl - > pv - > pe_count ) ;
2010-09-22 13:45:21 +00:00
return 1 ;
out :
dm_pool_free ( vg - > vgmem , pvl ) ;
return 0 ;
}
2011-03-10 12:43:29 +00:00
static struct volume_group * _format1_vg_read ( struct format_instance * fid ,
const char * vg_name ,
2012-02-29 02:35:35 +00:00
struct metadata_area * mda __attribute__ ( ( unused ) ) ,
2015-03-18 23:43:02 +00:00
struct cached_vg_fmtdata * * vg_fmtdata __attribute__ ( ( unused ) ) ,
unsigned * use_previous_vg __attribute__ ( ( unused ) ) ,
2012-02-29 02:35:35 +00:00
int single_device __attribute__ ( ( unused ) ) )
2001-10-04 17:48:55 +00:00
{
2011-03-10 12:43:29 +00:00
struct volume_group * vg ;
2001-10-09 17:09:46 +00:00
struct disk_list * dl ;
2011-03-10 12:43:29 +00:00
DM_LIST_INIT ( pvs ) ;
/* Strip dev_dir if present */
2012-02-27 11:23:15 +00:00
if ( vg_name )
vg_name = strip_dir ( vg_name , fid - > fmt - > cmd - > dev_dir ) ;
2011-03-10 12:43:29 +00:00
if ( ! ( vg = alloc_vg ( " format1_vg_read " , fid - > fmt - > cmd , NULL ) ) )
return_NULL ;
2001-10-09 17:09:46 +00:00
2011-03-10 12:43:29 +00:00
if ( ! read_pvs_in_vg ( fid - > fmt , vg_name , fid - > fmt - > cmd - > filter ,
vg - > vgmem , & pvs ) )
2007-04-26 16:44:59 +00:00
goto_bad ;
2001-10-04 17:48:55 +00:00
2011-03-10 12:43:29 +00:00
if ( dm_list_empty ( & pvs ) )
2007-04-26 16:44:59 +00:00
goto_bad ;
2001-10-09 17:09:46 +00:00
2011-03-10 12:43:29 +00:00
if ( ! _check_vgs ( & pvs , vg ) )
2007-04-26 16:44:59 +00:00
goto_bad ;
2001-10-09 14:26:45 +00:00
2011-03-10 12:43:29 +00:00
dl = dm_list_item ( pvs . n , struct disk_list ) ;
2002-01-29 17:23:33 +00:00
2011-03-10 12:43:29 +00:00
if ( ! import_vg ( vg - > vgmem , vg , dl ) )
2007-04-26 16:44:59 +00:00
goto_bad ;
2001-10-04 17:48:55 +00:00
2011-03-10 12:43:29 +00:00
if ( ! import_pvs ( fid - > fmt , vg - > vgmem , vg , & pvs ) )
2007-04-26 16:44:59 +00:00
goto_bad ;
2001-10-04 17:48:55 +00:00
2011-03-10 12:43:29 +00:00
if ( ! import_lvs ( vg - > vgmem , vg , & pvs ) )
2007-04-26 16:44:59 +00:00
goto_bad ;
2001-10-04 17:48:55 +00:00
2011-03-10 12:43:29 +00:00
if ( ! import_extents ( fid - > fmt - > cmd , vg , & pvs ) )
2007-04-26 16:44:59 +00:00
goto_bad ;
2001-10-08 09:45:16 +00:00
2013-04-21 23:53:53 +02:00
/* FIXME: workaround - temporary assignment of fid */
vg - > fid = fid ;
if ( ! import_snapshots ( vg - > vgmem , vg , & pvs ) ) {
vg - > fid = NULL ;
2007-04-26 16:44:59 +00:00
goto_bad ;
2013-04-21 23:53:53 +02:00
}
vg - > fid = NULL ;
2002-02-13 11:43:29 +00:00
2010-09-22 13:45:21 +00:00
/* Fix extents counts by adding missing PV if partial VG */
2011-03-10 12:43:29 +00:00
if ( ( vg - > status & PARTIAL_VG ) & & ! _fix_partial_vg ( vg , & pvs ) )
2010-09-22 13:45:21 +00:00
goto_bad ;
2013-04-21 21:36:08 +02:00
vg_set_fid ( vg , fid ) ;
2001-10-04 17:48:55 +00:00
return vg ;
2009-04-10 09:59:18 +00:00
bad :
2012-02-13 10:53:31 +00:00
release_vg ( vg ) ;
2011-03-10 12:43:29 +00:00
2009-04-10 09:59:18 +00:00
return NULL ;
2001-10-04 17:48:55 +00:00
}
2004-03-08 17:25:59 +00:00
static struct disk_list * _flatten_pv ( struct format_instance * fid ,
2005-10-16 23:03:59 +00:00
struct dm_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
{
2005-10-16 23:03:59 +00:00
struct disk_list * dl = dm_pool_alloc ( mem , sizeof ( * dl ) ) ;
2001-10-08 12:11:33 +00:00
2008-01-30 13:19:47 +00:00
if ( ! dl )
return_NULL ;
2001-10-09 10:47:52 +00:00
dl - > mem = mem ;
dl - > dev = pv - > dev ;
2008-11-03 22:14:30 +00:00
dm_list_init ( & dl - > uuids ) ;
dm_list_init ( & dl - > lvds ) ;
2001-10-09 10:47:52 +00:00
2004-03-08 17:25:59 +00:00
if ( ! export_pv ( fid - > fmt - > cmd , 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 ) | |
2002-11-18 14:01:16 +00:00
! export_lvs ( dl , vg , pv , dev_dir ) | | ! calculate_layout ( dl ) ) {
2005-10-16 23:03:59 +00:00
dm_pool_free ( mem , dl ) ;
2008-01-30 13:19:47 +00:00
return_NULL ;
2001-10-09 10:47:52 +00:00
}
return dl ;
2001-10-05 16:36:53 +00:00
}
2005-10-16 23:03:59 +00:00
static int _flatten_vg ( struct format_instance * fid , struct dm_pool * mem ,
2002-04-24 18:20:51 +00:00
struct volume_group * vg ,
2008-11-03 22:14:30 +00:00
struct dm_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-09 10:47:52 +00:00
struct pv_list * pvl ;
2001-10-05 16:36:53 +00:00
struct disk_list * data ;
2008-11-03 22:14:30 +00:00
dm_list_iterate_items ( pvl , & vg - > pvs ) {
2008-01-30 13:19:47 +00:00
if ( ! ( data = _flatten_pv ( fid , mem , vg , pvl - > pv , dev_dir ) ) )
return_0 ;
2001-10-05 16:36:53 +00:00
2008-11-03 22:14:30 +00:00
dm_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
2008-01-30 13:19:47 +00:00
if ( ! export_vg_number ( fid , pvds , vg - > name , filter ) )
return_0 ;
2001-10-11 14:10:18 +00:00
2001-10-05 16:36:53 +00:00
return 1 ;
}
2006-04-19 15:33:07 +00:00
static int _format1_vg_write ( struct format_instance * fid , struct volume_group * vg ,
2010-07-09 15:34:40 +00:00
struct metadata_area * mda __attribute__ ( ( unused ) ) )
2001-10-05 16:36:53 +00:00
{
2009-04-10 09:59:18 +00:00
struct dm_pool * mem = dm_pool_create ( " lvm1 vg_write " , VG_MEMPOOL_CHUNK ) ;
2008-11-03 22:14:30 +00:00
struct dm_list pvds ;
2001-10-05 16:36:53 +00:00
int r = 0 ;
2008-01-30 13:19:47 +00:00
if ( ! mem )
return_0 ;
2001-10-05 16:36:53 +00:00
2008-11-03 22:14:30 +00:00
dm_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 ) & &
2013-03-25 14:30:40 +01:00
write_disks ( fid - > fmt , & pvds , 1 ) ) ;
2002-11-18 14:01:16 +00:00
2008-03-17 16:51:31 +00:00
lvmcache_update_vg ( vg , 0 ) ;
2005-10-16 23:03:59 +00:00
dm_pool_destroy ( mem ) ;
2001-10-05 16:36:53 +00:00
return r ;
}
2001-10-04 17:48:55 +00:00
2006-04-19 15:33:07 +00:00
static int _format1_pv_read ( const struct format_type * fmt , const char * pv_name ,
2011-02-21 12:15:59 +00:00
struct physical_volume * pv , int scan_label_only __attribute__ ( ( unused ) ) )
2001-10-09 08:58:52 +00:00
{
2005-10-16 23:03:59 +00:00
struct dm_pool * mem = dm_pool_create ( " lvm1 pv_read " , 1024 ) ;
2001-10-09 08:58:52 +00:00
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-11-18 14:01:16 +00:00
log_very_verbose ( " Reading physical volume data %s from disk " , pv_name ) ;
2001-10-17 15:29:31 +00:00
2008-01-30 13:19:47 +00:00
if ( ! mem )
return_0 ;
2001-10-09 08:58:52 +00:00
2008-01-30 13:19:47 +00:00
if ( ! ( dev = dev_cache_get ( pv_name , fmt - > cmd - > filter ) ) )
goto_out ;
2001-10-09 16:05:34 +00:00
2008-01-30 13:19:47 +00:00
if ( ! ( dl = read_disk ( fmt , dev , mem , NULL ) ) )
goto_out ;
2001-10-09 08:58:52 +00:00
2008-01-30 13:19:47 +00:00
if ( ! import_pv ( fmt , fmt - > cmd - > mem , dl - > dev , NULL , pv , & dl - > pvd , & dl - > vgd ) )
goto_out ;
2001-10-09 08:58:52 +00:00
2002-11-18 14:01:16 +00:00
pv - > fmt = fmt ;
2002-04-24 18:20:51 +00:00
r = 1 ;
2001-10-09 08:58:52 +00:00
2002-04-24 18:20:51 +00:00
out :
2005-10-16 23:03:59 +00:00
dm_pool_destroy ( mem ) ;
2002-04-24 18:20:51 +00:00
return r ;
2001-10-09 08:58:52 +00:00
}
2011-02-21 12:20:18 +00:00
static int _format1_pv_initialise ( const struct format_type * fmt ,
int64_t label_sector __attribute__ ( ( unused ) ) ,
unsigned long data_alignment __attribute__ ( ( unused ) ) ,
unsigned long data_alignment_offset __attribute__ ( ( unused ) ) ,
2013-02-18 12:47:31 +01:00
struct pvcreate_restorable_params * rp ,
2011-02-21 12:20:18 +00:00
struct physical_volume * pv )
2001-10-08 17:53:43 +00:00
{
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-11-18 14:01:16 +00:00
log_error ( " Physical volumes cannot be bigger than %s " ,
2006-05-09 21:23:51 +00:00
display_size ( fmt - > cmd , ( uint64_t ) MAX_PV_SIZE ) ) ;
2002-02-15 01:26:16 +00:00
return 0 ;
}
2002-11-18 14:01:16 +00:00
/* Nothing more to do if extent size isn't provided */
2013-02-18 12:47:31 +01:00
if ( ! rp - > extent_size )
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 .
*/
2013-02-18 12:47:31 +01:00
if ( ! calculate_extent_count ( pv , rp - > extent_size , rp - > extent_count , rp - > pe_start ) )
2008-01-30 13:19:47 +00:00
return_0 ;
2001-10-10 10:55:55 +00:00
2002-11-18 14:01:16 +00:00
/* Retain existing extent locations exactly */
2013-02-18 12:47:31 +01:00
if ( ( ( rp - > pe_start | | rp - > extent_count ) & & ( rp - > pe_start ! = pv - > pe_start ) ) | |
( rp - > extent_count & & ( rp - > extent_count ! = pv - > pe_count ) ) ) {
2002-11-18 14:01:16 +00:00
log_error ( " Metadata would overwrite physical extents " ) ;
return 0 ;
}
2001-10-10 09:25:04 +00:00
return 1 ;
}
2011-02-21 12:20:18 +00:00
static int _format1_pv_setup ( const struct format_type * fmt ,
struct physical_volume * pv ,
2011-02-25 14:12:14 +00:00
struct volume_group * vg )
2011-02-21 12:20:18 +00:00
{
2014-07-11 12:24:15 +02:00
int r ;
2013-02-18 12:47:31 +01:00
struct pvcreate_restorable_params rp = { . restorefile = NULL ,
. id = { { 0 } } ,
. idp = NULL ,
2013-05-28 12:37:22 +02:00
. ba_start = 0 ,
. ba_size = 0 ,
2013-02-18 12:47:31 +01:00
. pe_start = 0 ,
. extent_count = 0 ,
. extent_size = vg - > extent_size } ;
2014-07-11 12:24:15 +02:00
if ( ( r = _format1_pv_initialise ( fmt , - 1 , 0 , 0 , & rp , pv ) ) )
pv - > status | = ALLOCATABLE_PV ;
return r ;
2011-02-21 12:20:18 +00:00
}
2006-04-19 15:33:07 +00:00
static int _format1_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 )
2003-11-06 16:58:38 +00:00
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-12-12 20:55:49 +00:00
log_error ( " logical volumes cannot be larger than %s " ,
2006-05-09 21:23:51 +00:00
display_size ( fid - > fmt - > cmd , max_size ) ) ;
2002-02-13 21:28:56 +00:00
return 0 ;
}
2002-01-24 17:16:36 +00:00
return 1 ;
}
2011-02-21 12:26:27 +00:00
static int _format1_pv_write ( const struct format_type * fmt , struct physical_volume * pv )
2001-10-09 17:44:58 +00:00
{
2005-10-16 23:03:59 +00:00
struct dm_pool * mem ;
2001-10-10 10:05:29 +00:00
struct disk_list * dl ;
2008-11-03 22:14:30 +00:00
struct dm_list pvs ;
2003-07-04 22:34:56 +00:00
struct lvmcache_info * info ;
2011-06-01 19:29:31 +00:00
int pe_count , pe_size , pe_start ;
int r = 1 ;
2001-10-10 10:05:29 +00:00
2003-07-04 22:34:56 +00:00
if ( ! ( info = lvmcache_add ( fmt - > labeller , ( char * ) & pv - > id , pv - > dev ,
2008-01-30 13:19:47 +00:00
pv - > vg_name , NULL , 0 ) ) )
return_0 ;
2010-12-20 14:06:33 +00:00
2012-02-10 01:28:27 +00:00
lvmcache_update_pv ( info , pv , fmt ) ;
lvmcache_del_mdas ( info ) ;
2012-02-23 13:11:07 +00:00
lvmcache_del_das ( info ) ;
2013-05-28 12:37:22 +02:00
lvmcache_del_bas ( info ) ;
2002-11-18 14:01:16 +00:00
2008-11-03 22:14:30 +00:00
dm_list_init ( & pvs ) ;
2001-10-10 10:05:29 +00:00
2011-06-01 19:29:31 +00:00
pe_count = pv - > pe_count ;
pe_size = pv - > pe_size ;
pe_start = pv - > pe_start ;
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 ;
2006-08-17 19:15:27 +00:00
pv - > pe_start = LVM1_PE_ALIGN ;
2001-10-18 16:55:19 +00:00
2008-01-30 13:19:47 +00:00
if ( ! ( mem = dm_pool_create ( " lvm1 pv_write " , 1024 ) ) )
return_0 ;
2001-10-10 10:05:29 +00:00
2007-04-26 16:44:59 +00:00
if ( ! ( dl = dm_pool_alloc ( mem , sizeof ( * dl ) ) ) )
goto_bad ;
2001-10-10 10:05:29 +00:00
dl - > mem = mem ;
dl - > dev = pv - > dev ;
2011-06-01 19:29:31 +00:00
dm_list_init ( & dl - > uuids ) ;
dm_list_init ( & dl - > lvds ) ;
2001-10-10 10:05:29 +00:00
2007-04-26 16:44:59 +00:00
if ( ! export_pv ( fmt - > cmd , mem , NULL , & dl - > pvd , pv ) )
goto_bad ;
2001-10-10 10:05:29 +00:00
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 ;
2006-08-17 19:15:27 +00:00
dl - > pvd . pe_on_disk . base = LVM1_PE_ALIGN < < SECTOR_SHIFT ;
2002-01-29 15:52:11 +00:00
2008-11-03 22:14:30 +00:00
dm_list_add ( & pvs , & dl - > list ) ;
2013-03-25 14:30:40 +01:00
if ( ! write_disks ( fmt , & pvs , 0 ) )
2007-04-26 16:44:59 +00:00
goto_bad ;
2001-10-10 10:05:29 +00:00
2011-06-01 19:29:31 +00:00
goto out ;
2001-10-10 10:05:29 +00:00
2002-04-24 18:20:51 +00:00
bad :
2011-06-01 19:29:31 +00:00
r = 0 ;
out :
pv - > pe_size = pe_size ;
pv - > pe_count = pe_count ;
pv - > pe_start = pe_start ;
2005-10-16 23:03:59 +00:00
dm_pool_destroy ( mem ) ;
2011-06-01 19:29:31 +00:00
return r ;
2001-10-09 17:44:58 +00:00
}
2006-04-19 15:33:07 +00:00
static int _format1_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 */
2003-11-06 20:33:34 +00:00
if ( ! vg - > max_lv | | vg - > max_lv > = MAX_LV )
2001-10-15 18:39:40 +00:00
vg - > max_lv = MAX_LV - 1 ;
2001-10-12 14:25:53 +00:00
2003-11-06 20:33:34 +00:00
if ( ! vg - > max_pv | | 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
2014-10-14 18:12:15 +01:00
if ( ! vg_check_new_extent_size ( vg - > fid - > fmt , vg - > extent_size ) )
return_0 ;
2001-10-15 20:29:15 +00:00
2015-03-04 00:44:15 +00:00
/* Generate lvm1_system_id if not yet set */
if ( ! * vg - > lvm1_system_id & &
! generate_lvm1_system_id ( vg - > cmd , vg - > lvm1_system_id , " " ) )
return_0 ;
2001-10-12 14:25:53 +00:00
return 1 ;
}
2001-10-09 17:44:58 +00:00
2010-07-09 15:34:40 +00:00
static int _format1_segtype_supported ( struct format_instance * fid __attribute__ ( ( unused ) ) ,
2006-05-09 21:23:51 +00:00
const struct segment_type * segtype )
2004-09-14 13:56:18 +00:00
{
2008-01-30 13:19:47 +00:00
if ( ! ( segtype - > flags & SEG_FORMAT1_SUPPORT ) )
return_0 ;
2004-09-14 13:56:18 +00:00
return 1 ;
}
2002-11-18 14:01:16 +00:00
static struct metadata_area_ops _metadata_format1_ops = {
2006-05-09 21:23:51 +00:00
. vg_read = _format1_vg_read ,
. vg_write = _format1_vg_write ,
2002-11-18 14:01:16 +00:00
} ;
2006-04-19 15:33:07 +00:00
static struct format_instance * _format1_create_instance ( const struct format_type * fmt ,
2011-02-21 12:05:49 +00:00
const struct format_instance_ctx * fic )
2002-04-24 18:20:51 +00:00
{
struct format_instance * fid ;
struct metadata_area * mda ;
2011-03-11 14:30:27 +00:00
if ( ! ( fid = alloc_fid ( fmt , fic ) ) )
2008-01-30 13:19:47 +00:00
return_NULL ;
2002-04-24 18:20:51 +00:00
/* Define a NULL metadata area */
2011-03-11 15:10:16 +00:00
if ( ! ( mda = dm_pool_zalloc ( fid - > mem , sizeof ( * mda ) ) ) ) {
2011-03-11 14:38:38 +00:00
log_error ( " Unable to allocate metadata area structure "
" for lvm1 format " ) ;
goto bad ;
2002-04-24 18:20:51 +00:00
}
2002-11-18 14:01:16 +00:00
mda - > ops = & _metadata_format1_ops ;
2002-04-24 18:20:51 +00:00
mda - > metadata_locn = NULL ;
2010-10-05 17:34:05 +00:00
mda - > status = 0 ;
2010-06-28 20:32:44 +00:00
dm_list_add ( & fid - > metadata_areas_in_use , & mda - > list ) ;
2002-04-24 18:20:51 +00:00
return fid ;
2011-03-11 14:38:38 +00:00
bad :
dm_pool_destroy ( fid - > mem ) ;
return NULL ;
2002-04-24 18:20:51 +00:00
}
2011-03-11 14:38:38 +00:00
static void _format1_destroy_instance ( struct format_instance * fid )
2001-10-04 17:48:55 +00:00
{
2011-03-11 14:38:38 +00:00
if ( - - fid - > ref_count < = 1 )
dm_pool_destroy ( fid - > mem ) ;
2001-10-08 12:11:33 +00:00
}
2010-12-20 13:32:49 +00:00
static void _format1_destroy ( struct format_type * fmt )
2002-04-24 18:20:51 +00:00
{
2012-02-13 11:03:59 +00:00
if ( fmt - > orphan_vg )
free_orphan_vg ( fmt - > orphan_vg ) ;
2012-02-10 02:53:03 +00:00
2010-12-20 13:32:49 +00:00
dm_free ( fmt ) ;
2002-04-24 18:20:51 +00:00
}
2001-10-08 12:11:33 +00:00
2001-11-12 12:16:57 +00:00
static struct format_handler _format1_ops = {
2006-05-09 21:23:51 +00:00
. pv_read = _format1_pv_read ,
2011-02-21 12:20:18 +00:00
. pv_initialise = _format1_pv_initialise ,
2006-05-09 21:23:51 +00:00
. pv_setup = _format1_pv_setup ,
. pv_write = _format1_pv_write ,
. lv_setup = _format1_lv_setup ,
. vg_setup = _format1_vg_setup ,
. segtype_supported = _format1_segtype_supported ,
. create_instance = _format1_create_instance ,
. destroy_instance = _format1_destroy_instance ,
. destroy = _format1_destroy ,
2001-11-12 12:16:57 +00:00
} ;
2001-10-30 17:53:21 +00:00
2002-11-18 14:01:16 +00:00
# ifdef LVM1_INTERNAL
struct format_type * init_lvm1_format ( struct cmd_context * cmd )
# else /* Shared */
2003-04-15 13:24:42 +00:00
struct format_type * init_format ( struct cmd_context * cmd ) ;
2002-11-18 14:01:16 +00:00
struct format_type * init_format ( struct cmd_context * cmd )
# endif
2001-11-12 12:16:57 +00:00
{
2005-10-16 23:03:59 +00:00
struct format_type * fmt = dm_malloc ( sizeof ( * fmt ) ) ;
2012-02-10 02:53:03 +00:00
struct format_instance_ctx fic ;
struct format_instance * fid ;
2001-10-08 12:11:33 +00:00
2012-02-13 10:56:31 +00:00
if ( ! fmt ) {
log_error ( " Failed to allocate format1 format type structure. " ) ;
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 ;
2002-11-18 14:01:16 +00:00
fmt - > alias = NULL ;
2008-02-06 15:47:28 +00:00
fmt - > orphan_vg_name = FMT_LVM1_ORPHAN_VG_NAME ;
2007-11-09 16:51:54 +00:00
fmt - > features = FMT_RESTRICTED_LVIDS | FMT_ORPHAN_ALLOCATABLE |
2015-02-23 23:03:52 +00:00
FMT_RESTRICTED_READAHEAD | FMT_OBSOLETE |
FMT_SYSTEMID_ON_PVS ;
2002-04-24 18:20:51 +00:00
fmt - > private = NULL ;
2001-10-08 12:11:33 +00:00
2012-02-23 13:11:07 +00:00
dm_list_init ( & fmt - > mda_ops ) ;
2002-11-18 14:01:16 +00:00
if ( ! ( fmt - > labeller = lvm1_labeller_create ( fmt ) ) ) {
log_error ( " Couldn't create lvm1 label handler. " ) ;
2012-02-08 10:49:36 +00:00
dm_free ( fmt ) ;
2002-11-18 14:01:16 +00:00
return NULL ;
}
2013-07-29 15:58:18 +02:00
if ( ! ( label_register_handler ( fmt - > labeller ) ) ) {
2002-11-18 14:01:16 +00:00
log_error ( " Couldn't register lvm1 label handler. " ) ;
2012-02-08 10:49:36 +00:00
fmt - > labeller - > ops - > destroy ( fmt - > labeller ) ;
dm_free ( fmt ) ;
2002-11-18 14:01:16 +00:00
return NULL ;
}
2012-02-15 01:44:58 +00:00
if ( ! ( fmt - > orphan_vg = alloc_vg ( " format1_orphan " , cmd , fmt - > orphan_vg_name ) ) ) {
2012-02-10 02:53:03 +00:00
log_error ( " Couldn't create lvm1 orphan VG. " ) ;
2012-02-13 10:56:31 +00:00
dm_free ( fmt ) ;
2012-02-10 02:53:03 +00:00
return NULL ;
}
2012-02-13 10:56:31 +00:00
2012-02-12 23:01:19 +00:00
fic . type = FMT_INSTANCE_AUX_MDAS ;
2012-02-10 02:53:03 +00:00
fic . context . vg_ref . vg_name = fmt - > orphan_vg_name ;
fic . context . vg_ref . vg_id = NULL ;
2012-02-13 10:56:31 +00:00
2012-02-10 02:53:03 +00:00
if ( ! ( fid = _format1_create_instance ( fmt , & fic ) ) ) {
2012-02-13 10:56:31 +00:00
_format1_destroy ( fmt ) ;
return_NULL ;
2012-02-10 02:53:03 +00:00
}
2012-02-13 10:56:31 +00:00
2012-02-10 02:53:03 +00:00
vg_set_fid ( fmt - > orphan_vg , fid ) ;
2004-09-14 17:37:51 +00:00
log_very_verbose ( " Initialised format: %s " , fmt - > name ) ;
2002-04-24 18:20:51 +00:00
return fmt ;
2001-10-04 17:48:55 +00:00
}