2021-07-09 15:44:07 +03:00
#!/bin/bash
#
2023-05-11 21:07:37 +03:00
# Copyright (C) 2021-2023 Red Hat, Inc. All rights reserved.
2021-07-09 15:44:07 +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 General Public License v.2.
#
# You should have received a copy of the GNU 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
#
# Author: Zdenek Kabelac <zkabelac at redhat.com>
#
2021-08-23 15:02:42 +03:00
# Script for importing VDO volumes to lvm2 managed VDO LVs
2021-07-09 15:44:07 +03:00
#
# Needed utilities:
# lvm, dmsetup,
2021-08-23 15:02:42 +03:00
# vdo,
2023-05-11 21:07:37 +03:00
# grep, awk, sed, blockdev, readlink, stat, mkdir, truncate
2021-07-09 15:44:07 +03:00
#
# Conversion is using 'vdo convert' support from VDO manager to move
# existing VDO header by 2M which makes space to place in PV header
# and VG metadata area, and then create VDOPOOL LV and VDO LV in such VG.
#
set -euE -o pipefail
2021-08-26 17:53:33 +03:00
TOOL = lvm_import_vdo
2023-05-11 21:07:37 +03:00
IMPORT_NAME = " VDO_ ${ TOOL } _ ${ RANDOM } $$ "
test ${# IMPORT_NAME } -lt 100 || error " Random name \" $IMPORT_NAME \" is too long! "
TEMPDIR = " ${ TMPDIR :- /tmp } / $IMPORT_NAME "
2021-07-09 15:44:07 +03:00
_SAVEPATH = $PATH
PATH = " /sbin:/usr/sbin:/bin:/usr/sbin: $PATH "
# user may override lvm location by setting LVM_BINARY
LVM = ${ LVM_BINARY :- lvm }
VDO = ${ VDO_BINARY :- vdo }
BLOCKDEV = "blockdev"
2023-05-11 21:07:37 +03:00
LOSETUP = "losetup"
2021-07-09 15:44:07 +03:00
READLINK = "readlink"
READLINK_E = "-e"
2021-09-01 16:46:04 +03:00
STAT = "stat"
2021-07-09 15:44:07 +03:00
MKDIR = "mkdir"
2023-05-11 21:07:37 +03:00
TRUNCATE = "truncate"
2021-08-31 21:52:26 +03:00
DMSETUP = "dmsetup"
2021-07-09 15:44:07 +03:00
DM_DEV_DIR = " ${ DM_DEV_DIR :- /dev } "
2023-05-11 21:07:37 +03:00
DM_UUID_PREFIX = " ${ DM_UUID_PREFIX :- } "
DM_VG_NAME =
DM_LV_NAME =
2023-06-29 14:05:55 +03:00
VDO_CONFIG = ${ VDO_CONFIG :- } # can be overridden with --vdo-config
2023-05-11 21:07:37 +03:00
VDOCONF =
test -n " $VDO_CONFIG " && VDOCONF = " -f $VDO_CONFIG "
DEVICE =
VGNAME =
LVNAME =
2021-08-31 21:52:26 +03:00
DEVMAJOR = 0
DEVMINOR = 0
2021-09-06 16:06:32 +03:00
PROMPTING = ""
2023-05-11 21:07:37 +03:00
USE_VDO_DM_SNAPSHOT = 1
VDO_DM_SNAPSHOT_NAME =
VDO_DM_SNAPSHOT_DEVICE =
VDO_SNAPSHOT_LOOP =
2021-08-31 21:52:26 +03:00
2021-07-09 15:44:07 +03:00
DRY = 0
VERB = ""
FORCE = ""
YES = ""
2023-05-11 21:07:37 +03:00
ABORT_AFTER_VDO_CONVERT = 0
VDO_ALLOCATION_PARAMS =
2021-07-09 15:44:07 +03:00
# default name for converted VG and its VDO LV
2021-09-06 16:06:32 +03:00
DEFAULT_NAME = "vdovg/vdolvol"
NAME = ""
2021-07-09 15:44:07 +03:00
# help message
tool_usage( ) {
echo " ${ TOOL } : Utility to convert VDO volume to VDO LV. "
echo
echo " ${ TOOL } [options] <vdo_device_path> "
echo
echo " Options:"
echo " -f | --force Bypass sanity checks"
echo " -h | --help Show this help message"
echo " -n | --name Specifies VG/LV name for converted VDO volume"
echo " -v | --verbose Be verbose"
echo " -y | --yes Answer \"yes\" at any prompts"
2021-09-17 17:49:17 +03:00
echo " --dry-run Print verbosely commands without running them"
2023-05-11 21:07:37 +03:00
echo " --no-snapshot Do not use snapshot for converted VDO device"
echo " --uuid-prefix Prefix for DM snapshot uuid"
echo " --vdo-config Configuration file for VDO manager"
2021-07-09 15:44:07 +03:00
exit
}
verbose( ) {
test -z " $VERB " || echo " $TOOL : " " $@ "
}
# Support multi-line error messages
error( ) {
for i in " $@ " ; do
echo " $TOOL : $i " >& 2
done
cleanup 1
}
dry( ) {
if [ " $DRY " -ne 0 ] ; then
verbose "Dry execution" " $@ "
return 0
fi
verbose "Executing" " $@ "
" $@ "
}
cleanup( ) {
trap '' 2
2023-05-11 21:07:37 +03:00
test -n " $VDO_DM_SNAPSHOT_NAME " && { " $DMSETUP " remove " $VDO_DM_SNAPSHOT_NAME " || true ; }
test -n " $VDO_SNAPSHOT_LOOP " && { " $LOSETUP " -d " $VDO_SNAPSHOT_LOOP " || true ; }
2021-09-06 16:06:32 +03:00
test -z " $PROMPTING " || echo "No"
2023-05-11 21:07:37 +03:00
rm -rf " $TEMPDIR " || true
2021-07-09 15:44:07 +03:00
# error exit status for break
exit " ${ 1 :- 1 } "
}
2023-05-11 21:07:37 +03:00
# Create snapshot target like for persistent snapshot with 16KiB chunksize
snapshot_target_line_( ) {
echo " 0 $( " $BLOCKDEV " --getsize " $1 " ) snapshot ${ 3 :- } $1 $2 P 32 "
}
snapshot_create_( ) {
VDO_DM_SNAPSHOT_NAME = " ${ IMPORT_NAME } _snap "
local file = " $TEMPDIR / $VDO_DM_SNAPSHOT_NAME "
# TODO: maybe use ramdisk via 'brd' device ?)
" $TRUNCATE " -s 20M " $file "
VDO_SNAPSHOT_LOOP = $( " $LOSETUP " -f --show " $file " )
" $DMSETUP " create " $VDO_DM_SNAPSHOT_NAME " -u " ${ DM_UUID_PREFIX } - ${ VDO_DM_SNAPSHOT_NAME } -priv " --table " $( snapshot_target_line_ " $1 " " $VDO_SNAPSHOT_LOOP " ) "
VDO_DM_SNAPSHOT_DEVICE = " $DM_DEV_DIR /mapper/ $VDO_DM_SNAPSHOT_NAME "
verbose " Snapshot of VDO device $1 created: $VDO_DM_SNAPSHOT_DEVICE . "
}
snapshot_merge_( ) {
local status
2023-06-29 14:05:55 +03:00
local initial_status
2023-05-11 21:07:37 +03:00
initial_status = ( $( " $DMSETUP " status " $VDO_DM_SNAPSHOT_NAME " ) )
" $DMSETUP " reload " $VDO_DM_SNAPSHOT_NAME " --table " $( snapshot_target_line_ " $1 " " $VDO_SNAPSHOT_LOOP " -merge) "
" $DMSETUP " suspend " $VDO_DM_SNAPSHOT_NAME " || {
error "ABORTING: Failed to initialize snapshot merge! Origin volume is unchanged."
}
verbose "Merging converted VDO volume..."
# Running merging
" $DMSETUP " resume " $VDO_DM_SNAPSHOT_NAME "
#du -h "$TEMPDIR/$VDO_DM_SNAPSHOT_NAME"
# Loop for a while, till the snapshot is merged.
2023-06-29 14:05:55 +03:00
# Should be nearly instantaneous.
2023-05-11 21:07:37 +03:00
# FIXME: Recovery when something prevents merging is hard
for i in $( seq 1 20) ; do
status = ( $( " $DMSETUP " status " $VDO_DM_SNAPSHOT_NAME " ) )
# Check if merging is finished
test " ${ status [3]%/* } " = " ${ status [4] } " && break
# Wait a bit and retry
sleep .2
done
test " ${ status [3]%/* } " = " ${ status [4] } " || {
# FIXME: Now what shall we do ??? Help....
# Keep snapshot in table for possible analysis...
VDO_DM_SNAPSHOT_NAME =
VDO_SNAPSHOT_LOOP =
echo " Initial snapshot status ${ initial_status [*] } "
echo " Failing merge snapshot status ${ status [*] } "
error "ABORTING: Snapshot failed to merge! (Administrator required...)"
}
sync
" $DMSETUP " remove " $VDO_DM_SNAPSHOT_NAME " || {
sleep 1 # sleep and retry once more
" $DMSETUP " remove " $VDO_DM_SNAPSHOT_NAME " || {
error " ABORTING: Cannot remove snapshot $VDO_DM_SNAPSHOT_NAME ! (check volume autoactivation...) "
}
}
VDO_DM_SNAPSHOT_NAME =
" $LOSETUP " -d " $VDO_SNAPSHOT_LOOP "
VDO_SNAPSHOT_LOOP =
}
2021-07-09 15:44:07 +03:00
get_enabled_value_( ) {
case " $1 " in
enabled) echo "1" ; ;
*) echo "0" ; ;
esac
}
get_kb_size_with_unit_( ) {
case " $1 " in
*[ kK] ) echo $(( ${ 1 %[kK] } )) ; ;
*[ mM] ) echo $(( ${ 1 %[mM] } * 1024 )) ; ;
*[ gG] ) echo $(( ${ 1 %[gG] } * 1024 * 1024 )) ; ;
*[ tT] ) echo $(( ${ 1 %[tT] } * 1024 * 1024 * 1024 )) ; ;
*[ pP] ) echo $(( ${ 1 %[pP] } * 1024 * 1024 * 1024 * 1024 )) ; ;
esac
}
# Figure out largest possible extent size usable for VG
# $1 physical size
# $2 logical size
get_largest_extent_size_( ) {
local max = 4
local i
local d
for i in 8 16 32 64 128 256 512 1024 2048 4096 ; do
d = $(( $1 / i ))
test $(( d * i )) -eq " $1 " || break
d = $(( $2 / i ))
test $(( d * i )) -eq " $2 " || break
max = $i
done
echo " $max "
}
# detect LV on the given device
2023-06-29 14:05:55 +03:00
# deference device name if it is symbolic link
2021-07-09 15:44:07 +03:00
detect_lv_( ) {
local DEVICE = $1
local SYSVOLUME
local MAJORMINOR
DEVICE = ${ 1 /# " ${ DM_DEV_DIR } / " / }
2021-09-01 16:46:04 +03:00
DEVICE = $( " $READLINK " $READLINK_E " $DM_DEV_DIR / $DEVICE " || true )
test -n " $DEVICE " || error " Readlink cannot access device \" $1 \". "
2021-07-09 15:44:07 +03:00
RDEVICE = $DEVICE
case " $RDEVICE " in
# hardcoded /dev since udev does not create these entries elsewhere
/dev/dm-[ 0-9] *)
read -r <" /sys/block/ ${ RDEVICE #/dev/ } /dm/name " SYSVOLUME 2>& 1 && DEVICE = " $DM_DEV_DIR /mapper/ $SYSVOLUME "
read -r <" /sys/block/ ${ RDEVICE #/dev/ } /dev " MAJORMINOR 2>& 1 || error " Cannot get major:minor for \" $DEVICE \". "
2021-08-31 21:52:26 +03:00
DEVMAJOR = ${ MAJORMINOR %% : * }
DEVMINOR = ${ MAJORMINOR ##* : }
2021-07-09 15:44:07 +03:00
; ;
*)
2021-09-01 16:46:04 +03:00
RSTAT = $( " $STAT " --format "DEVMAJOR=\$((0x%t)) DEVMINOR=\$((0x%T))" " $RDEVICE " || true )
test -n " $RSTAT " || error " Cannot get major:minor for \" $DEVICE \". "
eval " $RSTAT "
2021-07-09 15:44:07 +03:00
; ;
esac
2021-09-06 16:06:32 +03:00
test " $DEVMAJOR " != " $( grep device-mapper /proc/devices | cut -f1 -d' ' ) " && return
DEV = " $( " $DMSETUP " info -c -j " $DEVMAJOR " -m " $DEVMINOR " -o uuid,name --noheadings --nameprefixes --separator ' ' ) "
2021-08-31 21:52:26 +03:00
case " $DEV " in
Device*) ; ; # no devices
*) eval " $DEV " ; ;
esac
2021-07-09 15:44:07 +03:00
}
# parse yaml config files into 'prefix_yaml_part_names=("value")' strings
parse_yaml_( ) {
local yaml_file = $1
local prefix = $2
local s
local w
local fs
s = '[[:space:]]*'
w = '[a-zA-Z0-9_.-]*'
fs = " $( echo @| tr @ '\034' ) "
(
sed -ne '/^--/s|--||g; s|\"|\\\"|g; s/[[:space:]]*$//g;' \
-e 's/\$/\\\$/g' \
-e "/#.*[\"\']/!s| #.*||g; /^#/s|#.*||g;" \
-e " s|^\( $s \)\( $w \) $s : $s \"\(.*\)\" $s \$|\1 $fs \2 $fs \3|p " \
-e " s|^\( $s \)\( $w \) ${ s } [:-] $s \(.*\) $s \$|\1 $fs \2 $fs \3|p " |
awk -F" $fs " ' {
indent = length( $1 ) /2;
if ( length( $2 ) = = 0) { conj[ indent] = "+" ; } else { conj[ indent] = "" ; }
vname[ indent] = $2 ;
for ( i in vname) { if ( i > indent) { delete vname[ i] } }
if ( length( $3 ) > 0) {
vn = "" ; for ( i = 0; i<indent; i++) { vn = ( vn) ( vname[ i] ) ( "_" ) }
printf( "%s%s%s%s=(\"%s\")\n" , "'" $prefix "'" ,vn, $2 , conj[ indent-1] , $3 ) ;
}
} ' |
sed -e 's/_=/+=/g' |
awk ' BEGIN {
FS = "=" ;
OFS = "="
}
/( -| \. ) .*= / {
gsub( "-|\\." , "_" , $1 )
}
{ print } '
) < " $yaml_file "
}
2023-05-11 21:07:37 +03:00
#
# Convert VDO volume on LV to VDOPool within this VG
#
# This conversion requires the size of VDO virtual volume has to be expressed in the VG's extent size.
# Currently this enforces a user to reduce the VG extent size to the smaller size (up to 4KiB).
#
# TODO: We may eventually relax this condition just like we are doing rounding for convert_non_lv_()
2023-06-29 14:05:55 +03:00
# Let's if there would be any singly user requiring this feature.
2023-05-11 21:07:37 +03:00
# It may allow to better use larger VDO volume size (in TiB ranges).
#
convert_lv_( ) {
local vdo_logicalSize = $1
local extent_size
local pvfree
pvfree = $( " $LVM " lvs -o size --units b --nosuffix --noheadings " $DM_VG_NAME / $DM_LV_NAME " )
pvfree = $(( pvfree / 1024 )) # to KiB
# select largest possible extent size that can exactly express both sizes
extent_size = $( get_largest_extent_size_ " $pvfree " " $vdo_logicalSize " )
# validate existing VG extent_size can express virtual VDO size
vg_extent_size = $( " $LVM " vgs -o vg_extent_size --units b --nosuffix --noheadings " $VGNAME " )
vg_extent_size = $(( vg_extent_size / 1024 ))
test " $vg_extent_size " -le " $extent_size " || {
error " Please vgchange extent_size to at most $extent_size KiB or extend and align virtual size of VDO device on $vg_extent_size KiB before retrying conversion. "
}
verbose "Renaming existing LV to be used as _vdata volume for VDO pool LV."
dry " $LVM " lvrename $YES $VERB " $VGNAME / $DM_LV_NAME " " $VGNAME / ${ LVNAME } _vpool " || {
error " Rename of LV \" $VGNAME / $DM_LV_NAME \" failed, while VDO header has been already moved! "
}
verbose "Converting to VDO pool."
dry " $LVM " lvconvert $YES $VERB $FORCE --config " $VDO_ALLOCATION_PARAMS " -Zn -V " ${ vdo_logicalSize } k " -n " $LVNAME " --type vdo-pool " $VGNAME / ${ LVNAME } _vpool "
verbose "Removing now unused VDO entry from VDO configuration."
dry " $VDO " remove $VDOCONF $VERB --force --name " $VDONAME "
}
#
# Convert VDO volume on a device to VG with VDOPool LV
#
# Convert device with the use of snapshot on top of original VDO volume (can be optionally disabled)
2023-06-29 14:05:55 +03:00
# Once the whole conversion is finished, snapshot is merged (During the short period time of merging
2023-05-11 21:07:37 +03:00
# user must ensure there will be no power-off!)
#
# For best use the latest version of vdoprepareforlvm tool is required.
convert_non_lv_( ) {
local vdo_logicalSize = $1
2023-06-23 19:01:32 +03:00
local vdo_logicalSizeRounded
2023-05-11 21:07:37 +03:00
local extent_size
local output
local pvfree
if [ " $USE_VDO_DM_SNAPSHOT " = "1" ] ; then
dry snapshot_create_ " $DEVICE "
sed " s: $DEVICE : $VDO_DM_SNAPSHOT_DEVICE : " " $TEMPDIR /vdoconf.yml " > " $TEMPDIR /vdo_snap.yml "
# Let VDO manager operate on snapshot volume
VDOCONF = " -f $TEMPDIR /vdo_snap.yml "
fi
verbose "Moving VDO header."
2023-06-29 14:06:18 +03:00
output = $( dry " $VDO " convert $VDOCONF $VERB --force --name " $VDONAME " )
2023-05-11 21:07:37 +03:00
if [ " $ABORT_AFTER_VDO_CONVERT " != "0" ] ; then
2023-06-29 14:05:55 +03:00
verbose "Aborting VDO conversion after moving VDO header, volume is useless!"
2023-05-11 21:07:37 +03:00
cleanup 0
fi
# Parse result from VDO preparation/conversion tool
2023-06-29 14:05:55 +03:00
# New version of the tool provides output with alignment and offset
2023-05-11 21:07:37 +03:00
local vdo_length = 0
local vdo_aligned = 0
local vdo_offset = 0
local vdo_non_converted = 0
while IFS = read -r line ; do
2023-06-29 14:06:18 +03:00
# trim leading spaces
case " $( echo $line ) " in
2023-05-11 21:07:37 +03:00
"Non converted" *) vdo_non_converted = 1 ; ;
"Length" *) vdo_length = ${ line ##* = } ; ;
"Conversion completed" *)
vdo_aligned = ${ line ##*aligned on }
vdo_aligned = ${ vdo_aligned %%[!0-9]* }
vdo_offset = ${ line ##*offset }
# backward compatibility with report from older version
vdo_offset = ${ vdo_offset ##*by }
vdo_offset = ${ vdo_offset %%[!0-9]* }
; ;
esac
done <<< " $output "
# In case we operation with snapshot, all lvm2 operation will also run on top of snapshot
local devices = ${ VDO_DM_SNAPSHOT_DEVICE :- $DEVICE }
dry " $LVM " pvcreate $YES --devices " $devices " --dataalignment " $vdo_offset " b " $devices " || {
error " Creation of PV on \" $DEVICE \" failed, while VDO header has been already moved! "
}
# Obtain free space in this new PV
# after 'vdo convert' call there is ~(1-2)M free space at the front of the device
pvfree = $( " $BLOCKDEV " --getsize64 " $DEVICE " )
pvfree = $(( ( pvfree - vdo_offset ) / 1024 )) # to KiB
if [ -n " $vdo_aligned " ] && [ " $vdo_aligned " != "0" ] ; then
extent_size = $(( vdo_aligned / 1024 ))
else
extent_size = $( get_largest_extent_size_ " $pvfree " " $vdo_logicalSize " )
fi
# Round virtual size to the LOWER size expressed in extent units.
# lvm is parsing VDO metadata and can read real full size and use it instead of this smaller value.
# To precisely byte-synchronize the size of VDO LV, user can lvresize such VDO LV later.
2023-06-23 19:01:32 +03:00
vdo_logicalSizeRounded = $(( ( vdo_logicalSize / extent_size ) * extent_size ))
2023-05-11 21:07:37 +03:00
verbose " Creating VG \" ${ NAME %/* } \" with extent size $extent_size KiB. "
dry " $LVM " vgcreate $YES $VERB --devices " $devices " -s " ${ extent_size } k " " $VGNAME " " $devices " || {
error " Creation of VG \" $VGNAME \" failed, while VDO header has been already moved! "
}
verbose " Creating VDO pool data LV from all extents in volume group $VGNAME . "
dry " $LVM " lvcreate -Zn -Wn -an $YES $VERB --devices " $devices " -l100%VG -n " ${ LVNAME } _vpool " " $VGNAME " " $devices "
verbose "Converting to VDO pool."
2023-06-23 19:01:32 +03:00
dry " $LVM " lvconvert $YES $VERB $FORCE --devices " $devices " --config " $VDO_ALLOCATION_PARAMS " -Zn -V " ${ vdo_logicalSizeRounded } k " -n " $LVNAME " --type vdo-pool " $VGNAME / ${ LVNAME } _vpool "
if [ " $vdo_logicalSizeRounded " -lt " $vdo_logicalSize " ] ; then
2023-06-29 14:05:55 +03:00
# need to extend virtual size to be covering all the converted area
2023-06-23 19:01:32 +03:00
# let lvm2 to round to the proper virtual size of VDO LV
dry " $LVM " lvextend $YES $VERB --fs ignore --devices " $devices " -L " $vdo_logicalSize " k " $VGNAME / $LVNAME "
fi
2023-05-11 21:07:37 +03:00
dry " $LVM " vgchange -an $VERB $FORCE --devices " $devices " " $VGNAME "
if [ " $USE_VDO_DM_SNAPSHOT " = "1" ] ; then
if [ -z " $YES " ] ; then
PROMPTING = yes
echo "Warning: Do not interrupt merging process once it starts (VDO data may become irrecoverable)!"
echo -n " Do you want to merge converted VDO device \" $DEVICE \" to VDO LV \" $VGNAME / $LVNAME \"? [y|N]: "
read -r -n 1 -s ANSWER
case " ${ ANSWER : 0 : 1 } " in
y| Y ) echo "Yes" ; ;
* ) echo "No" ; PROMPTING = "" ; cleanup 1 ; ;
esac
PROMPTING = ""
YES = "-y" # From now, now prompting
fi
dry snapshot_merge_ " $DEVICE "
if [ -e " $TEMPDIR /vdo_snap.yml " ] ; then
dry cp " $TEMPDIR /vdo_snap.yml " " $VDO_CONFIG "
else
dry rm -f " $VDO_CONFIG "
fi
verbose "Merging of VDO device finished."
fi
dry " $LVM " lvchange -ay $VERB $FORCE " $VGNAME / $LVNAME "
}
# Convert existing VDO volume into lvm2 volume
2021-07-09 15:44:07 +03:00
convert2lvm_( ) {
local VDONAME
local TRVDONAME
2021-08-31 21:52:26 +03:00
local FOUND = ""
local MAJOR = 0
local MINOR = 0
2021-07-09 15:44:07 +03:00
2023-05-11 21:07:37 +03:00
VGNAME = ${ NAME %/* }
LVNAME = ${ NAME #*/ }
2021-07-09 15:44:07 +03:00
DM_UUID = ""
detect_lv_ " $DEVICE "
case " $DM_UUID " in
2021-08-31 21:52:26 +03:00
LVM-*) eval " $( " $DMSETUP " splitname --nameprefixes --noheadings --separator ' ' " $DM_NAME " ) "
2021-09-06 16:06:32 +03:00
if [ -z " $VGNAME " ] || [ " $VGNAME " = " $LVNAME " ] ; then
2021-07-09 15:44:07 +03:00
VGNAME = $DM_VG_NAME
2023-05-11 21:07:37 +03:00
verbose " Using existing volume group name \" $VGNAME \". "
2021-09-06 16:06:32 +03:00
test -n " $LVNAME " || LVNAME = $DM_LV_NAME
2023-05-11 21:07:37 +03:00
elif [ " $VGNAME " != " $DM_VG_NAME " ] ; then
2021-09-06 16:06:32 +03:00
error " Volume group name \" $VGNAME \" does not match name \" $DM_VG_NAME \" for VDO device \" $DEVICE \". "
2021-07-09 15:44:07 +03:00
fi
; ;
2023-05-11 21:07:37 +03:00
*)
2021-09-06 16:06:32 +03:00
# Check if we need to generate unused $VGNANE
if [ -z " $VGNAME " ] || [ " $VGNAME " = " $LVNAME " ] ; then
VGNAME = ${ DEFAULT_NAME %/* }
# Find largest matching VG name to our 'default' vgname
2023-05-11 21:07:37 +03:00
LASTVGNAME = $( LC_ALL = C " $LVM " vgs -oname -O-name --noheadings -S name = ~" ${ VGNAME } " | grep -E " ${ VGNAME } [0-9]? ? " | head -1 || true )
if [ -n " $LASTVGNAME " ] ; then
2023-02-09 18:04:54 +03:00
LASTVGNAME = ${ LASTVGNAME #* " ${ VGNAME } " }
2021-09-06 16:06:32 +03:00
# If the number is becoming too high, try some random number
test " $LASTVGNAME " -gt 99999999 2>/dev/null && LASTVGNAME = $RANDOM
# Generate new unused VG name
2023-02-09 18:04:54 +03:00
VGNAME = " ${ VGNAME } $(( LASTVGNAME + 1 )) "
2023-05-11 21:07:37 +03:00
verbose " Selected unused volume group name \" $VGNAME \". "
2021-09-06 16:06:32 +03:00
fi
fi
# New VG is created, LV name should be always unused.
test -n " $LVNAME " || LVNAME = ${ DEFAULT_NAME #*/ }
2021-08-31 21:52:26 +03:00
" $LVM " vgs " $VGNAME " >/dev/null 2>& 1 && error " Cannot use already existing volume group name \" $VGNAME \". "
2021-07-09 15:44:07 +03:00
; ;
esac
2023-05-11 21:07:37 +03:00
verbose " Checked whether device \" $DEVICE \" is already logical volume. "
2021-07-09 15:44:07 +03:00
" $MKDIR " -p -m 0000 " $TEMPDIR " || error " Failed to create $TEMPDIR . "
2023-06-29 14:05:55 +03:00
# TODO: might use directly /etc/vdoconf.yml (avoiding need of 'vdo' manager)
2021-07-09 15:44:07 +03:00
verbose "Getting YAML VDO configuration."
" $VDO " printConfigFile $VDOCONF >" $TEMPDIR /vdoconf.yml "
2023-05-11 21:07:37 +03:00
test -s " $TEMPDIR /vdoconf.yml " || error "Cannot work without VDO configuration"
2021-07-09 15:44:07 +03:00
2021-08-31 21:52:26 +03:00
# Check list of devices in VDO configure file for their major:minor
# and match with given $DEVICE devmajor:devminor
for i in $( awk '/.*device:/ {print $2}' " $TEMPDIR /vdoconf.yml " ) ; do
local DEV
DEV = $( " $READLINK " $READLINK_E " $i " ) || continue
2021-09-01 16:46:04 +03:00
RSTAT = $( " $STAT " --format "MAJOR=\$((0x%t)) MINOR=\$((0x%T))" " $DEV " 2>/dev/null) || continue
eval " $RSTAT "
2021-08-31 21:52:26 +03:00
test " $MAJOR " = " $DEVMAJOR " && test " $MINOR " = " $DEVMINOR " && {
test -z " $FOUND " || error " VDO configuration contains duplicate entries $FOUND and $i "
FOUND = $i
}
done
2023-05-11 21:07:37 +03:00
test -n " $FOUND " || error "Can't find matching device in VDO configuration file."
verbose " Found matching device $FOUND $MAJOR : $MINOR . "
2021-08-31 21:52:26 +03:00
VDONAME = $( awk -v DNAME = " $FOUND " '/.*VDOService$/ {VNAME=substr($1, 0, length($1) - 1)} /[[:space:]]*device:/ { if ($2 ~ DNAME) {print VNAME}}' " $TEMPDIR /vdoconf.yml " )
2021-07-09 15:44:07 +03:00
TRVDONAME = $( echo " $VDONAME " | tr '-' '_' )
# When VDO volume is 'active', check it's not mounted/being used
2021-08-31 21:52:26 +03:00
DM_OPEN = " $( " $DMSETUP " info -c -o open " $VDONAME " --noheadings --nameprefixes 2>/dev/null || true ) "
case " $DM_OPEN " in
Device*) ; ; # no devices
2021-09-01 16:46:04 +03:00
*) eval " $DM_OPEN "
2021-09-20 16:41:14 +03:00
test " ${ DM_OPEN :- 0 } " -eq 0 || error " Cannot convert in use VDO volume \" $VDONAME \"! "
2021-08-31 21:52:26 +03:00
; ;
esac
2021-07-09 15:44:07 +03:00
#parse_yaml_ "$TEMPDIR/vdoconf.yml" _
eval " $( parse_yaml_ " $TEMPDIR /vdoconf.yml " _ | grep " $TRVDONAME " | sed -e " s/_config_vdos_ $TRVDONAME /vdo/g " ) "
vdo_logicalSize = $( get_kb_size_with_unit_ " $vdo_logicalSize " )
vdo_physicalSize = $( get_kb_size_with_unit_ " $vdo_physicalSize " )
2021-08-31 21:52:26 +03:00
verbose " Converted VDO device has logical/physical size $vdo_logicalSize / $vdo_physicalSize KiB. "
2021-07-09 15:44:07 +03:00
2023-05-11 21:07:37 +03:00
VDO_ALLOCATION_PARAMS = $( cat <<EOF
2021-07-09 15:44:07 +03:00
allocation {
vdo_use_compression = $( get_enabled_value_ " $vdo_compression " )
vdo_use_deduplication = $( get_enabled_value_ " $vdo_deduplication " )
vdo_use_metadata_hints = 1
vdo_minimum_io_size = $vdo_logicalBlockSize
2022-05-31 23:48:38 +03:00
vdo_block_map_cache_size_mb = $(( $( get_kb_size_with_unit_ " $vdo_blockMapCacheSize " ) / 1024 ))
2021-07-09 15:44:07 +03:00
vdo_block_map_period = $vdo_blockMapPeriod
vdo_use_sparse_index = $( get_enabled_value_ " $vdo_indexSparse " )
vdo_index_memory_size_mb = $( awk " BEGIN {print $vdo_indexMemory * 1024} " )
2022-07-11 02:07:24 +03:00
vdo_slab_size_mb = $(( $( get_kb_size_with_unit_ " $vdo_slabSize " ) / 1024 ))
2021-07-09 15:44:07 +03:00
vdo_ack_threads = $vdo_ackThreads
vdo_bio_threads = $vdo_bioThreads
vdo_bio_rotation = $vdo_bioRotationInterval
vdo_cpu_threads = $vdo_cpuThreads
vdo_hash_zone_threads = $vdo_hashZoneThreads
vdo_logical_threads = $vdo_logicalThreads
vdo_physical_threads = $vdo_physicalThreads
vdo_write_policy = $vdo_writePolicy
2021-09-06 15:57:43 +03:00
vdo_max_discard = $(( $( get_kb_size_with_unit_ " $vdo_maxDiscardSize " ) / 4 ))
2021-07-09 15:44:07 +03:00
vdo_pool_header_size = 0
}
EOF
)
2023-05-11 21:07:37 +03:00
verbose " VDO conversion parameters: $VDO_ALLOCATION_PARAMS "
verbose "Stopping VDO volume."
2023-06-29 14:06:18 +03:00
dry " $VDO " stop $VDOCONF --name " $VDONAME " $VERB
2021-07-09 15:44:07 +03:00
2021-09-06 16:06:32 +03:00
# If user has not provided '--yes', prompt before conversion
2023-05-11 21:07:37 +03:00
if [ -z " $YES " ] && [ " $USE_VDO_DM_SNAPSHOT " != "1" ] ; then
2021-09-06 16:06:32 +03:00
PROMPTING = yes
echo -n " Convert VDO device \" $DEVICE \" to VDO LV \" $VGNAME / $LVNAME \"? [y|N]: "
2023-05-11 21:07:37 +03:00
read -r -n 1 -s ANSWER
2021-09-06 16:06:32 +03:00
case " ${ ANSWER : 0 : 1 } " in
2023-05-11 21:07:37 +03:00
y| Y ) echo "Yes" ; ;
* ) echo "No" ; PROMPTING = "" ; cleanup 1 ; ;
2021-09-06 16:06:32 +03:00
esac
PROMPTING = ""
2023-05-11 21:07:37 +03:00
YES = "-y" # From now, no prompting
2021-09-06 16:06:32 +03:00
fi
2023-05-11 21:07:37 +03:00
# Make a backup of the existing VDO yaml configuration file
test -e " $VDO_CONFIG " && dry cp -a " $VDO_CONFIG " " ${ VDO_CONFIG } .backup "
2021-07-09 15:44:07 +03:00
2023-05-11 21:07:37 +03:00
case " $DM_UUID " in
LVM-*) convert_lv_ " $vdo_logicalSize " ; ;
*) convert_non_lv_ " $vdo_logicalSize " ; ;
esac
2021-07-09 15:44:07 +03:00
}
#############################
# start point of this script
# - parsing parameters
#############################
trap "cleanup 2" 2
test " $# " -eq 0 && tool_usage
while [ " $# " -ne 0 ]
do
case " $1 " in
"" ) ; ;
"-f" | "--force" ) FORCE = "-f" ; ;
"-h" | "--help" ) tool_usage ; ;
"-n" | "--name" ) shift; NAME = $1 ; ;
"-v" | "--verbose" ) VERB = "-v" ; ;
"-y" | "--yes" ) YES = "-y" ; ;
2023-05-11 21:07:37 +03:00
"--abort-after-vdo-convert" ) ABORT_AFTER_VDO_CONVERT = 1; USE_VDO_DM_SNAPSHOT = 0 ; ; # For testing only
2021-09-17 17:49:17 +03:00
"--dry-run" ) DRY = "1" ; VERB = "-v" ; ;
2023-05-11 21:07:37 +03:00
"--no-snapshot" ) USE_VDO_DM_SNAPSHOT = 0 ; ;
"--uuid-prefix" ) shift; DM_UUID_PREFIX = $1 ; ; # For testing only
"--vdo-config" ) shift; VDO_CONFIG = $1 ; VDOCONF = " -f $VDO_CONFIG " ; ;
2021-07-09 15:44:07 +03:00
"-*" ) error " Wrong argument \" $1 \". (see: $TOOL --help) " ; ;
2023-05-11 21:07:37 +03:00
*) DEVICE = $1 ; ; # device name does not start with '-'
2021-07-09 15:44:07 +03:00
esac
shift
done
2023-05-11 21:07:37 +03:00
test -n " $DEVICE " || error " Device name is not specified. (see: $TOOL --help) "
convert2lvm_
2021-08-31 21:52:26 +03:00
2023-05-11 21:07:37 +03:00
cleanup 0