2019-03-11 00:35:19 -07:00
|
|
|
#!/usr/bin/env bash
|
2023-03-27 19:23:50 +02:00
|
|
|
set -eu
|
|
|
|
|
2023-03-28 18:42:02 +02:00
|
|
|
/run/server.sh 5000 > /dev/null &
|
2023-03-27 19:23:50 +02:00
|
|
|
|
2023-03-28 18:42:02 +02:00
|
|
|
if /run/install.sh; then
|
|
|
|
echo "Starting DSM for Docker..."
|
|
|
|
else
|
2023-03-28 22:00:23 +02:00
|
|
|
echo "Installation failed (code $?)" && exit 80
|
2023-03-26 23:22:02 +02:00
|
|
|
fi
|
|
|
|
|
2023-03-28 22:00:23 +02:00
|
|
|
IMG="/storage"
|
|
|
|
FILE="$IMG/boot.img"
|
|
|
|
[ ! -f "$FILE" ] && echo "ERROR: Synology DSM boot-image does not exist ($FILE)" && exit 81
|
|
|
|
|
|
|
|
FILE="$IMG/system.img"
|
|
|
|
[ ! -f "$FILE" ] && echo "ERROR: Synology DSM system-image does not exist ($FILE)" && exit 82
|
|
|
|
|
|
|
|
FILE="$IMG/data.img"
|
|
|
|
if [ ! -f "$FILE" ]; then
|
|
|
|
truncate -s $DISK_SIZE $FILE
|
|
|
|
mkfs.ext4 -q $FILE
|
|
|
|
fi
|
|
|
|
|
|
|
|
[ ! -f "$FILE" ] && echo "ERROR: Synology DSM data-image does not exist ($FILE)" && exit 83
|
|
|
|
|
2019-03-11 00:35:19 -07:00
|
|
|
# A bridge of this name will be created to host the TAP interface created for
|
|
|
|
# the VM
|
|
|
|
QEMU_BRIDGE='qemubr0'
|
|
|
|
|
|
|
|
# DHCPD must have an IP address to run, but that address doesn't have to
|
|
|
|
# be valid. This is the dummy address dhcpd is configured to use.
|
|
|
|
DUMMY_DHCPD_IP='10.0.0.1'
|
|
|
|
|
|
|
|
# These scripts configure/deconfigure the VM interface on the bridge.
|
|
|
|
QEMU_IFUP='/run/qemu-ifup'
|
|
|
|
QEMU_IFDOWN='/run/qemu-ifdown'
|
|
|
|
|
|
|
|
# The name of the dhcpd config file we make
|
|
|
|
DHCPD_CONF_FILE='dhcpd.conf'
|
|
|
|
|
|
|
|
function default_intf() {
|
|
|
|
ip -json route show |
|
|
|
|
jq -r '.[] | select(.dst == "default") | .dev'
|
|
|
|
}
|
|
|
|
|
|
|
|
# First step, we run the things that need to happen before we start mucking
|
|
|
|
# with the interfaces. We start by generating the DHCPD config file based
|
|
|
|
# on our current address/routes. We "steal" the container's IP, and lease
|
|
|
|
# it to the VM once it starts up.
|
2023-03-26 20:25:00 +02:00
|
|
|
/run/generate-dhcpd-conf $QEMU_BRIDGE > $DHCPD_CONF_FILE
|
2019-03-11 00:35:19 -07:00
|
|
|
default_dev=`default_intf`
|
|
|
|
|
|
|
|
# Now we start modifying the networking configuration. First we clear out
|
|
|
|
# the IP address of the default device (will also have the side-effect of
|
|
|
|
# removing the default route)
|
|
|
|
ip addr flush dev $default_dev
|
|
|
|
|
|
|
|
# Next, we create our bridge, and add our container interface to it.
|
|
|
|
ip link add $QEMU_BRIDGE type bridge
|
|
|
|
ip link set dev $default_dev master $QEMU_BRIDGE
|
|
|
|
|
|
|
|
# Then, we toggle the interface and the bridge to make sure everything is up
|
|
|
|
# and running.
|
|
|
|
ip link set dev $default_dev up
|
|
|
|
ip link set dev $QEMU_BRIDGE up
|
|
|
|
|
2023-03-26 20:25:00 +02:00
|
|
|
# Prevent error about missing file
|
|
|
|
touch /var/lib/misc/udhcpd.leases
|
|
|
|
|
2019-03-11 00:35:19 -07:00
|
|
|
# Finally, start our DHCPD server
|
2023-03-27 22:44:39 +02:00
|
|
|
udhcpd -I $DUMMY_DHCPD_IP -f $DHCPD_CONF_FILE 2>&1 &
|
2019-03-11 00:35:19 -07:00
|
|
|
|
2023-03-28 06:48:35 +02:00
|
|
|
# Start the Serial Emulator
|
|
|
|
|
|
|
|
HOST_SERIAL=$(/run/serial.sh)
|
|
|
|
GUEST_SERIAL=$(/run/serial.sh)
|
|
|
|
|
|
|
|
./run/serial.bin -cpu=1 \
|
|
|
|
-vmmversion="2.6.1-12139" \
|
|
|
|
-buildnumber=42962 \
|
|
|
|
-vmmts="1679863686" \
|
|
|
|
-cpu_arch string="VirtualDSM" \
|
|
|
|
-guestsn="$GUEST_SERIAL" \
|
|
|
|
-hostsn="$HOST_SERIAL" \
|
|
|
|
-guestuuid="ba13a19a-c0c1-4fef-9346-915ed3b98341" > /dev/null 2>&1 &
|
2023-03-27 19:23:50 +02:00
|
|
|
|
|
|
|
# Stop the webserver
|
|
|
|
pkill -f server.sh
|
2023-03-27 01:20:30 +02:00
|
|
|
|
2023-03-28 22:00:23 +02:00
|
|
|
[ ! -e /dev/fuse ] && echo "Error: FUSE interface not available..." && exit 84
|
|
|
|
[ ! -e /dev/net/tun ] && echo "Error: TUN interface not available..." && exit 85
|
2023-03-28 18:42:02 +02:00
|
|
|
|
|
|
|
if [ -e /dev/kvm ] && sh -c 'echo -n > /dev/kvm' &> /dev/null; then
|
|
|
|
echo "Booting DSM image..."
|
|
|
|
else
|
2023-03-28 22:00:23 +02:00
|
|
|
echo "Error: KVM not available..." && exit 86
|
2023-03-28 18:42:02 +02:00
|
|
|
fi
|
|
|
|
|
2023-03-28 06:48:35 +02:00
|
|
|
# Configure QEMU for graceful shutdown
|
|
|
|
|
|
|
|
QEMU_MONPORT=7100
|
|
|
|
QEMU_POWERDOWN_TIMEOUT=30
|
|
|
|
|
|
|
|
_graceful_shutdown() {
|
|
|
|
|
|
|
|
local COUNT=0
|
|
|
|
local QEMU_MONPORT="${QEMU_MONPORT:-7100}"
|
|
|
|
local QEMU_POWERDOWN_TIMEOUT="${QEMU_POWERDOWN_TIMEOUT:-120}"
|
|
|
|
|
|
|
|
set +e
|
|
|
|
echo "Trying to shut down the VM gracefully"
|
|
|
|
echo 'system_powerdown' | nc -q 1 localhost ${QEMU_MONPORT}>/dev/null 2>&1
|
|
|
|
echo ""
|
|
|
|
while echo 'info version'|nc -q 1 localhost ${QEMU_MONPORT:-7100}>/dev/null 2>&1 && [ "${COUNT}" -lt "${QEMU_POWERDOWN_TIMEOUT}" ]; do
|
|
|
|
let COUNT++
|
|
|
|
echo "QEMU still running. Retrying... (${COUNT}/${QEMU_POWERDOWN_TIMEOUT})"
|
|
|
|
sleep 1
|
|
|
|
done
|
|
|
|
|
|
|
|
if echo 'info version'|nc -q 1 localhost ${QEMU_MONPORT:-7100}>/dev/null 2>&1; then
|
|
|
|
echo "Killing the VM"
|
|
|
|
echo 'quit' | nc -q 1 localhost ${QEMU_MONPORT}>/dev/null 2>&1 || true
|
|
|
|
fi
|
|
|
|
echo "Exiting..."
|
|
|
|
}
|
|
|
|
|
|
|
|
trap _graceful_shutdown SIGINT SIGTERM SIGHUP
|
|
|
|
|
2019-03-11 00:35:19 -07:00
|
|
|
# And run the VM! A brief explaination of the options here:
|
2023-03-28 07:04:22 +02:00
|
|
|
# -accel=kvm: use KVM for this VM (much faster for our case).
|
2019-03-11 00:35:19 -07:00
|
|
|
# -nographic: disable SDL graphics.
|
|
|
|
# -serial mon:stdio: use "monitored stdio" as our serial output.
|
2023-03-28 06:48:35 +02:00
|
|
|
exec qemu-system-x86_64 -name Synology -m $RAM_SIZE -machine accel=kvm -cpu host -nographic -serial mon:stdio \
|
|
|
|
-monitor telnet:localhost:${QEMU_MONPORT:-7100},server,nowait,nodelay \
|
2023-03-27 01:20:30 +02:00
|
|
|
-device virtio-serial-pci,id=virtio-serial0,bus=pci.0,addr=0x3 -chardev pty,id=charserial0 \
|
|
|
|
-device isa-serial,chardev=charserial0,id=serial0 -chardev socket,id=charchannel0,host=127.0.0.1,port=12345,reconnect=10 \
|
|
|
|
-device virtserialport,bus=virtio-serial0.0,nr=1,chardev=charchannel0,id=channel0,name=vchannel \
|
2023-03-26 20:25:00 +02:00
|
|
|
-device virtio-net,netdev=tap0 -netdev tap,id=tap0,ifname=Tap,script=$QEMU_IFUP,downscript=$QEMU_IFDOWN \
|
2023-03-28 06:48:35 +02:00
|
|
|
-device virtio-scsi-pci,id=hw-synoboot,bus=pci.0,addr=0xa -drive file=$IMG/boot.img,if=none,id=drive-synoboot,format=raw,cache=none,aio=native,detect-zeroes=on \
|
2023-03-26 20:25:00 +02:00
|
|
|
-device scsi-hd,bus=hw-synoboot.0,channel=0,scsi-id=0,lun=0,drive=drive-synoboot,id=synoboot0,bootindex=1 \
|
2023-03-28 06:48:35 +02:00
|
|
|
-device virtio-scsi-pci,id=hw-synosys,bus=pci.0,addr=0xb -drive file=$IMG/system.img,if=none,id=drive-synosys,format=raw,cache=none,aio=native,detect-zeroes=on \
|
2023-03-26 20:25:00 +02:00
|
|
|
-device scsi-hd,bus=hw-synosys.0,channel=0,scsi-id=0,lun=0,drive=drive-synosys,id=synosys0,bootindex=2 \
|
2023-03-28 06:48:35 +02:00
|
|
|
-device virtio-scsi-pci,id=hw-userdata,bus=pci.0,addr=0xc -drive file=$IMG/data.img,if=none,id=drive-userdata,format=raw,cache=none,aio=native,detect-zeroes=on \
|
2023-03-27 01:20:30 +02:00
|
|
|
-device scsi-hd,bus=hw-userdata.0,channel=0,scsi-id=0,lun=0,drive=drive-userdata,id=userdata0,bootindex=3 \
|
2023-03-28 06:48:35 +02:00
|
|
|
-device piix3-usb-uhci,id=usb,bus=pci.0,addr=0x1.0x2 &
|
2023-03-27 22:44:39 +02:00
|
|
|
|
2023-03-28 06:48:35 +02:00
|
|
|
wait $!
|