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-11 23:50:53 +03:00
int lvcreate ( struct cmd_context * cmd , int argc , char * * argv )
2001-11-06 22:02:26 +03:00
{
int zero ;
uint32_t read_ahead = 0 ;
int stripes = 1 ;
int stripesize = 0 ;
int opt = 0 ;
uint32_t status = 0 ;
uint32_t size = 0 ;
2001-12-03 19:27:16 +03:00
uint32_t size_rest ;
2001-11-06 22:02:26 +03:00
uint32_t extents = 0 ;
struct volume_group * vg ;
struct logical_volume * lv ;
2002-01-21 17:28:12 +03:00
struct list * pvh ;
2001-11-06 22:02:26 +03:00
char * lv_name = NULL ;
char * vg_name ;
char * st ;
2002-02-11 23:50:53 +03:00
if ( arg_count ( cmd , snapshot_ARG ) | | arg_count ( cmd , chunksize_ARG ) ) {
2001-11-06 22:02:26 +03:00
log_error ( " Snapshots are not yet supported in LVM2. " ) ;
return EINVALID_CMD_LINE ;
}
/* mutually exclusive */
2002-02-11 23:50:53 +03:00
if ( ( arg_count ( cmd , zero_ARG ) & & arg_count ( cmd , snapshot_ARG ) ) | |
( arg_count ( cmd , extents_ARG ) & & arg_count ( cmd , size_ARG ) ) ) {
2001-11-06 22:02:26 +03:00
log_error ( " Invalid combination of arguments " ) ;
return EINVALID_CMD_LINE ;
}
2002-02-11 23:50:53 +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 " ) ;
return EINVALID_CMD_LINE ;
}
2002-02-11 23:50:53 +03:00
if ( strcmp ( arg_str_value ( cmd , contiguous_ARG , " n " ) , " n " ) )
2001-11-06 22:02:26 +03:00
status | = ALLOC_CONTIGUOUS ;
2001-11-12 22:28:50 +03:00
else
status | = ALLOC_SIMPLE ;
2001-11-06 22:02:26 +03:00
2002-02-11 23:50:53 +03:00
zero = strcmp ( arg_str_value ( cmd , zero_ARG , " y " ) , " n " ) ;
2001-11-06 22:02:26 +03:00
2002-02-11 23:50:53 +03:00
if ( arg_count ( cmd , stripes_ARG ) ) {
stripes = arg_int_value ( cmd , stripes_ARG , 1 ) ;
2001-11-06 22:02:26 +03:00
if ( stripes = = 1 )
log_print ( " Redundant stripes argument: default is 1 " ) ;
}
2002-02-11 23:50:53 +03:00
if ( arg_count ( cmd , stripesize_ARG ) )
stripesize = 2 * arg_int_value ( cmd , stripesize_ARG , 0 ) ;
2001-11-06 22:02:26 +03:00
if ( stripes = = 1 & & stripesize ) {
log_print ( " Ignoring stripesize argument with single stripe " ) ;
stripesize = 0 ;
}
2001-12-03 19:27:16 +03:00
if ( stripes > 1 & & ! stripesize ) {
stripesize = 2 * STRIPE_SIZE_DEFAULT ;
log_print ( " Using default stripesize %dKB " , stripesize / 2 ) ;
}
2002-02-11 23:50:53 +03:00
if ( arg_count ( cmd , permission_ARG ) )
status | = arg_int_value ( cmd , permission_ARG , 0 ) ;
2001-11-08 01:47:43 +03:00
else
status | = LVM_READ | LVM_WRITE ;
2001-11-06 22:02:26 +03:00
2002-02-11 23:50:53 +03:00
if ( arg_count ( cmd , readahead_ARG ) )
read_ahead = arg_int_value ( cmd , readahead_ARG , 0 ) ;
2001-11-06 22:02:26 +03:00
2002-02-11 23:50:53 +03:00
if ( arg_count ( cmd , extents_ARG ) )
extents = arg_int_value ( cmd , extents_ARG , 0 ) ;
2001-11-06 22:02:26 +03:00
/* Size returned in kilobyte units; held in sectors */
2002-02-11 23:50:53 +03:00
if ( arg_count ( cmd , size_ARG ) )
size = arg_int_value ( cmd , size_ARG , 0 ) ;
2001-11-06 22:02:26 +03:00
2002-02-11 23:50:53 +03:00
if ( arg_count ( cmd , name_ARG ) )
lv_name = arg_value ( cmd , name_ARG ) ;
2001-11-12 22:28:50 +03:00
2001-11-06 22:02:26 +03:00
/* If VG not on command line, try -n arg and then environment */
if ( ! argc ) {
2002-02-11 23:50:53 +03:00
if ( ! ( vg_name = extract_vgname ( cmd - > fid , lv_name ) ) ) {
2001-11-06 22:02:26 +03:00
log_error ( " Please provide a volume group name " ) ;
return EINVALID_CMD_LINE ;
}
2002-01-21 17:28:12 +03:00
2001-11-06 22:02:26 +03:00
} else {
/* Ensure lv_name doesn't contain a different VG! */
2001-11-14 15:07:37 +03:00
if ( lv_name & & strchr ( lv_name , ' / ' ) ) {
2002-02-11 23:50:53 +03:00
if ( ! ( vg_name = extract_vgname ( cmd - > fid , lv_name ) ) )
2001-11-06 22:02:26 +03:00
return EINVALID_CMD_LINE ;
if ( strcmp ( vg_name , argv [ 0 ] ) ) {
log_error ( " Inconsistent volume group names "
2002-01-30 18:04:48 +03:00
" given: \" %s \" and \" %s \" " ,
vg_name , argv [ 0 ] ) ;
2001-11-06 22:02:26 +03:00
return EINVALID_CMD_LINE ;
}
}
vg_name = argv [ 0 ] ;
argv + + ;
argc - - ;
}
2001-11-14 15:07:37 +03:00
if ( lv_name & & ( st = strrchr ( lv_name , ' / ' ) ) )
2001-11-06 22:02:26 +03:00
lv_name = st + 1 ;
/* does VG exist? */
2002-01-30 18:04:48 +03:00
log_verbose ( " Finding volume group \" %s \" " , vg_name ) ;
2002-02-11 18:42:34 +03:00
if ( ! lock_vol ( vg_name , LCK_VG | LCK_WRITE ) ) {
log_error ( " Can't get lock for %s " , vg_name ) ;
return ECMD_FAILED ;
}
2002-02-11 23:50:53 +03:00
if ( ! ( vg = cmd - > fid - > ops - > vg_read ( cmd - > fid , vg_name ) ) ) {
2002-01-30 18:04:48 +03:00
log_error ( " Volume group \" %s \" doesn't exist " , vg_name ) ;
2002-02-11 18:42:34 +03:00
goto error ;
2001-11-06 22:02:26 +03:00
}
2002-01-29 20:23:33 +03:00
if ( vg - > status & EXPORTED_VG ) {
2002-01-30 18:04:48 +03:00
log_error ( " Volume group \" %s \" is exported " , vg_name ) ;
2002-02-11 18:42:34 +03:00
goto error ;
2002-01-29 20:23:33 +03:00
}
if ( ! ( vg - > status & LVM_WRITE ) ) {
2002-01-30 18:04:48 +03:00
log_error ( " Volume group \" %s \" is read-only " , vg_name ) ;
2002-02-11 18:42:34 +03:00
goto error ;
2002-01-29 20:23:33 +03:00
}
2002-01-21 17:28:12 +03:00
if ( lv_name & & find_lv_in_vg ( vg , lv_name ) ) {
2002-01-30 18:04:48 +03:00
log_error ( " Logical volume \" %s \" already exists in "
" volume group \" %s \" " , lv_name , vg_name ) ;
2002-02-11 18:42:34 +03:00
goto error ;
2001-11-06 22:02:26 +03:00
}
2002-01-21 19:05:23 +03:00
if ( ! argc )
2001-11-06 22:02:26 +03:00
/* Use full list from VG */
pvh = & vg - > pvs ;
2002-01-21 19:05:23 +03:00
else {
2002-02-11 23:50:53 +03:00
if ( ! ( pvh = create_pv_list ( cmd - > mem , vg ,
2002-01-21 19:05:23 +03:00
argc - opt , argv + opt ) ) ) {
stack ;
2002-02-11 18:42:34 +03:00
goto error ;
2002-01-21 19:05:23 +03:00
}
2001-11-06 22:02:26 +03:00
}
2001-12-03 19:27:16 +03:00
if ( argc & & argc < stripes ) {
log_error ( " Too few physical volumes on "
2001-11-06 22:02:26 +03:00
" command line for %d-way striping " , stripes ) ;
2002-02-11 18:42:34 +03:00
goto error_cmdline ;
2001-11-06 22:02:26 +03:00
}
2001-12-03 19:27:16 +03:00
if ( stripes < 1 | | stripes > MAX_STRIPES ) {
2002-01-21 19:05:23 +03:00
log_error ( " Number of stripes (%d) must be between %d and %d " ,
2001-12-03 19:27:16 +03:00
stripes , 1 , MAX_STRIPES ) ;
2002-02-11 18:42:34 +03:00
goto error_cmdline ;
2001-11-06 22:02:26 +03:00
}
2001-12-03 19:27:16 +03:00
if ( stripes > 1 & & ( stripesize < STRIPE_SIZE_MIN | |
stripesize > STRIPE_SIZE_MAX | |
stripesize & ( stripesize - 1 ) ) ) {
log_error ( " Invalid stripe size %d " , stripesize ) ;
2002-02-11 18:42:34 +03:00
goto error_cmdline ;
2001-11-06 22:02:26 +03:00
}
2001-12-03 19:27:16 +03:00
if ( stripesize > vg - > extent_size ) {
log_error ( " Setting stripe size %d KB to physical extent "
" size %u KB " ,
stripesize / 2 , vg - > extent_size / 2 ) ;
stripesize = vg - > extent_size ;
2001-11-06 22:02:26 +03:00
}
if ( size ) {
/* No of 512-byte sectors */
extents = size * 2 ;
if ( extents % vg - > extent_size ) {
char * s1 ;
extents + = vg - > extent_size - extents % vg - > extent_size ;
log_print ( " Rounding up size to full physical extent %s " ,
2001-11-13 20:53:06 +03:00
( s1 = display_size ( extents / 2 , SIZE_SHORT ) ) ) ;
2001-11-06 22:02:26 +03:00
dbg_free ( s1 ) ;
}
extents / = vg - > extent_size ;
}
2001-12-03 19:27:16 +03:00
if ( ( size_rest = extents % stripes ) ) {
log_print ( " Rounding size (%d extents) up to stripe boundary "
" size (%d extents) " , extents ,
extents - size_rest + stripes ) ;
extents = extents - size_rest + stripes ;
}
2001-11-06 22:02:26 +03:00
2002-01-09 16:17:14 +03:00
if ( ! archive ( vg ) )
2002-02-11 18:42:34 +03:00
goto error ;
2002-01-09 16:17:14 +03:00
2002-02-11 23:50:53 +03:00
if ( ! ( lv = lv_create ( cmd - > fid , lv_name , status ,
2002-01-24 20:15:24 +03:00
stripes , stripesize , extents ,
2002-01-21 19:05:23 +03:00
vg , pvh ) ) )
2002-02-11 18:42:34 +03:00
goto error ;
2001-11-06 22:02:26 +03:00
2002-02-11 23:50:53 +03:00
if ( arg_count ( cmd , readahead_ARG ) ) {
2001-11-06 22:02:26 +03:00
log_verbose ( " Setting read ahead sectors " ) ;
lv - > read_ahead = read_ahead ;
}
2002-02-11 23:50:53 +03:00
if ( arg_count ( cmd , minor_ARG ) ) {
2002-02-01 20:54:39 +03:00
lv - > status | = FIXED_MINOR ;
2002-02-11 23:50:53 +03:00
lv - > minor = arg_int_value ( cmd , minor_ARG , - 1 ) ;
2002-02-01 20:54:39 +03:00
log_verbose ( " Setting minor number to %d " , lv - > minor ) ;
}
2002-02-11 23:50:53 +03:00
if ( arg_count ( cmd , persistent_ARG ) ) {
if ( ! strcmp ( arg_str_value ( cmd , persistent_ARG , " n " ) , " n " ) )
2002-02-01 20:54:39 +03:00
lv - > status & = ~ FIXED_MINOR ;
else
2002-02-11 23:50:53 +03:00
if ( ! arg_count ( cmd , minor_ARG ) ) {
2002-02-01 20:54:39 +03:00
log_error ( " Please specify minor number with "
" --minor when using -My " ) ;
2002-02-11 18:42:34 +03:00
goto error ;
2002-02-01 20:54:39 +03:00
}
lv - > status | = FIXED_MINOR ;
}
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-11 18:42:34 +03:00
goto error ;
2001-11-06 22:02:26 +03:00
2002-01-07 14:12:11 +03:00
backup ( vg ) ;
2002-01-01 00:27:39 +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
2001-11-06 22:02:26 +03:00
if ( ! lv_activate ( lv ) )
2002-02-11 18:42:34 +03:00
goto error ;
2001-11-06 22:02:26 +03:00
2001-11-14 16:52:38 +03:00
if ( zero ) {
2001-11-06 22:02:26 +03:00
struct device * dev ;
2001-11-16 18:38:52 +03:00
char * name ;
2001-11-06 22:02:26 +03:00
2002-02-11 23:50:53 +03:00
if ( ! ( name = pool_alloc ( cmd - > mem , PATH_MAX ) ) ) {
2001-11-16 18:38:52 +03:00
log_error ( " Name allocation failed - device not zeroed " ) ;
2002-02-11 18:42:34 +03:00
goto error ;
2001-11-16 18:38:52 +03:00
}
2001-11-06 22:02:26 +03:00
2002-02-11 23:50:53 +03:00
if ( lvm_snprintf ( name , PATH_MAX , " %s%s/%s " , cmd - > dev_dir ,
2001-12-31 22:09:51 +03:00
lv - > vg - > name , lv - > name ) < 0 ) {
2002-01-21 19:05:23 +03:00
log_error ( " Name too long - device not zeroed (%s) " ,
2001-12-31 22:09:51 +03:00
lv - > name ) ;
2002-02-11 18:42:34 +03:00
goto error ;
2001-12-31 22:09:51 +03:00
}
2001-11-06 22:02:26 +03:00
2002-01-30 18:04:48 +03:00
log_verbose ( " Zeroing start of logical volume \" %s \" " , name ) ;
2001-11-16 18:38:52 +03:00
if ( ! ( dev = dev_cache_get ( name , NULL ) ) ) {
2002-01-30 18:04:48 +03:00
log_error ( " \" %s \" not found: device not zeroed " , name ) ;
2002-02-11 18:42:34 +03:00
goto error ;
2001-11-06 22:02:26 +03:00
}
2001-11-16 18:38:52 +03:00
if ( ! ( dev_open ( dev , O_WRONLY ) ) )
2002-02-11 18:42:34 +03:00
goto error ;
2001-11-16 18:38:52 +03:00
dev_zero ( dev , 0 , 4096 ) ;
dev_close ( dev ) ;
2001-11-06 22:02:26 +03:00
} else
2002-01-30 18:04:48 +03:00
log_print ( " WARNING: \" %s \" not zeroed " , lv - > name ) ;
2001-11-06 22:02:26 +03:00
2002-02-11 18:42:34 +03:00
lock_vol ( vg_name , LCK_VG | LCK_NONE ) ;
2001-11-06 22:02:26 +03:00
return 0 ;
2002-02-11 18:42:34 +03:00
error :
lock_vol ( vg_name , LCK_VG | LCK_NONE ) ;
return ECMD_FAILED ;
error_cmdline :
lock_vol ( vg_name , LCK_VG | LCK_NONE ) ;
return EINVALID_CMD_LINE ;
2001-11-06 22:02:26 +03:00
}