2001-11-06 22:02:26 +03:00
/*
2001-11-08 01:47:43 +03:00
* Copyright ( C ) 2001 Sistina Software
2001-11-06 22:02:26 +03:00
*
2001-11-08 01:47:43 +03:00
* This file is released under the GPL .
2001-11-06 22:02:26 +03:00
*
*/
# include "tools.h"
2002-01-01 00:27:39 +03:00
2001-11-16 18:38:52 +03:00
# include <fcntl.h>
2001-11-06 22:02:26 +03:00
2002-02-15 14:53:22 +03:00
struct lvcreate_params {
/* flags */
int snapshot ;
2001-11-06 22:02:26 +03:00
int zero ;
2002-02-15 14:53:22 +03:00
int contiguous ;
2002-02-21 22:04:37 +03:00
int minor ;
2001-11-06 22:02:26 +03:00
2002-02-15 14:53:22 +03:00
char * origin ;
2001-11-06 22:02:26 +03:00
char * vg_name ;
2002-02-15 14:53:22 +03:00
char * lv_name ;
2001-11-06 22:02:26 +03:00
2002-02-15 14:53:22 +03:00
uint32_t stripes ;
uint32_t stripe_size ;
uint32_t chunk_size ;
/* size */
uint32_t extents ;
uint64_t size ;
uint32_t permission ;
uint32_t read_ahead ;
int pv_count ;
char * * pvs ;
} ;
static int _read_name_params ( struct lvcreate_params * lp ,
2002-03-04 18:25:52 +03:00
struct cmd_context * cmd , int * pargc , char * * * pargv )
2002-02-15 14:53:22 +03:00
{
int argc = * pargc ;
char * * argv = * pargv , * ptr ;
if ( arg_count ( cmd , name_ARG ) )
lp - > lv_name = arg_value ( cmd , name_ARG ) ;
if ( arg_count ( cmd , snapshot_ARG ) ) {
lp - > snapshot = 1 ;
if ( ! argc ) {
log_err ( " Please specify a logical volume to act as "
" the snapshot origin. " ) ;
return 0 ;
}
lp - > origin = argv [ 0 ] ;
( * pargv ) + + , ( * pargc ) - - ;
if ( ! ( lp - > vg_name = extract_vgname ( cmd - > fid , lp - > origin ) ) ) {
log_err ( " The origin name should include the "
" volume group. " ) ;
return 0 ;
}
2002-02-18 13:59:51 +03:00
/* Strip the volume group from the origin */
if ( ( ptr = strrchr ( lp - > origin , ( int ) ' / ' ) ) )
lp - > origin = ptr + 1 ;
2002-02-15 14:53:22 +03:00
} else {
/*
* If VG not on command line , try - n arg and then
* environment .
*/
if ( ! argc ) {
if ( ! ( lp - > vg_name =
extract_vgname ( cmd - > fid , lp - > lv_name ) ) ) {
log_err ( " Please provide a volume group name " ) ;
return 0 ;
}
} else {
/*
* Ensure lv_name doesn ' t contain a
* different VG .
*/
if ( lp - > lv_name & & strchr ( lp - > lv_name , ' / ' ) ) {
if ( ! ( lp - > vg_name =
extract_vgname ( cmd - > fid , lp - > lv_name ) ) )
2002-03-04 18:25:52 +03:00
return 0 ;
2002-02-15 14:53:22 +03:00
if ( strcmp ( lp - > vg_name , argv [ 0 ] ) ) {
log_error ( " Inconsistent volume group "
" names "
" given: \" %s \" and \" %s \" " ,
lp - > vg_name , argv [ 0 ] ) ;
return 0 ;
}
}
lp - > vg_name = argv [ 0 ] ;
( * pargv ) + + , ( * pargc ) - - ;
}
2001-11-06 22:02:26 +03:00
}
2002-02-15 14:53:22 +03:00
if ( lp - > lv_name & & ( ptr = strrchr ( lp - > lv_name , ' / ' ) ) )
lp - > lv_name = ptr + 1 ;
return 1 ;
}
static int _read_size_params ( struct lvcreate_params * lp ,
2002-03-04 18:25:52 +03:00
struct cmd_context * cmd , int * pargc , char * * * pargv )
2002-02-15 14:53:22 +03:00
{
/*
* There are two mutually exclusive ways of specifying
* the size . . .
*/
if ( arg_count ( cmd , extents_ARG ) & & arg_count ( cmd , size_ARG ) ) {
2001-11-06 22:02:26 +03:00
log_error ( " Invalid combination of arguments " ) ;
2002-02-15 14:53:22 +03:00
return 0 ;
2001-11-06 22:02:26 +03:00
}
2002-02-15 14:53:22 +03:00
/*
* . . . you must use one of them .
*/
2002-02-12 00:00:35 +03:00
if ( arg_count ( cmd , size_ARG ) + arg_count ( cmd , extents_ARG ) = = 0 ) {
2001-11-06 22:02:26 +03:00
log_error ( " Please indicate size using option -l or -L " ) ;
2002-02-15 14:53:22 +03:00
return 0 ;
2001-11-06 22:02:26 +03:00
}
2002-02-15 14:53:22 +03:00
if ( arg_count ( cmd , extents_ARG ) )
lp - > extents = arg_int_value ( cmd , extents_ARG , 0 ) ;
/* Size returned in kilobyte units; held in sectors */
if ( arg_count ( cmd , size_ARG ) )
2002-02-18 13:59:51 +03:00
lp - > size = arg_int64_value ( cmd , size_ARG , 0 ) * 2ull ;
2002-02-15 14:53:22 +03:00
return 1 ;
}
static int _read_stripe_params ( struct lvcreate_params * lp ,
struct cmd_context * cmd ,
int * pargc , char * * * pargv )
{
int argc = * pargc ;
2001-11-06 22:02:26 +03:00
2002-02-15 14:53:22 +03:00
lp - > stripes = 1 ;
2001-11-06 22:02:26 +03:00
2002-02-12 00:00:35 +03:00
if ( arg_count ( cmd , stripes_ARG ) ) {
2002-02-15 14:53:22 +03:00
lp - > stripes = arg_int_value ( cmd , stripes_ARG , 1 ) ;
if ( lp - > stripes = = 1 )
2001-11-06 22:02:26 +03:00
log_print ( " Redundant stripes argument: default is 1 " ) ;
}
2002-02-12 00:00:35 +03:00
if ( arg_count ( cmd , stripesize_ARG ) )
2002-02-15 14:53:22 +03:00
lp - > stripe_size = 2 * arg_int_value ( cmd , stripesize_ARG , 0 ) ;
2001-11-06 22:02:26 +03:00
2002-02-15 14:53:22 +03:00
if ( lp - > stripes = = 1 & & lp - > stripe_size ) {
2001-11-06 22:02:26 +03:00
log_print ( " Ignoring stripesize argument with single stripe " ) ;
2002-02-15 14:53:22 +03:00
lp - > stripe_size = 0 ;
2001-11-06 22:02:26 +03:00
}
2002-02-15 14:53:22 +03:00
if ( lp - > stripes > 1 & & ! lp - > stripe_size ) {
lp - > stripe_size = 2 * STRIPE_SIZE_DEFAULT ;
2002-03-04 18:25:52 +03:00
log_print ( " Using default stripesize %dKB " , lp - > stripe_size / 2 ) ;
2001-12-03 19:27:16 +03:00
}
2002-02-15 14:53:22 +03:00
if ( argc & & argc < lp - > stripes ) {
log_error ( " Too few physical volumes on "
" command line for %d-way striping " , lp - > stripes ) ;
return 0 ;
}
2001-11-06 22:02:26 +03:00
2002-02-15 14:53:22 +03:00
if ( lp - > stripes < 1 | | lp - > stripes > MAX_STRIPES ) {
log_error ( " Number of stripes (%d) must be between %d and %d " ,
lp - > stripes , 1 , MAX_STRIPES ) ;
return 0 ;
}
2001-11-06 22:02:26 +03:00
2002-02-15 14:53:22 +03:00
if ( lp - > stripes > 1 & & ( lp - > stripe_size < STRIPE_SIZE_MIN | |
lp - > stripe_size > STRIPE_SIZE_MAX | |
lp - > stripe_size & ( lp - > stripe_size - 1 ) ) ) {
log_error ( " Invalid stripe size %d " , lp - > stripe_size ) ;
return 0 ;
}
2001-11-06 22:02:26 +03:00
2002-02-15 14:53:22 +03:00
return 1 ;
}
2001-11-06 22:02:26 +03:00
2002-02-15 14:53:22 +03:00
static int _read_params ( struct lvcreate_params * lp , struct cmd_context * cmd ,
int argc , char * * argv )
{
/*
* Set the defaults .
*/
memset ( lp , 0 , sizeof ( * lp ) ) ;
lp - > chunk_size = 256 ;
if ( ! _read_name_params ( lp , cmd , & argc , & argv ) | |
! _read_size_params ( lp , cmd , & argc , & argv ) | |
! _read_stripe_params ( lp , cmd , & argc , & argv ) )
2002-02-21 13:15:54 +03:00
return 0 ;
2001-11-12 22:28:50 +03:00
2002-02-15 14:53:22 +03:00
/*
* Should we zero the lv .
*/
2002-02-18 18:52:48 +03:00
lp - > zero = strcmp ( arg_str_value ( cmd , zero_ARG , " y " ) , " n " ) ;
2002-01-21 17:28:12 +03:00
2002-02-15 14:53:22 +03:00
/*
* Contiguous ?
*/
lp - > contiguous = strcmp ( arg_str_value ( cmd , contiguous_ARG , " n " ) , " n " ) ;
/*
* Read ahead .
*/
if ( arg_count ( cmd , readahead_ARG ) )
lp - > read_ahead = arg_int_value ( cmd , readahead_ARG , 0 ) ;
/*
* Permissions .
*/
if ( arg_count ( cmd , permission_ARG ) )
lp - > permission = arg_int_value ( cmd , permission_ARG , 0 ) ;
else
lp - > permission = LVM_READ | LVM_WRITE ;
2002-02-21 22:04:37 +03:00
lp - > minor = arg_int_value ( cmd , minor_ARG , - 1 ) ;
2002-02-15 14:53:22 +03:00
2002-02-21 22:04:37 +03:00
/* Persistent minor */
2002-02-15 14:53:22 +03:00
if ( arg_count ( cmd , persistent_ARG ) ) {
2002-02-21 22:04:37 +03:00
if ( ! strcmp ( arg_str_value ( cmd , persistent_ARG , " n " ) , " y " ) ) {
if ( lp - > minor = = - 1 ) {
log_error ( " Please specify minor number with "
" --minor when using -My " ) ;
return 0 ;
}
} else {
if ( lp - > minor ! = - 1 ) {
log_error ( " --minor not possible with -Mn " ) ;
return 0 ;
}
2001-11-06 22:02:26 +03:00
}
}
2002-02-15 14:53:22 +03:00
lp - > pv_count = argc ;
lp - > pvs = argv ;
2001-11-06 22:02:26 +03:00
2002-02-15 14:53:22 +03:00
return 1 ;
}
2002-03-04 16:46:37 +03:00
/*
* Volumes may be zeroed to remove old application data .
*/
static int _zero_lv ( struct cmd_context * cmd , struct logical_volume * lv )
{
struct device * dev ;
char * name ;
/*
* FIXME :
* < clausen > also , more than 4 k
* < clausen > say , reiserfs puts it ' s superblock 32 k in , IIRC
* < ejt_ > k , I ' ll drop a fixme to that effect
* ( I know the device is at least 4 k , but not 32 k )
*/
if ( ! ( name = pool_alloc ( cmd - > mem , PATH_MAX ) ) ) {
log_error ( " Name allocation failed - device not zeroed " ) ;
return 0 ;
}
if ( lvm_snprintf ( name , PATH_MAX , " %s%s/%s " , cmd - > dev_dir ,
lv - > vg - > name , lv - > name ) < 0 ) {
2002-03-04 18:25:52 +03:00
log_error ( " Name too long - device not zeroed (%s) " , lv - > name ) ;
2002-03-04 16:46:37 +03:00
return 0 ;
}
log_verbose ( " Zeroing start of logical volume \" %s \" " , lv - > name ) ;
if ( ! ( dev = dev_cache_get ( name , NULL ) ) ) {
2002-03-05 23:03:09 +03:00
log_error ( " %s: not found: device not zeroed " , name ) ;
2002-03-04 16:46:37 +03:00
return 0 ;
}
if ( ! ( dev_open ( dev , O_WRONLY ) ) )
return 0 ;
dev_zero ( dev , 0 , 4096 ) ;
dev_close ( dev ) ;
return 1 ;
}
2002-03-04 18:25:52 +03:00
static int _lvcreate ( struct cmd_context * cmd , struct lvcreate_params * lp )
2002-02-15 14:53:22 +03:00
{
uint32_t size_rest ;
2002-02-21 22:04:37 +03:00
uint32_t status = 0 ;
2002-02-15 14:53:22 +03:00
struct volume_group * vg ;
2002-02-18 13:59:51 +03:00
struct logical_volume * lv , * org ;
2002-02-15 14:53:22 +03:00
struct list * pvh ;
if ( lp - > contiguous )
status | = ALLOC_CONTIGUOUS ;
else
status | = ALLOC_SIMPLE ;
status | = lp - > permission ;
/* does VG exist? */
log_verbose ( " Finding volume group \" %s \" " , lp - > vg_name ) ;
if ( ! ( vg = cmd - > fid - > ops - > vg_read ( cmd - > fid , lp - > vg_name ) ) ) {
log_error ( " Volume group \" %s \" doesn't exist " , lp - > vg_name ) ;
return 0 ;
2001-11-06 22:02:26 +03:00
}
2002-01-29 20:23:33 +03:00
if ( vg - > status & EXPORTED_VG ) {
2002-02-15 14:53:22 +03:00
log_error ( " Volume group \" %s \" is exported " , lp - > vg_name ) ;
return 0 ;
2002-01-29 20:23:33 +03:00
}
if ( ! ( vg - > status & LVM_WRITE ) ) {
2002-02-15 14:53:22 +03:00
log_error ( " Volume group \" %s \" is read-only " , lp - > vg_name ) ;
return 0 ;
2002-01-29 20:23:33 +03:00
}
2002-02-15 14:53:22 +03:00
if ( lp - > lv_name & & find_lv_in_vg ( vg , lp - > lv_name ) ) {
2002-01-30 18:04:48 +03:00
log_error ( " Logical volume \" %s \" already exists in "
2002-02-15 14:53:22 +03:00
" volume group \" %s \" " , lp - > lv_name , lp - > vg_name ) ;
return 0 ;
2001-11-06 22:02:26 +03:00
}
2002-02-15 14:53:22 +03:00
/*
* Create the pv list .
*/
if ( lp - > pv_count ) {
2002-02-11 23:50:53 +03:00
if ( ! ( pvh = create_pv_list ( cmd - > mem , vg ,
2002-02-15 14:53:22 +03:00
lp - > pv_count , lp - > pvs ) ) ) {
2002-01-21 19:05:23 +03:00
stack ;
2002-02-15 14:53:22 +03:00
return 0 ;
2002-01-21 19:05:23 +03:00
}
2002-02-15 14:53:22 +03:00
} else
pvh = & vg - > pvs ;
2001-11-06 22:02:26 +03:00
2002-02-15 14:53:22 +03:00
if ( lp - > stripe_size > vg - > extent_size ) {
2001-12-03 19:27:16 +03:00
log_error ( " Setting stripe size %d KB to physical extent "
2002-02-15 14:53:22 +03:00
" size %u KB " , lp - > stripe_size / 2 ,
vg - > extent_size / 2 ) ;
lp - > stripe_size = vg - > extent_size ;
2001-11-06 22:02:26 +03:00
}
2002-02-15 14:53:22 +03:00
if ( lp - > size ) {
2001-11-06 22:02:26 +03:00
/* No of 512-byte sectors */
2002-02-18 13:59:51 +03:00
lp - > extents = lp - > size ;
2001-11-06 22:02:26 +03:00
2002-02-15 14:53:22 +03:00
if ( lp - > extents % vg - > extent_size ) {
2001-11-06 22:02:26 +03:00
char * s1 ;
2002-02-15 14:53:22 +03:00
lp - > extents + = vg - > extent_size - lp - > extents %
2002-03-04 18:25:52 +03:00
vg - > extent_size ;
2002-02-15 14:53:22 +03:00
log_print ( " Rounding up size to full physical "
" extent %s " ,
( s1 = display_size ( lp - > extents / 2 ,
SIZE_SHORT ) ) ) ;
2001-11-06 22:02:26 +03:00
dbg_free ( s1 ) ;
}
2002-02-15 14:53:22 +03:00
lp - > extents / = vg - > extent_size ;
2001-11-06 22:02:26 +03:00
}
2002-02-15 14:53:22 +03:00
if ( ( size_rest = lp - > extents % lp - > stripes ) ) {
2002-02-12 00:00:35 +03:00
log_print ( " Rounding size (%d extents) up to stripe boundary "
2002-02-15 14:53:22 +03:00
" size (%d extents) " , lp - > extents ,
lp - > extents - size_rest + lp - > stripes ) ;
lp - > extents = lp - > extents - size_rest + lp - > stripes ;
2002-02-12 00:00:35 +03:00
}
2001-11-06 22:02:26 +03:00
2002-02-18 13:59:51 +03:00
if ( lp - > snapshot & & ! ( org = find_lv ( vg , lp - > origin ) ) ) {
log_err ( " Couldn't find origin volume '%s'. " , lp - > origin ) ;
return 0 ;
}
2002-02-15 14:53:22 +03:00
if ( ! ( lv = lv_create ( cmd - > fid , lp - > lv_name , status ,
lp - > stripes , lp - > stripe_size , lp - > extents ,
2002-03-04 18:25:52 +03:00
vg , pvh ) ) ) return 0 ;
2002-02-18 13:59:51 +03:00
2002-02-15 14:53:22 +03:00
if ( lp - > read_ahead ) {
2001-11-06 22:02:26 +03:00
log_verbose ( " Setting read ahead sectors " ) ;
2002-02-15 14:53:22 +03:00
lv - > read_ahead = lp - > read_ahead ;
2001-11-06 22:02:26 +03:00
}
2002-02-15 14:53:22 +03:00
if ( lp - > minor > = 0 ) {
lv - > minor = lp - > minor ;
2002-02-21 22:04:37 +03:00
lv - > status | = FIXED_MINOR ;
2002-02-01 20:54:39 +03:00
log_verbose ( " Setting minor number to %d " , lv - > minor ) ;
}
2002-03-04 18:25:52 +03:00
if ( ! archive ( vg ) )
return 0 ;
2001-11-06 22:02:26 +03:00
/* store vg on disk(s) */
2002-02-11 23:50:53 +03:00
if ( ! cmd - > fid - > ops - > vg_write ( cmd - > fid , vg ) )
2002-02-15 14:53:22 +03:00
return 0 ;
2001-11-06 22:02:26 +03:00
2002-03-05 23:03:09 +03:00
if ( ! lock_vol ( cmd , lv - > lvid . s , LCK_LV_ACTIVATE ) )
2002-03-04 18:25:52 +03:00
return 0 ;
if ( lp - > zero | | lp - > snapshot )
_zero_lv ( cmd , lv ) ;
else
log_print ( " WARNING: \" %s \" not zeroed " , lv - > name ) ;
if ( lp - > snapshot ) {
2002-03-05 23:03:09 +03:00
if ( ! lock_vol ( cmd , lv - > lvid . s , LCK_LV_DEACTIVATE ) ) {
2002-03-04 18:25:52 +03:00
log_err ( " Couldn't lock snapshot. " ) ;
return 0 ;
}
if ( ! vg_add_snapshot ( org , lv , 1 , lp - > chunk_size ) ) {
log_err ( " Couldn't create snapshot. " ) ;
return 0 ;
}
/* store vg on disk(s) */
if ( ! cmd - > fid - > ops - > vg_write ( cmd - > fid , vg ) )
return 0 ;
2002-03-05 23:03:09 +03:00
if ( ! lock_vol ( cmd , lv - > lvid . s , LCK_LV_ACTIVATE ) )
2002-03-04 18:25:52 +03:00
return 0 ;
}
2002-01-07 14:12:11 +03:00
backup ( vg ) ;
2002-03-04 18:25:52 +03:00
2002-01-30 18:04:48 +03:00
log_print ( " Logical volume \" %s \" created " , lv - > name ) ;
2001-11-14 17:12:01 +03:00
2002-03-04 18:25:52 +03:00
/*
* FIXME : as a sanity check we could try reading the
* last block of the device ?
*/
2002-02-15 14:53:22 +03:00
return 1 ;
}
2001-11-06 22:02:26 +03:00
2002-02-15 14:53:22 +03:00
int lvcreate ( struct cmd_context * cmd , int argc , char * * argv )
{
int r = ECMD_FAILED ;
struct lvcreate_params lp ;
2002-02-21 22:04:37 +03:00
memset ( & lp , 0 , sizeof ( lp ) ) ;
2002-02-15 14:53:22 +03:00
if ( ! _read_params ( & lp , cmd , argc , argv ) )
return - EINVALID_CMD_LINE ;
2002-02-27 15:26:41 +03:00
if ( ! lock_vol ( cmd , lp . vg_name , LCK_VG_WRITE ) ) {
2002-02-15 14:53:22 +03:00
log_error ( " Can't get lock for %s " , lp . vg_name ) ;
return 0 ;
}
2002-03-04 18:25:52 +03:00
if ( ! _lvcreate ( cmd , & lp ) ) {
2002-02-15 14:53:22 +03:00
stack ;
goto out ;
}
r = 0 ;
2002-02-11 18:42:34 +03:00
2002-03-04 18:25:52 +03:00
out :
2002-02-27 15:26:41 +03:00
lock_vol ( cmd , lp . vg_name , LCK_VG_UNLOCK ) ;
2002-02-15 14:53:22 +03:00
return r ;
2001-11-06 22:02:26 +03:00
}