blob: 9f3b587536e23faeb38cad2a505e3ba041e91592 [file] [log] [blame]
#!/bin/sh
echo update: "$@"
echoerr() {
echo 1>&2 "ERROR: $*"
}
cd /
if ! test -r /proc/mounts || ! test -f /proc/mounts
then
mkdir -p /proc
mount -t proc proc proc
fi
if ! test -d /sys/class
then
mkdir -p /sys
mount -t sysfs sys sys
fi
if ! test -c /dev/null
then
mkdir -p /dev
mount -t devtmpfs dev dev
fi
# mtd number N with mtd name Name can be mounted via mtdN, or mtd:Name
# (with a mtd aware fs) or by /dev/mtdblockN (with a mtd or block fs).
mtdismounted() {
m=${1##mtd}
if grep -s "mtdblock$m " /proc/mounts || grep -s "mtd$m " /proc/mounts
then
return 0
fi
n=$(cat "/sys/class/mtd/mtd$m/name")
if test -n "$n" && grep -s "mtd:$n " /proc/mounts
then
return 0
fi
return 1
}
# Detect child partitions when the whole flash is to be updated.
# Ignore mtdNro and mtdblockN names in the class subsystem directory.
childmtds() {
for m in "/sys/class/mtd/$1/mtd"*
do
m=${m##*/}
if test "${m%ro}" = "${m#mtdblock}"
then
echo "$m"
fi
done
}
toobig() {
if test "$(stat -L -c "%s" "$1")" -gt "$(cat "/sys/class/mtd/$2/size")"
then
return 0
fi
return 1
}
findmtd() {
m=$(grep -xl "$1" /sys/class/mtd/*/name)
m=${m%/name}
m=${m##*/}
echo "$m"
}
blkid_fs_type() {
# Emulate util-linux's `blkid -s TYPE -o value $1`
# Example busybox blkid output:
# # blkid /dev/mtdblock5
# /dev/mtdblock5: TYPE="squashfs"
# Process output to extract TYPE value "squashfs".
blkid "$1" | sed -e 's/^.*TYPE="//' -e 's/".*$//'
}
probe_fs_type() {
fst=$(blkid_fs_type "$1")
echo "${fst:=jffs2}"
}
rwfs=$(findmtd rwfs)
rwdev=/dev/mtdblock${rwfs#mtd}
rwopts=rw
rorwopts=ro${rwopts#rw}
rwdir=/run/initramfs/rw
upper=$rwdir/cow
save=/run/save/${upper##*/}
mounted=
doflash=y
doclean=
dosave=y
dorestore=y
toram=
checksize=y
checkmount=y
whitelist=/run/initramfs/whitelist
image=/run/initramfs/image-
imglist=
while test "$1" != "${1#-}"
do
case "$1" in
--help)
cat <<HERE
Usage: $0 [options] -- Write images in /run/initramfs to flash (/dev/mtd*)
--help Show this message
--no-flash Don't attempt to write images to flash
--ignore-size Don't compare image size to mtd device size
--ignore-mount Don't check if destination is mounted
--save-files Copy whitelisted files to save directory in RAM
--no-save-files Don't copy whitelisted files to save directory
--copy-files Copy files from save directory to rwfs mountpoint
--restore-files Restore files from save directory to rwfs layer
--no-restore-files Don't restore saved files from ram to rwfs layer
--clean-saved-files Delete saved whitelisted files from RAM
--no-clean-saved-files Retain saved whitelisted files in RAM
HERE
exit 0 ;;
--no-clean-saved-files)
doclean=
shift ;;
--clean-saved-files)
doclean=y
shift ;;
--no-save-files)
dosave=
shift ;;
--save-files)
dosave=y
shift ;;
--no-restore-files)
dorestore=
shift ;;
--restore-files)
dorestore=y
shift ;;
--no-flash)
doflash=
shift ;;
--ignore-size)
checksize=
shift ;;
--ignore-mount)
checkmount=
doflash=
shift ;;
--copy-files)
toram=y
shift ;;
*)
echoerr "Unknown option $1. Try $0 --help."
exit 1 ;;
esac
done
if test "$dosave" = "y"
then
if test ! -d $upper -a -n "$rwfs"
then
mkdir -p $rwdir
mount "$rwdev" $rwdir -t "$(probe_fs_type "$rwdev")" -o "$rorwopts"
mounted=$rwdir
fi
while read -r f
do
# Entries shall start with /, no trailing /.. or embedded /../
if test "/${f#/}" != "$f" -o "${f%/..}" != "${f#*/../}"
then
echo 1>&2 "WARNING: Skipping bad whitelist entry $f."
continue
fi
if ! test -e "$upper/$f"
then
continue
fi
d="$save/$f"
while test "${d%/}" != "${d%/.}"
do
d="${d%/.}"
d="${d%/}"
done
mkdir -p "${d%/*}"
cp -rp "$upper/$f" "${d%/*}/"
done < $whitelist
if test -n "$mounted"
then
umount $mounted
fi
fi
imglist=$(echo $image*)
if test "$imglist" = "$image*" -a ! -e "$imglist"
then
# shell didn't expand the wildcard, so no files exist
echo "No images found to update."
imglist=
fi
for f in $imglist
do
m=$(findmtd "${f#"$image"}")
if test -z "$m"
then
echoerr "Unable to find mtd partition for ${f##*/}."
exit 1
fi
if test -n "$checksize" && toobig "$f" "$m"
then
echoerr "Image ${f##*/} too big for $m."
exit 1
fi
for s in $m $(childmtds "$m")
do
if test -n "$checkmount" && mtdismounted "$s"
then
echoerr "Device $s is mounted, ${f##*/} is busy."
exit 1
fi
done
done
if test -n "$doflash"
then
for f in $imglist
do
if test ! -s "$f"
then
echo "Skipping empty update of ${f#"$image"}."
rm "$f"
continue
fi
m=$(findmtd "${f#"$image"}")
echo "Updating ${f#"$image"}..."
flashcp -v "$f" "/dev/$m" && rm "$f"
done
fi
if test -d "$save" -a "$toram" = "y"
then
mkdir -p $upper
cp -rp "$save/." "$upper/"
fi
if test -d "$save" -a "$dorestore" = "y"
then
odir=$rwdir
rwdir=/run/rw
upper=$rwdir${upper#"$odir"}
mkdir -p $rwdir
mount "$rwdev" $rwdir -t "$(probe_fs_type "$rwdev")" -o $rwopts
mkdir -p "$upper"
cp -rp "$save/." "$upper/"
umount $rwdir
rmdir $rwdir
fi
if test "$doclean" = "y"
then
rm -rf "$save"
fi
exit