2010-05-14 15:33:20 +04:00
#!/bin/bash
2007-12-17 15:31:50 +03:00
#
2010-10-08 16:35:56 +04:00
# Copyright (C) 2007-2010 Red Hat, Inc. All rights reserved.
2007-12-17 15:31:50 +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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
#
2008-01-08 19:45:43 +03:00
# Author: Zdenek Kabelac <zkabelac at redhat.com>
2007-12-17 15:31:50 +03:00
#
# Script for resizing devices (usable for LVM resize)
#
# Needed utilities:
# mount, umount, grep, readlink, blockdev, blkid, fsck, xfs_check
#
2009-02-04 15:47:05 +03:00
# ext2/ext3/ext4: resize2fs, tune2fs
2007-12-17 15:31:50 +03:00
# reiserfs: resize_reiserfs, reiserfstune
# xfs: xfs_growfs, xfs_info
#
TOOL = fsadm
PATH = /sbin:/usr/sbin:/bin:/usr/sbin:$PATH
# utilities
TUNE_EXT = tune2fs
RESIZE_EXT = resize2fs
TUNE_REISER = reiserfstune
RESIZE_REISER = resize_reiserfs
TUNE_XFS = xfs_info
RESIZE_XFS = xfs_growfs
MOUNT = mount
UMOUNT = umount
MKDIR = mkdir
2008-01-08 19:45:43 +03:00
RMDIR = rmdir
BLOCKDEV = blockdev
BLKID = blkid
2007-12-17 15:31:50 +03:00
GREP = grep
READLINK = readlink
2008-04-29 19:25:28 +04:00
READLINK_E = "-e"
2007-12-17 15:31:50 +03:00
FSCK = fsck
XFS_CHECK = xfs_check
2009-06-09 19:31:36 +04:00
# user may override lvm location by setting LVM_BINARY
2010-10-08 18:49:25 +04:00
LVM = ${ LVM_BINARY :- lvm }
2008-02-06 15:45:32 +03:00
2010-10-08 17:49:20 +04:00
YES = ${ _FSADM_YES }
2007-12-17 15:31:50 +03:00
DRY = 0
2008-02-06 15:45:32 +03:00
VERB =
2007-12-17 15:31:50 +03:00
FORCE =
EXTOFF = 0
2008-02-06 15:45:32 +03:00
DO_LVRESIZE = 0
2007-12-17 15:31:50 +03:00
FSTYPE = unknown
VOLUME = unknown
TEMPDIR = " ${ TMPDIR :- /tmp } / ${ TOOL } _ ${ RANDOM } $$ /m "
BLOCKSIZE =
BLOCKCOUNT =
MOUNTPOINT =
MOUNTED =
REMOUNT =
2010-10-08 18:55:19 +04:00
PROCMOUNTS = "/proc/mounts"
2007-12-17 15:31:50 +03:00
IFS_OLD = $IFS
2009-06-09 19:31:36 +04:00
# without bash $'\n'
NL = '
'
2007-12-17 15:31:50 +03:00
tool_usage( ) {
2007-12-17 17:47:22 +03:00
echo " ${ TOOL } : Utility to resize or check the filesystem on a device "
echo
echo " ${ TOOL } [options] check device "
echo " - Check the filesystem on device using fsck"
echo
2008-01-08 19:45:43 +03:00
echo " ${ TOOL } [options] resize device [new_size[BKMGTPE]] "
2007-12-17 17:47:22 +03:00
echo " - Change the size of the filesystem on device to new_size"
echo
echo " Options:"
echo " -h | --help Show this help message"
echo " -v | --verbose Be verbose"
2009-02-04 15:47:05 +03:00
echo " -e | --ext-offline unmount filesystem before ext2/ext3/ext4 resize"
2007-12-17 17:47:22 +03:00
echo " -f | --force Bypass sanity checks"
echo " -n | --dry-run Print commands without running them"
2008-02-06 15:45:32 +03:00
echo " -l | --lvresize Resize given device (if it is LVM device)"
2007-12-17 17:47:22 +03:00
echo " -y | --yes Answer \"yes\" at any prompts"
echo
echo " new_size - Absolute number of filesystem blocks to be in the filesystem,"
echo " or an absolute size using a suffix (in powers of 1024)."
echo " If new_size is not supplied, the whole device is used."
2007-12-17 15:31:50 +03:00
exit
}
verbose( ) {
2008-02-06 15:45:32 +03:00
test -n " $VERB " && echo " $TOOL : $@ " || true
2007-12-17 15:31:50 +03:00
}
error( ) {
echo " $TOOL : $@ " >& 2
cleanup 1
}
dry( ) {
2008-01-08 19:45:43 +03:00
if [ " $DRY " -ne 0 ] ; then
verbose " Dry execution $@ "
return 0
fi
2007-12-17 15:31:50 +03:00
verbose " Executing $@ "
$@
}
cleanup( ) {
trap '' 2
# reset MOUNTPOINT - avoid recursion
test " $MOUNTPOINT " = " $TEMPDIR " && MOUNTPOINT = "" temp_umount
if [ -n " $REMOUNT " ] ; then
verbose "Remounting unmounted filesystem back"
dry $MOUNT " $VOLUME " " $MOUNTED "
fi
IFS = $IFS_OLD
trap 2
2008-02-06 15:45:32 +03:00
2010-10-08 17:47:10 +04:00
if [ " $DO_LVRESIZE " -eq 2 ] ; then
# start LVRESIZE with the filesystem modification flag
# and allow recursive call of fsadm
2010-10-08 17:49:20 +04:00
_FSADM_YES = $YES
export _FSADM_YES
2010-10-08 17:47:10 +04:00
unset FSADM_RUNNING
dry exec $LVM lvresize $VERB $FORCE -r -L${ NEWSIZE } b $VOLUME_ORIG
fi
2010-10-08 16:35:56 +04:00
# error exit status for break
exit ${ 1 :- 1 }
2007-12-17 15:31:50 +03:00
}
2008-01-08 19:45:43 +03:00
# convert parameter from Exa/Peta/Tera/Giga/Mega/Kilo/Bytes and blocks
# (2^(60/50/40/30/20/10/0))
2007-12-17 15:31:50 +03:00
decode_size( ) {
case " $1 " in
2008-01-08 19:45:43 +03:00
*[ eE] ) NEWSIZE = $(( ${ 1 %[eE] } * 1152921504606846976 )) ; ;
*[ pP] ) NEWSIZE = $(( ${ 1 %[pP] } * 1125899906842624 )) ; ;
2007-12-17 15:31:50 +03:00
*[ tT] ) NEWSIZE = $(( ${ 1 %[tT] } * 1099511627776 )) ; ;
*[ gG] ) NEWSIZE = $(( ${ 1 %[gG] } * 1073741824 )) ; ;
*[ mM] ) NEWSIZE = $(( ${ 1 %[mM] } * 1048576 )) ; ;
*[ kK] ) NEWSIZE = $(( ${ 1 %[kK] } * 1024 )) ; ;
*[ bB] ) NEWSIZE = ${ 1 %[bB] } ; ;
*) NEWSIZE = $(( $1 * $2 )) ; ;
esac
#NEWBLOCKCOUNT=$(round_block_size $NEWSIZE $2)
NEWBLOCKCOUNT = $(( $NEWSIZE / $2 ))
2008-02-06 15:45:32 +03:00
if [ $DO_LVRESIZE -eq 1 ] ; then
# start lvresize, but first cleanup mounted dirs
DO_LVRESIZE = 2
cleanup 0
fi
2007-12-17 15:31:50 +03:00
}
# detect filesystem on the given device
# dereference device name if it is symbolic link
detect_fs( ) {
2010-10-08 18:55:19 +04:00
VOLUME_ORIG = $1
VOLUME = ${ 1 #/dev/ }
2009-06-09 19:31:36 +04:00
VOLUME = $( $READLINK $READLINK_E " /dev/ $VOLUME " ) || error " Cannot get readlink $1 "
2010-10-08 18:55:19 +04:00
RVOLUME = $VOLUME
case " $RVOLUME " in
/dev/dm-[ 0-9] *)
read </sys/block/${ RVOLUME #/dev/ } /dm/name SYSVOLUME 2>& 1 && VOLUME = " /dev/mapper/ $SYSVOLUME "
; ;
esac
2007-12-17 15:31:50 +03:00
# use /dev/null as cache file to be sure about the result
2009-02-06 17:28:06 +03:00
# not using option '-o value' to be compatible with older version of blkid
FSTYPE = $( $BLKID -c /dev/null -s TYPE " $VOLUME " ) || error " Cannot get FSTYPE of \" $VOLUME \" "
FSTYPE = ${ FSTYPE ##*TYPE= \" } # cut quotation marks
FSTYPE = ${ FSTYPE %% \" * }
2007-12-17 15:31:50 +03:00
verbose " \" $FSTYPE \" filesystem found on \" $VOLUME \" "
}
# check if the given device is already mounted and where
2010-10-08 18:55:19 +04:00
# FIXME: resolve swap usage and device stacking
2007-12-17 15:31:50 +03:00
detect_mounted( ) {
2010-10-08 18:55:19 +04:00
test -e $PROCMOUNTS || error " Cannot detect mounted device $VOLUME "
MOUNTED = $( $GREP ^" $VOLUME " $PROCMOUNTS )
# for empty string try again with real volume name
test -z " $MOUNTED " && MOUNTED = $( $GREP ^" $RVOLUME " $PROCMOUNTS )
# cut device name prefix and trim everything past mountpoint
# echo translates \040 to spaces
MOUNTED = ${ MOUNTED #* }
MOUNTED = $( echo -n -e ${ MOUNTED %% * } )
2007-12-17 15:31:50 +03:00
test -n " $MOUNTED "
}
# get the full size of device in bytes
detect_device_size( ) {
2008-04-29 19:25:28 +04:00
# check if blockdev supports getsize64
2009-02-24 18:48:00 +03:00
$BLOCKDEV 2>& 1 | $GREP getsize64 >/dev/null
if test $? -eq 0; then
2008-04-29 19:25:28 +04:00
DEVSIZE = $( $BLOCKDEV --getsize64 " $VOLUME " ) || error " Cannot read size of device \" $VOLUME \" "
else
DEVSIZE = $( $BLOCKDEV --getsize " $VOLUME " ) || error " Cannot read size of device \" $VOLUME \" "
SSSIZE = $( $BLOCKDEV --getss " $VOLUME " ) || error " Cannot block size read device \" $VOLUME \" "
DEVSIZE = $(( $DEVSIZE * $SSSIZE ))
fi
2007-12-17 15:31:50 +03:00
}
# round up $1 / $2
# could be needed to gaurantee 'at least given size'
# but it makes many troubles
round_up_block_size( ) {
echo $(( ( $1 + $2 - 1 ) / $2 ))
}
temp_mount( ) {
dry $MKDIR -p -m 0000 " $TEMPDIR " || error " Failed to create $TEMPDIR "
dry $MOUNT " $VOLUME " " $TEMPDIR " || error " Failed to mount $TEMPDIR "
}
temp_umount( ) {
2008-01-08 19:45:43 +03:00
dry $UMOUNT " $TEMPDIR " || error " Failed to umount $TEMPDIR "
dry $RMDIR " ${ TEMPDIR } " || error " Failed to remove $TEMPDIR "
dry $RMDIR " ${ TEMPDIR %%m } " || error " Failed to remove ${ TEMPDIR %%m } "
2007-12-17 15:31:50 +03:00
}
yes_no( ) {
echo -n " $@ ? [Y|n] "
2009-02-24 18:48:00 +03:00
2007-12-17 15:31:50 +03:00
if [ -n " $YES " ] ; then
2009-02-24 18:48:00 +03:00
echo y ; return 0
2007-12-17 15:31:50 +03:00
fi
2009-02-24 18:48:00 +03:00
while read -r -s -n 1 ANS ; do
case " $ANS " in
"y" | "Y" | "" ) echo y ; return 0 ; ;
"n" | "N" ) echo n ; return 1 ; ;
esac
done
2007-12-17 15:31:50 +03:00
}
try_umount( ) {
yes_no " Do you want to unmount \" $MOUNTED \" " && dry $UMOUNT " $MOUNTED " && return 0
2010-10-08 15:18:29 +04:00
error " Cannot proceed with mounted filesystem \" $MOUNTED \" "
2007-12-17 15:31:50 +03:00
}
validate_parsing( ) {
test -n " $BLOCKSIZE " -a -n " $BLOCKCOUNT " || error " Cannot parse $1 output "
}
####################################
2009-02-04 15:47:05 +03:00
# Resize ext2/ext3/ext4 filesystem
2007-12-17 15:31:50 +03:00
# - unmounted or mounted for upsize
# - unmounted for downsize
####################################
resize_ext( ) {
verbose " Parsing $TUNE_EXT -l \" $VOLUME \" "
for i in $( $TUNE_EXT -l " $VOLUME " ) ; do
case " $i " in
"Block size" *) BLOCKSIZE = ${ i ##* } ; ;
"Block count" *) BLOCKCOUNT = ${ i ##* } ; ;
esac
done
validate_parsing $TUNE_EXT
decode_size $1 $BLOCKSIZE
FSFORCE = $FORCE
2008-01-08 19:45:43 +03:00
if [ " $NEWBLOCKCOUNT " -lt " $BLOCKCOUNT " -o " $EXTOFF " -eq 1 ] ; then
2007-12-17 15:31:50 +03:00
detect_mounted && verbose " $RESIZE_EXT needs unmounted filesystem " && try_umount
REMOUNT = $MOUNTED
# CHECKME: after umount resize2fs requires fsck or -f flag.
FSFORCE = "-f"
fi
2009-02-24 18:48:00 +03:00
verbose " Resizing filesystem on device \" $VOLUME \" to $NEWSIZE bytes ( $BLOCKCOUNT -> $NEWBLOCKCOUNT blocks of $BLOCKSIZE bytes) "
2007-12-17 15:31:50 +03:00
dry $RESIZE_EXT $FSFORCE " $VOLUME " $NEWBLOCKCOUNT
}
#############################
# Resize reiserfs filesystem
# - unmounted for upsize
# - unmounted for downsize
#############################
resize_reiser( ) {
2009-02-24 18:48:00 +03:00
detect_mounted && verbose "ReiserFS resizes only unmounted filesystem" && try_umount
REMOUNT = $MOUNTED
2007-12-17 15:31:50 +03:00
verbose " Parsing $TUNE_REISER \" $VOLUME \" "
for i in $( $TUNE_REISER " $VOLUME " ) ; do
case " $i " in
"Blocksize" *) BLOCKSIZE = ${ i ##* : } ; ;
"Count of blocks" *) BLOCKCOUNT = ${ i ##* : } ; ;
esac
done
validate_parsing $TUNE_REISER
decode_size $1 $BLOCKSIZE
verbose " Resizing \" $VOLUME \" $BLOCKCOUNT -> $NEWBLOCKCOUNT blocks ( $NEWSIZE bytes, bs: $NEWBLOCKCOUNT ) "
if [ -n " $YES " ] ; then
2010-10-08 19:00:06 +04:00
echo y | dry $RESIZE_REISER -s $NEWSIZE " $VOLUME "
2007-12-17 15:31:50 +03:00
else
dry $RESIZE_REISER -s $NEWSIZE " $VOLUME "
fi
}
########################
# Resize XFS filesystem
# - mounted for upsize
2010-10-08 15:18:29 +04:00
# - cannot downsize
2007-12-17 15:31:50 +03:00
########################
resize_xfs( ) {
detect_mounted
MOUNTPOINT = $MOUNTED
if [ -z " $MOUNTED " ] ; then
MOUNTPOINT = $TEMPDIR
temp_mount || error "Cannot mount Xfs filesystem"
fi
verbose " Parsing $TUNE_XFS \" $MOUNTPOINT \" "
for i in $( $TUNE_XFS " $MOUNTPOINT " ) ; do
case " $i " in
"data" *) BLOCKSIZE = ${ i ##*bsize= } ; BLOCKCOUNT = ${ i ##*blocks= } ; ;
esac
done
BLOCKSIZE = ${ BLOCKSIZE %%[^0-9]* }
BLOCKCOUNT = ${ BLOCKCOUNT %%[^0-9]* }
validate_parsing $TUNE_XFS
decode_size $1 $BLOCKSIZE
if [ $NEWBLOCKCOUNT -gt $BLOCKCOUNT ] ; then
verbose " Resizing Xfs mounted on \" $MOUNTPOINT \" to fill device \" $VOLUME \" "
dry $RESIZE_XFS $MOUNTPOINT
elif [ $NEWBLOCKCOUNT -eq $BLOCKCOUNT ] ; then
verbose "Xfs filesystem already has the right size"
else
error "Xfs filesystem shrinking is unsupported"
fi
}
####################
# Resize filesystem
####################
resize( ) {
2008-02-06 15:45:32 +03:00
NEWSIZE = $2
2007-12-17 15:31:50 +03:00
detect_fs " $1 "
detect_device_size
2009-02-24 18:48:00 +03:00
verbose " Device \" $VOLUME \" size is $DEVSIZE bytes "
2007-12-17 15:31:50 +03:00
# if the size parameter is missing use device size
2008-02-06 15:45:32 +03:00
#if [ -n "$NEWSIZE" -a $NEWSIZE <
2008-01-08 19:45:43 +03:00
test -z " $NEWSIZE " && NEWSIZE = ${ DEVSIZE } b
2007-12-17 15:31:50 +03:00
trap cleanup 2
2009-06-09 19:31:36 +04:00
IFS = $NL
2007-12-17 15:31:50 +03:00
case " $FSTYPE " in
2009-02-04 15:47:05 +03:00
"ext3" | "ext2" | "ext4" ) resize_ext $NEWSIZE ; ;
2007-12-17 15:31:50 +03:00
"reiserfs" ) resize_reiser $NEWSIZE ; ;
"xfs" ) resize_xfs $NEWSIZE ; ;
*) error " Filesystem \" $FSTYPE \" on device \" $VOLUME \" is not supported by this tool " ; ;
esac || error " Resize $FSTYPE failed "
2008-02-06 15:45:32 +03:00
cleanup 0
2007-12-17 15:31:50 +03:00
}
###################
# Check filesystem
###################
check( ) {
detect_fs " $1 "
2010-10-08 15:18:29 +04:00
detect_mounted && error " Cannot fsck device \" $VOLUME \", filesystem is mounted on $MOUNTED "
2007-12-17 15:31:50 +03:00
case " $FSTYPE " in
"xfs" ) dry $XFS_CHECK " $VOLUME " ; ;
2010-10-08 19:02:05 +04:00
*) # check if executed from interactive shell environment
case " $- " in
*i*) dry $FSCK $YES $FORCE " $VOLUME " ; ;
*) dry $FSCK $FORCE -p " $VOLUME " ; ;
esac
2007-12-17 15:31:50 +03:00
esac
}
#############################
# start point of this script
# - parsing parameters
#############################
2008-01-08 19:45:43 +03:00
2008-02-06 15:45:32 +03:00
# test if we are not invoked recursively
test -n " $FSADM_RUNNING " && exit 0
2008-01-08 19:45:43 +03:00
# test some prerequisities
test -n " $TUNE_EXT " -a -n " $RESIZE_EXT " -a -n " $TUNE_REISER " -a -n " $RESIZE_REISER " \
-a -n " $TUNE_XFS " -a -n " $RESIZE_XFS " -a -n " $MOUNT " -a -n " $UMOUNT " -a -n " $MKDIR " \
-a -n " $RMDIR " -a -n " $BLOCKDEV " -a -n " $BLKID " -a -n " $GREP " -a -n " $READLINK " \
2009-06-09 19:31:36 +04:00
-a -n " $FSCK " -a -n " $XFS_CHECK " -a -n "LVM" \
2008-02-06 15:45:32 +03:00
|| error "Required command definitions in the script are missing!"
2008-04-29 19:25:28 +04:00
2009-06-09 19:31:36 +04:00
$LVM version >/dev/null 2>& 1 || error " Could not run lvm binary ' $LVM ' "
$( $READLINK -e / >/dev/null 2>& 1) || READLINK_E = "-f"
2008-01-08 19:45:43 +03:00
TEST64BIT = $(( 1000 * 1000000000000 ))
test $TEST64BIT -eq 1000000000000000 || error "Shell does not handle 64bit arithmetic"
$( echo Y | $GREP Y >/dev/null) || error "Grep does not work properly"
2009-02-24 18:48:00 +03:00
if [ " $# " -eq 0 ] ; then
2007-12-17 15:31:50 +03:00
tool_usage
fi
2009-02-24 18:48:00 +03:00
while [ " $# " -ne 0 ]
2007-12-17 15:31:50 +03:00
do
2009-02-24 18:48:00 +03:00
case " $1 " in
"" ) ; ;
"-h" | "--help" ) tool_usage ; ;
"-v" | "--verbose" ) VERB = "-v" ; ;
"-n" | "--dry-run" ) DRY = 1 ; ;
"-f" | "--force" ) FORCE = "-f" ; ;
"-e" | "--ext-offline" ) EXTOFF = 1 ; ;
"-y" | "--yes" ) YES = "-y" ; ;
"-l" | "--lvresize" ) DO_LVRESIZE = 1 ; ;
"check" ) CHECK = " $2 " ; shift ; ;
"resize" ) RESIZE = " $2 " ; NEWSIZE = " $3 " ; shift 2 ; ;
*) error " Wrong argument \" $1 \". (see: $TOOL --help) "
2007-12-17 15:31:50 +03:00
esac
shift
done
if [ -n " $CHECK " ] ; then
check " $CHECK "
elif [ -n " $RESIZE " ] ; then
2008-02-06 15:45:32 +03:00
export FSADM_RUNNING = "fsadm"
2007-12-17 15:31:50 +03:00
resize " $RESIZE " " $NEWSIZE "
else
error " Missing command. (see: $TOOL --help) "
fi