Current File : //usr/local/jetapps/usr/share/rear/output/USB/Linux-i386/300_create_extlinux.sh |
# Create a suitable syslinux configuration based on capabilities
# Nothing to do here when GRUB2 is specified to be used as USB bootloader:
test "$USB_BOOTLOADER" = "grub" && return
function get_usb_syslinux_version {
for file in $BUILD_DIR/outputfs/{boot/syslinux,}/{ld,ext}linux.sys; do
if [[ -s "$file" ]]; then
strings $file | grep -E -m1 "^(EXT|SYS)LINUX " | cut -d' ' -f2
return 0
fi
done
return 1
}
function syslinux_needs_update {
local usb_syslinux_version=$(get_usb_syslinux_version)
local syslinux_version=$(get_syslinux_version)
Log "USB syslinux version: $usb_syslinux_version"
Log "System syslinux version: $syslinux_version"
if [[ "$usb_syslinux_version" ]] && version_newer "$usb_syslinux_version" "$syslinux_version"; then
Log "No need to update syslinux on USB media (at version $usb_syslinux_version)."
return 1
else
if [[ "$FEATURE_SYSLINUX_SUBMENU" ]]; then
Log "Beware that older entries may not appear in the syslinux menu."
fi
return 0
fi
}
function syslinux_has {
local file="$1"
if [[ -e "$BUILD_DIR/outputfs/$SYSLINUX_PREFIX/$file" ]]; then
if [[ "$SYSLINUX_NEEDS_UPDATE" ]]; then
if [[ -e "$SYSLINUX_DIR/$file" ]]; then
cp -f $v "$SYSLINUX_DIR/$file" "$BUILD_DIR/outputfs/$SYSLINUX_PREFIX/$file" >&2
else
# Make sure we don't have any older copies on USB media
rm -f $v "$BUILD_DIR/outputfs/$SYSLINUX_PREFIX/$file" >&2
return 1;
fi
else
return 0
fi
else
if [[ -e "$SYSLINUX_DIR/$file" ]]; then
cp $v "$SYSLINUX_DIR/$file" "$BUILD_DIR/outputfs/$SYSLINUX_PREFIX/$file" >&2
else
return 1
fi
fi
}
# FIXME: Syslinux older than 3.62 do have menu.c32 but not submenu support
# We simplify by disabling MENU support for everything older than 3.62
function syslinux_write {
if [[ "$*" ]]; then
echo "$@" | syslinux_write
elif [[ "$FEATURE_SYSLINUX_SUBMENU" ]]; then
cat >&4
else
awk '
BEGIN {
IGNORECASE=1
IN_TEXT=0
IGNORE=0
}
/DEFAULT MENU.C32/ { IGNORE=1 }
/ENDTEXT/ { IN_TEXT=0 }
/LABEL -/ { IGNORE=1 }
/MENU / { IGNORE=1 }
/TEXT HELP/ { IN_TEXT=1 }
{
if (IN_TEXT) { IGNORE=1 }
if (! IGNORE) {
print
# } else {
# print "#" $0
}
if (! IN_TEXT) { IGNORE=0 }
}' >&4
fi
}
if syslinux_needs_update; then
SYSLINUX_NEEDS_UPDATE="y"
fi
set_syslinux_features $(get_usb_syslinux_version)
case "$WORKFLOW" in
(mkbackup) usb_label_workflow="backup";;
(mkrescue) usb_label_workflow="rescue image";;
(*) BugError "Workflow $WORKFLOW should not run this script."
esac
USB_REAR_DIR="$BUILD_DIR/outputfs/$USB_PREFIX"
if [ ! -d "$USB_REAR_DIR" ]; then
mkdir -p $v "$USB_REAR_DIR" >/dev/null || Error "Could not create USB ReaR dir [$USB_REAR_DIR] !"
fi
# We generate a single syslinux.cfg for the current system
Log "Creating $USB_PREFIX/syslinux.cfg"
# FIXME: # type -a time
# time is a shell keyword
# time is /usr/bin/time
time=$(basename $USB_REAR_DIR)
if test $USB_SUFFIX ; then
# USB_SUFFIX specifies the last part of the backup directory on the USB medium
# and then basename $USB_REAR_DIR can be anything so that we use it as is:
menu_label="$time"
else
# When USB_SUFFIX is unset, empty, or contains only blanks
# basename $USB_REAR_DIR is a timestamp of the form YYYYMMDD.HHMM
menu_label="${time:0:4}-${time:4:2}-${time:6:2} ${time:9:2}:${time:11:2}"
fi
syslinux_write <<EOF 4>"$USB_REAR_DIR/syslinux.cfg"
label $HOSTNAME-$time
menu label $menu_label $usb_label_workflow
say $HOSTNAME-$time - Recover $HOSTNAME $usb_label_workflow ($time)
text help
Relax-and-Recover v$VERSION - $usb_label_workflow using kernel $(uname -r) ${IPADDR:+on $IPADDR}
${BACKUP:+BACKUP=$BACKUP} ${OUTPUT:+OUTPUT=$OUTPUT} ${BACKUP_URL:+BACKUP_URL=$BACKUP_URL}
endtext
kernel /$USB_PREFIX/kernel
append initrd=/$USB_PREFIX/$REAR_INITRD_FILENAME root=/dev/ram0 vga=normal rw $KERNEL_CMDLINE
label $HOSTNAME-$time
menu label $menu_label $usb_label_workflow - AUTOMATIC RECOVER
say $HOSTNAME-$time - Recover $HOSTNAME $usb_label_workflow ($time)
text help
Relax-and-Recover v$VERSION - $usb_label_workflow using kernel $(uname -r) ${IPADDR:+on $IPADDR}
${BACKUP:+BACKUP=$BACKUP} ${OUTPUT:+OUTPUT=$OUTPUT} ${BACKUP_URL:+BACKUP_URL=$BACKUP_URL}
endtext
kernel /$USB_PREFIX/kernel
append initrd=/$USB_PREFIX/$REAR_INITRD_FILENAME root=/dev/ram0 vga=normal rw $KERNEL_CMDLINE auto_recover
EOF
# Clean up older images of a given system, but keep USB_RETAIN_BACKUP_NR
# entries for backup and rescue when backup on USB works in default mode.
# It keeps by default the two (by default USB_RETAIN_BACKUP_NR=2)
# topmost directories of the "ls -dt $BUILD_DIR/outputfs/rear/$HOSTNAME/*"
# output ("ls -t" sorts by time, newest first)
# regardless whether or not a new backup or a new rescue system was created
# so when there is no new backup or no new rescue system
# it keeps the existing directories as is,
# cf. https://github.com/rear/rear/pull/2794#issuecomment-1106286485
# When USB_SUFFIX is set the compliance mode is used where
# backup on USB works in compliance with backup on NFS which means
# a fixed backup directory and no automated removal of backups or other stuff
# see https://github.com/rear/rear/issues/1164
# Use plain $USB_SUFFIX and not "$USB_SUFFIX" because when USB_SUFFIX contains only blanks
# test "$USB_SUFFIX" would result true because test " " results true:
if ! test $USB_SUFFIX ; then
backup_count=${USB_RETAIN_BACKUP_NR:-2}
rescue_count=${USB_RETAIN_BACKUP_NR:-2}
for rear_run in $(ls -dt $BUILD_DIR/outputfs/rear/$HOSTNAME/*); do
# This fails when the backup archive name is not
# ${BACKUP_PROG_ARCHIVE}${BACKUP_PROG_SUFFIX}${BACKUP_PROG_COMPRESS_SUFFIX}
# so that in particular it would fail for incremental/differential backups
# but incremental/differential backups on USB require USB_SUFFIX to be set:
backup_name=$rear_run/${BACKUP_PROG_ARCHIVE}${BACKUP_PROG_SUFFIX}${BACKUP_PROG_COMPRESS_SUFFIX}
if [[ -e $backup_name ]] ; then
backup_count=$((backup_count - 1))
if (( backup_count < 0 )); then
Log "Remove older backup directory $rear_run"
rm -rf $v $rear_run >/dev/null
fi
else
rescue_count=$((rescue_count - 1))
if (( rescue_count < 0 )); then
Log "Remove older rescue directory $rear_run"
rm -rf $v $rear_run >/dev/null
fi
fi
done
fi
# We generate a ReaR syslinux.cfg based on existing ReaR syslinux.cfg files.
Log "Creating /rear/syslinux.cfg"
{
syslinux_write <<EOF
label rear
say Relax-and-Recover - Recover $HOSTNAME from $time
menu hide
kernel $HOSTNAME-$time
EOF
oldsystem=
# TODO: Sort systems by name, but also sort timestamps in reverse order
for file in $(cd $BUILD_DIR/outputfs; find rear/*/* -name syslinux.cfg); do
dir=$(dirname $file)
# FIXME: # type -a time
# time is a shell keyword
# time is /usr/bin/time
time=$(basename $dir)
system=$(basename $(dirname $dir))
Log "Processing $file"
if [[ "$system" != "$oldsystem" ]]; then
if [[ "$oldsystem" ]]; then
# Close previous submenu
syslinux_write "menu end"
else
# Begin recovery header at top
syslinux_write <<EOF
label -
menu label Recovery images
menu disable
EOF
fi
# Begin submenu
syslinux_write <<EOF
menu begin $system
menu label $system
text help
Recover backup of $system to this system
endtext
EOF
fi
# Include entry
if [[ "$FEATURE_SYSLINUX_INCLUDE" ]]; then
syslinux_write " include /$file"
else
cat $BUILD_DIR/outputfs/$file >&4
fi
oldsystem=$system
done
if [[ "$oldsystem" ]]; then
# Close last submenu
syslinux_write <<EOF
menu separator
label -
menu label ^Back
menu default
text help
Return to the main menu
endtext
menu exit
menu end
EOF
fi
} 4>"$BUILD_DIR/outputfs/rear/syslinux.cfg"
if [ ! -d "$BUILD_DIR/outputfs/$SYSLINUX_PREFIX" ]; then
mkdir -p $v "$BUILD_DIR/outputfs/$SYSLINUX_PREFIX" >/dev/null || Error "Could not create USB syslinux dir [$BUILD_DIR/outputfs/$SYSLINUX_PREFIX] !"
fi
echo "$VERSION_INFO" >$BUILD_DIR/outputfs/$SYSLINUX_PREFIX/message
# We generate a main extlinux.conf in /boot/syslinux that consist of all
# default functionality
Log "Creating $SYSLINUX_PREFIX/extlinux.conf"
{
# Enable serial console:
# For the SERIAL directive to work properly, it must be the first directive in the configuration file,
# see "SERIAL port" at https://wiki.syslinux.org/wiki/index.php?title=SYSLINUX
# It may be useful to reduce it to exact one device since the last 'serial' line wins in SYSLINUX.
if is_true "$USE_SERIAL_CONSOLE" ; then
# When the user has specified SERIAL_CONSOLE_DEVICE_SYSLINUX use only that (no automatisms):
if test "$SERIAL_CONSOLE_DEVICE_SYSLINUX" ; then
# SERIAL_CONSOLE_DEVICE_SYSLINUX can be a character device node like "/dev/ttyS0"
# or a whole SYSLINUX 'serial' directive like "serial 1 9600" for /dev/ttyS1 at 9600 baud.
if test -c "$SERIAL_CONSOLE_DEVICE_SYSLINUX" ; then
# The port value for the SYSLINUX 'serial' directive
# is the trailing digits of the serial device node
# cf. the code of get_partition_number() in lib/layout-functions.sh
port=$( echo "$SERIAL_CONSOLE_DEVICE_SYSLINUX" | grep -o -E '[0-9]+$' )
# E.g. for /dev/ttyS12 the unit would be 12 but
# https://wiki.syslinux.org/wiki/index.php?title=SYSLINUX
# reads in the section "SERIAL port [baudrate [flowcontrol]]"
# "port values from 0 to 3 mean the first four serial ports detected by the BIOS"
# which indicates port values should be less than 4 so we tell the user about it
# but we do not error out because the user may have tested that it does work for him:
test $port -lt 4 || LogPrintError "SERIAL_CONSOLE_DEVICE_SYSLINUX '$SERIAL_CONSOLE_DEVICE_SYSLINUX' may not work (only /dev/ttyS0 up to /dev/ttyS3 should work)"
if speed=$( get_serial_device_speed $SERIAL_CONSOLE_DEVICE_SYSLINUX ) ; then
syslinux_write "serial $port $speed"
else
syslinux_write "serial $port"
fi
else
# When SERIAL_CONSOLE_DEVICE_SYSLINUX is a whole SYSLINUX 'serial' directive use it as specified:
syslinux_write "$SERIAL_CONSOLE_DEVICE_SYSLINUX"
fi
else
for devnode in $( get_serial_console_devices ) ; do
# Add SYSLINUX serial console config for real serial devices:
if speed=$( get_serial_device_speed $devnode ) ; then
# The port value for the SYSLINUX 'serial' directive
# is the trailing digits of the serial device node
# cf. the code of get_partition_number() in lib/layout-functions.sh
port=$( echo "$devnode" | grep -o -E '[0-9]+$' )
test $port -lt 4 || LogPrintError "$devnode may not work as serial console for SYSLINUX (only /dev/ttyS0 up to /dev/ttyS3 should work)"
syslinux_write "serial $port $speed"
# Use the first one and skip the rest to avoid that the last 'serial' line wins in SYSLINUX:
break
fi
done
fi
fi
syslinux_write "display message"
# Add useful syslinux utilities, if present
syslinux_has "cat.c32"
syslinux_has "config.c32"
syslinux_has "cmd.c32"
syslinux_has "cpuid.c32"
syslinux_has "disk.c32"
syslinux_has "host.c32"
syslinux_has "kbdmap.c32"
syslinux_has "ls.c32"
syslinux_has "lua.c32"
syslinux_has "rosh.c32"
syslinux_has "sysdump.c32"
syslinux_has "vesamenu.c32"
# Add needed libraries for syslinux v5 and hdt
syslinux_has "ldlinux.c32"
syslinux_has "libcom32.c32"
syslinux_has "libgpl.c32"
syslinux_has "libmenu.c32"
syslinux_has "libutil.c32"
if [ -r $(get_template "rear.help") ]; then
cp $v $(get_template "rear.help") "$BUILD_DIR/outputfs/$SYSLINUX_PREFIX/rear.help" >/dev/null
syslinux_write <<EOF
say F1 - Show help
F1 /boot/syslinux/rear.help
menu tabmsg Press [Tab] to edit options or [F1] for help
EOF
fi
# Use menu system, if menu.c32 is available
if syslinux_has "menu.c32"; then
syslinux_write "default menu.c32"
fi
syslinux_write <<EOF
timeout 300
#noescape 1
menu title $PRODUCT v$VERSION
EOF
if [[ "$FEATURE_SYSLINUX_INCLUDE" ]]; then
syslinux_write <<EOF
### Add custom items to your configuration by creating custom.cfg
include custom.cfg
### Include generated configuration
include /rear/syslinux.cfg
EOF
else
cat "$BUILD_DIR/outputfs/rear/syslinux.cfg" >&4
fi
syslinux_write <<EOF
menu separator
label -
menu label Other actions
menu disable
EOF
if [[ "$FEATURE_SYSLINUX_MENU_HELP" && -r $(get_template "rear.help") ]]; then
syslinux_write <<EOF
label help
menu label ^Help for Relax-and-Recover
text help
Information about Relax-and-Recover and steps for recovering your system
endtext
menu help rear.help
EOF
fi
# Use chain booting for booting disk, if chain.c32 is available
if syslinux_has "chain.c32" ; then
# Boot from boothd0 (which is usually the same USB disk where this syslinux boot menu is currently shown)
# only as boot default when that was explicitly specified by the user (results usually a boot loop):
if test "boothd0" = "$USB_BIOS_BOOT_DEFAULT" ; then
syslinux_write <<EOF
ontimeout boothd0
label boothd0
say boothd0 - boot first local disk
menu label Boot ^First local disk (hd0)
text help
Usually hd0 is the USB disk wherefrom currently is booted
endtext
menu default
kernel chain.c32
append hd0
label boothd1
say boothd1 - boot second local disk
menu label Boot ^Second local disk (hd1)
text help
Usually hd1 is the local harddisk
endtext
kernel chain.c32
append hd1
EOF
else
# Boot from boothd1 (which is usually the local harddisk) by default (i.e. when USB_BIOS_BOOT_DEFAULT is not boothd0):
syslinux_write <<EOF
label boothd0
say boothd0 - boot first local disk
menu label Boot ^First local disk (hd0)
text help
Usually hd0 is the USB disk wherefrom currently is booted
endtext
kernel chain.c32
append hd0
ontimeout boothd1
label boothd1
say boothd1 - boot second local disk
menu label Boot ^Second local disk (hd1)
text help
Usually hd1 is the local harddisk
endtext
menu default
kernel chain.c32
append hd1
EOF
fi
syslinux_write <<EOF
label bootlocal
say bootlocal - boot second local bios disk
menu label Boot ^BIOS disk (0x81)
text help
Try this when booting from local disk does not work
endtext
localboot 0x81
EOF
else
# Fallback when chain.c32 is not available:
syslinux_write <<EOF
ontimeout bootlocal
label bootlocal
say bootlocal - boot second local bios disk
menu label Boot ^BIOS disk (0x81)
localboot 0x81
EOF
fi
syslinux_write <<EOF
label bootnext
menu label Boot ^Next device
text help
Boot from the next device in the BIOS boot order list
endtext
localboot -1
EOF
if syslinux_has "hdt.c32"; then
if [ -r "/usr/share/hwdata/pci.ids" ]; then
cp $v "/usr/share/hwdata/pci.ids" "$BUILD_DIR/outputfs/$SYSLINUX_PREFIX/pci.ids" >/dev/null
elif [ -r "/usr/share/pci.ids" ]; then
cp $v "/usr/share/pci.ids" "$BUILD_DIR/outputfs/$SYSLINUX_PREFIX/pci.ids" >/dev/null
fi
if [ -r "/lib/modules/$(uname -r)/modules.pcimap" ]; then
cp $v "/lib/modules/$KERNEL_VERSION/modules.pcimap" "$BUILD_DIR/outputfs/$SYSLINUX_PREFIX/modules.pcimap" >/dev/null
fi
syslinux_write <<EOF
label hdt
say hdt - Hardware Detection Tool
menu label Hardware ^Detection tool
text help
Information about your current hardware configuration
endtext
kernel hdt.c32
EOF
fi
# Because usr/sbin/rear sets 'shopt -s nullglob' the 'ls' command will list all files
# in the current working directory if nothing matches the globbing pattern '/boot/memtest86+-*'
# which results '.' in MEMTEST_BIN (the plain 'ls -d' output in the current working directory).
# You need the memtest86+ package installed for this to work
MEMTEST_BIN=$(ls -d /boot/memtest86+-* 2>/dev/null | tail -1)
if [[ "$MEMTEST_BIN" != "." && -r "$MEMTEST_BIN" ]]; then
cp $v "$MEMTEST_BIN" "$BUILD_DIR/outputfs/$SYSLINUX_PREFIX/memtest" >/dev/null
syslinux_write <<EOF
label memtest
say memtest - Run memtest86+
menu label ^Memory test
text help
Test your memory for problems
endtext
kernel memtest
append -
EOF
fi
# echo -e "label -\n menu label ^Exit menu\n menu quit\n" >&4
if syslinux_has "reboot.c32"; then
syslinux_write <<EOF
label reboot
say reboot - Reboot the system
menu label ^Reboot system
text help
Reboot the system now
endtext
kernel reboot.c32
EOF
fi
if syslinux_has "poweroff.com"; then
syslinux_write <<EOF
label poweroff
say poweroff - Power off the system
menu label ^Power off system
text help
Power off the system now
endtext
kernel poweroff.com
EOF
elif syslinux_has "poweroff.c32"; then
syslinux_write <<EOF
label poweroff
say poweroff - Power off the system
menu label ^Power off system
text help
Power off the system now
endtext
kernel poweroff.c32
EOF
fi
} 4>"$BUILD_DIR/outputfs/$SYSLINUX_PREFIX/extlinux.conf"
Log "Created extlinux configuration '$SYSLINUX_PREFIX/extlinux.conf'"
# vim: set et ts=4 sw=4