2008-08-18 18:49:11 +02:00
|
|
|
#!/usr/bin/perl
|
|
|
|
#
|
|
|
|
# This program 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 2
|
|
|
|
# of the License, or (at your option) any later version.
|
|
|
|
#
|
|
|
|
# This program 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 this program; if not, write to the Free Software
|
|
|
|
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
|
|
|
|
# 02110-1301, USA. }}}
|
|
|
|
#
|
|
|
|
use Bootloader::Tools;
|
|
|
|
use strict;
|
|
|
|
use Getopt::Long;
|
|
|
|
|
|
|
|
use constant FALSE => 0;
|
|
|
|
use constant TRUE => 1;
|
|
|
|
|
|
|
|
my $GRUBDIR = "/boot/grub";
|
|
|
|
my $GRUBDEFAULT = "$GRUBDIR/default";
|
|
|
|
my $debug = FALSE;
|
2008-10-06 19:04:19 +02:00
|
|
|
my $showHelp = FALSE;
|
2008-08-18 18:49:11 +02:00
|
|
|
|
|
|
|
#
|
|
|
|
# Prints the given stuff (variable number of arguments) if debugging has
|
|
|
|
# been enabled. Does nothing otherwise.
|
|
|
|
sub print_debug(@) # {{{
|
|
|
|
{
|
|
|
|
if ($debug) {
|
|
|
|
print STDERR @_;
|
|
|
|
print STDERR "\n";
|
|
|
|
}
|
|
|
|
} # }}}
|
|
|
|
|
2008-10-06 19:04:19 +02:00
|
|
|
#
|
|
|
|
# Displays help. Does not exit.
|
|
|
|
sub show_help()
|
|
|
|
{
|
|
|
|
print STDERR "kexec-bootloader\n";
|
|
|
|
print STDERR "Loads kexec kernel from bootloader configuration.\n\n";
|
|
|
|
print STDERR "Options:\n";
|
|
|
|
print STDERR " -h | --help Shows that help message.\n";
|
|
|
|
print STDERR " -D | --debug Prints debugging information.\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-08-18 18:49:11 +02:00
|
|
|
#
|
|
|
|
# Returns the value stored with "grubonce". If no value has been stored
|
|
|
|
# or the /boot/grub/default file is not readable, then -1 is returned.
|
2008-10-28 00:51:20 +01:00
|
|
|
#
|
|
|
|
# Also emulate the behaviour when using GRUB which resets the 'magic once' flag
|
|
|
|
# when booting. Because we use kexec, we have to reset that 'magic once' flag
|
|
|
|
# ourselves.
|
|
|
|
sub get_grubonce_and_reset_magic() # {{{
|
2008-08-18 18:49:11 +02:00
|
|
|
{
|
|
|
|
# no /boot/grub/default file
|
|
|
|
if (! -f $GRUBDEFAULT) {
|
2008-10-28 00:51:20 +01:00
|
|
|
print_debug("get_grubonce_and_reset_magic(): No $GRUBDEFAULT.");
|
2008-08-18 18:49:11 +02:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
# read /boot/grub/default
|
|
|
|
open(FH, $GRUBDEFAULT) or return -1;
|
|
|
|
my $value;
|
|
|
|
my $ret = sysread(FH, $value, 10);
|
2008-10-28 00:51:20 +01:00
|
|
|
close(FH);
|
2008-08-18 18:49:11 +02:00
|
|
|
|
|
|
|
# only if we have read 4 bytes it's valid
|
|
|
|
if ($ret != 10) {
|
2008-10-28 00:51:20 +01:00
|
|
|
print_debug("get_grubonce_and_reset_magic(): ".
|
|
|
|
"Read returned $ret instead of 4.");
|
2008-08-18 18:49:11 +02:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
$value =~ s/\n//g;
|
|
|
|
my $once = int($value);
|
|
|
|
|
|
|
|
# 0x4000 is the "magic once flag"
|
2008-10-28 00:51:20 +01:00
|
|
|
unless ($once & 0x4000) {
|
|
|
|
print_debug("get_grubonce_and_reset_magic(): No magic 0x40000.");
|
2008-08-18 18:49:11 +02:00
|
|
|
return -1;
|
|
|
|
}
|
2008-10-28 00:51:20 +01:00
|
|
|
|
|
|
|
my $defaultno = $once & ~0x4000;
|
|
|
|
my $buf = $defaultno . "\0" . "\n" x 9;
|
|
|
|
|
|
|
|
# now reset the grubonce flag
|
|
|
|
open(FH, ">$GRUBDEFAULT") or return $defaultno;
|
|
|
|
$ret = syswrite(FH, $buf, 10);
|
|
|
|
close(FH);
|
|
|
|
|
|
|
|
return $defaultno;
|
2008-08-18 18:49:11 +02:00
|
|
|
} # }}}
|
|
|
|
|
|
|
|
#
|
|
|
|
# Parses command line options and sets global variables.
|
|
|
|
sub parse_options() # {{{
|
|
|
|
{
|
|
|
|
GetOptions(
|
2008-10-06 19:04:19 +02:00
|
|
|
"D|debug" => \$debug,
|
|
|
|
"h|help" => \$showHelp
|
2008-08-18 18:49:11 +02:00
|
|
|
);
|
|
|
|
} # }}}
|
|
|
|
|
|
|
|
|
|
|
|
parse_options();
|
2008-10-06 19:04:19 +02:00
|
|
|
if ($showHelp) {
|
|
|
|
show_help();
|
|
|
|
exit(0);
|
|
|
|
}
|
2008-08-18 18:49:11 +02:00
|
|
|
Bootloader::Tools::InitLibrary();
|
|
|
|
my $loader = Bootloader::Tools::GetBootloader();
|
|
|
|
my $default = -1;
|
|
|
|
|
|
|
|
if ($loader =~ m/GRUB/i) {
|
2008-10-28 00:51:20 +01:00
|
|
|
$default = get_grubonce_and_reset_magic();
|
2008-08-18 18:49:11 +02:00
|
|
|
print_debug("GRUB Default: $default");
|
|
|
|
}
|
|
|
|
|
|
|
|
my $section = undef;
|
|
|
|
# do we have a default?
|
2008-10-22 18:09:16 +02:00
|
|
|
if ($default >= 0) {
|
2008-08-18 18:49:11 +02:00
|
|
|
my @sections = Bootloader::Tools::GetSectionList();
|
|
|
|
print_debug("Number of sections: " . $#sections);
|
|
|
|
|
|
|
|
if ($#sections < 0 || $#sections < $default) {
|
|
|
|
print STDERR "WARNING: grubonce default number ($default) is invalid.\n";
|
|
|
|
print STDERR " Falling back to the default GRUB section.\n";
|
|
|
|
} else {
|
|
|
|
my $sect_name = $sections[$default];
|
|
|
|
$section = Bootloader::Tools::GetSection($sect_name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
# use the default section if we didn't get any default otherwise
|
|
|
|
if (!$section) {
|
|
|
|
$section = Bootloader::Tools::GetDefaultSection();
|
|
|
|
}
|
|
|
|
|
2008-10-09 18:48:54 +02:00
|
|
|
if (!$section) {
|
|
|
|
print STDERR "Unable to get default section of bootloader configuration.\n";
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
2014-02-21 18:41:59 +01:00
|
|
|
my $image=$section->{"image"};
|
|
|
|
my $initrd=$section->{"initrd"};
|
|
|
|
|
|
|
|
# handle btfs /@ -> /
|
|
|
|
if (substr ($image, 0, 3) eq "/@/") {
|
|
|
|
$image = substr ($image, 2);
|
|
|
|
}
|
|
|
|
if ($initrd && substr ($initrd, 0, 3) eq "/@/") {
|
|
|
|
$initrd = substr ($initrd, 2);
|
|
|
|
}
|
|
|
|
|
2008-08-18 18:49:11 +02:00
|
|
|
if ($debug) {
|
|
|
|
print "Type : " . $section->{"type"}."\n";
|
|
|
|
print "Name : " . $section->{"name"}."\n";
|
2014-02-21 18:41:59 +01:00
|
|
|
print "Image : " . $image."\n";
|
|
|
|
print "Initrd : " . $initrd."\n";
|
2008-08-18 18:49:11 +02:00
|
|
|
print "VGA : " . $section->{"vgamode"}."\n";
|
|
|
|
print "Append : " . $section->{"append"}."\n";
|
|
|
|
print "Root : " . $section->{"root"}."\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($section->{"type"} ne "image") {
|
2008-10-09 18:48:54 +02:00
|
|
|
print STDERR "Default boot section is no image.\n";
|
2008-08-18 18:49:11 +02:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
2014-02-21 18:41:59 +01:00
|
|
|
|
2008-08-18 18:49:11 +02:00
|
|
|
# check if the image exists
|
2014-02-21 18:41:59 +01:00
|
|
|
if (! -f $image) {
|
|
|
|
print STDERR "Image '" . $image . "' does not exist.\n";
|
2008-08-18 18:49:11 +02:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
# check if the initrd exists
|
2014-02-21 18:41:59 +01:00
|
|
|
if ($initrd && ! -f $initrd) {
|
|
|
|
print STDERR "Initrd '" . $initrd . "' does not exist.\n";
|
2008-08-18 18:49:11 +02:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
# Do we have kexec?
|
|
|
|
if (system("which kexec &>/dev/null") != 0) {
|
|
|
|
print STDERR "kexec not available. Install kexec-tools.\n";
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
# build the command list
|
|
|
|
my $cmd = "kexec";
|
|
|
|
|
|
|
|
# append image
|
2014-02-21 18:41:59 +01:00
|
|
|
$cmd .= " -l '" . $image . "'";
|
2008-08-18 18:49:11 +02:00
|
|
|
|
|
|
|
# append initrd if available
|
2014-02-21 18:41:59 +01:00
|
|
|
if ($initrd) {
|
|
|
|
$cmd .= " --initrd='" . $initrd . "'";
|
2008-08-18 18:49:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
# build append line
|
|
|
|
my $append = "";
|
|
|
|
if ($section->{"root"}) {
|
|
|
|
$append .= "root=" . $section->{"root"};
|
|
|
|
}
|
|
|
|
if ($section->{"vga"}) {
|
|
|
|
$append .= " vga=" . $section->{"vga"};
|
|
|
|
}
|
|
|
|
if ($section->{"append"}) {
|
|
|
|
$append .= " " . $section->{"append"};
|
|
|
|
}
|
|
|
|
|
|
|
|
# and tell that kexec
|
|
|
|
$cmd .= " --append='" . $append . "'";
|
|
|
|
|
|
|
|
print_debug("Kexec call: " . $cmd);
|
|
|
|
if (system($cmd) != 0) {
|
|
|
|
print STDERR "kexec($cmd) failed.";
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
exit(0);
|
|
|
|
|
|
|
|
# :vim set ts=4 sw=4 et fdm=markers: :collapseFolds=1:
|