2016-04-12 05:05:00 +02:00
|
|
|
#!/bin/bash
|
2016-07-10 05:45:54 +02:00
|
|
|
|
2016-11-24 15:38:11 +01:00
|
|
|
usage() { echo "Usage: $0 [-s] [-i] imagefile.img [newimagefile.img]"; exit -1; }
|
2016-07-10 05:45:54 +02:00
|
|
|
|
|
|
|
should_skip_autoexpand=false
|
2016-11-24 15:38:11 +01:00
|
|
|
force_inplace_and_copy_after=false
|
2016-07-10 05:45:54 +02:00
|
|
|
|
2016-11-24 15:38:11 +01:00
|
|
|
while getopts ":s:i" opt; do
|
2016-08-22 10:11:37 +02:00
|
|
|
case "${opt}" in
|
|
|
|
s) should_skip_autoexpand=true ;;
|
2016-11-24 15:38:11 +01:00
|
|
|
i) force_inplace_and_copy_after=true ;;
|
2016-08-22 10:11:37 +02:00
|
|
|
*) usage ;;
|
|
|
|
esac
|
2016-07-10 05:45:54 +02:00
|
|
|
done
|
|
|
|
shift $((OPTIND-1))
|
|
|
|
|
2016-04-12 05:05:00 +02:00
|
|
|
#Args
|
|
|
|
img=$1
|
|
|
|
|
|
|
|
#Usage checks
|
|
|
|
if [[ -z $img ]]; then
|
2016-07-10 05:45:54 +02:00
|
|
|
usage
|
2016-04-12 05:05:00 +02:00
|
|
|
fi
|
|
|
|
if [[ ! -e $img ]]; then
|
|
|
|
echo "ERROR: $img is not a file..."
|
|
|
|
exit -2
|
|
|
|
fi
|
|
|
|
if (( EUID != 0 )); then
|
2016-08-22 10:11:37 +02:00
|
|
|
echo "ERROR: You need to be running as root."
|
|
|
|
exit -3
|
2016-04-12 05:05:00 +02:00
|
|
|
fi
|
|
|
|
|
2016-04-26 18:03:43 +02:00
|
|
|
#Check that what we need is installed
|
|
|
|
A=`which parted 2>&1`
|
|
|
|
if (( $? != 0 )); then
|
2016-08-22 10:11:37 +02:00
|
|
|
echo "ERROR: parted is not installed."
|
|
|
|
exit -4
|
2016-04-26 18:03:43 +02:00
|
|
|
fi
|
|
|
|
|
2016-04-28 07:35:12 +02:00
|
|
|
#Copy to new file if requested
|
2016-11-24 15:38:11 +01:00
|
|
|
if [ -n "$2" -a "$force_inplace_and_copy_after" = false ]; then
|
2016-04-28 07:35:12 +02:00
|
|
|
echo "Copying $1 to $2..."
|
2016-11-24 15:37:39 +01:00
|
|
|
if [[ -f $img ]]; then
|
|
|
|
cp --reflink=auto --sparse=always "$1" "$2"
|
|
|
|
else
|
|
|
|
imgsize=`parted -m $img unit B print | tail -1 | cut -d ':' -f 3 | tr -d 'B\n'`
|
|
|
|
imgsize=`expr $imgsize + 1`
|
|
|
|
dd if="$1" of="$2" conv=sparse count=$imgsize iflag=count_bytes bs=1M
|
|
|
|
fi
|
2016-04-28 07:35:12 +02:00
|
|
|
if (( $? != 0 )); then
|
2016-08-22 10:11:37 +02:00
|
|
|
echo "ERROR: Could not copy file..."
|
|
|
|
exit -5
|
2016-04-28 07:35:12 +02:00
|
|
|
fi
|
|
|
|
img=$2
|
|
|
|
fi
|
|
|
|
|
2016-04-12 05:05:00 +02:00
|
|
|
#Gather info
|
|
|
|
beforesize=`ls -lah $img | cut -d ' ' -f 5`
|
2016-04-27 10:33:21 +02:00
|
|
|
partnum=`parted -m $img unit B print | tail -n 1 | cut -d ':' -f 1 | tr -d '\n'`
|
2016-04-27 08:10:23 +02:00
|
|
|
partstart=`parted -m $img unit B print | tail -n 1 | cut -d ':' -f 2 | tr -d 'B\n'`
|
2016-04-12 05:05:00 +02:00
|
|
|
loopback=`losetup -f --show -o $partstart $img`
|
2016-07-07 07:28:32 +02:00
|
|
|
currentsize=`tune2fs -l $loopback | grep 'Block count' | tr -d ' ' | cut -d ':' -f 2 | tr -d '\n'`
|
|
|
|
blocksize=`tune2fs -l $loopback | grep 'Block size' | tr -d ' ' | cut -d ':' -f 2 | tr -d '\n'`
|
2016-04-12 05:05:00 +02:00
|
|
|
|
2016-07-10 05:45:54 +02:00
|
|
|
#Check if we should make pi expand rootfs on next boot
|
|
|
|
if [ "$should_skip_autoexpand" = false ]; then
|
|
|
|
#Make pi expand rootfs on next boot
|
|
|
|
mountdir=`mktemp -d`
|
|
|
|
mount $loopback $mountdir
|
2016-04-19 10:21:22 +02:00
|
|
|
|
2016-10-31 06:42:34 +01:00
|
|
|
if [ `md5sum $mountdir/etc/rc.local | cut -d ' ' -f 1` != "c4eb22d9aa99915af319287d194d4529" ]; then
|
2016-07-10 05:45:54 +02:00
|
|
|
echo Creating new /etc/rc.local
|
|
|
|
mv $mountdir/etc/rc.local $mountdir/etc/rc.local.bak
|
2016-08-22 10:11:37 +02:00
|
|
|
###Do not touch the following 6 lines including EOF###
|
2016-10-31 06:42:34 +01:00
|
|
|
cat <<\EOFE > $mountdir/etc/rc.local
|
2016-08-18 03:01:14 +02:00
|
|
|
#!/bin/bash
|
2016-10-31 06:42:34 +01:00
|
|
|
do_expand_rootfs() {
|
|
|
|
ROOT_PART=$(cat /proc/cmdline | tr -s ' ' '\n' | grep root= | cut -d '=' -f 2)
|
|
|
|
if [[ "$ROOT_PART" != /dev/mmcblk0* ]] ; then
|
|
|
|
echo $ROOT_PART is not an SD card...
|
|
|
|
return 0
|
|
|
|
fi
|
|
|
|
PART_NUM=$(echo $ROOT_PART | cut -d 'p' -f 2)
|
|
|
|
|
|
|
|
# Get the starting offset of the root partition
|
|
|
|
PART_START=$(fdisk -l /dev/mmcblk0 | grep $ROOT_PART | tr -s ' ' | cut -d ' ' -f 2)
|
|
|
|
[ "$PART_START" ] || return 1
|
|
|
|
# Return value will likely be error for fdisk as it fails to reload the
|
|
|
|
# partition table because the root fs is mounted
|
|
|
|
fdisk /dev/mmcblk0 <<EOF
|
|
|
|
p
|
|
|
|
d
|
|
|
|
$PART_NUM
|
|
|
|
n
|
|
|
|
p
|
|
|
|
$PART_NUM
|
|
|
|
$PART_START
|
|
|
|
|
|
|
|
w
|
|
|
|
|
2016-04-12 05:05:00 +02:00
|
|
|
EOF
|
2016-10-31 06:42:34 +01:00
|
|
|
|
|
|
|
cat <<EOF > /etc/rc.local &&
|
|
|
|
#!/bin/sh
|
|
|
|
resize2fs $ROOT_PART
|
|
|
|
rm -f /etc/rc.local; cp -f /etc/rc.local.bak /etc/rc.local; /etc/rc.local
|
|
|
|
|
|
|
|
EOF
|
|
|
|
reboot
|
|
|
|
exit
|
|
|
|
}
|
|
|
|
do_expand_rootfs
|
|
|
|
rm -f /etc/rc.local; cp -f /etc/rc.local.bak /etc/rc.local; /etc/rc.local
|
|
|
|
exit 0
|
|
|
|
EOFE
|
2016-08-22 10:11:37 +02:00
|
|
|
###End no touch zone###
|
2016-07-10 05:45:54 +02:00
|
|
|
chmod +x $mountdir/etc/rc.local
|
|
|
|
fi
|
|
|
|
umount $mountdir
|
|
|
|
else
|
|
|
|
echo Skipping autoexpanding process...
|
2016-04-19 10:21:22 +02:00
|
|
|
fi
|
2016-04-12 05:05:00 +02:00
|
|
|
|
2016-07-07 07:28:32 +02:00
|
|
|
#Make sure filesystem is ok
|
2016-04-12 05:05:00 +02:00
|
|
|
e2fsck -f $loopback
|
2016-07-07 07:28:32 +02:00
|
|
|
minsize=`resize2fs -P $loopback | cut -d ':' -f 2 | tr -d ' ' | tr -d '\n'`
|
|
|
|
if [[ $currentsize -eq $minsize ]]; then
|
|
|
|
echo ERROR: Image already shrunk to smallest size
|
|
|
|
exit -6
|
|
|
|
fi
|
|
|
|
|
|
|
|
#Add some free space to the end of the filesystem
|
|
|
|
if [[ `expr $currentsize - $minsize - 5000` -gt 0 ]]; then
|
|
|
|
minsize=`expr $minsize + 5000 | tr -d '\n'`
|
|
|
|
elif [[ `expr $currentsize - $minsize - 1000` -gt 0 ]]; then
|
|
|
|
minsize=`expr $minsize + 1000 | tr -d '\n'`
|
|
|
|
elif [[ `expr $currentsize - $minsize - 100` -gt 0 ]]; then
|
|
|
|
minsize=`expr $minsize + 100 | tr -d '\n'`
|
|
|
|
fi
|
|
|
|
|
|
|
|
#Shrink filesystem
|
2016-04-12 05:05:00 +02:00
|
|
|
resize2fs -p $loopback $minsize
|
2016-04-19 10:21:22 +02:00
|
|
|
if [[ $? != 0 ]]; then
|
|
|
|
echo ERROR: resize2fs failed...
|
|
|
|
mount $loopback $mountdir
|
|
|
|
mv $mountdir/etc/rc.local.bak $mountdir/etc/rc.local
|
|
|
|
umount $mountdir
|
|
|
|
losetup -d $loopback
|
|
|
|
exit $rc
|
|
|
|
fi
|
2016-04-12 05:05:00 +02:00
|
|
|
sleep 1
|
|
|
|
|
|
|
|
#Shrink partition
|
|
|
|
losetup -d $loopback
|
2016-07-07 07:28:32 +02:00
|
|
|
partnewsize=`expr $minsize \* $blocksize | tr -d '\n'`
|
2016-04-27 10:32:22 +02:00
|
|
|
newpartend=`expr $partstart + $partnewsize | tr -d '\n'`
|
2016-04-27 10:33:21 +02:00
|
|
|
part1=`parted $img rm $partnum`
|
2016-04-12 05:05:00 +02:00
|
|
|
part2=`parted $img unit B mkpart primary $partstart $newpartend`
|
|
|
|
|
|
|
|
#Truncate the file
|
2016-11-24 11:59:51 +01:00
|
|
|
if [[ -f $img ]]; then
|
|
|
|
endresult=`parted -m $img unit B print free | tail -1 | cut -d ':' -f 2 | tr -d 'B\n'`
|
|
|
|
truncate -s $endresult $img
|
|
|
|
aftersize=`ls -lah $img | cut -d ' ' -f 5`
|
|
|
|
echo "Shrunk $img from $beforesize to $aftersize"
|
|
|
|
fi
|
2016-11-24 15:38:11 +01:00
|
|
|
|
|
|
|
#Copy to new file if requested
|
|
|
|
if [ -n "$2" -a "$force_inplace_and_copy_after" = true ]; then
|
|
|
|
echo "Copying $1 to $2..."
|
|
|
|
if [[ -f $img ]]; then
|
|
|
|
cp --reflink=auto --sparse=always "$1" "$2"
|
|
|
|
else
|
|
|
|
imgsize=`parted -m $img unit B print | tail -1 | cut -d ':' -f 3 | tr -d 'B\n'`
|
|
|
|
imgsize=`expr $imgsize + 1`
|
|
|
|
dd if="$1" of="$2" conv=sparse count=$imgsize iflag=count_bytes bs=1M
|
|
|
|
fi
|
|
|
|
|
|
|
|
if (( $? != 0 )); then
|
|
|
|
echo "ERROR: Could not copy file..."
|
|
|
|
exit -5
|
|
|
|
fi
|
|
|
|
img=$2
|
|
|
|
fi
|