#!/bin/sh # This file will be in /init_functions.sh inside the initramfs. IP=172.16.42.1 TELNET_PORT=23 # Redirect stdout and stderr to logfile setup_log() { # Bail out if PMOS_NO_OUTPUT_REDIRECT is set echo "### postmarketOS initramfs ###" grep -q PMOS_NO_OUTPUT_REDIRECT /proc/cmdline && return # Print a message about what is going on to the normal output echo "NOTE: All output from the initramfs gets redirected to:" echo "/pmOS_init.log" echo "If you want to disable this behavior (e.g. because you're" echo "debugging over serial), please add this to your kernel" echo "command line: PMOS_NO_OUTPUT_REDIRECT" # Start redirect, print the first line again exec >/pmOS_init.log 2>&1 echo "### postmarketOS initramfs ###" } mount_proc_sys_dev() { # mdev mount -t proc -o nodev,noexec,nosuid proc /proc mount -t sysfs -o nodev,noexec,nosuid sysfs /sys mkdir /config mount -t configfs -o nodev,noexec,nosuid configfs /config # /dev/pts (needed for telnet) mkdir -p /dev/pts mount -t devpts devpts /dev/pts } setup_mdev() { echo /sbin/mdev >/proc/sys/kernel/hotplug mdev -s } mount_subpartitions() { for i in /dev/mmcblk*; do case "$(kpartx -l "$i" 2>/dev/null | wc -l)" in 2) echo "Mount subpartitions of $i" kpartx -afs "$i" break ;; *) continue ;; esac done } find_root_partition() { # The partition layout is one of the following: # a) boot, root partitions on sdcard # b) boot, root partition on the "system" partition (which has its # own partition header! so we have partitions on partitions!) # # mount_subpartitions() must get executed before calling # find_root_partition(), so partitions from b) also get found. # # However, after executing mount_subpartitions(), the partitions # from a) get mounted to /dev/mapper - and then you can only use # the ones from /dev/mapper, not the original partition paths (they # will appear as busy when trying to mount them). This is an # unwanted side-effect, that we must deal with. # The subpartitions from b) get mounted to /dev/mapper, and this is # what we want. # # To deal with the side-effect, we use the partitions from # /dev/mapper first, and then fall back to partitions with all paths # (in case the user inserted an SD card after mount_subpartitions() # ran!). # Try the partitions in /dev/mapper first. for id in pmOS_root crypto_LUKS; do DEVICE="$(blkid | grep /dev/mapper | grep "$id" \ | cut -d ":" -f 1)" [ -z "$DEVICE" ] || break done # Then try all devices if [ -z "$DEVICE" ]; then for id in pmOS_root crypto_LUKS; do DEVICE="$(blkid | grep "$id" | cut -d ":" -f 1)" [ -z "$DEVICE" ] || break done fi echo "$DEVICE" } find_boot_partition() { findfs LABEL="pmOS_boot" } mount_boot_partition() { partition=$(find_boot_partition) if [ -z "$partition" ]; then echo "ERROR: boot partition not found!" show_splash /splash-noboot.ppm.gz loop_forever fi echo "Mount boot partition ($partition)" mount -r -t ext2 "$partition" /boot } # $1: initramfs-extra path extract_initramfs_extra() { initramfs_extra="$1" if [ ! -e "$initramfs_extra" ]; then echo "ERROR: initramfs-extra not found!" show_splash /splash-noinitramfsextra.ppm.gz loop_forever fi echo "Extract $initramfs_extra" gzip -d -c "$initramfs_extra" | cpio -i } wait_root_partition() { while [ -z "$(find_root_partition)" ]; do show_splash /splash-nosystem.ppm.gz echo "Could not find the root partition." echo "Maybe you need to insert the sdcard, if your device has" echo "any? Trying again in one second..." sleep 1 done } resize_root_partition() { partition=$(find_root_partition) # Only resize the partition if it's inside the device-mapper, which means # that the partition is stored as a subpartition inside another one. # In this case we want to resize it to use all the unused space of the # external partition. if [ -z "${partition##"/dev/mapper/"*}" ]; then # Get physical device partition_dev=$(dmsetup deps -o devname "$partition" | \ awk -F "[()]" '{print "/dev/"$2}') # Check if there is unallocated space at the end of the device if parted -s "$partition_dev" print free | tail -n2 | \ head -n1 | grep -qi "free space"; then echo "Resize root partition ($partition)" # unmount subpartition, resize and remount it kpartx -d "$partition" parted -s "$partition_dev" resizepart 2 100% kpartx -afs "$partition_dev" fi fi # Detect and resize root partition on QEMU if [ -z "${partition##"/dev/hda"*}" ]; then echo "Resize root partition ($partition)" parted -s /dev/hda resizepart 2 100% partprobe fi } unlock_root_partition() { partition="$(find_root_partition)" if cryptsetup isLuks "$partition"; then until cryptsetup status root | grep -qwi active; do start_usb_unlock cryptsetup luksOpen "$partition" root || continue done # Show again the loading splashscreen show_splash /splash-loading.ppm.gz fi } resize_root_filesystem() { partition="$(find_root_partition)" touch /etc/mtab # see https://bugs.debian.org/cgi-bin/bugreport.cgi?bug=673323 echo "Check/repair root filesystem ($partition)" e2fsck -f -y "$partition" echo "Resize root filesystem ($partition)" resize2fs -f "$partition" } mount_root_partition() { partition="$(find_root_partition)" echo "Mount root partition ($partition)" mount -w -t ext4 "$partition" /sysroot if ! [ -e /sysroot/usr ]; then echo "ERROR: unable to mount root partition!" show_splash /splash-mounterror.ppm.gz loop_forever fi } setup_usb_network_android() { # Only run, when we have the android usb driver SYS=/sys/class/android_usb/android0 [ -e "$SYS" ] || return # Do the setup printf "%s" "0" >"$SYS/enable" printf "%s" "18D1" >"$SYS/idVendor" printf "%s" "D001" >"$SYS/idProduct" printf "%s" "rndis" >"$SYS/functions" printf "%s" "1" >"$SYS/enable" } setup_usb_network_configfs() { CONFIGFS=/config/usb_gadget [ -e "$CONFIGFS" ] || return mkdir $CONFIGFS/g1 printf "%s" "18D1" >"$CONFIGFS/g1/idVendor" printf "%s" "D001" >"$CONFIGFS/g1/idProduct" mkdir $CONFIGFS/g1/strings/0x409 mkdir $CONFIGFS/g1/functions/rndis.usb0 mkdir $CONFIGFS/g1/configs/c.1 mkdir $CONFIGFS/g1/configs/c.1/strings/0x409 printf "%s" "rndis" > $CONFIGFS/g1/configs/c.1/strings/0x409/configuration ln -s $CONFIGFS/g1/functions/rndis.usb0 $CONFIGFS/g1/configs/c.1 # See also: #338 # shellcheck disable=SC2005 echo "$(ls /sys/class/udc)" > $CONFIGFS/g1/UDC } setup_usb_network() { # Only run once _marker="/tmp/_setup_usb_network" [ -e "$_marker" ] && return touch "$_marker" echo "Setup usb network" # Run all usb network setup functions (add more below!) setup_usb_network_android setup_usb_network_configfs } start_udhcpd() { # Only run once [ -e /etc/udhcpd.conf ] && return # Get usb interface INTERFACE="" ifconfig rndis0 "$IP" && INTERFACE=rndis0 if [ -z $INTERFACE ]; then ifconfig usb0 "$IP" && INTERFACE=usb0 fi # Create /etc/udhcpd.conf { echo "start 172.16.42.2" echo "end 172.16.42.254" echo "lease_file /var/udhcpd.leases" echo "interface $INTERFACE" echo "option subnet 255.255.255.0" } >/etc/udhcpd.conf echo "Start the dhcpcd daemon (forks into background)" udhcpd } start_usb_unlock() { # Only run once _marker="/tmp/_start_usb_unlock" [ -e "$_marker" ] && return touch "$_marker" # Set up networking setup_usb_network start_udhcpd # Telnet splash show_splash /splash-telnet.ppm.gz echo "Start the telnet daemon (unlock encrypted partition)" { echo '#!/bin/sh' echo '. /init_functions.sh' echo 'unlock_root_partition' echo 'echo_connect_ssh_message' echo 'killall cryptsetup' echo "pkill -f telnetd.*:${TELNET_PORT}" } >/telnet_connect.sh chmod +x /telnet_connect.sh telnetd -b "${IP}:${TELNET_PORT}" -l /telnet_connect.sh } # $1: path to ppm.gz file show_splash() { gzip -c -d "$1" >/tmp/splash.ppm fbsplash -s /tmp/splash.ppm } echo_connect_ssh_message() { echo "Your root partition has been decrypted successfully!" echo "You can connect to your device using SSH in a few seconds:" echo "ssh user@$IP" } loop_forever() { while true; do sleep 1 done }