351 lines
8.7 KiB
Text
351 lines
8.7 KiB
Text
# Helper library for grub-mkconfig
|
|
# Copyright (C) 2007,2008,2009,2010 Free Software Foundation, Inc.
|
|
#
|
|
# GRUB is free software: you can redistribute it and/or modify
|
|
# it under the terms of the GNU General Public License as published by
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
# (at your option) any later version.
|
|
#
|
|
# GRUB is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
# GNU General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License
|
|
# along with GRUB. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
transform="@program_transform_name@"
|
|
|
|
prefix="@prefix@"
|
|
exec_prefix="@exec_prefix@"
|
|
datarootdir="@datarootdir@"
|
|
datadir="@datadir@"
|
|
bindir="@bindir@"
|
|
sbindir="@sbindir@"
|
|
pkgdatadir="${datadir}/@PACKAGE@"
|
|
|
|
if test "x$grub_probe" = x; then
|
|
grub_probe="${sbindir}/`echo grub-probe | sed "${transform}"`"
|
|
fi
|
|
if test "x$grub_mkrelpath" = x; then
|
|
grub_mkrelpath="${bindir}/`echo grub-mkrelpath | sed "${transform}"`"
|
|
fi
|
|
|
|
if which gettext >/dev/null 2>/dev/null; then
|
|
:
|
|
else
|
|
gettext () {
|
|
echo -n "$@"
|
|
}
|
|
fi
|
|
|
|
grub_warn ()
|
|
{
|
|
echo "$(gettext "Warning:")" "$@" >&2
|
|
}
|
|
|
|
make_system_path_relative_to_its_root ()
|
|
{
|
|
"${grub_mkrelpath}" "$1"
|
|
}
|
|
|
|
is_path_readable_by_grub ()
|
|
{
|
|
path="$1"
|
|
|
|
# abort if path doesn't exist
|
|
if test -e "$path" ; then : ;else
|
|
return 1
|
|
fi
|
|
|
|
# abort if file is in a filesystem we can't read
|
|
if "${grub_probe}" -t fs "$path" > /dev/null 2>&1 ; then : ; else
|
|
return 1
|
|
fi
|
|
|
|
# ... or if we can't figure out the abstraction module, for example if
|
|
# memberlist fails on an LVM volume group.
|
|
if abstractions="`"${grub_probe}" -t abstraction "$path"`" 2>&1 ; then
|
|
:
|
|
else
|
|
return 1
|
|
fi
|
|
|
|
if [ x$GRUB_CRYPTODISK_ENABLE = xy ]; then
|
|
return 0
|
|
fi
|
|
|
|
for abstraction in $abstractions; do
|
|
if [ "x$abstraction" = xcryptodisk ]; then
|
|
return 1
|
|
fi
|
|
done
|
|
|
|
return 0
|
|
}
|
|
|
|
convert_system_path_to_grub_path ()
|
|
{
|
|
path="$1"
|
|
|
|
grub_warn "convert_system_path_to_grub_path() is deprecated. Use prepare_grub_to_access_device() instead."
|
|
|
|
# abort if GRUB can't access the path
|
|
if is_path_readable_by_grub "${path}" ; then : ; else
|
|
return 1
|
|
fi
|
|
|
|
if drive="`"${grub_probe}" -t drive "$path"`" ; then : ; else
|
|
return 1
|
|
fi
|
|
|
|
if relative_path="`make_system_path_relative_to_its_root "$path"`" ; then : ; else
|
|
return 1
|
|
fi
|
|
|
|
echo "${drive}${relative_path}"
|
|
}
|
|
|
|
save_default_entry ()
|
|
{
|
|
if [ "x${GRUB_SAVEDEFAULT}" = "xtrue" ] ; then
|
|
cat << EOF
|
|
savedefault
|
|
EOF
|
|
fi
|
|
}
|
|
|
|
prepare_grub_to_access_device ()
|
|
{
|
|
device="$1"
|
|
|
|
partmap="`"${grub_probe}" --device "${device}" --target=partmap`"
|
|
for module in ${partmap} ; do
|
|
case "${module}" in
|
|
netbsd | openbsd)
|
|
echo "insmod part_bsd";;
|
|
*)
|
|
echo "insmod part_${module}";;
|
|
esac
|
|
done
|
|
|
|
# Abstraction modules aren't auto-loaded.
|
|
abstraction="`"${grub_probe}" --device "${device}" --target=abstraction`"
|
|
for module in ${abstraction} ; do
|
|
echo "insmod ${module}"
|
|
done
|
|
|
|
fs="`"${grub_probe}" --device "${device}" --target=fs`"
|
|
for module in ${fs} ; do
|
|
echo "insmod ${module}"
|
|
done
|
|
|
|
if [ x$GRUB_CRYPTODISK_ENABLE = xy ]; then
|
|
for uuid in "`"${grub_probe}" --device "${device}" --target=cryptodisk_uuid`"; do
|
|
echo "cryptomount -u $uuid"
|
|
done
|
|
fi
|
|
|
|
# If there's a filesystem UUID that GRUB is capable of identifying, use it;
|
|
# otherwise set root as per value in device.map.
|
|
fs_hint="`"${grub_probe}" --device "${device}" --target=compatibility_hint`"
|
|
if [ "x$fs_hint" != x ]; then
|
|
echo "set root='$fs_hint'"
|
|
fi
|
|
if fs_uuid="`"${grub_probe}" --device "${device}" --target=fs_uuid 2> /dev/null`" ; then
|
|
hints="`"${grub_probe}" --device "${device}" --target=hints_string 2> /dev/null`"
|
|
echo "if [ x\$feature_platform_search_hint = xy ]; then"
|
|
echo " search --no-floppy --fs-uuid --set=root ${hints} ${fs_uuid}"
|
|
echo "else"
|
|
echo " search --no-floppy --fs-uuid --set=root ${fs_uuid}"
|
|
echo "fi"
|
|
fi
|
|
}
|
|
|
|
grub_get_device_id ()
|
|
{
|
|
device="$1"
|
|
if fs_uuid="`"${grub_probe}" --device "${device}" --target=fs_uuid 2> /dev/null`" ; then
|
|
echo "$fs_uuid";
|
|
else
|
|
echo "$device"
|
|
fi
|
|
}
|
|
|
|
grub_file_is_not_garbage ()
|
|
{
|
|
if test -f "$1" ; then
|
|
case "$1" in
|
|
*.dpkg-*) return 1 ;; # debian dpkg
|
|
*.rpmsave|*.rpmnew) return 1 ;;
|
|
README*|*/README*) return 1 ;; # documentation
|
|
esac
|
|
else
|
|
return 1
|
|
fi
|
|
return 0
|
|
}
|
|
|
|
version_sort ()
|
|
{
|
|
case $version_sort_sort_has_v in
|
|
yes)
|
|
LC_ALL=C sort -V;;
|
|
no)
|
|
LC_ALL=C sort -n;;
|
|
*)
|
|
if sort -V </dev/null > /dev/null 2>&1; then
|
|
version_sort_sort_has_v=yes
|
|
LC_ALL=C sort -V
|
|
else
|
|
version_sort_sort_has_v=no
|
|
LC_ALL=C sort -n
|
|
fi;;
|
|
esac
|
|
}
|
|
|
|
version_test_numeric ()
|
|
{
|
|
version_test_numeric_a="$1"
|
|
version_test_numeric_cmp="$2"
|
|
version_test_numeric_b="$3"
|
|
if [ "$version_test_numeric_a" = "$version_test_numeric_b" ] ; then
|
|
case "$version_test_numeric_cmp" in
|
|
ge|eq|le) return 0 ;;
|
|
gt|lt) return 1 ;;
|
|
esac
|
|
fi
|
|
if [ "$version_test_numeric_cmp" = "lt" ] ; then
|
|
version_test_numeric_c="$version_test_numeric_a"
|
|
version_test_numeric_a="$version_test_numeric_b"
|
|
version_test_numeric_b="$version_test_numeric_c"
|
|
fi
|
|
if (echo "$version_test_numeric_a" ; echo "$version_test_numeric_b") | version_sort | head -n 1 | grep -qx "$version_test_numeric_b" ; then
|
|
return 0
|
|
else
|
|
return 1
|
|
fi
|
|
}
|
|
|
|
version_test_gt ()
|
|
{
|
|
version_test_gt_a="`echo "$1" | sed -e "s/[^-]*-//"`"
|
|
version_test_gt_b="`echo "$2" | sed -e "s/[^-]*-//"`"
|
|
version_test_gt_cmp=gt
|
|
if [ "x$version_test_gt_b" = "x" ] ; then
|
|
return 0
|
|
fi
|
|
case "$version_test_gt_a:$version_test_gt_b" in
|
|
*.old:*.old) ;;
|
|
*.old:*) version_test_gt_a="`echo -n "$version_test_gt_a" | sed -e 's/\.old$//'`" ; version_test_gt_cmp=gt ;;
|
|
*:*.old) version_test_gt_b="`echo -n "$version_test_gt_b" | sed -e 's/\.old$//'`" ; version_test_gt_cmp=ge ;;
|
|
esac
|
|
version_test_numeric "$version_test_gt_a" "$version_test_gt_cmp" "$version_test_gt_b"
|
|
return "$?"
|
|
}
|
|
|
|
version_find_latest ()
|
|
{
|
|
version_find_latest_a=""
|
|
for i in "$@" ; do
|
|
if version_test_gt "$i" "$version_find_latest_a" ; then
|
|
version_find_latest_a="$i"
|
|
fi
|
|
done
|
|
echo "$version_find_latest_a"
|
|
}
|
|
|
|
# One layer of quotation is eaten by "", the second by sed, and the third by
|
|
# printf; so this turns ' into \'. Note that you must use the output of
|
|
# this function in a printf format string.
|
|
|
|
grub_quote () {
|
|
sed "s/'/'\\\\\\\\''/g"
|
|
}
|
|
|
|
gettext_quoted () {
|
|
gettext "$@" | sed "s/'/'\\\\\\\\''/g"
|
|
}
|
|
|
|
# Run the first argument through gettext_quoted, and then pass that and all
|
|
# remaining arguments to printf. This is a useful abbreviation and tends to
|
|
# be easier to type.
|
|
gettext_printf () {
|
|
gettext_printf_format="$1"
|
|
shift
|
|
printf "$(gettext "$gettext_printf_format")" "$@"
|
|
}
|
|
|
|
uses_abstraction () {
|
|
device="$1"
|
|
|
|
abstraction="`"${grub_probe}" --device "${device}" --target=abstraction`"
|
|
for module in ${abstraction}; do
|
|
if test "x${module}" = "x$2"; then
|
|
return 0
|
|
fi
|
|
done
|
|
return 1
|
|
}
|
|
|
|
print_option_help () {
|
|
if test x$print_option_help_wc = x; then
|
|
if wc -L < /dev/null > /dev/null; then
|
|
print_option_help_wc=-L
|
|
elif wc -m < /dev/null > /dev/null; then
|
|
print_option_help_wc=-m
|
|
else
|
|
print_option_help_wc=-b
|
|
fi
|
|
fi
|
|
if test x$grub_have_fmt = x; then
|
|
if fmt -w 40 < /dev/null > /dev/null; then
|
|
grub_have_fmt=y;
|
|
else
|
|
grub_have_fmt=n;
|
|
fi
|
|
fi
|
|
print_option_help_lead=" $1"
|
|
print_option_help_lspace="$(echo "$print_option_help_lead" | wc $print_option_help_wc)"
|
|
print_option_help_fill="$((26 - print_option_help_lspace))"
|
|
echo -n "$print_option_help_lead"
|
|
if test $print_option_help_fill -le 0; then
|
|
print_option_help_nl=y
|
|
echo
|
|
else
|
|
print_option_help_i=0;
|
|
while test $print_option_help_i -lt $print_option_help_fill; do
|
|
echo -n " "
|
|
print_option_help_i=$((print_option_help_i+1))
|
|
done
|
|
print_option_help_nl=n
|
|
fi
|
|
if test x$grub_have_fmt = xy; then
|
|
print_option_help_split="$(echo "$2" | fmt -w 50)"
|
|
else
|
|
print_option_help_split="$2"
|
|
fi
|
|
if test x$print_option_help_nl = xy; then
|
|
echo "$print_option_help_split" | awk \
|
|
'{ print " " $0; }'
|
|
else
|
|
echo "$print_option_help_split" | awk 'BEGIN { n = 0 }
|
|
{ if (n == 1) print " " $0; else print $0; n = 1 ; }'
|
|
fi
|
|
}
|
|
|
|
grub_fmt () {
|
|
if test x$grub_have_fmt = x; then
|
|
if fmt -w 40 < /dev/null > /dev/null; then
|
|
grub_have_fmt=y;
|
|
else
|
|
grub_have_fmt=n;
|
|
fi
|
|
fi
|
|
|
|
if test x$grub_have_fmt = xy; then
|
|
fmt
|
|
else
|
|
cat
|
|
fi
|
|
}
|