2017-04-29 17:40:34 +00:00
|
|
|
#!/bin/sh
|
2017-07-12 04:17:45 +00:00
|
|
|
set -e -o pipefail
|
|
|
|
bootdir="$1"
|
|
|
|
file="$2"
|
2017-07-13 04:33:49 +00:00
|
|
|
|
|
|
|
if [ -z "$bootdir" -o -z "$file" ]; then
|
2017-07-17 16:43:14 +00:00
|
|
|
die "Usage: $0 /boot /boot/grub/grub.cfg"
|
2017-07-13 04:33:49 +00:00
|
|
|
fi
|
2017-04-29 17:40:34 +00:00
|
|
|
|
|
|
|
reset_entry() {
|
|
|
|
name=""
|
|
|
|
kexectype="elf"
|
|
|
|
kernel=""
|
|
|
|
initrd=""
|
|
|
|
modules=""
|
|
|
|
append=""
|
|
|
|
}
|
|
|
|
|
2017-07-13 04:33:49 +00:00
|
|
|
filedir=`dirname $file`
|
2017-07-22 18:25:39 +00:00
|
|
|
bootdir="${bootdir%%/}"
|
|
|
|
bootlen="${#bootdir}"
|
|
|
|
appenddir="${filedir:$bootlen}"
|
2017-07-12 04:17:45 +00:00
|
|
|
|
|
|
|
fix_path() {
|
|
|
|
path="$@"
|
|
|
|
if [ "${path:0:1}" != "/" ]; then
|
|
|
|
path="$appenddir/$path"
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
2022-11-10 15:13:38 +00:00
|
|
|
# GRUB kernel lines (linux/multiboot) can include a command line. Check whether
|
|
|
|
# the file path exists in $bootdir.
|
|
|
|
check_path() {
|
|
|
|
local checkpath firstval
|
|
|
|
checkpath="$1"
|
|
|
|
firstval="$(echo "$checkpath" | cut -d\ -f1)"
|
|
|
|
if ! [ -r "$bootdir$firstval" ]; then return 1; fi
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
2017-04-29 17:40:34 +00:00
|
|
|
echo_entry() {
|
2022-11-10 15:26:20 +00:00
|
|
|
if [ -z "$kernel" ]; then return; fi
|
2017-04-29 17:40:34 +00:00
|
|
|
|
2022-11-10 15:26:20 +00:00
|
|
|
fix_path $kernel
|
|
|
|
# The kernel must exist - if it doesn't, ignore this entry, it
|
|
|
|
# wouldn't work anyway. This could happen if there was a
|
|
|
|
# GRUB variable in the kernel path, etc.
|
|
|
|
if ! check_path "$path"; then return; fi
|
|
|
|
entry="$name|$kexectype|kernel $path"
|
2017-07-02 18:27:02 +00:00
|
|
|
|
2022-11-10 15:26:20 +00:00
|
|
|
case "$kexectype" in
|
|
|
|
elf)
|
|
|
|
if [ -n "$initrd" ]; then
|
|
|
|
for init in $(echo $initrd | tr ',' ' '); do
|
|
|
|
fix_path $init
|
|
|
|
# The initrd must also exist
|
|
|
|
if ! check_path "$path"; then return; fi
|
|
|
|
entry="$entry|initrd $path"
|
|
|
|
done
|
|
|
|
fi
|
|
|
|
if [ -n "$append" ]; then
|
|
|
|
entry="$entry|append $append"
|
|
|
|
fi
|
|
|
|
;;
|
|
|
|
multiboot|xen)
|
|
|
|
entry="$entry$modules"
|
|
|
|
;;
|
|
|
|
*)
|
|
|
|
return
|
|
|
|
;;
|
|
|
|
esac
|
2017-04-29 17:40:34 +00:00
|
|
|
|
2022-11-10 15:26:20 +00:00
|
|
|
# Double-expand here in case there are variables in the kernel
|
|
|
|
# parameters - some configs do this and can boot with empty
|
|
|
|
# expansions (Debian Live ISOs use this for loopback boots)
|
|
|
|
echo $(eval "echo \"$entry\"")
|
2017-04-29 17:40:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
search_entry() {
|
|
|
|
case $line in
|
|
|
|
menuentry* | MENUENTRY* )
|
|
|
|
state="grub"
|
|
|
|
reset_entry
|
|
|
|
name=`echo $line | tr "'" "\"" | cut -d\" -f 2`
|
|
|
|
;;
|
|
|
|
|
|
|
|
label* | LABEL* )
|
|
|
|
state="syslinux"
|
|
|
|
reset_entry
|
2017-07-04 23:49:14 +00:00
|
|
|
name=`echo $line | cut -c6- `
|
2017-04-29 17:40:34 +00:00
|
|
|
esac
|
|
|
|
}
|
|
|
|
|
|
|
|
grub_entry() {
|
|
|
|
if [ "$line" = "}" ]; then
|
|
|
|
echo_entry
|
|
|
|
state="search"
|
|
|
|
return
|
|
|
|
fi
|
|
|
|
|
|
|
|
# add info to menuentry
|
|
|
|
trimcmd=`echo $line | tr '\t ' ' ' | tr -s ' '`
|
|
|
|
cmd=`echo $trimcmd | cut -d\ -f1`
|
|
|
|
val=`echo $trimcmd | cut -d\ -f2-`
|
|
|
|
case $cmd in
|
|
|
|
multiboot*)
|
2017-07-04 23:49:14 +00:00
|
|
|
# TODO: differentiate between Xen and other multiboot kernels
|
|
|
|
kexectype="xen"
|
2017-04-29 17:40:34 +00:00
|
|
|
kernel="$val"
|
|
|
|
;;
|
|
|
|
module*)
|
2017-04-30 01:50:10 +00:00
|
|
|
case $val in
|
|
|
|
--nounzip*) val=`echo $val | cut -d\ -f2-` ;;
|
|
|
|
esac
|
2017-07-12 04:17:45 +00:00
|
|
|
fix_path $val
|
|
|
|
modules="$modules|module $path"
|
2017-04-29 17:40:34 +00:00
|
|
|
;;
|
|
|
|
linux*)
|
2022-11-10 15:09:16 +00:00
|
|
|
# Some configs have a device specification in the kernel
|
|
|
|
# or initrd path. Assume this would be /boot and remove
|
|
|
|
# it. Keep the '/' following the device, since this
|
|
|
|
# path is relative to the device root, not the config
|
|
|
|
# location.
|
|
|
|
kernel=`echo $trimcmd | sed "s/([^)]*)//g" | cut -d\ -f2`
|
2017-04-29 17:40:34 +00:00
|
|
|
append=`echo $trimcmd | cut -d\ -f3-`
|
|
|
|
;;
|
|
|
|
initrd*)
|
2022-11-10 15:09:16 +00:00
|
|
|
# Trim off device specification as above
|
|
|
|
initrd="$(echo "$val" | sed "s/([^)]*)//g")"
|
2017-04-29 17:40:34 +00:00
|
|
|
;;
|
|
|
|
esac
|
|
|
|
}
|
|
|
|
|
|
|
|
syslinux_end() {
|
|
|
|
# finish menuentry
|
|
|
|
|
|
|
|
# attempt to parse out of append if missing initrd
|
|
|
|
if [ -z "$initrd" ]; then
|
|
|
|
newappend=""
|
|
|
|
for param in $append; do
|
|
|
|
case $param in
|
|
|
|
initrd=*)
|
|
|
|
initrd=`echo $param | cut -d\= -f2`
|
|
|
|
;;
|
|
|
|
*) newappend="$newappend $param" ;;
|
|
|
|
esac
|
|
|
|
done
|
2017-07-13 04:33:49 +00:00
|
|
|
append="${newappend##' '}"
|
2017-04-29 17:40:34 +00:00
|
|
|
fi
|
|
|
|
|
2020-06-26 21:23:43 +00:00
|
|
|
appenddir="$(echo $appenddir | cut -d\/ -f -2)"
|
2017-04-29 17:40:34 +00:00
|
|
|
echo_entry
|
|
|
|
state="search"
|
|
|
|
}
|
|
|
|
|
|
|
|
syslinux_multiboot_append() {
|
|
|
|
splitval=`echo "${val// --- /|}" | tr '|' '\n'`
|
|
|
|
while read line
|
|
|
|
do
|
|
|
|
if [ -z "$kernel" ]; then
|
|
|
|
kernel="$line"
|
|
|
|
else
|
2017-07-12 04:17:45 +00:00
|
|
|
fix_path $line
|
|
|
|
modules="$modules|module $path"
|
2017-04-29 17:40:34 +00:00
|
|
|
fi
|
|
|
|
done << EOF
|
|
|
|
$splitval
|
|
|
|
EOF
|
|
|
|
}
|
|
|
|
|
|
|
|
syslinux_entry() {
|
2017-04-30 01:50:10 +00:00
|
|
|
case $line in
|
|
|
|
"")
|
|
|
|
syslinux_end
|
|
|
|
return
|
|
|
|
;;
|
|
|
|
label* | LABEL* )
|
|
|
|
syslinux_end
|
|
|
|
search_entry
|
|
|
|
return
|
|
|
|
;;
|
|
|
|
esac
|
2017-04-29 17:40:34 +00:00
|
|
|
|
|
|
|
# add info to menuentry
|
|
|
|
trimcmd=`echo $line | tr '\t ' ' ' | tr -s ' '`
|
|
|
|
cmd=`echo $trimcmd | cut -d\ -f1`
|
|
|
|
val=`echo $trimcmd | cut -d\ -f2-`
|
|
|
|
case $trimcmd in
|
|
|
|
menu* | MENU* )
|
|
|
|
cmd2=`echo $trimcmd | cut -d \ -f2`
|
|
|
|
if [ "$cmd2" = "label" -o "$cmd2" = "LABEL" ]; then
|
2017-07-04 23:49:14 +00:00
|
|
|
name=`echo $trimcmd | cut -c11- | tr -d '^'`
|
2017-04-29 17:40:34 +00:00
|
|
|
fi
|
|
|
|
;;
|
|
|
|
linux* | LINUX* | kernel* | KERNEL* )
|
|
|
|
case $val in
|
2017-07-04 23:49:14 +00:00
|
|
|
# TODO: differentiate between Xen and other multiboot kernels
|
|
|
|
*mboot.c32) kexectype="xen" ;;
|
2017-04-29 17:40:34 +00:00
|
|
|
*.c32)
|
|
|
|
# skip this entry
|
|
|
|
state="search"
|
|
|
|
;;
|
|
|
|
*)
|
2020-08-24 20:42:33 +00:00
|
|
|
kernel="${val#"$bootdir"}"
|
2017-04-29 17:40:34 +00:00
|
|
|
esac
|
|
|
|
;;
|
|
|
|
initrd* | INITRD* )
|
2020-08-24 20:42:33 +00:00
|
|
|
initrd="${val#"$bootdir"}"
|
2017-04-29 17:40:34 +00:00
|
|
|
;;
|
|
|
|
append* | APPEND* )
|
2017-07-04 23:49:14 +00:00
|
|
|
if [ "$kexectype" = "multiboot" -o "$kexectype" = "xen" ]; then
|
2017-04-29 17:40:34 +00:00
|
|
|
syslinux_multiboot_append
|
|
|
|
else
|
|
|
|
append="$val"
|
|
|
|
fi
|
|
|
|
;;
|
|
|
|
esac
|
|
|
|
}
|
|
|
|
|
|
|
|
state="search"
|
|
|
|
while read line
|
|
|
|
do
|
|
|
|
case $state in
|
|
|
|
search)
|
|
|
|
search_entry
|
|
|
|
;;
|
|
|
|
grub)
|
|
|
|
grub_entry
|
|
|
|
;;
|
|
|
|
syslinux)
|
|
|
|
syslinux_entry
|
|
|
|
;;
|
|
|
|
esac
|
2017-07-13 04:33:49 +00:00
|
|
|
done < "$file"
|
2017-04-29 17:40:34 +00:00
|
|
|
|
|
|
|
# handle EOF case
|
|
|
|
if [ "$state" = "syslinux" ]; then
|
|
|
|
syslinux_end
|
|
|
|
fi
|