[Pm-utils] Some thoughts about some of the hooks
Stefan Seyfried
seife at suse.de
Wed Oct 11 06:06:40 PDT 2006
On Tue, Oct 10, 2006 at 07:53:40PM -0400, Peter Jones wrote:
> > And since every distro has a little bit different tweaks for grub (or
> > maybe just plain simple naming conventions for the boot entries or such),
> > i think the grub hook belongs in the grub package.
>
> For now, I'd just as soon leave this here, but make it call some script
> we can mutually decide on to set/unset the next boot option.
On SUSE i can just copy /boot/grub/default to a safe place and restore it
during thaw(), and everything will be fine. I have however no idea if this
is a solution that only works due to some suse-specific grub patch or if
this is a generic solution.
> > I have a pretty complicated logic to find out which kernel to boot next
> > in powersaved scripts, that is apparently obsoleted by /sbin/grubby on
> > fedora and that also will not work on other distros due to different
> > naming conventions for the kernel binary.
>
> Yeah, this way is just how we do it on Fedora; I'm perfectly willing to
> change it to be more generic.
I'll attach my current grub hook which does:
(during suspend)
- parse /boot/grub/menu.lst to find the available menu entries and
put them into an array
- check if there is an entry that matches /boot/vmlinuz-$(uname -r)
- if there is none, touch /var/run/pm-utils.inhibit, this is checked later
to prevent suspending when we already know we cannot resume.
- copy /boot/grub/default to /var/run/pm-utils.grubonce.default
- call grubonce with the entry number found before
(during resume)
- move /var/run/pm-utils.grubonce.default to /boot/grub/default
I have my doubts that this will work anywhere besides a SUSE system since
it makes some assumptions about the layout of /boot/grub/menu.lst and about
the names of the kernel binaries, and i do not think that these are valid on
other systems.
This is also a pretty raw port of the corresponding powersave script, since
pm-utils do not have any logging / debugging infrastructure yet, every hook
has to roll its own for now.
So this is just a "look how i think that it could be done" and not a "please
include" :-)
Regards,
Stefan
--
Stefan Seyfried
QA / R&D Team Mobile Devices | "Any ideas, John?"
SUSE LINUX Products GmbH, Nürnberg | "Well, surrounding them's out."
-------------- next part --------------
#!/bin/bash
#
# Stefan Seyfried, SUSE Linux Products GmbH 2006, GPL v2
# mostly taken from the powersave project.
GRUBONCE="/usr/sbin/grubonce"
GRUBDEFAULT="/boot/grub/default"
GRUBDEFSAVE="/var/run/pm-utils.grubonce.default"
LOGFILE="/var/log/pm-utils.$1.log"
INHIBIT="/var/run/pm-utils.inhibit"
#####################################################################
# gets a list of available kernels from /boot/grub/menu.lst
# kernels are in the array $KERNELS, output to stdout to be eval-ed.
getkernels()
{
# DEBUG "Running getkernels()" INFO
local MENU_LST="/boot/grub/menu.lst"
local I DUMMY
declare -i I=0 J=-1
# build an array KERNELS with all the kernels in /boot/grub/menu.lst
# the array MENU_ENTRIES contains the corresponding menu entry numbers
# DEFAULT_BOOT contains the default entry.
while read LINE; do
case $LINE in
title*)
let J++ # increase for every menu entry, even for non-linux
# DEBUG "Found grub menu entry #${J}: '${LINE}'" INFO
;;
default*)
DUMMY=($LINE) # "default 0 #maybe a comment"
echo "DEFAULT_BOOT=${DUMMY[1]}" # ^^[0]^^ 1 ^^[2]^ 3 ^^[4]^^
# DEBUG "Default boot entry is '${DUMMY[1]}'" INFO
;;
kernel*)
DUMMY=($LINE) # kernel (hd0,1)/boot/vmlinuz-ABC root=/dev/hda2
echo "KERNELS[$I]='${DUMMY[1]##*/}'" # vmlinuz-ABC
echo "MENU_ENTRIES[$I]=$J"
# DEBUG "Found kernel entry #${I}: '${DUMMY[1]##*/}'" INFO
let I++
;;
*) ;;
esac
done < $MENU_LST
}
#############################################################
# runs grubonce from the grub package to select which kernel
# to boot on next startup
grub-once()
{
if [ -x "$GRUBONCE" ]; then
rm -f "$GRUBDEFSAVE"
if [ -e "$GRUBDEFAULT" ]; then
echo "saving original $GRUBDEFAULT" >> $LOGFILE
cp "$GRUBDEFAULT" "$GRUBDEFSAVE"
fi
echo "running '$GRUBONCE $1'" >> $LOGFILE
$GRUBONCE $1
else
echo "$GRUBONCE not found, not preparing bootloader" >> $LOGFILE
fi
}
#############################################################
# restore grub default after (eventually failed) resume
grub-once-restore()
{
rm -f "$GRUBDEFAULT"
if [ -e "$GRUBDEFSAVE" ]; then
echo "restoring original $GRUBDEFAULT" >> $LOGFILE
mv "$GRUBDEFSAVE" "$GRUBDEFAULT"
fi
}
#############################################################################
# try to find a kernel image that matches the actually running kernel.
# We need this, if more than one kernel is installed. This works reasonably
# well with grub, if all kernels are named "vmlinuz-`uname -r`" and are
# located in /boot. If they are not, good luck ;-)
find-kernel-entry()
{
NEXT_BOOT=-1
ARCH=`uname -m`
declare -i I=0
# DEBUG "running kernel: $RUNNING" DIAG
while [ -n "${KERNELS[$I]}" ]; do
BOOTING="${KERNELS[$I]}"
if IMAGE=`readlink /boot/$BOOTING` && [ -e "/boot/${IMAGE##*/}" ]; then
# DEBUG "Found kernel symlink $BOOTING => $IMAGE" INFO
BOOTING=$IMAGE
fi
case $ARCH in
ppc*) BOOTING="${BOOTING#*vmlinux-}" ;;
*) BOOTING="${BOOTING#*vmlinuz-}" ;;
esac
if [ "$RUNNING" == "$BOOTING" ]; then
NEXT_BOOT=${MENU_ENTRIES[$I]}
# DEBUG "running kernel is grub menu entry $NEXT_BOOT" DIAG
# DEBUG "kernel filename: '${KERNELS[$I]}'" DIAG
echo "running kernel is grub menu entry $NEXT_BOOT (${KERNELS[$I]})" \
>> $LOGFILE
break
fi
let I++
done
# if we have not found a kernel, issue a warning.
# if we have found a kernel, we'll do "grub-once" later, after
# prepare_suspend finished.
if [ $NEXT_BOOT -eq -1 ]; then
# DEBUG "no kernelfile matching the running kernel found" WARN
echo "no kernelfile matching the running kernel found" >> $LOGFILE
fi
}
#############################################################################
# if we did not find a kernel (or BOOT_LOADER is not GRUB) check,
# if the running kernel is still the one that will (probably) be booted for
# resume (default entry in menu.lst or, if there is none, the kernel file
# /boot/vmlinuz points to.)
# This will only work, if you use "original" SUSE kernels.
# you can always override with the config variable set to "yes"
prepare-grub()
{
eval `getkernels`
RUNNING=`uname -r`
find-kernel-entry
RET=0
echo "next boot: $NEXT_BOOT"
if [ $NEXT_BOOT -eq -1 ]; then
# which kernel is booted with the default entry?
BOOTING="${KERNELS[$DEFAULT_BOOT]}"
# if there is no default entry (no menu.lst?) we fall back to
# the default of /boot/vmlinuz.
[ -z "$BOOTING" ] && BOOTING="vmlinuz"
if IMAGE=`readlink /boot/$BOOTING` && [ -e "/boot/${IMAGE##*/}" ]; then
BOOTING=$IMAGE
fi
BOOTING="${BOOTING#*vmlinuz-}"
# DEBUG "running kernel: '$RUNNING', booting kernel: '$BOOTING'" DIAG
echo "running kernel: '$RUNNING', probably booting kernel: '$BOOTING'" \
>> $LOGFILE
if [ "$BOOTING" != "$RUNNING" ]; then
echo "kernel version mismatch, cannot suspend to disk" >> $LOGFILE
echo "running: $RUNNING booting: $BOOTING" >> $INHIBIT
RET=1
fi
else
# set the bootloader to the running kernel
echo "preparing boot-loader: selecting entry $NEXT_BOOT, kernel /boot/$BOOTING" \
>> $LOGFILE
T1=`date +"%s%N"`
sync; sync; sync # this is needed to speed up grub-once on reiserfs
T2=`date +"%s%N"`
grub-once $NEXT_BOOT > /dev/null 2>&1
T3=`date +"%s%N"`
S=$(((T2-T1)/100000000)); S="$((S/10)).${S:0-1}"
G=$(((T3-T2)/100000000)); G="$((G/10)).${G:0-1}"
echo " time needed for sync: $S seconds, time needed for grub: $G seconds." \
>> $LOGFILE
fi
return $RET
}
###### main()
case $1 in
hibernate)
prepare-grub
;;
thaw)
grub-once-restore
;;
esac
More information about the Pm-utils
mailing list