forked from Mirror/pmbootstrap
Recovery installer zip (#404)
Supports flashing with TWRP and other Android recovery OS through adb sideload, as well as exporting a generated recovery zip file. See also: https://github.com/postmarketOS/pmbootstrap/wiki/deviceinfo_flash_methods#recovery-mode-adb
This commit is contained in:
parent
28fa7eeaf0
commit
fbe968f1ab
17 changed files with 552 additions and 54 deletions
33
aports/main/postmarketos-android-recovery-installer/APKBUILD
Normal file
33
aports/main/postmarketos-android-recovery-installer/APKBUILD
Normal file
|
@ -0,0 +1,33 @@
|
||||||
|
pkgname=postmarketos-android-recovery-installer
|
||||||
|
pkgver=0.0.1
|
||||||
|
pkgrel=1
|
||||||
|
pkgdesc="TWRP compatible postmarketOS installer script"
|
||||||
|
url="https://github.com/postmarketOS"
|
||||||
|
# multipath-tools: kpartx
|
||||||
|
depends="busybox-extras lddtree cryptsetup multipath-tools device-mapper parted zip"
|
||||||
|
source="build_zip.sh
|
||||||
|
update-binary
|
||||||
|
pmos_install
|
||||||
|
pmos_install_functions
|
||||||
|
pmos_setpw"
|
||||||
|
arch="noarch"
|
||||||
|
license="GPL3"
|
||||||
|
|
||||||
|
package() {
|
||||||
|
install -Dm755 "$srcdir/build_zip.sh" \
|
||||||
|
"$pkgdir/sbin/build-recovery-zip"
|
||||||
|
mkdir -p "$pkgdir/var/lib/postmarketos-android-recovery-installer/META-INF/com/google/android/"
|
||||||
|
install -Dm644 "$srcdir"/update-binary \
|
||||||
|
"$pkgdir/var/lib/postmarketos-android-recovery-installer/META-INF/com/google/android/update-binary"
|
||||||
|
mkdir "$pkgdir/var/lib/postmarketos-android-recovery-installer/bin/"
|
||||||
|
for file in pmos_install pmos_install_functions pmos_setpw; do
|
||||||
|
install -Dm755 "$srcdir/$file" \
|
||||||
|
"$pkgdir/var/lib/postmarketos-android-recovery-installer/bin/$file"
|
||||||
|
done
|
||||||
|
mkdir "$pkgdir/var/lib/postmarketos-android-recovery-installer/lib/"
|
||||||
|
}
|
||||||
|
sha512sums="9c7a90965aeb7f19ac166282066063510eeba6691ae695b2821e1a9e050463378c56492a27b3bfd4c8155380e6f24adc558dd0c98fc308ee7335b00c7b12fc0b build_zip.sh
|
||||||
|
874d7505f9940d98a67fd8e5881ab0b93ae9fd0c46d7f4004468a2e9bbe4853f4bf6db64380c27684a66ebbd45ebe3399219b3910799de24003b8399ab2a4497 update-binary
|
||||||
|
5647a089c95d291d5662bbe6931a01f8591823d63b0226832897a046f351121c2c5d6ebfc83dcf9762ac50774920393fea37c05a92f2079e9688d6fe58711e49 pmos_install
|
||||||
|
dba3da4d2c18a480fda3bda233052f946bfd5a5f4fe05115341d4dc1466519584930e116719c5338ef4309a51dfea7d2e531ed133723f59c8d6d0c5a4f73a26b pmos_install_functions
|
||||||
|
1969a467bc1e0f04ed445dd78db4eb1ad77b769a6e04c35211ad2c45cb8293243f864e499cdecf6016292d1accb26e6f62073b2afab023a20a79e0ea3dc15bd9 pmos_setpw"
|
|
@ -0,0 +1,52 @@
|
||||||
|
#!/bin/ash
|
||||||
|
|
||||||
|
# Copyright 2017 Attila Szöllősi
|
||||||
|
#
|
||||||
|
# This file is part of postmarketos-android-recovery-installer.
|
||||||
|
#
|
||||||
|
# postmarketos-android-recovery-installer is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU General Public License as published by
|
||||||
|
# the Free Software Foundation, either version 3 of the License, or
|
||||||
|
# (at your option) any later version.
|
||||||
|
#
|
||||||
|
# postmarketos-android-recovery-installer is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License
|
||||||
|
# along with postmarketos-android-recovery-installer. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
set -e
|
||||||
|
|
||||||
|
# Copy files to the destination specified
|
||||||
|
# $1: files
|
||||||
|
# $2: destination
|
||||||
|
copy_files()
|
||||||
|
{
|
||||||
|
for file in $1; do
|
||||||
|
filename=$(basename "$file")
|
||||||
|
install -Dm755 "$file" "$2"/"$filename"
|
||||||
|
done
|
||||||
|
}
|
||||||
|
|
||||||
|
check_whether_exists()
|
||||||
|
{
|
||||||
|
if [ ! -e "$1" ]
|
||||||
|
then
|
||||||
|
echo "$1 not found"
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
# shellcheck disable=SC1091
|
||||||
|
. ./install_options
|
||||||
|
|
||||||
|
BINARIES="/sbin/cryptsetup /sbin/kpartx /usr/sbin/parted /usr/sbin/partprobe"
|
||||||
|
# shellcheck disable=SC2086
|
||||||
|
LIBRARIES=$(lddtree -l $BINARIES | awk '/lib/ {print}' | sort -u)
|
||||||
|
copy_files "$BINARIES" bin/
|
||||||
|
copy_files "$LIBRARIES" lib/
|
||||||
|
check_whether_exists rootfs.tar.gz
|
||||||
|
[ "$FLASH_BOOT" = "true" ] && check_whether_exists boot.img
|
||||||
|
zip -0 -r "pmos-$DEVICE.zip" .
|
86
aports/main/postmarketos-android-recovery-installer/pmos_install
Executable file
86
aports/main/postmarketos-android-recovery-installer/pmos_install
Executable file
|
@ -0,0 +1,86 @@
|
||||||
|
#!/sbin/ash
|
||||||
|
|
||||||
|
# Copyright 2017 Attila Szöllősi
|
||||||
|
#
|
||||||
|
# This file is part of postmarketos-android-recovery-installer.
|
||||||
|
#
|
||||||
|
# postmarketos-android-recovery-installer is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU General Public License as published by
|
||||||
|
# the Free Software Foundation, either version 3 of the License, or
|
||||||
|
# (at your option) any later version.
|
||||||
|
#
|
||||||
|
# postmarketos-android-recovery-installer is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License
|
||||||
|
# along with postmarketos-android-recovery-installer. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
# shellcheck source=pmos_install_functions
|
||||||
|
. /tmp/postmarketos/bin/pmos_install_functions "$1" "$2"
|
||||||
|
# shellcheck source=/dev/null
|
||||||
|
. "$WORKING_DIR"/install_options
|
||||||
|
|
||||||
|
exec > "$WORKING_DIR"/pmos.log 2>&1
|
||||||
|
set -ex
|
||||||
|
|
||||||
|
ui_print "postmarketOS recovery installer"
|
||||||
|
|
||||||
|
ui_print "Extracting partition info from fstab..."
|
||||||
|
extract_partition_table
|
||||||
|
ui_print "Entering working directory..."
|
||||||
|
cd "$WORKING_DIR"
|
||||||
|
ui_print "Extracting files..."
|
||||||
|
unzip -o "$ZIP"
|
||||||
|
mkdir /lib
|
||||||
|
ui_print "Symlinking .so files to /lib/..."
|
||||||
|
ln -s "$WORKING_DIR"/lib/* /lib/
|
||||||
|
ui_print "Unmounting /$INSTALL_PARTITION..."
|
||||||
|
umount_install_partition
|
||||||
|
ui_print "Creating partition table on $INSTALL_DEVICE..."
|
||||||
|
# parted returns nonzero even when command executed successfully
|
||||||
|
partition_install_device || :
|
||||||
|
if [ "$FDE" = "true" ]
|
||||||
|
then
|
||||||
|
ui_print "Generating temporary keyfile with random data..."
|
||||||
|
dd bs=512 count=4 if=/dev/urandom of="$WORKING_DIR"/lukskey
|
||||||
|
ui_print "Initializing LUKS device..."
|
||||||
|
cryptsetup luksFormat --use-urandom -c "$CIPHER" -q "$ROOT_PARTITION" "$WORKING_DIR"/lukskey
|
||||||
|
ui_print "Opening LUKS partition..."
|
||||||
|
cryptsetup luksOpen -d "$WORKING_DIR"/lukskey "$ROOT_PARTITION" pm_crypt
|
||||||
|
ui_print "Formatting LUKS partition..."
|
||||||
|
make_ext4fs -L 'pmOS_root' /dev/mapper/pm_crypt
|
||||||
|
ui_print "Mounting LUKS partition..."
|
||||||
|
mount -t ext4 -rw /dev/mapper/pm_crypt /"$INSTALL_PARTITION"
|
||||||
|
else
|
||||||
|
ui_print "Formatting root partition..."
|
||||||
|
make_ext4fs -L 'pmOS_root' "$ROOT_PARTITION"
|
||||||
|
ui_print "Mounting root partition..."
|
||||||
|
mount -t ext4 -rw "$ROOT_PARTITION" /"$INSTALL_PARTITION"
|
||||||
|
fi
|
||||||
|
ui_print "Formatting pmOS_boot..."
|
||||||
|
mkfs.ext2 -q -L 'pmOS_boot' "$PMOS_BOOT"
|
||||||
|
ui_print "Mounting pmOS_boot..."
|
||||||
|
mkdir /"$INSTALL_PARTITION"/boot
|
||||||
|
mount -t ext2 -rw "$PMOS_BOOT" /"$INSTALL_PARTITION"/boot || {
|
||||||
|
ui_print "Failed to format/mount ext2 partition."
|
||||||
|
ui_print "Trying ext4..."
|
||||||
|
make_ext4fs -L 'pmOS_boot' "$PMOS_BOOT"
|
||||||
|
mount -t ext4 -rw "$PMOS_BOOT" /"$INSTALL_PARTITION"/boot
|
||||||
|
}
|
||||||
|
ui_print "Installing rootfs..."
|
||||||
|
tar -xf rootfs.tar.gz -C /"$INSTALL_PARTITION"
|
||||||
|
if [ "$FLASH_BOOTIMG" = "true" ]
|
||||||
|
then
|
||||||
|
ui_print "Flashing boot.img..."
|
||||||
|
dd if=boot.img of="$BOOT"
|
||||||
|
fi
|
||||||
|
if [ "$FDE" = "true" ]
|
||||||
|
then
|
||||||
|
ui_print "Creating a symlink for password setting script in /sbin/..."
|
||||||
|
ln -s "$WORKING_DIR"/bin/pmos_setpw /sbin/
|
||||||
|
ui_print "Do not forget to add a password to the LUKS partition!"
|
||||||
|
ui_print "Run the command: pmos_setpw from the terminal/adb shell!"
|
||||||
|
fi
|
||||||
|
ui_print "Installation done."
|
98
aports/main/postmarketos-android-recovery-installer/pmos_install_functions
Executable file
98
aports/main/postmarketos-android-recovery-installer/pmos_install_functions
Executable file
|
@ -0,0 +1,98 @@
|
||||||
|
#!/sbin/ash
|
||||||
|
|
||||||
|
# Copyright 2017 Attila Szöllősi
|
||||||
|
#
|
||||||
|
# This file is part of postmarketos-android-recovery-installer.
|
||||||
|
#
|
||||||
|
# postmarketos-android-recovery-installer is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU General Public License as published by
|
||||||
|
# the Free Software Foundation, either version 3 of the License, or
|
||||||
|
# (at your option) any later version.
|
||||||
|
#
|
||||||
|
# postmarketos-android-recovery-installer is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License
|
||||||
|
# along with postmarketos-android-recovery-installer. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
export OUTFD=$1
|
||||||
|
export ZIP=$2
|
||||||
|
export WORKING_DIR="/tmp/postmarketos"
|
||||||
|
export PATH=$PATH:"$WORKING_DIR"/bin
|
||||||
|
|
||||||
|
# shellcheck source=/dev/null
|
||||||
|
. "$WORKING_DIR"/install_options
|
||||||
|
|
||||||
|
# taken from https://github.com/Debuffer-XDA/Gov-Tuner/blob/master/META-INF/com/google/android/update-binary
|
||||||
|
# Copyright (c) 2016 - 2017 Debuffer
|
||||||
|
ui_print() {
|
||||||
|
echo -n -e "ui_print $1\n" > /proc/self/fd/"$OUTFD"
|
||||||
|
echo -n -e "ui_print\n" > /proc/self/fd/"$OUTFD"
|
||||||
|
}
|
||||||
|
|
||||||
|
extract_partition_table() {
|
||||||
|
case "$INSTALL_PARTITION" in
|
||||||
|
"system")
|
||||||
|
# We need to resolve symlinks, to make set_subpartitions() work.
|
||||||
|
_INSTALL_DEVICE=$(readlink -fn "$(awk '/^\/system/ {print $3}' /etc/recovery.fstab)")
|
||||||
|
;;
|
||||||
|
"external_sd")
|
||||||
|
_INSTALL_DEVICE=$(readlink -fn "$(awk '/^\/external_sd/ {print $4}' /etc/recovery.fstab)")
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
echo "No support for flashing $INSTALL_PARTITION."
|
||||||
|
return 1
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
if [ ! -z "$_INSTALL_DEVICE" ]
|
||||||
|
then
|
||||||
|
echo "install device found at $_INSTALL_DEVICE"
|
||||||
|
export INSTALL_DEVICE=$_INSTALL_DEVICE
|
||||||
|
else
|
||||||
|
echo "Couldn't find /$INSTALL_PARTITION/ in fstab."
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
_BOOT=$(awk '/^\/boot/ {print $3}' /etc/recovery.fstab)
|
||||||
|
if [ ! -z "$_BOOT" ]
|
||||||
|
then
|
||||||
|
echo "boot partition found at $_BOOT"
|
||||||
|
export BOOT=$_BOOT
|
||||||
|
else
|
||||||
|
echo "Couldn't find /boot/ in fstab."
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
partition_install_device() {
|
||||||
|
for command in "mktable msdos" \
|
||||||
|
"mkpart primary ext2 2048s 100M" \
|
||||||
|
"mkpart primary 100M 100%" \
|
||||||
|
"set 1 boot on"
|
||||||
|
do
|
||||||
|
parted -s "$INSTALL_DEVICE" "$command"
|
||||||
|
done
|
||||||
|
partprobe
|
||||||
|
if [ "$INSTALL_PARTITION" = "system" ]
|
||||||
|
then
|
||||||
|
kpartx -afs "$INSTALL_DEVICE"
|
||||||
|
ln -s /dev/mapper/* /dev/block/
|
||||||
|
fi
|
||||||
|
set_subpartitions
|
||||||
|
}
|
||||||
|
|
||||||
|
set_subpartitions() {
|
||||||
|
export PMOS_BOOT="$INSTALL_DEVICE"p1
|
||||||
|
export ROOT_PARTITION="$INSTALL_DEVICE"p2
|
||||||
|
}
|
||||||
|
|
||||||
|
umount_install_partition() {
|
||||||
|
if mountpoint -q "/$INSTALL_PARTITION/"
|
||||||
|
then
|
||||||
|
umount /"$INSTALL_PARTITION"/
|
||||||
|
else
|
||||||
|
echo 'Continuing...'
|
||||||
|
return 0
|
||||||
|
fi
|
||||||
|
}
|
29
aports/main/postmarketos-android-recovery-installer/pmos_setpw
Executable file
29
aports/main/postmarketos-android-recovery-installer/pmos_setpw
Executable file
|
@ -0,0 +1,29 @@
|
||||||
|
#!/sbin/ash -e
|
||||||
|
|
||||||
|
# Copyright 2017 Attila Szöllősi
|
||||||
|
#
|
||||||
|
# This file is part of postmarketos-android-recovery-installer.
|
||||||
|
#
|
||||||
|
# postmarketos-android-recovery-installer is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU General Public License as published by
|
||||||
|
# the Free Software Foundation, either version 3 of the License, or
|
||||||
|
# (at your option) any later version.
|
||||||
|
#
|
||||||
|
# postmarketos-android-recovery-installer is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License
|
||||||
|
# along with postmarketos-android-recovery-installer. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
# shellcheck source=pmos_install_functions
|
||||||
|
. /tmp/postmarketos/bin/pmos_install_functions
|
||||||
|
|
||||||
|
extract_partition_table
|
||||||
|
set_subpartitions
|
||||||
|
echo "Set the password of the encrypted rootfs!"
|
||||||
|
cryptsetup luksAddKey -d "$WORKING_DIR"/lukskey "$ROOT_PARTITION"
|
||||||
|
# Remove temporary keyfile
|
||||||
|
cryptsetup luksRemoveKey "$ROOT_PARTITION" "$WORKING_DIR"/lukskey
|
||||||
|
echo "Successfully added key to the LUKS device."
|
|
@ -0,0 +1,31 @@
|
||||||
|
#!/sbin/ash
|
||||||
|
|
||||||
|
# Copyright 2017 Attila Szöllősi
|
||||||
|
#
|
||||||
|
# This file is part of postmarketos-android-recovery-installer.
|
||||||
|
#
|
||||||
|
# postmarketos-android-recovery-installer is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU General Public License as published by
|
||||||
|
# the Free Software Foundation, either version 3 of the License, or
|
||||||
|
# (at your option) any later version.
|
||||||
|
#
|
||||||
|
# postmarketos-android-recovery-installer is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License
|
||||||
|
# along with postmarketos-android-recovery-installer. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
# Print fail information
|
||||||
|
OUTFD=$2
|
||||||
|
fail_info() {
|
||||||
|
FAIL_MSG="Failed. Check /tmp/postmarketos/pmos.log for more info!"
|
||||||
|
echo -n -e "ui_print $FAIL_MSG\n" > /proc/self/fd/"$OUTFD"
|
||||||
|
echo -n -e "ui_print\n" > /proc/self/fd/"$OUTFD"
|
||||||
|
}
|
||||||
|
# Create working directory
|
||||||
|
mkdir /tmp/postmarketos/
|
||||||
|
# Extract and start the installer script
|
||||||
|
unzip "$3" "bin/pmos_install" "bin/pmos_install_functions" "install_options" -d /tmp/postmarketos/
|
||||||
|
/tmp/postmarketos/bin/pmos_install "$2" "$3" || { fail_info ; exit 1 ; }
|
|
@ -1,6 +1,6 @@
|
||||||
pkgname=postmarketos-mkinitfs
|
pkgname=postmarketos-mkinitfs
|
||||||
pkgver=0.3.3
|
pkgver=0.3.4
|
||||||
pkgrel=6
|
pkgrel=0
|
||||||
pkgdesc="Tool to generate initramfs images for postmarketOS"
|
pkgdesc="Tool to generate initramfs images for postmarketOS"
|
||||||
url="https://github.com/postmarketOS"
|
url="https://github.com/postmarketOS"
|
||||||
# multipath-tools: kpartx
|
# multipath-tools: kpartx
|
||||||
|
@ -26,5 +26,5 @@ package() {
|
||||||
mkdir -p "$pkgdir/etc/postmarketos-mkinitfs/hooks/"
|
mkdir -p "$pkgdir/etc/postmarketos-mkinitfs/hooks/"
|
||||||
}
|
}
|
||||||
sha512sums="3d0215d61a34e846c6c3e4ff1911742a620cd1c6ff1de3cf26eaa4cb7643467da72bf9abc6a53992cc750bb76340be820149b25b806152b70fc0d40e0f8aa310 init.sh.in
|
sha512sums="3d0215d61a34e846c6c3e4ff1911742a620cd1c6ff1de3cf26eaa4cb7643467da72bf9abc6a53992cc750bb76340be820149b25b806152b70fc0d40e0f8aa310 init.sh.in
|
||||||
2331fe9a89ba58348b41fbfdeb6f4daeff3f6ef161d1b7582c3e900baba377fa9411efa0b052ea5c2ae22f75bc48f6b8f38dafad0bd836a0319906e70482898c init_functions.sh
|
a47398cdbb5e68a34086038cf6d72df91f6e58dcae7ff1ea8a375cd44f21e4573a944122ca5a32dda7b002bb14ec5826435edd2512c3db198dc9c0c3756e3cbe init_functions.sh
|
||||||
ef1481ef45e786486fb8e9939f756afb1d873a92546468d3dda3065ef46404be7e9847ab1f630fa6cf3e4ab99bdb116401093bbb1bbc882ea85ea824cdf7e389 mkinitfs.sh"
|
ef1481ef45e786486fb8e9939f756afb1d873a92546468d3dda3065ef46404be7e9847ab1f630fa6cf3e4ab99bdb116401093bbb1bbc882ea85ea824cdf7e389 mkinitfs.sh"
|
||||||
|
|
|
@ -105,7 +105,7 @@ mount_boot_partition() {
|
||||||
loop_forever
|
loop_forever
|
||||||
fi
|
fi
|
||||||
echo "Mount boot partition ($partition)"
|
echo "Mount boot partition ($partition)"
|
||||||
mount -r -t ext2 "$partition" /boot
|
mount -r "$partition" /boot
|
||||||
}
|
}
|
||||||
|
|
||||||
# $1: initramfs-extra path
|
# $1: initramfs-extra path
|
||||||
|
|
|
@ -19,6 +19,8 @@ along with pmbootstrap. If not, see <http://www.gnu.org/licenses/>.
|
||||||
import logging
|
import logging
|
||||||
import glob
|
import glob
|
||||||
import os
|
import os
|
||||||
|
import socket
|
||||||
|
from contextlib import closing
|
||||||
|
|
||||||
import pmb.chroot
|
import pmb.chroot
|
||||||
import pmb.chroot.distccd
|
import pmb.chroot.distccd
|
||||||
|
@ -27,6 +29,16 @@ import pmb.install.losetup
|
||||||
import pmb.parse.arch
|
import pmb.parse.arch
|
||||||
|
|
||||||
|
|
||||||
|
def kill_adb(args):
|
||||||
|
"""
|
||||||
|
Kill adb daemon if it's running.
|
||||||
|
"""
|
||||||
|
port = 5038
|
||||||
|
with closing(socket.socket(socket.AF_INET, socket.SOCK_STREAM)) as sock:
|
||||||
|
if sock.connect_ex(("127.0.0.1", port)) == 0:
|
||||||
|
pmb.chroot.root(args, ["adb", "-P", str(port), "kill-server"])
|
||||||
|
|
||||||
|
|
||||||
def shutdown_cryptsetup_device(args, name):
|
def shutdown_cryptsetup_device(args, name):
|
||||||
"""
|
"""
|
||||||
:param name: cryptsetup device name, usually "pm_crypt" in pmbootstrap
|
:param name: cryptsetup device name, usually "pm_crypt" in pmbootstrap
|
||||||
|
@ -55,6 +67,9 @@ def shutdown_cryptsetup_device(args, name):
|
||||||
def shutdown(args, only_install_related=False):
|
def shutdown(args, only_install_related=False):
|
||||||
pmb.chroot.distccd.stop(args)
|
pmb.chroot.distccd.stop(args)
|
||||||
|
|
||||||
|
# Stop adb server
|
||||||
|
kill_adb(args)
|
||||||
|
|
||||||
# Umount installation-related paths (order is important!)
|
# Umount installation-related paths (order is important!)
|
||||||
pmb.helpers.mount.umount_all(args, args.work +
|
pmb.helpers.mount.umount_all(args, args.work +
|
||||||
"/chroot_native/mnt/install")
|
"/chroot_native/mnt/install")
|
||||||
|
|
|
@ -230,6 +230,7 @@ install_device_packages = [
|
||||||
# chroot, before the flash programs get started.
|
# chroot, before the flash programs get started.
|
||||||
flash_mount_bind = [
|
flash_mount_bind = [
|
||||||
"/sys/bus/usb/devices/",
|
"/sys/bus/usb/devices/",
|
||||||
|
"/sys/dev/",
|
||||||
"/sys/devices/",
|
"/sys/devices/",
|
||||||
"/dev/bus/usb/"
|
"/dev/bus/usb/"
|
||||||
]
|
]
|
||||||
|
@ -289,6 +290,17 @@ flashers = {
|
||||||
["heimdall", "flash", "--$PARTITION_KERNEL", "$BOOT/boot.img-$FLAVOR"]],
|
["heimdall", "flash", "--$PARTITION_KERNEL", "$BOOT/boot.img-$FLAVOR"]],
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
"adb": {
|
||||||
|
"depends": ["android-tools"],
|
||||||
|
"actions":
|
||||||
|
{
|
||||||
|
"list_devices": [["adb", "-P", "5038", "devices"]],
|
||||||
|
"sideload": [["echo", "< wait for any device >"],
|
||||||
|
["adb", "-P", "5038", "wait-for-usb-sideload"],
|
||||||
|
["adb", "-P", "5038", "sideload",
|
||||||
|
"$RECOVERY_ZIP"]],
|
||||||
|
}
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
#
|
#
|
||||||
|
|
|
@ -48,13 +48,18 @@ def symlinks(args, flavor, folder):
|
||||||
"uImage-" + flavor: "Kernel, legacy u-boot image format",
|
"uImage-" + flavor: "Kernel, legacy u-boot image format",
|
||||||
"vmlinuz-" + flavor: "Linux kernel",
|
"vmlinuz-" + flavor: "Linux kernel",
|
||||||
args.device + ".img": "System partition",
|
args.device + ".img": "System partition",
|
||||||
|
"pmos-" + args.device + ".zip": "Android recovery flashable zip",
|
||||||
}
|
}
|
||||||
|
|
||||||
# Generate a list of patterns
|
# Generate a list of patterns
|
||||||
path_native = args.work + "/chroot_native"
|
path_native = args.work + "/chroot_native"
|
||||||
path_boot = args.work + "/chroot_rootfs_" + args.device + "/boot"
|
path_boot = args.work + "/chroot_rootfs_" + args.device + "/boot"
|
||||||
|
path_buildroot = args.work + "/chroot_buildroot_" + args.deviceinfo["arch"]
|
||||||
patterns = [path_boot + "/*-" + flavor,
|
patterns = [path_boot + "/*-" + flavor,
|
||||||
path_native + "/home/user/rootfs/" + args.device + ".img"]
|
path_native + "/home/user/rootfs/" + args.device + ".img",
|
||||||
|
path_buildroot +
|
||||||
|
"/var/lib/postmarketos-android-recovery-installer/pmos-" +
|
||||||
|
args.device + ".zip"]
|
||||||
|
|
||||||
# Generate a list of files from the patterns
|
# Generate a list of files from the patterns
|
||||||
files = []
|
files = []
|
||||||
|
|
|
@ -95,6 +95,25 @@ def list_devices(args):
|
||||||
pmb.flasher.run(args, "list_devices")
|
pmb.flasher.run(args, "list_devices")
|
||||||
|
|
||||||
|
|
||||||
|
def sideload(args):
|
||||||
|
# Mount the buildroot
|
||||||
|
suffix = "buildroot_" + args.deviceinfo["arch"]
|
||||||
|
mountpoint = "/mnt/" + suffix
|
||||||
|
pmb.helpers.mount.bind(args, args.work + "/chroot_" + suffix,
|
||||||
|
args.work + "/chroot_native/" + mountpoint)
|
||||||
|
|
||||||
|
# Missing recovery zip error
|
||||||
|
zip_path = ("/var/lib/postmarketos-android-recovery-installer/pmos-" +
|
||||||
|
args.device + ".zip")
|
||||||
|
if not os.path.exists(args.work + "/chroot_native" + mountpoint +
|
||||||
|
zip_path):
|
||||||
|
raise RuntimeError("The recovery zip has not been generated yet,"
|
||||||
|
" please run 'pmbootstrap install' with the"
|
||||||
|
" '--android-recovery-zip' parameter first!")
|
||||||
|
|
||||||
|
pmb.flasher.run(args, "sideload")
|
||||||
|
|
||||||
|
|
||||||
def export(args):
|
def export(args):
|
||||||
# Create the export folder
|
# Create the export folder
|
||||||
if not os.path.exists(args.export_folder):
|
if not os.path.exists(args.export_folder):
|
||||||
|
@ -123,5 +142,7 @@ def frontend(args):
|
||||||
list_flavors(args)
|
list_flavors(args)
|
||||||
if action == "list_devices":
|
if action == "list_devices":
|
||||||
list_devices(args)
|
list_devices(args)
|
||||||
|
if action == "sideload":
|
||||||
|
sideload(args)
|
||||||
if action == "export":
|
if action == "export":
|
||||||
export(args)
|
export(args)
|
||||||
|
|
|
@ -24,7 +24,7 @@ def run(args, action, flavor=None):
|
||||||
pmb.flasher.init(args)
|
pmb.flasher.init(args)
|
||||||
|
|
||||||
# Verify action
|
# Verify action
|
||||||
method = args.deviceinfo["flash_methods"]
|
method = args.flash_method or args.deviceinfo["flash_methods"]
|
||||||
cfg = pmb.config.flashers[method]
|
cfg = pmb.config.flashers[method]
|
||||||
if action not in cfg["actions"]:
|
if action not in cfg["actions"]:
|
||||||
raise RuntimeError("action " + action + " is not"
|
raise RuntimeError("action " + action + " is not"
|
||||||
|
@ -42,6 +42,9 @@ def run(args, action, flavor=None):
|
||||||
"$KERNEL_CMDLINE": _cmdline,
|
"$KERNEL_CMDLINE": _cmdline,
|
||||||
"$PARTITION_INITFS": args.deviceinfo["flash_heimdall_partition_initfs"],
|
"$PARTITION_INITFS": args.deviceinfo["flash_heimdall_partition_initfs"],
|
||||||
"$PARTITION_KERNEL": args.deviceinfo["flash_heimdall_partition_kernel"],
|
"$PARTITION_KERNEL": args.deviceinfo["flash_heimdall_partition_kernel"],
|
||||||
|
"$RECOVERY_ZIP": "/mnt/buildroot_" + args.deviceinfo["arch"] +
|
||||||
|
"/var/lib/postmarketos-android-recovery-installer"
|
||||||
|
"/pmos-" + args.device + ".zip",
|
||||||
}
|
}
|
||||||
|
|
||||||
# Run the commands of each action
|
# Run the commands of each action
|
||||||
|
|
|
@ -28,14 +28,17 @@ import pmb.config
|
||||||
import pmb.helpers.run
|
import pmb.helpers.run
|
||||||
import pmb.install.blockdevice
|
import pmb.install.blockdevice
|
||||||
import pmb.install.file
|
import pmb.install.file
|
||||||
|
import pmb.install.recovery
|
||||||
import pmb.install
|
import pmb.install
|
||||||
|
|
||||||
|
|
||||||
def mount_device_rootfs(args):
|
def mount_device_rootfs(args, suffix="native"):
|
||||||
# Mount the device rootfs
|
"""
|
||||||
|
Mount the device rootfs.
|
||||||
|
"""
|
||||||
mountpoint = "/mnt/rootfs_" + args.device
|
mountpoint = "/mnt/rootfs_" + args.device
|
||||||
pmb.helpers.mount.bind(args, args.work + "/chroot_rootfs_" + args.device,
|
pmb.helpers.mount.bind(args, args.work + "/chroot_rootfs_" + args.device,
|
||||||
args.work + "/chroot_native" + mountpoint)
|
args.work + "/chroot_" + suffix + mountpoint)
|
||||||
return mountpoint
|
return mountpoint
|
||||||
|
|
||||||
|
|
||||||
|
@ -153,45 +156,7 @@ def setup_keymap(args):
|
||||||
logging.info("NOTE: No valid keymap specified for device")
|
logging.info("NOTE: No valid keymap specified for device")
|
||||||
|
|
||||||
|
|
||||||
def install(args):
|
def install_system_image(args):
|
||||||
# Install required programs in native chroot
|
|
||||||
logging.info("*** (1/5) PREPARE NATIVE CHROOT ***")
|
|
||||||
pmb.chroot.apk.install(args, pmb.config.install_native_packages,
|
|
||||||
build=False)
|
|
||||||
|
|
||||||
# List all packages to be installed (including the ones specified by --add)
|
|
||||||
# and upgrade the installed packages/apkindexes
|
|
||||||
logging.info("*** (2/5) CREATE DEVICE ROOTFS (" + args.device + ") ***")
|
|
||||||
install_packages = (pmb.config.install_device_packages +
|
|
||||||
["device-" + args.device])
|
|
||||||
if args.ui.lower() != "none":
|
|
||||||
install_packages += ["postmarketos-ui-" + args.ui]
|
|
||||||
suffix = "rootfs_" + args.device
|
|
||||||
pmb.chroot.apk.upgrade(args, suffix)
|
|
||||||
|
|
||||||
# Explicitly call build on the install packages, to re-build them or any
|
|
||||||
# dependency, in case the version increased
|
|
||||||
if args.extra_packages.lower() != "none":
|
|
||||||
install_packages += args.extra_packages.split(",")
|
|
||||||
if args.add:
|
|
||||||
install_packages += args.add.split(",")
|
|
||||||
for pkgname in install_packages:
|
|
||||||
pmb.build.package(args, pkgname, args.deviceinfo["arch"])
|
|
||||||
|
|
||||||
# Install all packages to device rootfs chroot (and rebuild the initramfs,
|
|
||||||
# because that doesn't always happen automatically yet, e.g. when the user
|
|
||||||
# installed a hook without pmbootstrap - see #69 for more info)
|
|
||||||
pmb.chroot.apk.install(args, install_packages, suffix)
|
|
||||||
pmb.install.file.write_os_release(args, suffix)
|
|
||||||
for flavor in pmb.chroot.other.kernel_flavors_installed(args, suffix):
|
|
||||||
pmb.chroot.initfs.build(args, flavor, suffix)
|
|
||||||
|
|
||||||
# Set the user password
|
|
||||||
set_user_password(args)
|
|
||||||
|
|
||||||
# Set the keymap if the device requires it
|
|
||||||
setup_keymap(args)
|
|
||||||
|
|
||||||
# Partition and fill image/sdcard
|
# Partition and fill image/sdcard
|
||||||
logging.info("*** (3/5) PREPARE INSTALL BLOCKDEVICE ***")
|
logging.info("*** (3/5) PREPARE INSTALL BLOCKDEVICE ***")
|
||||||
pmb.chroot.shutdown(args, True)
|
pmb.chroot.shutdown(args, True)
|
||||||
|
@ -249,3 +214,72 @@ def install(args):
|
||||||
logging.info("* If the above steps do not work, you can also create"
|
logging.info("* If the above steps do not work, you can also create"
|
||||||
" symlinks to the generated files with 'pmbootstrap flasher"
|
" symlinks to the generated files with 'pmbootstrap flasher"
|
||||||
" export [export_folder]' and flash outside of pmbootstrap.")
|
" export [export_folder]' and flash outside of pmbootstrap.")
|
||||||
|
|
||||||
|
|
||||||
|
def install_recovery_zip(args):
|
||||||
|
logging.info("*** (3/4) CREATING RECOVERY-FLASHABLE ZIP ***")
|
||||||
|
suffix = "buildroot_" + args.deviceinfo["arch"]
|
||||||
|
mount_device_rootfs(args, suffix)
|
||||||
|
pmb.install.recovery.create_zip(args, suffix)
|
||||||
|
|
||||||
|
# Flash information
|
||||||
|
logging.info("*** (4/4) FLASHING TO DEVICE ***")
|
||||||
|
logging.info("Run the following to flash your installation to the"
|
||||||
|
" target device:")
|
||||||
|
logging.info("* pmbootstrap flasher --method adb sideload")
|
||||||
|
logging.info(" Flashes the installer zip to your device:")
|
||||||
|
|
||||||
|
# Export information
|
||||||
|
logging.info("* If this does not work, you can also create a"
|
||||||
|
" symlink to the generated zip with 'pmbootstrap flasher"
|
||||||
|
" export --android-recovery-zip [export_folder]' and"
|
||||||
|
" flash outside of pmbootstrap.")
|
||||||
|
|
||||||
|
|
||||||
|
def install(args):
|
||||||
|
# Number of steps for the different installation methods.
|
||||||
|
steps = 4 if args.android_recovery_zip else 5
|
||||||
|
|
||||||
|
# Install required programs in native chroot
|
||||||
|
logging.info("*** (1/{}) PREPARE NATIVE CHROOT ***".format(steps))
|
||||||
|
pmb.chroot.apk.install(args, pmb.config.install_native_packages,
|
||||||
|
build=False)
|
||||||
|
|
||||||
|
# List all packages to be installed (including the ones specified by --add)
|
||||||
|
# and upgrade the installed packages/apkindexes
|
||||||
|
logging.info('*** (2/{0}) CREATE DEVICE ROOTFS ("{1}") ***'.format(steps,
|
||||||
|
args.device))
|
||||||
|
install_packages = (pmb.config.install_device_packages +
|
||||||
|
["device-" + args.device])
|
||||||
|
if args.ui.lower() != "none":
|
||||||
|
install_packages += ["postmarketos-ui-" + args.ui]
|
||||||
|
suffix = "rootfs_" + args.device
|
||||||
|
pmb.chroot.apk.upgrade(args, suffix)
|
||||||
|
|
||||||
|
# Explicitly call build on the install packages, to re-build them or any
|
||||||
|
# dependency, in case the version increased
|
||||||
|
if args.extra_packages.lower() != "none":
|
||||||
|
install_packages += args.extra_packages.split(",")
|
||||||
|
if args.add:
|
||||||
|
install_packages += args.add.split(",")
|
||||||
|
for pkgname in install_packages:
|
||||||
|
pmb.build.package(args, pkgname, args.deviceinfo["arch"])
|
||||||
|
|
||||||
|
# Install all packages to device rootfs chroot (and rebuild the initramfs,
|
||||||
|
# because that doesn't always happen automatically yet, e.g. when the user
|
||||||
|
# installed a hook without pmbootstrap - see #69 for more info)
|
||||||
|
pmb.chroot.apk.install(args, install_packages, suffix)
|
||||||
|
pmb.install.file.write_os_release(args, suffix)
|
||||||
|
for flavor in pmb.chroot.other.kernel_flavors_installed(args, suffix):
|
||||||
|
pmb.chroot.initfs.build(args, flavor, suffix)
|
||||||
|
|
||||||
|
# Set the user password
|
||||||
|
set_user_password(args)
|
||||||
|
|
||||||
|
# Set the keymap if the device requires it
|
||||||
|
setup_keymap(args)
|
||||||
|
|
||||||
|
if args.android_recovery_zip:
|
||||||
|
install_recovery_zip(args)
|
||||||
|
else:
|
||||||
|
install_system_image(args)
|
||||||
|
|
61
pmb/install/recovery.py
Normal file
61
pmb/install/recovery.py
Normal file
|
@ -0,0 +1,61 @@
|
||||||
|
"""
|
||||||
|
Copyright 2017 Attila Szollosi
|
||||||
|
|
||||||
|
This file is part of pmbootstrap.
|
||||||
|
|
||||||
|
pmbootstrap is free software: you can redistribute it and/or modify
|
||||||
|
it under the terms of the GNU General Public License as published by
|
||||||
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
|
(at your option) any later version.
|
||||||
|
|
||||||
|
pmbootstrap is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
GNU General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU General Public License
|
||||||
|
along with pmbootstrap. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
"""
|
||||||
|
import logging
|
||||||
|
|
||||||
|
import pmb.chroot
|
||||||
|
|
||||||
|
|
||||||
|
def create_zip(args, suffix):
|
||||||
|
"""
|
||||||
|
Create android recovery compatible installer zip.
|
||||||
|
"""
|
||||||
|
zip_root = "/var/lib/postmarketos-android-recovery-installer/"
|
||||||
|
rootfs = "/mnt/rootfs_" + args.device
|
||||||
|
|
||||||
|
# Install recovery installer package in buildroot
|
||||||
|
pmb.chroot.apk.install(args,
|
||||||
|
["postmarketos-android-recovery-installer"],
|
||||||
|
suffix)
|
||||||
|
|
||||||
|
logging.info("(" + suffix + ") create recovery zip")
|
||||||
|
|
||||||
|
# Create config file for the recovery installer
|
||||||
|
with open(args.work + "/chroot_" + suffix + "/tmp/install_options",
|
||||||
|
"w") as install_options:
|
||||||
|
install_options.write(
|
||||||
|
"\n".join(['DEVICE="{}"'.format(args.device),
|
||||||
|
'FLASH_BOOTIMG="{}"'.format(
|
||||||
|
str(args.recovery_flash_bootimg).lower()),
|
||||||
|
'INSTALL_PARTITION="{}"'.format(
|
||||||
|
args.recovery_install_partition),
|
||||||
|
'CIPHER="{}"'.format(args.cipher),
|
||||||
|
'FDE="{}"'.format(
|
||||||
|
str(args.full_disk_encryption).lower())]))
|
||||||
|
|
||||||
|
commands = [
|
||||||
|
# Move config file from /tmp/ to zip root
|
||||||
|
["mv", "/tmp/install_options", "install_options"],
|
||||||
|
# Copy boot.img to zip root
|
||||||
|
["cp", rootfs + "/boot/boot.img-" + args.device, "boot.img"],
|
||||||
|
# Create tar archive of the rootfs
|
||||||
|
["tar", "-pczf", "rootfs.tar.gz", "--exclude", "./home/user/*",
|
||||||
|
"-C", rootfs, "."],
|
||||||
|
["build-recovery-zip"]]
|
||||||
|
for command in commands:
|
||||||
|
pmb.chroot.root(args, command, suffix, working_dir=zip_root)
|
|
@ -25,12 +25,8 @@ def arguments_flasher(subparser):
|
||||||
ret = subparser.add_parser("flasher", help="flash something to the"
|
ret = subparser.add_parser("flasher", help="flash something to the"
|
||||||
" target device")
|
" target device")
|
||||||
sub = ret.add_subparsers(dest="action_flasher")
|
sub = ret.add_subparsers(dest="action_flasher")
|
||||||
|
ret.add_argument("--method", help="override flash method",
|
||||||
# Other
|
dest="flash_method", default=None)
|
||||||
sub.add_parser("flash_system", help="flash the system partition")
|
|
||||||
sub.add_parser("list_flavors", help="list installed kernel flavors" +
|
|
||||||
" inside the device rootfs chroot on this computer")
|
|
||||||
sub.add_parser("list_devices", help="show connected devices")
|
|
||||||
|
|
||||||
# Boot, flash kernel, export
|
# Boot, flash kernel, export
|
||||||
boot = sub.add_parser("boot", help="boot a kernel once")
|
boot = sub.add_parser("boot", help="boot a kernel once")
|
||||||
|
@ -42,6 +38,13 @@ def arguments_flasher(subparser):
|
||||||
for action in [boot, flash_kernel, export]:
|
for action in [boot, flash_kernel, export]:
|
||||||
action.add_argument("--flavor", default=None)
|
action.add_argument("--flavor", default=None)
|
||||||
|
|
||||||
|
# Other
|
||||||
|
sub.add_parser("flash_system", help="flash the system partition")
|
||||||
|
sub.add_parser("list_flavors", help="list installed kernel flavors" +
|
||||||
|
" inside the device rootfs chroot on this computer")
|
||||||
|
sub.add_parser("list_devices", help="show connected devices")
|
||||||
|
sub.add_parser("sideload", help="sideload recovery zip")
|
||||||
|
|
||||||
# Export: additional arguments
|
# Export: additional arguments
|
||||||
export.add_argument("export_folder", help="export folder, defaults to"
|
export.add_argument("export_folder", help="export folder, defaults to"
|
||||||
" /tmp/postmarketOS-export",
|
" /tmp/postmarketOS-export",
|
||||||
|
@ -181,6 +184,16 @@ def arguments():
|
||||||
" added to the rootfs (e.g. 'vim,gcc')")
|
" added to the rootfs (e.g. 'vim,gcc')")
|
||||||
install.add_argument("--no-fde", help="do not use full disk encryption",
|
install.add_argument("--no-fde", help="do not use full disk encryption",
|
||||||
action="store_false", dest="full_disk_encryption")
|
action="store_false", dest="full_disk_encryption")
|
||||||
|
install.add_argument("--android-recovery-zip",
|
||||||
|
help="generate TWRP flashable zip",
|
||||||
|
action="store_true", dest="android_recovery_zip")
|
||||||
|
install.add_argument("--recovery-flash-bootimg",
|
||||||
|
help="include kernel in recovery flashable zip",
|
||||||
|
action="store_true", dest="recovery_flash_bootimg")
|
||||||
|
install.add_argument("--recovery-install-partition", default="system",
|
||||||
|
help="partition to flash from recovery,"
|
||||||
|
"eg. external_sd",
|
||||||
|
dest="recovery_install_partition")
|
||||||
|
|
||||||
# Action: menuconfig / parse_apkbuild
|
# Action: menuconfig / parse_apkbuild
|
||||||
menuconfig = sub.add_parser("menuconfig", help="run menuconfig on"
|
menuconfig = sub.add_parser("menuconfig", help="run menuconfig on"
|
||||||
|
|
|
@ -26,6 +26,11 @@ sh_files="
|
||||||
./aports/main/postmarketos-base/firmwareload.sh
|
./aports/main/postmarketos-base/firmwareload.sh
|
||||||
./aports/main/postmarketos-mkinitfs/init.sh.in
|
./aports/main/postmarketos-mkinitfs/init.sh.in
|
||||||
./aports/main/postmarketos-mkinitfs/init_functions.sh
|
./aports/main/postmarketos-mkinitfs/init_functions.sh
|
||||||
|
./aports/main/postmarketos-android-recovery-installer/build_zip.sh
|
||||||
|
./aports/main/postmarketos-android-recovery-installer/pmos_install
|
||||||
|
./aports/main/postmarketos-android-recovery-installer/pmos_install_functions
|
||||||
|
./aports/main/postmarketos-android-recovery-installer/pmos_setpw
|
||||||
|
./aports/main/postmarketos-android-recovery-installer/update-binary
|
||||||
$(find . -name '*.trigger')
|
$(find . -name '*.trigger')
|
||||||
"
|
"
|
||||||
for file in ${sh_files}; do
|
for file in ${sh_files}; do
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue