* Strip of any "-unsigned" on kernel version.
[rrq/rescue-boot.git] / minbase-strap.sh
1 #!/bin/bash
2 #
3 # Utility script to debootstrap a minbase filesystem, with additions
4 # of kernel, busybox and an init script. Then make an initrd.gz from
5 # that and prepare a boot image file with syslinux from that.
6 #
7 if [ $(id -u) -ne 0 ] ; then
8     exec sudo env http_proxy=$http_proxy $0 $*
9     echo "** Abort: must be root" >&2
10     exit 1
11 fi
12 set -e -x
13
14 ARCH="${1-$(dpkg-architecture -q DEB_HOST_ARCH)}"
15 DIST=${2-ceres}
16 IMG=${3-boot-$DIST-$ARCH.img}
17 FS=${4-FS}
18
19 # Prepare or keep FS
20 DEBSTRAP=true
21 USRMERGE=true
22 if [ -d "$FS" ] ; then
23     select x in "keep existing $FS and skip debootstrap" \
24                     "remove existing $FS and debootstrap anew" \
25                     "abort" ; do
26         [ "$x" = abort ] && exit 1
27         [ -z "$x" ] || break
28     done
29     echo "selection: $REPLY"
30     [ "$REPLY" = 1 ] && DEBSTRAP=false
31 fi
32
33 stupid_links() {
34     mkdir -p $1/usr
35     for d in bin sbin lib ; do mkdir $1/usr/$d ; ln -s usr/$d $1/$d ; done
36 }
37
38 if $DEBSTRAP ; then
39     ## Optionally add the stupid-links
40     read -n 1 -p "prime $FS with stupid-links? [Yn]" x
41     [ "$x" = "n" ] && USRMERGE=false
42
43     rm -rf "$FS" # remove if existin
44
45     $USRMERGE && stupid_links "$FS"
46
47     ## bootstrap a filesystem, with exclusions
48     echo "http_proxy=$http_proxy debootstrap ..."
49     debootstrap --exclude=logrotate,cron,cron-daemon-common \
50                 --arch=$ARCH $DIST "$FS" http://deb.devuan.org/merged
51     chroot "$FS" apt-get install -y logrotate cron
52
53     ## Select and add a kernel
54     KERNELS=( $(chroot "$FS" apt-cache search linux-image-\* | \
55                     sed 's/\s.*//' | sort -h ) )
56     echo "** Please select kernel **"
57     select KERNEL in "${KERNELS[@]}" ; do [ -n "$KERNEL" ] && break ; done
58     VERSION=${KERNEL#linux-image-}
59     VERSION=${VERSION%-unsigned}
60     chroot "$FS" apt-get install -y $KERNEL
61     chroot "$FS" depmod -a $VERSION
62
63     chroot "$FS" apt-get install -y busybox-static debootstrap
64     touch "$FS"/usr/bin{linuxrc,init} # block these
65     chroot "$FS" /usr/bin/busybox --install -s /usr/bin
66
67 fi # End of $DEBSTRAP actions
68
69 ## Pick boot kernel, if there are many
70 LINUXES=( $(cd "$FS"/boot ; ls vmlinuz* 2>/dev/null) )
71 if [ ${#LINUXES[@]} -gt 1 ] ; then
72     echo "** Please select boot kernel"
73     select LINUX in $LINUXES ; do [ -n "$LINUX" ] && break ; done
74 elif [ -n "$LINUXES" ] ; then
75     LINUX="$FS"/boot/${LINUXES[0]}
76 else
77     echo "** Oh No! There is no $FS/boot/vmlinux-* ... bailing out!" >&2
78     exit 1
79 fi
80
81 #============================================================
82 # Prepare a temporary directory tree with a boot kernel and $FS packed
83 # up into an initrd.gz
84 TMP=$(mktemp -d XXXX)
85 trap "rm -r $TMP" 0 2 15
86
87 mkdir -p $TMP/prep/{proc,dev,tmp,sys,mnt,usr}
88 stupid_links $TMP/prep
89
90 # copy FS tree(s) in full into the TMP/prep tree
91 copyFStree() {
92     local D
93     for D in $* ; do
94         tar cf - -C "$FS" $D \
95             | tar -xf - --skip-old-files --keep-directory-symlink -C $TMP/prep
96     done
97 }
98
99 # Copy FS binary(s) with all its libraries into the TMP/prep tree
100 copyFSbin() {
101     local B BIN
102     for B in $* ; do
103         BIN="$(chroot "$FS" which "$B")"
104         tar cf - --dereference -C "$FS" ${BIN#/} \
105             $(chroot "$FS" ldd "$BIN" | grep -oE '/[^ ]*'|sed 's|/||')\
106             | tar -xf - --keep-directory-symlink -C $TMP/prep
107     done
108 }
109
110 INITRD="$FS"/boot/initrd.img${LINUX#*/vmlinuz}
111 if [ -r "$INITRD" ] ; then
112     #select X in my FS ; do [ -n "$X" ] && break ; done
113     X=FS
114 else
115     X=my
116 fi
117 case "$X" in
118     FS)
119         copyFStree lib/modules
120         zstdcat $INITRD | ( cd $TMP/prep && cpio -i )
121         ;;
122     my)
123         copyFStree lib/modules lib/udev
124         copyFSbin busybox sh cttyhack setsid mount mkdir
125         copyFSbin udevd udevadm modprobe modinfo kmod blkid
126         ;;
127 esac
128
129 ## (re)install /init
130 echo "** Please select /init template"
131 select INIT in none $(echo init*.template) ; do [ -n "$INIT" ] && break ; done
132 if [ "$INIT" != none ] ; then
133     mv $TMP/prep/init{,.FS}
134     cp $INIT $TMP/prep/init
135     chmod a+x $TMP/prep/init
136     cp message.txt $TMP/prep/message.txt
137 fi
138
139 ( cd "$TMP/prep" && find -printf '%P\n' | cpio -o -H newc ) | \
140     gzip > $TMP/initrd.gz
141 cp $LINUX $TMP/vmlinuz
142 rm -rf $TMP/prep
143
144 #============================================================
145 # Add syslinux boot equipment for bios boot under $TMP/syslinux
146 # (also includes the syslinux image mastering further below)
147 echo "Please select boot console options"
148
149 CONOPTS=( none ttyS0,115200 )
150 select CON in "${CONOPTS[@]}" ; do [ -n "$CON" ] && break ; done
151 case "$CON" in none) CON= ;; *) CON="console=$CON" ;; esac
152
153 mkdir -p $TMP/syslinux
154 cp -t $TMP/syslinux /usr/lib/syslinux/modules/bios/*
155 cat <<EOF > $TMP/syslinux/syslinux.cfg
156 default menu.c32
157 label linux
158     kernel /vmlinuz
159     append initrd=/initrd.gz init=/init console=ttyS0,115200 
160 label rescue
161     kernel /vmlinuz
162     append initrd=/initrd.gz init=/init console=ttyS0,115200 rescue
163 EOF
164
165 #============================================================
166 [ -e "$IMG" ] && read -p "** Will overwrite $IMG (or ^C here and now)" x
167 rm -f "$IMG"
168
169 # First partition sectors (round up to a Mb number
170 DUM1=$(( ( $(du -sB1 $TMP | sed 's/\s.*//') / 1048576 + 2 ) ))
171 DUM2=$(( $(stat -c %s FS.sqfs) / 1048576 + 2 ))
172
173 # Estimate the required disk image size in Mib
174 DUM=$(( DUM1 + DUM2 + 1 ))
175
176 # Create the image, with the partition marked as EFI parition and
177 # bootable; acutal format is FAT{12,16,32}
178 dd if=/dev/zero of="$IMG" bs=${DUM}M count=0 seek=1 status=none
179 cat <<EOF | tee -a /proc/self/fd/2 | sfdisk -q "$IMG"
180 label: dos
181
182 2048 $(( DUM1 * 2048 )) 0xef *
183 - - 0x83
184 EOF
185 PART=( $(fdisk -l -ostart,sectors $IMG | tail -n 2) )
186 echo "${PART[*]}"
187
188 # First sector is 2048 by default, which is byte address 1048576
189 TMPIMG=$(mktemp XXXX)
190 fusefile $TMPIMG $IMG/0+$(( ( ${PART[0]} + ${PART[1]} ) * 512 ))
191 mkfs.fat --offset ${PART[0]} "$TMPIMG"  >/dev/null
192 umount $TMPIMG
193 rm $TMPIMG
194 mcopy -i "$IMG@@1048576" -s $TMP/* ::/
195 [ -z "$SUDO_USER" ] || chown $SUDO_USER: "$IMG"
196
197 dd if=FS.sqfs of=$IMG seek=${PART[2]} conv=notrunc status=none
198
199 #============================================================
200 # Final syslinux bios boot image mastering
201 dd if=/usr/lib/SYSLINUX/mbr.bin of="$IMG" bs=440 conv=notrunc status=none
202 syslinux -i -d syslinux -t 1048576 "$IMG"