Dan Pasanen | 894b352 | 2017-01-16 19:26:50 -0600 | [diff] [blame] | 1 | #!/system/bin/sh |
| 2 | # |
| 3 | # Backup and restore addon /system files |
| 4 | # |
| 5 | |
| 6 | export S=/system |
| 7 | export C=/postinstall/tmp/backupdir |
maxwen | 5b15888 | 2022-08-21 00:02:47 +0200 | [diff] [blame] | 8 | export V=13 |
Dan Pasanen | 894b352 | 2017-01-16 19:26:50 -0600 | [diff] [blame] | 9 | |
Alessandro Astone | 09bec2c | 2020-12-29 18:38:28 +0100 | [diff] [blame] | 10 | export ADDOND_VERSION=3 |
| 11 | |
| 12 | # Partitions to mount for backup/restore in V3 |
Alessandro Astone | 9c8972e | 2021-02-18 14:04:13 +0100 | [diff] [blame^] | 13 | export all_V3_partitions="vendor product system_ext" |
Dan Pasanen | 8320eb1 | 2018-01-26 10:26:47 -0600 | [diff] [blame] | 14 | |
Dan Pasanen | 894b352 | 2017-01-16 19:26:50 -0600 | [diff] [blame] | 15 | # Scripts in /system/addon.d expect to find backuptool.functions in /tmp |
| 16 | mkdir -p /postinstall/tmp/ |
Alessandro Astone | 3b6b6b8 | 2020-12-29 14:16:20 +0100 | [diff] [blame] | 17 | mountpoint /postinstall/tmp >/dev/null 2>&1 || mount -t tmpfs tmpfs /postinstall/tmp |
Dan Pasanen | 894b352 | 2017-01-16 19:26:50 -0600 | [diff] [blame] | 18 | cp -f /postinstall/system/bin/backuptool_ab.functions /postinstall/tmp/backuptool.functions |
| 19 | |
Alessandro Astone | 09bec2c | 2020-12-29 18:38:28 +0100 | [diff] [blame] | 20 | get_script_version() { |
| 21 | version=$(grep "^# ADDOND_VERSION=" $1 | cut -d= -f2) |
| 22 | [ -z "$version" ] && version=1 |
| 23 | echo $version |
| 24 | } |
| 25 | |
Dan Pasanen | 894b352 | 2017-01-16 19:26:50 -0600 | [diff] [blame] | 26 | # Preserve /system/addon.d in /tmp/addon.d |
| 27 | preserve_addon_d() { |
| 28 | if [ -d /system/addon.d/ ]; then |
| 29 | mkdir -p /postinstall/tmp/addon.d/ |
| 30 | cp -a /system/addon.d/* /postinstall/tmp/addon.d/ |
Marko Man | 4e25498 | 2018-10-17 16:08:23 +0200 | [diff] [blame] | 31 | rm -f /postinstall/tmp/addon.d/70-gapps.sh |
Dan Pasanen | 8320eb1 | 2018-01-26 10:26:47 -0600 | [diff] [blame] | 32 | |
Alessandro Astone | 09bec2c | 2020-12-29 18:38:28 +0100 | [diff] [blame] | 33 | # Discard any version 1 script, as it is not compatible with a/b |
Dan Pasanen | 8320eb1 | 2018-01-26 10:26:47 -0600 | [diff] [blame] | 34 | for f in /postinstall/tmp/addon.d/*sh; do |
Alessandro Astone | 09bec2c | 2020-12-29 18:38:28 +0100 | [diff] [blame] | 35 | if [ $(get_script_version $f) = 1 ]; then |
Dan Pasanen | 8320eb1 | 2018-01-26 10:26:47 -0600 | [diff] [blame] | 36 | rm $f |
| 37 | fi |
| 38 | done |
Dan Pasanen | 894b352 | 2017-01-16 19:26:50 -0600 | [diff] [blame] | 39 | chmod 755 /postinstall/tmp/addon.d/*.sh |
| 40 | fi |
| 41 | } |
| 42 | |
| 43 | # Restore /postinstall/system/addon.d from /postinstall/tmp/addon.d |
| 44 | restore_addon_d() { |
| 45 | if [ -d /postinstall/tmp/addon.d/ ]; then |
| 46 | mkdir -p /postinstall/system/addon.d/ |
| 47 | cp -a /postinstall/tmp/addon.d/* /postinstall/system/addon.d/ |
| 48 | rm -rf /postinstall/tmp/addon.d/ |
| 49 | fi |
| 50 | } |
| 51 | |
| 52 | # Proceed only if /system is the expected major and minor version |
| 53 | check_prereq() { |
| 54 | # If there is no build.prop file the partition is probably empty. |
| 55 | if [ ! -r /system/build.prop ]; then |
z3DD3r | 31285e5 | 2020-03-04 18:07:40 +0300 | [diff] [blame] | 56 | echo "Backup/restore is not possible. Partition is probably empty" |
| 57 | return 1 |
Dan Pasanen | 894b352 | 2017-01-16 19:26:50 -0600 | [diff] [blame] | 58 | fi |
| 59 | |
z3DD3r | 8ca31e6 | 2020-03-04 12:42:10 +0300 | [diff] [blame] | 60 | if ! grep -q "^ro.omni.version=$V.*" /product/build.prop; then |
z3DD3r | 31285e5 | 2020-03-04 18:07:40 +0300 | [diff] [blame] | 61 | echo "Backup/restore is not possible. Incompatible ROM version: $V" |
| 62 | return 2 |
z3DD3r | 8ca31e6 | 2020-03-04 12:42:10 +0300 | [diff] [blame] | 63 | fi |
z3DD3r | 31285e5 | 2020-03-04 18:07:40 +0300 | [diff] [blame] | 64 | return 0 |
Dan Pasanen | 894b352 | 2017-01-16 19:26:50 -0600 | [diff] [blame] | 65 | } |
| 66 | |
Alessandro Astone | 09bec2c | 2020-12-29 18:38:28 +0100 | [diff] [blame] | 67 | # Execute /system/addon.d/*.sh scripts with each $@ parameter |
| 68 | run_stages() { |
maxwen | 1e534e8 | 2020-03-29 16:52:37 +0200 | [diff] [blame] | 69 | log -t "update_engine" $1 |
| 70 | |
Dan Pasanen | 894b352 | 2017-01-16 19:26:50 -0600 | [diff] [blame] | 71 | if [ -d /postinstall/tmp/addon.d/ ]; then |
| 72 | for script in $(find /postinstall/tmp/addon.d/ -name '*.sh' |sort -n); do |
maxwen | 1e534e8 | 2020-03-29 16:52:37 +0200 | [diff] [blame] | 73 | log -t "update_engine" $script |
| 74 | |
Dan Pasanen | 894b352 | 2017-01-16 19:26:50 -0600 | [diff] [blame] | 75 | # we have no /sbin/sh in android, only recovery |
| 76 | # use /system/bin/sh here instead |
| 77 | sed -i '0,/#!\/sbin\/sh/{s|#!/sbin/sh|#!/system/bin/sh|}' $script |
Alessandro Astone | 3b6b6b8 | 2020-12-29 14:16:20 +0100 | [diff] [blame] | 78 | # we can't count on /tmp existing on an A/B device, so utilize /postinstall/tmp as tmpfs |
Dan Pasanen | 894b352 | 2017-01-16 19:26:50 -0600 | [diff] [blame] | 79 | sed -i 's|. /tmp/backuptool.functions|. /postinstall/tmp/backuptool.functions|g' $script |
Alessandro Astone | 09bec2c | 2020-12-29 18:38:28 +0100 | [diff] [blame] | 80 | v=$(get_script_version $script) |
| 81 | if [ $v -ge 3 ]; then |
| 82 | mount_extra $all_V3_partitions |
| 83 | else |
| 84 | umount_extra $all_V3_partitions |
| 85 | fi |
| 86 | |
| 87 | for stage in $@; do |
| 88 | if [ $v -ge 3 ]; then |
| 89 | $script $stage |
| 90 | else |
| 91 | ADDOND_VERSION=2 $script $stage |
| 92 | fi |
| 93 | done |
Dan Pasanen | 894b352 | 2017-01-16 19:26:50 -0600 | [diff] [blame] | 94 | done |
| 95 | fi |
| 96 | } |
| 97 | |
Alessandro Astone | 09bec2c | 2020-12-29 18:38:28 +0100 | [diff] [blame] | 98 | ##################### |
| 99 | ### Mount helpers ### |
| 100 | ##################### |
| 101 | get_block_for_mount_point() { |
| 102 | grep -v "^#" /vendor/etc/fstab.$(getprop ro.boot.hardware) | grep " $1 " | tail -n1 | tr -s ' ' | cut -d' ' -f1 |
| 103 | } |
| 104 | |
| 105 | find_block() { |
| 106 | local name="$1" |
| 107 | local fstab_entry=$(get_block_for_mount_point "/$name") |
| 108 | # P-SAR hacks |
| 109 | [ -z "$fstab_entry" ] && [ "$name" = "system" ] && fstab_entry=$(get_block_for_mount_point "/") |
| 110 | [ -z "$fstab_entry" ] && [ "$name" = "system" ] && fstab_entry=$(get_block_for_mount_point "/system_root") |
| 111 | |
| 112 | local dev |
| 113 | if [ "$DYNAMIC_PARTITIONS" = "true" ]; then |
| 114 | if [ -n "$fstab_entry" ]; then |
| 115 | dev="${BLK_PATH}/${fstab_entry}${SLOT_SUFFIX}" |
| 116 | else |
| 117 | dev="${BLK_PATH}/${name}${SLOT_SUFFIX}" |
| 118 | fi |
| 119 | else |
| 120 | if [ -n "$fstab_entry" ]; then |
| 121 | dev="${fstab_entry}${SLOT_SUFFIX}" |
| 122 | else |
| 123 | dev="${BLK_PATH}/${name}${SLOT_SUFFIX}" |
| 124 | fi |
| 125 | fi |
| 126 | |
| 127 | if [ -b "$dev" ]; then |
| 128 | echo "$dev" |
| 129 | fi |
| 130 | } |
| 131 | |
| 132 | DYNAMIC_PARTITIONS=$(getprop ro.boot.dynamic_partitions) |
| 133 | if [ "$DYNAMIC_PARTITIONS" = "true" ]; then |
| 134 | BLK_PATH="/dev/block/mapper" |
| 135 | else |
| 136 | BLK_PATH=/dev/block/bootdevice/by-name |
| 137 | fi |
| 138 | |
| 139 | CURRENTSLOT=$(getprop ro.boot.slot_suffix) |
| 140 | if [ ! -z "$CURRENTSLOT" ]; then |
| 141 | if [ "$CURRENTSLOT" = "_a" ]; then |
| 142 | # Opposite slot |
| 143 | SLOT_SUFFIX="_b" |
| 144 | else |
| 145 | SLOT_SUFFIX="_a" |
| 146 | fi |
| 147 | fi |
| 148 | |
| 149 | mount_extra() { |
| 150 | for partition in $@; do |
| 151 | mnt_point="/postinstall/$partition" |
| 152 | mountpoint "$mnt_point" >/dev/null 2>&1 && break |
| 153 | |
| 154 | blk_dev=$(find_block "$partition") |
| 155 | if [ -n "$blk_dev" ]; then |
| 156 | [ "$DYNAMIC_PARTITIONS" = "true" ] && blockdev --setrw "$blk_dev" |
| 157 | mount -o rw "$blk_dev" "$mnt_point" |
| 158 | fi |
| 159 | done |
| 160 | } |
| 161 | |
| 162 | umount_extra() { |
| 163 | for partition in $@; do |
| 164 | # Careful with unmounting. If the update has a partition less than the current system, |
| 165 | # /postinstall/$partition is a symlink to /system/$partition, which on the active slot |
| 166 | # is a symlink to /$partition which is a mountpoint we would end up unmounting! |
| 167 | [ ! -L "/postinstall/$partition" ] && umount -l "/postinstall/$partition" 2>/dev/null |
| 168 | done |
| 169 | } |
| 170 | |
Dan Pasanen | 894b352 | 2017-01-16 19:26:50 -0600 | [diff] [blame] | 171 | case "$1" in |
| 172 | backup) |
z3DD3r | 31285e5 | 2020-03-04 18:07:40 +0300 | [diff] [blame] | 173 | if check_prereq; then |
| 174 | mkdir -p $C |
| 175 | preserve_addon_d |
Alessandro Astone | 09bec2c | 2020-12-29 18:38:28 +0100 | [diff] [blame] | 176 | run_stages pre-backup backup post-backup |
Dan Pasanen | 894b352 | 2017-01-16 19:26:50 -0600 | [diff] [blame] | 177 | fi |
maxwen | 1e534e8 | 2020-03-29 16:52:37 +0200 | [diff] [blame] | 178 | log -t "update_engine" "backuptool_ab.sh backup" |
Dan Pasanen | 894b352 | 2017-01-16 19:26:50 -0600 | [diff] [blame] | 179 | ;; |
| 180 | restore) |
z3DD3r | 31285e5 | 2020-03-04 18:07:40 +0300 | [diff] [blame] | 181 | if check_prereq; then |
Alessandro Astone | 09bec2c | 2020-12-29 18:38:28 +0100 | [diff] [blame] | 182 | run_stages pre-restore restore post-restore |
| 183 | umount_extra $all_V3_partitions |
z3DD3r | 31285e5 | 2020-03-04 18:07:40 +0300 | [diff] [blame] | 184 | restore_addon_d |
| 185 | rm -rf $C |
Alessandro Astone | 3b6b6b8 | 2020-12-29 14:16:20 +0100 | [diff] [blame] | 186 | umount /postinstall/tmp |
z3DD3r | 31285e5 | 2020-03-04 18:07:40 +0300 | [diff] [blame] | 187 | rm -rf /postinstall/tmp |
| 188 | sync |
maxwen | 1e534e8 | 2020-03-29 16:52:37 +0200 | [diff] [blame] | 189 | log -t "update_engine" "backuptool_ab.sh restore" |
Dan Pasanen | 894b352 | 2017-01-16 19:26:50 -0600 | [diff] [blame] | 190 | ;; |
| 191 | *) |
| 192 | echo "Usage: $0 {backup|restore}" |
| 193 | exit 1 |
| 194 | esac |
| 195 | |
| 196 | exit 0 |