3 ################################################################################
5 # Init script to boot over network through NBD
7 # Contact: Stéphane Desneux <stephane.desneux@iot.bzh>
9 ################################################################################
11 # banner generator: echo "AGL - Netboot" | figlet -f slant -w 80 -c
13 ________________________________________________________________________________
14 ___ ________ _ __ __ __ __
15 / | / ____/ / / | / /__ / /_/ /_ ____ ____ / /_
16 / /| |/ / __/ / ______ / |/ / _ \/ __/ __ \/ __ \/ __ \/ __/
17 / ___ / /_/ / /___ /_____/ / /| / __/ /_/ /_/ / /_/ / /_/ / /_
18 /_/ |_\____/_____/ /_/ |_/\___/\__/_.___/\____/\____/\__/
19 ________________________________________________________________________________
31 # -------------------------------------------
33 log_info() { echo "$0[$$]: $@" >&2; }
34 log_error() { echo "$0[$$]: ERROR $@" >&2; }
37 log_info "mounting FS: $@"
38 [[ -e /proc/filesystems ]] && { grep -q "$1" /proc/filesystems || { log_error "Unknown filesystem"; return 1; } }
39 [[ -d "$2" ]] || mkdir -p "$2"
40 [[ -e /proc/mounts ]] && { grep -q -e "^$1 $2 $1" /proc/mounts && { log_info "$2 ($1) already mounted"; return 0; } }
41 mount -t "$1" "$1" "$2"
46 check_debug "Reboot will occur after exiting this shell."
47 log_info "Rebooting..."
60 find_active_interface() {
61 [[ ! -d /sys/class/net ]] && { log_error "find_active_interface: /sys/class/net doesn't exist"; return 2; }
63 for x in $(ls -d /sys/class/net/* 2>/dev/null); do
65 # find interfaces with:
66 # - type == 1 (ethernet)
70 [[ $(cat $x/type) != 1 ]] && continue
71 [[ -d $x/wireless ]] && continue
72 [[ $(cat $x/operstate) != "up" ]] && continue
74 log_info "find_active_interface: first active interface is $iface"
79 log_error "Unable to find any active network interface."
83 # -------------------------------------------
85 export PATH=/sbin:/usr/sbin:/bin:/usr/bin
87 log_info "starting initrd script"
89 do_mount_fs proc /proc
90 do_mount_fs sysfs /sys
91 do_mount_fs devtmpfs /dev
92 do_mount_fs devpts /dev/pts
93 do_mount_fs tmpfs /dev/shm
94 do_mount_fs tmpfs /tmp
95 do_mount_fs tmpfs /run
97 # parse kernel commandline to get NBD server
98 for x in $(cat /proc/cmdline); do
100 nbd.server=*) NBD_SERVER=${x/*=/};;
101 nbd.port=*) NBD_PORT=${x/*=/};;
102 nbd.dev=*) NBD_DEV=/dev/${x/*=/};;
103 nbd.namev3=*) NBD_NAMEV3=${x/*=/};;
104 nbd.debug=*) DEBUG=${x/*=/};;
108 check_debug "Debug point 1. Exit to continue initrd script (mount NBD device)."
110 log_info "NBD parameters: device $NBD_DEV, server $NBD_SERVER:$NBD_PORT"
112 # check if smack is active (and if so, mount smackfs)
113 grep -q smackfs /proc/filesystems && {
116 do_mount_fs smackfs /sys/fs/smackfs
118 # adjust current label and network label
119 echo System >/proc/self/attr/current
120 echo System >/sys/fs/smackfs/ambient
126 log_info "Starting NBD client"
127 if [ -z "${NBD_NAMEV3}" ]; then
128 nbd-client $NBD_SERVER $NBD_PORT $NBD_DEV && { log_info "NBD client successfully started"; break; }
129 log_info "NBD client failed"
131 nbd3-client $NBD_SERVER $NBD_DEV --name $NBD_NAMEV3 && { log_info "NBD3 client successfully started"; break; }
132 log_info "NBDv3 client failed"
134 [[ $try -gt 0 ]] && { log_info "Retrying ($try trie(s) left)..."; sleep 3; try=$(( try - 1 )); continue; }
136 bail_out "Unable to mount NBD device $NBD_DEV using server $NBD_SERVER:$NBD_PORT"
141 mount $NBD_DEV -o noatime /sysroot || bail_out "Unable to mount root NBD device"
143 # move mounted devices to new root
145 for x in dev proc sys tmp run; do
146 log_info "Moving /$x to new rootfs"
150 # switch to new rootfs
151 log_info "Switching to new rootfs"
152 mkdir -p boot/initramfs
153 pivot_root . boot/initramfs || bail_out "pivot_root failed."
155 # workaround for connman (avoid bringing down the network interface used for booting, disable DNS proxy)
156 if [[ -f /lib/systemd/system/connman.service ]]; then
158 iface=$(find_active_interface)
159 [[ -n "$iface" ]] && newopts="$newopts -I $iface"
161 log_info "Adjusting Connman command line. Will be: 'connmand $newopts'"
162 sed -i "s|connmand -n\$|connmand $newopts|g" /lib/systemd/system/connman.service
165 # also use /proc/net/pnp to generate /etc/resolv.conf
166 rm -f /etc/resolv.conf
167 grep -v bootserver /proc/net/pnp | sed 's/^domain/search/g' >/etc/resolv.conf
168 chsmack -A /etc/resolv.conf
170 # unmount tmp and run to let systemd remount them with correct smack labels (SPEC-2596)
171 log_info "Unmounting /tmp and /run"
175 # finally, run systemd
176 check_debug "Debug point 2. Exit to continue initrd script (run systemd)."
178 log_info "Exec'ing systemd"
179 # banner generator: echo "AGL Booting . . ." | figlet -f slant -w 80 -c
181 ________________________________________________________________________________
182 ___ ________ ____ __ _
183 / | / ____/ / / __ )____ ____ / /_(_)___ ____ _
184 / /| |/ / __/ / / __ / __ \/ __ \/ __/ / __ \/ __ `/
185 / ___ / /_/ / /___ / /_/ / /_/ / /_/ / /_/ / / / / /_/ / _ _ _
186 /_/ |_\____/_____/ /_____/\____/\____/\__/_/_/ /_/\__, / (_) (_) (_)
187 _____________________________________________________/____/_____________________
190 exec /lib/systemd/systemd </dev/console >/dev/console 2>&1