2018-06-29 12:11:14 +03:00
/*
2019-01-16 17:34:28 +03:00
* Copyright ( C ) 2018 - 2019 Red Hat , Inc . All rights reserved .
2018-06-29 12:11:14 +03: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
* of the GNU Lesser General Public License v .2 .1 .
*
* You should have received a copy of the GNU Lesser General Public License
* along with this program ; if not , write to the Free Software Foundation ,
* Inc . , 51 Franklin Street , Fifth Floor , Boston , MA 02110 - 1301 USA
*/
# include "lib/misc/lib.h"
# include "lib/metadata/metadata.h"
# include "lib/locking/locking.h"
# include "lib/misc/lvm-string.h"
# include "lib/commands/toolcontext.h"
# include "lib/display/display.h"
# include "lib/metadata/segtype.h"
# include "lib/activate/activate.h"
# include "lib/config/defaults.h"
# include "lib/misc/lvm-exec.h"
2023-12-14 16:06:54 +03:00
# include "lib/metadata/lv_alloc.h"
2018-06-29 12:11:14 +03:00
2022-07-09 00:33:29 +03:00
# include <sys/sysinfo.h> // sysinfo
2023-01-15 23:27:37 +03:00
# include <stdarg.h>
2022-07-09 00:33:29 +03:00
2018-07-02 22:58:18 +03:00
const char * get_vdo_compression_state_name ( enum dm_vdo_compression_state state )
{
switch ( state ) {
case DM_VDO_COMPRESSION_ONLINE :
return " online " ;
default :
log_debug ( INTERNAL_ERROR " Unrecognized compression state: %u. " , state ) ;
/* Fall through */
case DM_VDO_COMPRESSION_OFFLINE :
return " offline " ;
}
}
const char * get_vdo_index_state_name ( enum dm_vdo_index_state state )
{
switch ( state ) {
case DM_VDO_INDEX_ERROR :
return " error " ;
case DM_VDO_INDEX_CLOSED :
return " closed " ;
case DM_VDO_INDEX_OPENING :
return " opening " ;
case DM_VDO_INDEX_CLOSING :
return " closing " ;
case DM_VDO_INDEX_OFFLINE :
return " offline " ;
case DM_VDO_INDEX_ONLINE :
return " online " ;
default :
log_debug ( INTERNAL_ERROR " Unrecognized index state: %u. " , state ) ;
/* Fall through */
case DM_VDO_INDEX_UNKNOWN :
return " unknown " ;
}
}
const char * get_vdo_operating_mode_name ( enum dm_vdo_operating_mode mode )
{
switch ( mode ) {
case DM_VDO_MODE_RECOVERING :
return " recovering " ;
case DM_VDO_MODE_READ_ONLY :
return " read-only " ;
default :
log_debug ( INTERNAL_ERROR " Unrecognized operating mode: %u. " , mode ) ;
/* Fall through */
case DM_VDO_MODE_NORMAL :
return " normal " ;
}
}
2018-06-29 12:11:14 +03:00
2019-10-04 15:59:00 +03:00
const char * get_vdo_write_policy_name ( enum dm_vdo_write_policy policy )
{
switch ( policy ) {
case DM_VDO_WRITE_POLICY_SYNC :
return " sync " ;
case DM_VDO_WRITE_POLICY_ASYNC :
return " async " ;
2021-09-06 14:56:40 +03:00
case DM_VDO_WRITE_POLICY_ASYNC_UNSAFE :
return " async-unsafe " ;
2019-10-04 15:59:00 +03:00
default :
log_debug ( INTERNAL_ERROR " Unrecognized VDO write policy: %u. " , policy ) ;
/* Fall through */
case DM_VDO_WRITE_POLICY_AUTO :
return " auto " ;
}
}
2018-06-29 12:11:14 +03:00
/*
* Size of VDO virtual LV is adding header_size in front and back of device
2024-08-30 00:05:41 +03:00
* to avoid collision with blkid checks .
2018-06-29 12:11:14 +03:00
*/
static uint64_t _get_virtual_size ( uint32_t extents , uint32_t extent_size ,
uint32_t header_size )
{
return ( uint64_t ) extents * extent_size + 2 * header_size ;
}
uint64_t get_vdo_pool_virtual_size ( const struct lv_segment * vdo_pool_seg )
{
return _get_virtual_size ( vdo_pool_seg - > vdo_pool_virtual_extents ,
vdo_pool_seg - > lv - > vg - > extent_size ,
vdo_pool_seg - > vdo_pool_header_size ) ;
}
2019-01-16 17:34:28 +03:00
int update_vdo_pool_virtual_size ( struct lv_segment * vdo_pool_seg )
{
struct seg_list * sl ;
uint32_t extents = 0 ;
/* FIXME: as long as we have only SINGLE VDO with vdo-pool this works */
/* after adding support for multiple VDO LVs - this needs heavy rework */
dm_list_iterate_items ( sl , & vdo_pool_seg - > lv - > segs_using_this_lv )
extents + = sl - > seg - > len ;
/* Only growing virtual/logical VDO size */
if ( extents > vdo_pool_seg - > vdo_pool_virtual_extents )
vdo_pool_seg - > vdo_pool_virtual_extents = extents ;
return 1 ;
}
2022-07-09 00:38:34 +03:00
uint32_t get_vdo_pool_max_extents ( const struct dm_vdo_target_params * vtp ,
uint32_t extent_size )
{
uint64_t max_extents = ( DM_VDO_PHYSICAL_SIZE_MAXIMUM + extent_size - 1 ) / extent_size ;
uint64_t max_slab_extents = ( ( extent_size - 1 + DM_VDO_SLABS_MAXIMUM *
( ( uint64_t ) vtp - > slab_size_mb < < ( 20 - SECTOR_SHIFT ) ) ) /
extent_size ) ;
max_extents = ( max_slab_extents < max_extents ) ? max_slab_extents : max_extents ;
return ( max_extents > UINT32_MAX ) ? UINT32_MAX : ( uint32_t ) max_extents ;
}
2021-09-09 15:59:38 +03:00
static int _sysfs_get_kvdo_value ( const char * dm_name , const struct dm_info * dminfo ,
const char * vdo_param , uint64_t * value )
2018-07-02 18:20:30 +03:00
{
char path [ PATH_MAX ] ;
char temp [ 64 ] ;
int fd , size , r = 0 ;
2022-08-10 16:09:34 +03:00
if ( dm_snprintf ( path , sizeof ( path ) , " %sblock/dm-%d/vdo/%s " ,
2021-09-09 15:59:38 +03:00
dm_sysfs_dir ( ) , dminfo - > minor , vdo_param ) < 0 ) {
log_debug ( " Failed to build kvdo path. " ) ;
2018-10-15 16:20:45 +03:00
return 0 ;
2018-07-02 18:20:30 +03:00
}
if ( ( fd = open ( path , O_RDONLY ) ) < 0 ) {
2021-09-09 15:59:38 +03:00
/* try with older location */
if ( dm_snprintf ( path , sizeof ( path ) , " %skvdo/%s/%s " ,
dm_sysfs_dir ( ) , dm_name , vdo_param ) < 0 ) {
log_debug ( " Failed to build kvdo path. " ) ;
return 0 ;
}
if ( ( fd = open ( path , O_RDONLY ) ) < 0 ) {
2018-07-02 18:20:30 +03:00
log_sys_debug ( " open " , path ) ;
2021-09-09 15:59:38 +03:00
goto bad ;
}
2018-07-02 18:20:30 +03:00
}
if ( ( size = read ( fd , temp , sizeof ( temp ) - 1 ) ) < 0 ) {
2021-09-09 15:59:38 +03:00
log_sys_debug ( " read " , path ) ;
2018-07-02 18:20:30 +03:00
goto bad ;
}
temp [ size ] = 0 ;
errno = 0 ;
* value = strtoll ( temp , NULL , 0 ) ;
if ( errno ) {
2021-09-09 15:59:38 +03:00
log_sys_debug ( " strtool " , path ) ;
2018-07-02 18:20:30 +03:00
goto bad ;
}
r = 1 ;
bad :
if ( fd > = 0 & & close ( fd ) )
2021-09-09 15:59:38 +03:00
log_sys_debug ( " close " , path ) ;
2018-07-02 18:20:30 +03:00
return r ;
}
int parse_vdo_pool_status ( struct dm_pool * mem , const struct logical_volume * vdo_pool_lv ,
2021-09-09 15:59:38 +03:00
const char * params , const struct dm_info * dminfo ,
struct lv_status_vdo * status )
2018-07-02 18:20:30 +03:00
{
struct dm_vdo_status_parse_result result ;
char * dm_name ;
2024-05-07 14:24:01 +03:00
uint64_t blocks ;
2018-07-02 18:20:30 +03:00
status - > usage = DM_PERCENT_INVALID ;
status - > saving = DM_PERCENT_INVALID ;
2018-07-05 00:17:38 +03:00
status - > data_usage = DM_PERCENT_INVALID ;
2018-07-02 18:20:30 +03:00
if ( ! ( dm_name = dm_build_dm_name ( mem , vdo_pool_lv - > vg - > name ,
2019-09-14 02:13:33 +03:00
vdo_pool_lv - > name , lv_layer ( vdo_pool_lv ) ) ) ) {
2018-07-02 18:20:30 +03:00
log_error ( " Failed to build VDO DM name %s. " ,
display_lvname ( vdo_pool_lv ) ) ;
return 0 ;
}
if ( ! dm_vdo_status_parse ( mem , params , & result ) ) {
log_error ( " Cannot parse %s VDO pool status %s. " ,
display_lvname ( vdo_pool_lv ) , result . error ) ;
return 0 ;
}
status - > vdo = result . status ;
2021-09-09 15:59:38 +03:00
if ( ( result . status - > operating_mode = = DM_VDO_MODE_NORMAL ) & &
2024-04-05 14:09:10 +03:00
( ( status - > data_blocks_used ! = ULLONG_MAX ) | |
2024-02-19 16:13:20 +03:00
_sysfs_get_kvdo_value ( dm_name , dminfo , " statistics/data_blocks_used " ,
& status - > data_blocks_used ) ) & &
2024-04-05 14:09:10 +03:00
( ( status - > logical_blocks_used ! = ULLONG_MAX ) | |
2024-02-19 16:13:20 +03:00
_sysfs_get_kvdo_value ( dm_name , dminfo , " statistics/logical_blocks_used " ,
& status - > logical_blocks_used ) ) ) {
2018-07-02 18:20:30 +03:00
status - > usage = dm_make_percent ( result . status - > used_blocks ,
result . status - > total_blocks ) ;
status - > saving = dm_make_percent ( status - > logical_blocks_used - status - > data_blocks_used ,
status - > logical_blocks_used ) ;
2024-05-07 14:24:01 +03:00
/* coverity needs to use a local variable to handle check here */
status - > data_usage = dm_make_percent ( ( ( blocks = status - > data_blocks_used ) < ( ULLONG_MAX / DM_VDO_BLOCK_SIZE ) ) ?
( blocks * DM_VDO_BLOCK_SIZE ) : ULLONG_MAX ,
2018-07-05 00:17:38 +03:00
first_seg ( vdo_pool_lv ) - > vdo_pool_virtual_extents *
( uint64_t ) vdo_pool_lv - > vg - > extent_size ) ;
2018-07-02 18:20:30 +03:00
}
return 1 ;
}
2018-06-29 12:11:14 +03:00
/*
* Formats data LV for a use as a VDO pool LV .
*
* Calls tool ' vdoformat ' on the already active volume .
*/
static int _format_vdo_pool_data_lv ( struct logical_volume * data_lv ,
const struct dm_vdo_target_params * vtp ,
uint64_t * logical_size )
{
2022-07-09 01:42:01 +03:00
char * dpath , * c ;
2018-06-29 12:11:14 +03:00
struct pipe_data pdata ;
2022-07-09 01:42:01 +03:00
uint64_t logical_size_aligned = 1 ;
2018-06-29 12:11:14 +03:00
FILE * f ;
uint64_t lb ;
unsigned slabbits ;
2023-07-02 18:46:47 +03:00
unsigned reformatting = 0 ;
2023-07-16 21:03:15 +03:00
int args = 0 ;
2018-06-29 12:11:14 +03:00
char buf [ 256 ] ; /* buffer for short disk header (64B) */
2023-07-16 21:03:15 +03:00
char * buf_pos = buf ;
const char * argv [ DEFAULT_MAX_EXEC_ARGS + 9 ] = { /* Max supported args */
2018-06-29 12:11:14 +03:00
find_config_tree_str_allow_empty ( data_lv - > vg - > cmd , global_vdo_format_executable_CFG , NULL )
} ;
2023-07-16 21:03:15 +03:00
if ( ! prepare_exec_args ( data_lv - > vg - > cmd , argv , & args , global_vdo_format_options_CFG ) )
return_0 ;
2018-06-29 12:11:14 +03:00
if ( ! ( dpath = lv_path_dup ( data_lv - > vg - > cmd - > mem , data_lv ) ) ) {
2023-01-04 08:21:13 +03:00
log_error ( " Failed to build device path for VDO formatting of data volume %s. " ,
2018-06-29 12:11:14 +03:00
display_lvname ( data_lv ) ) ;
return 0 ;
}
if ( * logical_size ) {
2022-07-09 01:42:01 +03:00
logical_size_aligned = 0 ;
2018-06-29 12:11:14 +03:00
2023-07-16 21:03:15 +03:00
argv [ + + args ] = buf_pos ;
buf_pos + = 1 + dm_snprintf ( buf_pos , 30 , " --logical-size= " FMTu64 " K " ,
( * logical_size / 2 ) ) ;
2018-06-29 12:11:14 +03:00
}
2020-02-21 13:59:29 +03:00
slabbits = 31 - clz ( vtp - > slab_size_mb / DM_VDO_BLOCK_SIZE * 2 * 1024 ) ; /* to KiB / block_size */
2018-06-29 12:11:14 +03:00
log_debug ( " Slab size %s converted to %u bits. " ,
display_size ( data_lv - > vg - > cmd , vtp - > slab_size_mb * UINT64_C ( 2 * 1024 ) ) , slabbits ) ;
2023-07-16 21:03:15 +03:00
argv [ + + args ] = buf_pos ;
buf_pos + = 1 + dm_snprintf ( buf_pos , 30 , " --slab-bits=%u " , slabbits ) ;
2018-06-29 12:11:14 +03:00
/* Convert size to GiB units or one of these strings: 0.25, 0.50, 0.75 */
2023-07-16 21:03:15 +03:00
argv [ + + args ] = buf_pos ;
if ( vtp - > index_memory_size_mb > = 1024 )
buf_pos + = 1 + dm_snprintf ( buf_pos , 30 , " --uds-memory-size=%u " ,
vtp - > index_memory_size_mb / 1024 ) ;
else
buf_pos + = 1 + dm_snprintf ( buf_pos , 30 , " --uds-memory-size=0.%2u " ,
( vtp - > index_memory_size_mb < 512 ) ? 25 :
( vtp - > index_memory_size_mb < 768 ) ? 50 : 75 ) ;
if ( vtp - > use_sparse_index )
argv [ + + args ] = " --uds-sparse " ;
2018-06-29 12:11:14 +03:00
/* Only unused VDO data LV could be activated and wiped */
if ( ! dm_list_empty ( & data_lv - > segs_using_this_lv ) ) {
log_error ( INTERNAL_ERROR " Failed to wipe logical VDO data for volume %s. " ,
display_lvname ( data_lv ) ) ;
return 0 ;
}
argv [ args ] = dpath ;
if ( ! ( f = pipe_open ( data_lv - > vg - > cmd , argv , 0 , & pdata ) ) ) {
log_error ( " WARNING: Cannot read output from %s. " , argv [ 0 ] ) ;
return 0 ;
}
2020-02-10 17:41:30 +03:00
while ( ! feof ( f ) & & fgets ( buf , sizeof ( buf ) , f ) ) {
/* TODO: Watch out for locales */
if ( ! * logical_size )
2018-06-29 12:11:14 +03:00
if ( sscanf ( buf , " Logical blocks defaulted to " FMTu64 " blocks " , & lb ) = = 1 ) {
* logical_size = lb * DM_VDO_BLOCK_SIZE ;
log_verbose ( " Available VDO logical blocks " FMTu64 " (%s). " ,
lb , display_size ( data_lv - > vg - > cmd , * logical_size ) ) ;
2020-01-22 14:57:23 +03:00
}
2022-07-09 01:42:01 +03:00
if ( ( c = strchr ( buf , ' \n ' ) ) )
* c = 0 ; /* cut last '\n' away */
2022-08-15 14:08:59 +03:00
if ( buf [ 0 ] ) {
2023-07-02 18:46:47 +03:00
if ( reformatting )
2022-08-15 14:08:59 +03:00
log_verbose ( " %s " , buf ) ; /* Print vdo_format messages */
else
log_print_unless_silent ( " %s " , buf ) ; /* Print vdo_format messages */
}
2020-02-10 17:41:30 +03:00
}
2018-06-29 12:11:14 +03:00
if ( ! pipe_close ( & pdata ) ) {
log_error ( " Command %s failed. " , argv [ 0 ] ) ;
return 0 ;
}
2020-01-22 14:57:23 +03:00
if ( ! * logical_size ) {
log_error ( " Number of VDO logical blocks was not provided by vdo_format output. " ) ;
return 0 ;
}
2022-07-09 01:42:01 +03:00
if ( logical_size_aligned ) {
// align obtained size to extent size
logical_size_aligned = * logical_size / data_lv - > vg - > extent_size * data_lv - > vg - > extent_size ;
if ( * logical_size ! = logical_size_aligned ) {
2022-10-26 15:38:29 +03:00
log_debug ( " Using bigger VDO virtual size unaligned on extent size by %s. " ,
display_size ( data_lv - > vg - > cmd , * logical_size - logical_size_aligned ) ) ;
2022-07-09 01:42:01 +03:00
}
}
2018-06-29 12:11:14 +03:00
return 1 ;
}
/*
* convert_vdo_pool_lv
* @ data_lv
* @ vtp
* @ virtual_extents
*
* Convert given data LV and its target parameters into a VDO LV with VDO pool .
*
* Returns : old data LV on success ( passed data LV becomes VDO LV ) , NULL on failure
*/
2023-11-28 17:50:59 +03:00
int convert_vdo_pool_lv ( struct logical_volume * data_lv ,
const struct dm_vdo_target_params * vtp ,
uint32_t * virtual_extents ,
int format ,
uint64_t vdo_pool_header_size )
2018-06-29 12:11:14 +03:00
{
const uint32_t extent_size = data_lv - > vg - > extent_size ;
struct cmd_context * cmd = data_lv - > vg - > cmd ;
struct logical_volume * vdo_pool_lv = data_lv ;
const struct segment_type * vdo_pool_segtype ;
struct lv_segment * vdo_pool_seg ;
uint64_t vdo_logical_size = 0 ;
uint64_t adjust ;
if ( ! ( vdo_pool_segtype = get_segtype_from_string ( cmd , SEG_TYPE_NAME_VDO_POOL ) ) )
2023-11-28 17:50:59 +03:00
return_0 ;
2018-06-29 12:11:14 +03:00
adjust = ( * virtual_extents * ( uint64_t ) extent_size ) % DM_VDO_BLOCK_SIZE ;
if ( adjust ) {
* virtual_extents + = ( DM_VDO_BLOCK_SIZE - adjust ) / extent_size ;
log_print_unless_silent ( " Rounding size up to 4,00 KiB VDO logical extent boundary: %s. " ,
display_size ( data_lv - > vg - > cmd , * virtual_extents * ( uint64_t ) extent_size ) ) ;
}
if ( * virtual_extents )
vdo_logical_size =
2021-06-28 18:53:08 +03:00
_get_virtual_size ( * virtual_extents , extent_size , vdo_pool_header_size ) ;
2018-06-29 12:11:14 +03:00
if ( ! dm_vdo_validate_target_params ( vtp , vdo_logical_size ) )
return_0 ;
/* Format data LV as VDO volume */
2021-02-16 23:26:35 +03:00
if ( format ) {
2022-08-11 12:44:58 +03:00
if ( test_mode ( ) ) {
2023-01-04 08:21:13 +03:00
log_verbose ( " Test mode: Skipping formatting of VDO pool volume. " ) ;
2022-08-11 12:44:58 +03:00
} else if ( ! _format_vdo_pool_data_lv ( data_lv , vtp , & vdo_logical_size ) ) {
2021-02-16 23:26:35 +03:00
log_error ( " Cannot format VDO pool volume %s. " , display_lvname ( data_lv ) ) ;
2023-11-28 17:50:59 +03:00
return 0 ;
2021-02-16 23:26:35 +03:00
}
} else {
2023-07-02 18:46:47 +03:00
log_verbose ( " Skipping VDO formatting %s. " , display_lvname ( data_lv ) ) ;
2021-02-16 23:26:35 +03:00
/* TODO: parse existing VDO data and retrieve vdo_logical_size */
2021-06-28 19:12:19 +03:00
if ( ! * virtual_extents )
vdo_logical_size = data_lv - > size ;
2018-06-29 12:11:14 +03:00
}
if ( ! deactivate_lv ( data_lv - > vg - > cmd , data_lv ) ) {
2023-07-02 18:46:47 +03:00
log_error ( " Cannot deactivate formatted VDO pool volume %s. " ,
2020-09-15 23:45:49 +03:00
display_lvname ( data_lv ) ) ;
2023-11-28 17:50:59 +03:00
return 0 ;
2018-06-29 12:11:14 +03:00
}
2021-06-28 18:53:08 +03:00
vdo_logical_size - = 2 * vdo_pool_header_size ;
2018-06-29 12:11:14 +03:00
if ( vdo_logical_size < extent_size ) {
if ( ! * virtual_extents )
/* User has not specified size and at least 1 extent is necessary */
log_error ( " Cannot create fully fitting VDO volume, "
" --virtualsize has to be specified. " ) ;
log_error ( " Size %s for VDO volume cannot be smaller then extent size %s. " ,
display_size ( data_lv - > vg - > cmd , vdo_logical_size ) ,
display_size ( data_lv - > vg - > cmd , extent_size ) ) ;
2023-11-28 17:50:59 +03:00
return 0 ;
2018-06-29 12:11:14 +03:00
}
* virtual_extents = vdo_logical_size / extent_size ;
/* Move segments from existing data_lv into LV_vdata */
2024-05-27 14:24:06 +03:00
/* coverity[format_string_injection] lv name is already validated */
2018-06-29 12:11:14 +03:00
if ( ! ( data_lv = insert_layer_for_lv ( cmd , vdo_pool_lv , 0 , " _vdata " ) ) )
2023-11-28 17:50:59 +03:00
return_0 ;
2018-06-29 12:11:14 +03:00
vdo_pool_seg = first_seg ( vdo_pool_lv ) ;
vdo_pool_seg - > segtype = vdo_pool_segtype ;
vdo_pool_seg - > vdo_params = * vtp ;
2021-06-28 18:53:08 +03:00
vdo_pool_seg - > vdo_pool_header_size = vdo_pool_header_size ;
2018-06-29 12:11:14 +03:00
vdo_pool_seg - > vdo_pool_virtual_extents = * virtual_extents ;
vdo_pool_lv - > status | = LV_VDO_POOL ;
data_lv - > status | = LV_VDO_POOL_DATA ;
2023-11-28 17:50:59 +03:00
return 1 ;
2018-06-29 12:11:14 +03:00
}
2023-12-14 16:06:54 +03:00
/*
* Convert LV into vdopool data LV and build virtual VDO LV on top of it .
* After this it swaps these two LVs so the returned LV is VDO LV !
*/
struct logical_volume * convert_vdo_lv ( struct logical_volume * lv ,
const struct vdo_convert_params * vcp )
{
struct cmd_context * cmd = lv - > vg - > cmd ;
char vdopool_name [ NAME_LEN ] , vdopool_tmpl [ NAME_LEN ] ;
struct lvcreate_params lvc = {
. activate = vcp - > activate ,
. alloc = ALLOC_INHERIT ,
. lv_name = vcp - > lv_name ? : lv - > name , /* preserve the name */
. major = - 1 ,
. minor = - 1 ,
. permission = LVM_READ | LVM_WRITE ,
. pool_name = vdopool_name ,
. pvh = & lv - > vg - > pvs ,
. read_ahead = DM_READ_AHEAD_AUTO ,
. stripes = 1 ,
. suppress_zero_warn = 1 , /* suppress warning for this VDO */
. tags = DM_LIST_HEAD_INIT ( lvc . tags ) ,
. virtual_extents = vcp - > virtual_extents ? : lv - > le_count , /* same size for Pool and Virtual LV */
} ;
struct logical_volume * vdo_lv , tmp_lv = {
. segments = DM_LIST_HEAD_INIT ( tmp_lv . segments )
} ;
if ( ! ( lvc . segtype = get_segtype_from_string ( cmd , SEG_TYPE_NAME_VDO ) ) )
return_NULL ;
if ( activation ( ) & &
lvc . segtype - > ops - > target_present & &
! lvc . segtype - > ops - > target_present ( cmd , NULL , & lvc . target_attr ) ) {
log_error ( " %s: Required device-mapper target(s) not detected in your kernel. " ,
lvc . segtype - > name ) ;
return NULL ;
}
if ( ! vcp - > lv_name ) {
/* TODO: maybe _vpool would be sufficient */
if ( dm_snprintf ( vdopool_tmpl , sizeof ( vdopool_tmpl ) , " %s_vpool%%d " , lv - > name ) < 0 ) {
log_error ( " Can't prepare vdo pool name for %s. " , display_lvname ( lv ) ) ;
return NULL ;
}
if ( ! generate_lv_name ( lv - > vg , vdopool_tmpl , vdopool_name , sizeof ( vdopool_name ) ) ) {
log_error ( " Can't generate new name for %s. " , vdopool_tmpl ) ;
return NULL ;
}
/* Rename to use _vpool name and release the passed-in name here */
if ( ! lv_rename_update ( cmd , lv , vdopool_name , 1 ) )
return_NULL ;
} else
lvc . pool_name = lv - > name ;
if ( ! activate_lv ( cmd , lv ) ) {
log_error ( " Aborting. Failed to activate pool metadata %s. " ,
display_lvname ( lv ) ) ;
return NULL ;
}
if ( vcp - > do_zero ) {
if ( test_mode ( ) ) {
log_verbose ( " Test mode: Skipping activation, zeroing and signature wiping. " ) ;
} else if ( ! ( wipe_lv ( lv , ( struct wipe_params )
{
. do_zero = 1 ,
. do_wipe_signatures = vcp - > do_wipe_signatures ,
. yes = vcp - > yes ,
. force = vcp - > force
} ) ) ) {
log_error ( " Aborting. Failed to wipe VDO data store %s. " ,
display_lvname ( lv ) ) ;
return NULL ;
}
}
2024-01-17 19:13:26 +03:00
if ( ! convert_vdo_pool_lv ( lv , & vcp - > vdo_params , & lvc . virtual_extents , vcp - > do_zero , vcp - > header_size ) )
2023-12-14 16:06:54 +03:00
return_NULL ;
/* Create VDO LV with the name, we just release above */
if ( ! ( vdo_lv = lv_create_single ( lv - > vg , & lvc ) ) )
return_NULL ;
if ( vcp - > lv_name )
return vdo_lv ;
/* Swap vdo_lv and lv segment, so passed-in LV appears as virtual VDO_LV */
if ( ! move_lv_segments ( & tmp_lv , lv , 0 , 0 ) | |
! move_lv_segments ( lv , vdo_lv , 0 , 0 ) | |
! move_lv_segments ( vdo_lv , & tmp_lv , 0 , 0 ) )
return_NULL ;
/* Also swap naming, so the passed in LV keeps the passed-in name */
2024-10-24 17:05:28 +03:00
tmp_lv . name = lv - > name ;
lv_set_name ( lv , NULL ) ;
lv_set_name ( vdo_lv , tmp_lv . name ) ;
lv_set_name ( lv , lvc . lv_name ) ;
2023-12-14 16:06:54 +03:00
/* Swap segment referencing */
if ( ! remove_seg_from_segs_using_this_lv ( lv , first_seg ( lv ) ) )
return_NULL ;
if ( ! set_lv_segment_area_lv ( first_seg ( lv ) , 0 , vdo_lv , 0 , 0 ) )
return_NULL ;
return lv ;
}
2019-10-04 15:59:25 +03:00
int set_vdo_write_policy ( enum dm_vdo_write_policy * vwp , const char * policy )
2018-06-29 12:11:14 +03:00
{
if ( strcasecmp ( policy , " sync " ) = = 0 )
* vwp = DM_VDO_WRITE_POLICY_SYNC ;
else if ( strcasecmp ( policy , " async " ) = = 0 )
* vwp = DM_VDO_WRITE_POLICY_ASYNC ;
2021-09-06 14:56:40 +03:00
else if ( strcasecmp ( policy , " async-unsafe " ) = = 0 )
* vwp = DM_VDO_WRITE_POLICY_ASYNC_UNSAFE ;
2018-06-29 12:11:14 +03:00
else if ( strcasecmp ( policy , " auto " ) = = 0 )
* vwp = DM_VDO_WRITE_POLICY_AUTO ;
else {
log_error ( " Unknown VDO write policy %s. " , policy ) ;
return 0 ;
}
2024-09-06 12:39:17 +03:00
if ( * vwp ! = DM_VDO_WRITE_POLICY_AUTO )
log_info ( " Deprecated VDO setting write_policy specified. " ) ;
2018-06-29 12:11:14 +03:00
return 1 ;
}
2018-06-29 14:16:08 +03:00
int fill_vdo_target_params ( struct cmd_context * cmd ,
struct dm_vdo_target_params * vtp ,
2021-06-28 18:53:08 +03:00
uint64_t * vdo_pool_header_size ,
2018-06-29 14:16:08 +03:00
struct profile * profile )
{
const char * policy ;
// TODO: Postpone filling data to the moment when VG is known with profile.
// TODO: Maybe add more lvm cmdline switches to set profile settings.
vtp - > use_compression =
find_config_tree_int ( cmd , allocation_vdo_use_compression_CFG , profile ) ;
vtp - > use_deduplication =
find_config_tree_int ( cmd , allocation_vdo_use_deduplication_CFG , profile ) ;
2018-12-20 15:17:30 +03:00
vtp - > use_metadata_hints =
find_config_tree_int ( cmd , allocation_vdo_use_metadata_hints_CFG , profile ) ;
vtp - > minimum_io_size =
2019-10-04 15:58:18 +03:00
find_config_tree_int ( cmd , allocation_vdo_minimum_io_size_CFG , profile ) > > SECTOR_SHIFT ;
2018-06-29 14:16:08 +03:00
vtp - > block_map_cache_size_mb =
find_config_tree_int64 ( cmd , allocation_vdo_block_map_cache_size_mb_CFG , profile ) ;
2018-12-20 15:17:30 +03:00
vtp - > block_map_era_length =
find_config_tree_int ( cmd , allocation_vdo_block_map_era_length_CFG , profile ) ;
2018-06-29 14:16:08 +03:00
vtp - > use_sparse_index =
find_config_tree_int ( cmd , allocation_vdo_use_sparse_index_CFG , profile ) ;
vtp - > index_memory_size_mb =
find_config_tree_int64 ( cmd , allocation_vdo_index_memory_size_mb_CFG , profile ) ;
vtp - > slab_size_mb =
find_config_tree_int ( cmd , allocation_vdo_slab_size_mb_CFG , profile ) ;
vtp - > ack_threads =
find_config_tree_int ( cmd , allocation_vdo_ack_threads_CFG , profile ) ;
vtp - > bio_threads =
find_config_tree_int ( cmd , allocation_vdo_bio_threads_CFG , profile ) ;
vtp - > bio_rotation =
find_config_tree_int ( cmd , allocation_vdo_bio_rotation_CFG , profile ) ;
vtp - > cpu_threads =
find_config_tree_int ( cmd , allocation_vdo_cpu_threads_CFG , profile ) ;
vtp - > hash_zone_threads =
find_config_tree_int ( cmd , allocation_vdo_hash_zone_threads_CFG , profile ) ;
vtp - > logical_threads =
find_config_tree_int ( cmd , allocation_vdo_logical_threads_CFG , profile ) ;
vtp - > physical_threads =
find_config_tree_int ( cmd , allocation_vdo_physical_threads_CFG , profile ) ;
2018-12-20 15:17:30 +03:00
vtp - > max_discard =
find_config_tree_int ( cmd , allocation_vdo_max_discard_CFG , profile ) ;
2018-06-29 14:16:08 +03:00
policy = find_config_tree_str ( cmd , allocation_vdo_write_policy_CFG , profile ) ;
2019-10-04 15:59:25 +03:00
if ( ! set_vdo_write_policy ( & vtp - > write_policy , policy ) )
2018-06-29 14:16:08 +03:00
return_0 ;
2021-06-28 18:53:08 +03:00
* vdo_pool_header_size = 2 * find_config_tree_int64 ( cmd , allocation_vdo_pool_header_size_CFG , profile ) ;
2024-09-06 12:39:17 +03:00
if ( vtp - > use_metadata_hints )
log_info ( " Deprecated VDO setting use_metadata_hints specified. " ) ;
2018-06-29 14:16:08 +03:00
return 1 ;
}
2022-07-09 00:33:29 +03:00
static int _get_sysinfo_memory ( uint64_t * total_mb , uint64_t * available_mb )
{
struct sysinfo si = { 0 } ;
* total_mb = * available_mb = UINT64_MAX ;
if ( sysinfo ( & si ) ! = 0 )
return 0 ;
2023-09-06 16:28:29 +03:00
log_debug ( " Sysinfo free:%llu bufferram:%llu sharedram:%llu freehigh:%llu unit:%u. " ,
( unsigned long long ) si . freeram > > 20 , ( unsigned long long ) si . bufferram > > 20 , ( unsigned long long ) si . sharedram > > 20 ,
( unsigned long long ) si . freehigh > > 20 , si . mem_unit ) ;
2022-07-09 00:33:29 +03:00
* available_mb = ( ( uint64_t ) ( si . freeram + si . bufferram ) * si . mem_unit ) > > 30 ;
* total_mb = si . totalram > > 30 ;
return 1 ;
}
typedef struct mem_table_s {
const char * name ;
uint64_t * value ;
} mem_table_t ;
static int _compare_mem_table_s ( const void * a , const void * b ) {
return strcmp ( ( ( const mem_table_t * ) a ) - > name , ( ( const mem_table_t * ) b ) - > name ) ;
}
2024-04-15 01:31:30 +03:00
static int _get_memory_info ( struct cmd_context * cmd , uint64_t * total_mb , uint64_t * available_mb )
2022-07-09 00:33:29 +03:00
{
2023-08-24 16:29:22 +03:00
uint64_t anon_pages = 0 , mem_available = 0 , mem_free = 0 , mem_total = 0 , shmem = 0 , swap_free = 0 ;
2022-07-09 00:33:29 +03:00
uint64_t can_swap ;
mem_table_t mt [ ] = {
{ " AnonPages " , & anon_pages } ,
{ " MemAvailable " , & mem_available } ,
{ " MemFree " , & mem_free } ,
{ " MemTotal " , & mem_total } ,
{ " Shmem " , & shmem } ,
{ " SwapFree " , & swap_free } ,
} ;
char line [ 128 ] , namebuf [ 32 ] , * e , * tail ;
2024-04-15 01:31:30 +03:00
char proc_meminfo [ PATH_MAX ] ;
2022-07-09 00:33:29 +03:00
FILE * fp ;
mem_table_t findme = { namebuf , NULL } ;
mem_table_t * found ;
2024-04-15 01:31:30 +03:00
if ( ( dm_snprintf ( proc_meminfo , sizeof ( proc_meminfo ) ,
" %s/meminfo " , cmd - > proc_dir ) < 0 ) | |
! ( fp = fopen ( proc_meminfo , " r " ) ) )
2022-07-09 00:33:29 +03:00
return _get_sysinfo_memory ( total_mb , available_mb ) ;
while ( fgets ( line , sizeof ( line ) , fp ) ) {
if ( ! ( e = strchr ( line , ' : ' ) ) )
break ;
2023-07-13 13:32:44 +03:00
if ( ( unsigned ) ( + + e - line ) > sizeof ( namebuf ) )
2022-07-09 00:33:29 +03:00
continue ; // something too long
2024-04-04 18:03:30 +03:00
dm_strncpy ( ( char * ) findme . name , line , e - line ) ;
2022-07-09 00:33:29 +03:00
found = bsearch ( & findme , mt , DM_ARRAY_SIZE ( mt ) , sizeof ( mem_table_t ) ,
_compare_mem_table_s ) ;
if ( ! found )
continue ; // not interesting
2022-08-18 14:55:29 +03:00
errno = 0 ;
2022-07-09 00:33:29 +03:00
* ( found - > value ) = ( uint64_t ) strtoull ( e , & tail , 10 ) ;
if ( ( e = = tail ) | | errno )
log_debug ( " Failing to parse value from %s. " , line ) ;
else
log_debug ( " Parsed %s = " FMTu64 " KiB. " , found - > name , * ( found - > value ) ) ;
}
( void ) fclose ( fp ) ;
// use at most 2/3 of swap space to keep machine usable
can_swap = ( anon_pages + shmem ) * 2 / 3 ;
swap_free = swap_free * 2 / 3 ;
if ( can_swap > swap_free )
can_swap = swap_free ;
// TODO: add more constrains, i.e. 3/4 of physical RAM...
* total_mb = mem_total > > 10 ;
* available_mb = ( mem_available + can_swap ) > > 10 ;
return 1 ;
}
static uint64_t _round_1024 ( uint64_t s )
{
return ( s + ( ( 1 < < 10 ) - 1 ) ) > > 10 ;
}
static uint64_t _round_sectors_to_tib ( uint64_t s )
{
return ( s + ( ( UINT64_C ( 1 ) < < ( 40 - SECTOR_SHIFT ) ) - 1 ) ) > > ( 40 - SECTOR_SHIFT ) ;
}
2023-01-15 23:27:37 +03:00
__attribute__ ( ( format ( printf , 3 , 4 ) ) )
static int _vdo_snprintf ( char * * buf , size_t * bufsize , const char * format , . . . )
2022-07-09 00:33:29 +03:00
{
2023-01-15 23:27:37 +03:00
int n ;
va_list ap ;
va_start ( ap , format ) ;
n = vsnprintf ( * buf , * bufsize , format , ap ) ;
va_end ( ap ) ;
if ( n < 0 | | ( ( unsigned ) n > = * bufsize ) )
return - 1 ;
* buf + = n ;
* bufsize - = n ;
return n ;
}
int check_vdo_constrains ( struct cmd_context * cmd , const struct vdo_pool_size_config * cfg )
{
2024-05-03 17:37:27 +03:00
static const char _vdo_split [ ] [ 4 ] = { " " , " and " , " , " , " , " } ;
2022-07-09 00:33:29 +03:00
uint64_t req_mb , total_mb , available_mb ;
2023-01-15 23:27:37 +03:00
uint64_t phy_mb = _round_sectors_to_tib ( UINT64_C ( 268 ) * cfg - > physical_size ) ; // 268 MiB per 1 TiB of physical size
uint64_t virt_mb = _round_1024 ( UINT64_C ( 1638 ) * _round_sectors_to_tib ( cfg - > virtual_size ) ) ; // 1.6 MiB per 1 TiB
uint64_t cache_mb = _round_1024 ( UINT64_C ( 1177 ) * cfg - > block_map_cache_size_mb ) ; // 1.15 MiB per 1 MiB cache size
2022-07-09 00:33:29 +03:00
char msg [ 512 ] ;
2023-01-15 23:27:37 +03:00
size_t mlen = sizeof ( msg ) ;
char * pmsg = msg ;
int cnt , has_cnt ;
2022-07-09 00:33:29 +03:00
2023-01-15 23:27:37 +03:00
if ( cfg - > block_map_cache_size_mb & & ( cache_mb < 150 ) )
2022-07-09 00:33:29 +03:00
cache_mb = 150 ; // always at least 150 MiB for block map
// total required memory for VDO target
2023-01-15 23:27:37 +03:00
req_mb = 38 + cfg - > index_memory_size_mb + virt_mb + phy_mb + cache_mb ;
2022-07-09 00:33:29 +03:00
2024-04-15 01:31:30 +03:00
_get_memory_info ( cmd , & total_mb , & available_mb ) ;
2022-07-09 00:33:29 +03:00
2023-01-15 23:27:37 +03:00
has_cnt = cnt = ( phy_mb ? 1 : 0 ) +
( virt_mb ? 1 : 0 ) +
( cfg - > block_map_cache_size_mb ? 1 : 0 ) +
( cfg - > index_memory_size_mb ? 1 : 0 ) ;
if ( phy_mb )
( void ) _vdo_snprintf ( & pmsg , & mlen , " %s RAM for physical volume size %s%s " ,
display_size ( cmd , phy_mb < < ( 20 - SECTOR_SHIFT ) ) ,
2024-05-03 17:37:27 +03:00
display_size ( cmd , cfg - > physical_size ) , _vdo_split [ - - cnt ] ) ;
2023-01-15 23:27:37 +03:00
if ( virt_mb )
( void ) _vdo_snprintf ( & pmsg , & mlen , " %s RAM for virtual volume size %s%s " ,
display_size ( cmd , virt_mb < < ( 20 - SECTOR_SHIFT ) ) ,
2024-05-03 17:37:27 +03:00
display_size ( cmd , cfg - > virtual_size ) , _vdo_split [ - - cnt ] ) ;
2023-01-15 23:27:37 +03:00
if ( cfg - > block_map_cache_size_mb )
( void ) _vdo_snprintf ( & pmsg , & mlen , " %s RAM for block map cache size %s%s " ,
display_size ( cmd , cache_mb < < ( 20 - SECTOR_SHIFT ) ) ,
display_size ( cmd , ( ( uint64_t ) cfg - > block_map_cache_size_mb ) < < ( 20 - SECTOR_SHIFT ) ) ,
2024-05-03 17:37:27 +03:00
_vdo_split [ - - cnt ] ) ;
2023-01-15 23:27:37 +03:00
if ( cfg - > index_memory_size_mb )
( void ) _vdo_snprintf ( & pmsg , & mlen , " %s RAM for index memory " ,
display_size ( cmd , ( ( uint64_t ) cfg - > index_memory_size_mb ) < < ( 20 - SECTOR_SHIFT ) ) ) ;
2022-07-09 00:33:29 +03:00
if ( req_mb > available_mb ) {
log_error ( " Not enough free memory for VDO target. %s RAM is required, but only %s RAM is available. " ,
display_size ( cmd , req_mb < < ( 20 - SECTOR_SHIFT ) ) ,
display_size ( cmd , available_mb < < ( 20 - SECTOR_SHIFT ) ) ) ;
2023-01-15 23:27:37 +03:00
if ( has_cnt )
log_print_unless_silent ( " VDO configuration needs%s. " , msg ) ;
2022-07-09 00:33:29 +03:00
return 0 ;
}
log_debug ( " VDO requires %s RAM, currently available %s RAM. " ,
display_size ( cmd , req_mb < < ( 20 - SECTOR_SHIFT ) ) ,
display_size ( cmd , available_mb < < ( 20 - SECTOR_SHIFT ) ) ) ;
2023-01-15 23:27:37 +03:00
if ( has_cnt )
log_verbose ( " VDO configuration needs%s. " , msg ) ;
2022-07-09 00:33:29 +03:00
return 1 ;
}