2021-07-09 15:44:07 +03:00
#!/bin/bash
#
# Copyright (C) 2021 Red Hat, Inc. All rights reserved.
#
# 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,
2021-09-01 16:46:04 +03:00
# grep, awk, sed, blockdev, readlink, stat, mkdir
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
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 }
VDOCONF = ${ VDOCONF :- }
BLOCKDEV = "blockdev"
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"
2021-08-31 21:52:26 +03:00
DMSETUP = "dmsetup"
2021-07-09 15:44:07 +03:00
TEMPDIR = " ${ TMPDIR :- /tmp } / ${ TOOL } _ ${ RANDOM } $$ "
DM_DEV_DIR = " ${ DM_DEV_DIR :- /dev } "
2021-08-31 21:52:26 +03:00
DEVICENAME = ""
DEVMAJOR = 0
DEVMINOR = 0
2021-09-06 16:06:32 +03:00
PROMPTING = ""
2021-08-31 21:52:26 +03:00
2021-07-09 15:44:07 +03:00
DRY = 0
VERB = ""
FORCE = ""
YES = ""
# 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"
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
2021-09-06 16:06:32 +03:00
test -z " $PROMPTING " || echo "No"
2021-07-09 15:44:07 +03:00
rm -rf " $TEMPDIR "
# error exit status for break
exit " ${ 1 :- 1 } "
}
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
}
get_mb_size_with_unit_( ) {
case " $1 " in
*[ mM] ) echo $(( ${ 1 %[mM] } )) ; ;
*[ gG] ) echo $(( ${ 1 %[gG] } * 1024 )) ; ;
*[ tT] ) echo $(( ${ 1 %[tT] } * 1024 * 1024 )) ; ;
*[ pP] ) echo $(( ${ 1 %[pP] } * 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
# dereference device name if it is symbolic link
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 "
}
# convert existing VDO volume into lvm2 volume
convert2lvm_( ) {
local DEVICE = $1
local VGNAME = ${ NAME %/* }
local LVNAME = ${ NAME #*/ }
local VDONAME
local TRVDONAME
local EXTENTSZ
local IS_LV = 1
2021-08-31 21:52:26 +03:00
local FOUND = ""
local MAJOR = 0
local MINOR = 0
local DM_VG_NAME
local DM_LV_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
2021-09-06 16:06:32 +03:00
verbose " Using existing volume group name $VGNAME . "
test -n " $LVNAME " || LVNAME = $DM_LV_NAME
2021-07-09 15:44:07 +03:00
elif test " $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
; ;
2021-09-06 16:06:32 +03:00
*) IS_LV = 0
# 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
2021-09-09 19:10:13 +03:00
LASTVGNAME = $( LC_ALL = C " $LVM " vgs -oname -O-name --noheadings -S name = ~${ VGNAME } | grep -E " $VGNAME [0-9]? ? " | head -1 || true )
2021-09-06 16:06:32 +03:00
if test -n " $LASTVGNAME " ; then
LASTVGNAME = ${ LASTVGNAME #* ${ VGNAME } }
# 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
VGNAME = " ${ VGNAME } $(( ${ LASTVGNAME } + 1 )) "
verbose " Selected unused volume group name $VGNAME . "
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
verbose " Checked whether device $1 is already LV ( $IS_LV ). "
" $MKDIR " -p -m 0000 " $TEMPDIR " || error " Failed to create $TEMPDIR . "
2021-08-31 21:52:26 +03:00
# TODO: might use directly /etc/vdoconf.yml (avoding need of 'vdo' manager)
2021-07-09 15:44:07 +03:00
verbose "Getting YAML VDO configuration."
" $VDO " printConfigFile $VDOCONF >" $TEMPDIR /vdoconf.yml "
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
test -n " $FOUND " || error "Can't find matching device in vdo configuration file."
verbose " Found matching device $FOUND $MAJOR : $MINOR "
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
PARAMS = $( cat <<EOF
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
vdo_block_map_cache_size_mb = $( get_mb_size_with_unit_ " $vdo_blockMapCacheSize " )
vdo_block_map_period = $vdo_blockMapPeriod
vdo_check_point_frequency = $vdo_indexCfreq
vdo_use_sparse_index = $( get_enabled_value_ " $vdo_indexSparse " )
vdo_index_memory_size_mb = $( awk " BEGIN {print $vdo_indexMemory * 1024} " )
vdo_slab_size_mb = $( get_mb_size_with_unit_ " $vdo_blockMapCacheSize " )
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
)
verbose " VDO conversion paramaters: $PARAMS "
2021-09-06 16:06:32 +03:00
# If user has not provided '--yes', prompt before conversion
if test -z " $YES " ; then
PROMPTING = yes
echo -n " Convert VDO device \" $DEVICE \" to VDO LV \" $VGNAME / $LVNAME \"? [y|N]: "
read -n 1 -s ANSWER
case " ${ ANSWER : 0 : 1 } " in
y| Y ) echo "Yes" ; ;
2021-09-10 23:39:23 +03:00
* ) echo "No" ; PROMPTING = "" ; exit 1 ; ;
2021-09-06 16:06:32 +03:00
esac
PROMPTING = ""
YES = "-y" # From now, now prompting
fi
2021-07-09 15:44:07 +03:00
verbose "Stopping VDO volume."
dry " $VDO " stop $VDOCONF --name " $VDONAME "
if [ " $IS_LV " = "0" ] ; then
verbose "Moving VDO header by 2MiB."
dry " $VDO " convert $VDOCONF --force --name " $VDONAME "
dry " $LVM " pvcreate $YES --dataalignment 2M " $DEVICE " || {
error " Creation of PV on \" $DEVICE \" failed, while VDO header has been already moved! "
}
# Obtain free space in this new PV
2021-08-23 15:02:42 +03:00
# after 'vdo convert' call there is +2M free space at the front of the device
2021-07-09 15:44:07 +03:00
case " $DRY " in
0) pvfree = $( " $LVM " pvs -o devsize --units b --nosuffix --noheadings " $DEVICE " ) ; ;
*) pvfree = $( " $BLOCKDEV " --getsize64 " $DEVICE " ) ; ;
esac
pvfree = $(( pvfree / 1024 - 2048 )) # to KiB
else
2021-08-31 21:52:26 +03:00
pvfree = $( " $LVM " lvs -o size --units b --nosuffix --noheadings " $DM_VG_NAME / $DM_LV_NAME " )
2021-07-09 15:44:07 +03:00
pvfree = $(( pvfree / 1024 )) # to KiB
fi
# select largest possible extent size that can exactly express both sizes
EXTENTSZ = $( get_largest_extent_size_ " $pvfree " " $vdo_logicalSize " )
if [ " $IS_LV " = "0" ] ; then
verbose " Creating VG \" ${ NAME %/* } \" with extent size $EXTENTSZ KiB. "
dry " $LVM " vgcreate $YES $VERB -s " ${ EXTENTSZ } k " " $VGNAME " " $DEVICE " || {
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 $YES $VERB -l100%VG -n " ${ LVNAME } _vpool " " $VGNAME "
else
# validate existing VG extent_size can express virtual VDO size
vg_extent_size = $( " $LVM " vgs -o vg_extent_size --units b --nosuffix --noheadings " $VGNAME " || true )
vg_extent_size = $(( vg_extent_size / 1024 ))
test " $vg_extent_size " -le " $EXTENTSZ " || {
2021-08-31 21:52:26 +03:00
error " Please vgchange extent_size to at most $EXTENTSZ KiB or extend and align virtual size of VDO device on $vg_extent_size KiB. "
2021-07-09 15:44:07 +03:00
}
verbose "Renaming existing LV to be used as _vdata volume for VDO pool LV."
2021-08-31 21:52:26 +03:00
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! "
2021-07-09 15:44:07 +03:00
}
fi
verbose "Converting to VDO pool."
dry " $LVM " lvconvert $YES $VERB $FORCE --config " $PARAMS " -Zn -V " ${ vdo_logicalSize } k " -n " $LVNAME " --type vdo-pool " $VGNAME / ${ LVNAME } _vpool "
rm -fr " $TEMPDIR "
}
#############################
# 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" ; ;
2021-09-17 17:49:17 +03:00
"--dry-run" ) DRY = "1" ; VERB = "-v" ; ;
2021-07-09 15:44:07 +03:00
"-*" ) error " Wrong argument \" $1 \". (see: $TOOL --help) " ; ;
*) DEVICENAME = $1 ; ; # device name does not start with '-'
esac
shift
done
2021-08-31 21:52:26 +03:00
test -n " $DEVICENAME " || error " Device name is not specified. (see: $TOOL --help) "
2021-07-09 15:44:07 +03:00
# do conversion
convert2lvm_ " $DEVICENAME "