| #!/system/bin/sh |
| # |
| # Backup and restore addon /system files |
| # |
| |
| export S=/system |
| export C=/postinstall/tmp/backupdir |
| export V=13 |
| |
| export ADDOND_VERSION=3 |
| |
| # Partitions to mount for backup/restore in V3 |
| export all_V3_partitions="vendor product system_ext" |
| |
| # Scripts in /system/addon.d expect to find backuptool.functions in /tmp |
| mkdir -p /postinstall/tmp/ |
| mountpoint /postinstall/tmp >/dev/null 2>&1 || mount -t tmpfs tmpfs /postinstall/tmp |
| cp -f /postinstall/system/bin/backuptool_ab.functions /postinstall/tmp/backuptool.functions |
| |
| get_script_version() { |
| version=$(grep "^# ADDOND_VERSION=" $1 | cut -d= -f2) |
| [ -z "$version" ] && version=1 |
| echo $version |
| } |
| |
| # Preserve /system/addon.d in /tmp/addon.d |
| preserve_addon_d() { |
| if [ -d /system/addon.d/ ]; then |
| mkdir -p /postinstall/tmp/addon.d/ |
| cp -a /system/addon.d/* /postinstall/tmp/addon.d/ |
| rm -f /postinstall/tmp/addon.d/70-gapps.sh |
| |
| # Discard any version 1 script, as it is not compatible with a/b |
| for f in /postinstall/tmp/addon.d/*sh; do |
| if [ $(get_script_version $f) = 1 ]; then |
| rm $f |
| fi |
| done |
| chmod 755 /postinstall/tmp/addon.d/*.sh |
| fi |
| } |
| |
| # Restore /postinstall/system/addon.d from /postinstall/tmp/addon.d |
| restore_addon_d() { |
| if [ -d /postinstall/tmp/addon.d/ ]; then |
| mkdir -p /postinstall/system/addon.d/ |
| cp -a /postinstall/tmp/addon.d/* /postinstall/system/addon.d/ |
| rm -rf /postinstall/tmp/addon.d/ |
| fi |
| } |
| |
| # Proceed only if /system is the expected major and minor version |
| check_prereq() { |
| # If there is no build.prop file the partition is probably empty. |
| if [ ! -r /system/build.prop ]; then |
| echo "Backup/restore is not possible. Partition is probably empty" |
| return 1 |
| fi |
| |
| if ! grep -q "^ro.omni.version=$V.*" /system/build.prop; then |
| echo "Backup/restore is not possible. Incompatible ROM version: $V" |
| return 2 |
| fi |
| return 0 |
| } |
| |
| # Execute /system/addon.d/*.sh scripts with each $@ parameter |
| run_stages() { |
| log -t "update_engine" $1 |
| |
| if [ -d /postinstall/tmp/addon.d/ ]; then |
| for script in $(find /postinstall/tmp/addon.d/ -name '*.sh' |sort -n); do |
| log -t "update_engine" $script |
| |
| # we have no /sbin/sh in android, only recovery |
| # use /system/bin/sh here instead |
| sed -i '0,/#!\/sbin\/sh/{s|#!/sbin/sh|#!/system/bin/sh|}' $script |
| # we can't count on /tmp existing on an A/B device, so utilize /postinstall/tmp as tmpfs |
| sed -i 's|. /tmp/backuptool.functions|. /postinstall/tmp/backuptool.functions|g' $script |
| v=$(get_script_version $script) |
| if [ $v -ge 3 ]; then |
| mount_extra $all_V3_partitions |
| else |
| umount_extra $all_V3_partitions |
| fi |
| |
| for stage in $@; do |
| if [ $v -ge 3 ]; then |
| $script $stage |
| else |
| ADDOND_VERSION=2 $script $stage |
| fi |
| done |
| done |
| fi |
| } |
| |
| ##################### |
| ### Mount helpers ### |
| ##################### |
| get_block_for_mount_point() { |
| grep -v "^#" /vendor/etc/fstab.$(getprop ro.boot.hardware) | grep "[[:blank:]]$1[[:blank:]]" | tail -n1 | tr -s [:blank:] ' ' | cut -d' ' -f1 |
| } |
| |
| find_block() { |
| local name="$1" |
| local fstab_entry=$(get_block_for_mount_point "/$name") |
| # P-SAR hacks |
| [ -z "$fstab_entry" ] && [ "$name" = "system" ] && fstab_entry=$(get_block_for_mount_point "/") |
| [ -z "$fstab_entry" ] && [ "$name" = "system" ] && fstab_entry=$(get_block_for_mount_point "/system_root") |
| |
| local dev |
| if [ "$DYNAMIC_PARTITIONS" = "true" ]; then |
| if [ -n "$fstab_entry" ]; then |
| dev="${BLK_PATH}/${fstab_entry}${SLOT_SUFFIX}" |
| else |
| dev="${BLK_PATH}/${name}${SLOT_SUFFIX}" |
| fi |
| else |
| if [ -n "$fstab_entry" ]; then |
| dev="${fstab_entry}${SLOT_SUFFIX}" |
| else |
| dev="${BLK_PATH}/${name}${SLOT_SUFFIX}" |
| fi |
| fi |
| |
| if [ -b "$dev" ]; then |
| echo "$dev" |
| fi |
| } |
| |
| DYNAMIC_PARTITIONS=$(getprop ro.boot.dynamic_partitions) |
| if [ "$DYNAMIC_PARTITIONS" = "true" ]; then |
| BLK_PATH="/dev/block/mapper" |
| else |
| BLK_PATH=/dev/block/bootdevice/by-name |
| fi |
| |
| CURRENTSLOT=$(getprop ro.boot.slot_suffix) |
| if [ ! -z "$CURRENTSLOT" ]; then |
| if [ "$CURRENTSLOT" = "_a" ]; then |
| # Opposite slot |
| SLOT_SUFFIX="_b" |
| else |
| SLOT_SUFFIX="_a" |
| fi |
| fi |
| |
| mount_extra() { |
| for partition in $@; do |
| mnt_point="/postinstall/$partition" |
| mountpoint "$mnt_point" >/dev/null 2>&1 && continue |
| [ -L "$mnt_point" ] && continue |
| |
| blk_dev=$(find_block "$partition") |
| if [ -n "$blk_dev" ]; then |
| [ "$DYNAMIC_PARTITIONS" = "true" ] && blockdev --setrw "$blk_dev" |
| mount -o rw "$blk_dev" "$mnt_point" |
| fi |
| done |
| } |
| |
| umount_extra() { |
| for partition in $@; do |
| # Careful with unmounting. If the update has a partition less than the current system, |
| # /postinstall/$partition is a symlink to /system/$partition, which on the active slot |
| # is a symlink to /$partition which is a mountpoint we would end up unmounting! |
| [ ! -L "/postinstall/$partition" ] && umount -l "/postinstall/$partition" 2>/dev/null |
| done |
| } |
| |
| cleanup() { |
| umount_extra $all_V3_partitions |
| umount /postinstall/tmp |
| rm -rf /postinstall/tmp |
| } |
| |
| case "$1" in |
| backup) |
| if check_prereq; then |
| mkdir -p $C |
| preserve_addon_d |
| run_stages pre-backup backup post-backup |
| umount_extra $all_V3_partitions |
| else |
| cleanup |
| fi |
| log -t "update_engine" "backuptool_ab.sh backup" |
| ;; |
| restore) |
| if check_prereq; then |
| run_stages pre-restore restore post-restore |
| restore_addon_d |
| cleanup |
| sync |
| else |
| cleanup |
| fi |
| log -t "update_engine" "backuptool_ab.sh restore" |
| ;; |
| *) |
| echo "Usage: $0 {backup|restore}" |
| exit 1 |
| esac |
| |
| exit 0 |