summaryrefslogtreecommitdiff
path: root/sys-boot/grub-handler/files
diff options
context:
space:
mode:
Diffstat (limited to 'sys-boot/grub-handler/files')
-rwxr-xr-xsys-boot/grub-handler/files/grub-handler318
1 files changed, 318 insertions, 0 deletions
diff --git a/sys-boot/grub-handler/files/grub-handler b/sys-boot/grub-handler/files/grub-handler
new file mode 100755
index 00000000..4034ba1d
--- /dev/null
+++ b/sys-boot/grub-handler/files/grub-handler
@@ -0,0 +1,318 @@
+#!/usr/bin/python2
+# -*- coding: utf-8 -*-
+"""
+ Kernel grub.conf configuration script
+
+ Copyright (C) 2009 Fabio Erculiani
+
+ 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.
+
+"""
+import os
+import sys
+import subprocess
+import shutil
+BOOT_MOUNT = False
+NO_SYS_ROOT_BOOT_DIR = "/boot"
+if os.path.ismount(NO_SYS_ROOT_BOOT_DIR):
+ BOOT_MOUNT = True
+SYS_ROOT = os.getenv("ROOT","")
+GRUB_CONF = SYS_ROOT+"/boot/grub/grub.conf"
+FSTAB_CONF = SYS_ROOT+"/etc/fstab"
+DISTRO_NAME = "Rogentos Linux"
+
+def getstatusoutput(cmd):
+ """Return (status, output) of executing cmd in a shell."""
+ pipe = os.popen('{ ' + cmd + '; } 2>&1', 'r')
+ text = pipe.read()
+ sts = pipe.close()
+ if sts is None: sts = 0
+ if text[-1:] == '\n': text = text[:-1]
+ return sts, text
+
+def get_kernel_grub_line(kernel):
+ return "title=%s (%s)\n" % (DISTRO_NAME, os.path.basename(kernel),)
+
+def configure_boot_grub(kernel, initramfs):
+
+ grub_dir = os.path.dirname(GRUB_CONF)
+ if not os.path.isdir(grub_dir):
+ os.makedirs(grub_dir)
+
+ if os.access(GRUB_CONF, os.R_OK | os.F_OK):
+
+ # open in append
+ grub = open(GRUB_CONF,"aw")
+ shutil.copy2(GRUB_CONF, GRUB_CONF+".add")
+ # get boot dev
+ boot_dev = get_grub_boot_dev()
+ # test if entry has been already added
+ grubtest = open(GRUB_CONF,"r")
+ content = grubtest.readlines()
+ content = [unicode(x,'raw_unicode_escape') for x in content]
+ for line in content:
+
+ if line.find(get_kernel_grub_line(kernel)) != -1:
+ grubtest.close()
+ print "** Kernel already in configuration => ", line.strip()
+ return
+
+ # also check if we have the same kernel listed
+ if (line.find("kernel") != 1) and \
+ (line.find(os.path.basename(kernel)) != -1) and not \
+ line.strip().startswith("#") \
+ and (line.find("safe mode") == -1):
+
+ grubtest.close()
+ print "** Kernel already in configuration (2) => ", line.strip()
+ return
+ else:
+
+ # create
+ boot_dev = get_grub_boot_dev()
+ grub = open(GRUB_CONF,"w")
+ # write header - guess (hd0,0)... since it is weird
+ # having a running system without a bootloader, at least, grub.
+ grub.write("default=0\ntimeout=10\n")
+
+ cmdline = ''
+ if os.access("/proc/cmdline", os.R_OK):
+ cmdline_f = open("/proc/cmdline","r")
+ cmdline = " "+cmdline_f.readline().strip()
+ cmdline_f.close()
+
+ grub.write(get_kernel_grub_line(kernel))
+ grub.write("\troot "+boot_dev+"\n")
+ grub.write("\tkernel "+kernel+cmdline+"\n")
+ if initramfs:
+ grub.write("\tinitrd "+initramfs+"\n")
+ grub.write("\tsavedefault\n")
+ grub.write("\n")
+ grub.flush()
+ grub.close()
+
+def remove_boot_grub(kernel, initramfs):
+
+ grub_dir = os.path.dirname(GRUB_CONF)
+ if not os.path.isdir(grub_dir):
+ os.makedirs(grub_dir)
+
+ if os.path.isdir(grub_dir) and os.access(GRUB_CONF, os.R_OK | os.F_OK):
+
+ shutil.copy2(GRUB_CONF, GRUB_CONF+".remove")
+ grub_f = open(GRUB_CONF,"r")
+ grub_conf = grub_f.readlines()
+ grub_f.close()
+
+ content = [unicode(x,'raw_unicode_escape') for x in grub_conf]
+ if not isinstance(kernel, unicode):
+ kernel = unicode(kernel,'raw_unicode_escape')
+ if not isinstance(initramfs, unicode):
+ initramfs = unicode(initramfs,'raw_unicode_escape')
+
+ new_conf = []
+ skip = False
+ for line in content:
+
+ kernel_grub_line = get_kernel_grub_line(kernel)
+ if (line.find(kernel_grub_line) != -1):
+ skip = True
+ continue
+
+ if line.strip().startswith("title"):
+ skip = False
+
+ if not skip or line.strip().startswith("#"):
+ new_conf.append(line)
+
+ grub_tmp_f = open(GRUB_CONF+".tmp","w")
+ for line in new_conf:
+ try:
+ grub_tmp_f.write(line)
+ except UnicodeEncodeError:
+ grub_tmp_f.write(line.encode('utf-8'))
+ grub_tmp_f.flush()
+ grub_tmp_f.close()
+ os.rename(GRUB_CONF+".tmp", GRUB_CONF)
+
+def boot_device_translation(boot_dev):
+
+ # actually disabled due to buggy grub.conf entry
+ if os.access(GRUB_CONF, os.R_OK | os.F_OK) and 0:
+
+ f_grub = open(GRUB_CONF, "r")
+ stored_boot_dev = [x.strip() for x in f_grub.readlines() if \
+ x.strip().startswith("#boot=")]
+ f_grub.close()
+ if stored_boot_dev:
+ stored_boot_dev = stored_boot_dev[0]
+ boot_dev = "/dev/" + stored_boot_dev[len("#boot="):]
+
+ if boot_dev.startswith("/dev/md"):
+
+ boot_dev = os.path.realpath(boot_dev)
+ md_dev = os.path.basename(boot_dev)
+
+ if os.access("/proc/mdstat", os.R_OK | os.F_OK):
+
+ f_mdstat = open("/proc/mdstat","r")
+ stored_boot_dev = [x.split() for x in f_mdstat.readlines() if \
+ x.startswith(md_dev)]
+ f_mdstat.close()
+
+ if stored_boot_dev:
+ stored_boot_dev = stored_boot_dev[0]
+ for elem in stored_boot_dev:
+ if elem.endswith("[0]"):
+ boot_dev = "/dev/" + elem[:-len("[0]")]
+ break
+
+ return boot_dev
+
+def resolve_device(device):
+ if device.startswith("/dev/"):
+ return device
+ if device.startswith("UUID=") or device.startswith("LABEL="):
+ print "resolving UUID/LABEL to device", device
+ rc, outstring = getstatusoutput("blkid -lt %s" % (device,))
+ if rc != 0:
+ print "cannot resolve UUID/LABEL for", device
+ return None # argh!
+ device = outstring.split(":")[0]
+ print "UUID/LABEL resolved to", device
+ return device
+
+def get_grub_boot_dev():
+
+ grub_avail = subprocess.call("which grub &> /dev/null", shell = True)
+ if grub_avail != 0:
+ print "** Cannot find grub. Cannot properly configure kernel"
+ return "(hd0,0)"
+
+ # load currently mounted partitions
+ if not os.access(FSTAB_CONF, os.R_OK | os.F_OK):
+ print "** Cannot find %s. Cannot properly configure kernel" % (
+ FSTAB_CONF,)
+ return "(hd0,0)"
+
+ f_fstab = open(FSTAB_CONF, "r")
+ mount_data = [x.split() for x in f_fstab.readlines()]
+ f_fstab.close()
+ # filter out bogus devices
+ mount_data = [x for x in mount_data if x]
+ mount_data = [x for x in mount_data if x[0].startswith("/") or \
+ x[0].startswith("UUID=") or x[0].startswith('LABEL=')]
+
+ mount_hash = {}
+ for item in mount_data:
+ solved_dev = resolve_device(item[0])
+ if not solved_dev:
+ continue
+ mount_hash[item[1]] = solved_dev
+ boot_dev = mount_hash.get(NO_SYS_ROOT_BOOT_DIR, mount_hash.get("/"))
+ if boot_dev == None:
+ print "** Cannot determine boot device. Cannot properly configure" \
+ " kernel"
+ return "(hd0,0)"
+
+ # translate boot device, if needed
+ boot_dev = boot_device_translation(boot_dev)
+
+ # load grub map file
+ map_file = "grub.map"
+ subprocess.call('echo "quit" | grub --no-floppy --no-config-file ' \
+ '--no-curses --batch --device-map=grub.map &> /dev/null', shell = True)
+ if not os.access(map_file, os.R_OK | os.F_OK):
+ print "** Cannot find grub. Cannot properly configure kernel"
+ return "(hd0,0)"
+
+ f_map = open(map_file)
+ map_data = [x.split() for x in f_map.readlines()]
+ f_map.close()
+ os.remove(map_file)
+ map_data = dict(((y, x) for x, y in map_data))
+
+ map_data_devs = map_data.keys()
+ grub_dev = None
+ linux_dev = None
+ for dev in map_data_devs:
+ if boot_dev.startswith(dev):
+ grub_dev = map_data.get(dev)
+ linux_dev = dev
+ break
+
+ if grub_dev == None:
+ print "** Cannot match grub device. Cannot properly configure kernel"
+ return "(hd0,0)"
+
+ device_number = boot_dev.replace(linux_dev,'')
+ try:
+ device_number = int(device_number)
+ except ValueError:
+ print "** Cannot get device number for '%s' => '%s' | '%s'. Cannot properly configure kernel" % (
+ device_number, boot_dev, linux_dev,)
+ return "(hd0,0)"
+
+ device_number -= 1
+ grub_boot_dev = grub_dev.replace(')',',%s)' % (device_number,))
+ return grub_boot_dev
+
+def print_help():
+ print "%s %s %s %s" % (sys.argv[0], "[add/remove]",
+ "<kernel>", "<initramfs or 'none'>",)
+
+def add_kernel(kernel, initramfs):
+
+ boot_len = len(NO_SYS_ROOT_BOOT_DIR)
+ if BOOT_MOUNT:
+ kernel = kernel[boot_len:]
+ if initramfs:
+ initramfs = initramfs[boot_len:]
+
+ # configure GRUB
+ print "** Configuring GRUB bootloader. Adding the new kernel ..."
+ configure_boot_grub(kernel, initramfs)
+
+def remove_kernel(kernel, initramfs):
+
+ boot_len = len(NO_SYS_ROOT_BOOT_DIR)
+ if BOOT_MOUNT:
+ kernel = kernel[boot_len:]
+ if initramfs:
+ initramfs = initramfs[boot_len:]
+
+ # configure GRUB
+ print "** Configuring GRUB bootloader. Removing the selected kernel ..."
+ remove_boot_grub(kernel, initramfs)
+
+
+
+if __name__ == "__main__":
+
+ args = sys.argv[1:]
+ if len(args) < 3:
+ print_help()
+ raise SystemExit(1)
+
+ cmd = args[0]
+ if cmd not in ("add", "remove",):
+ print_help()
+ raise SystemExit(1)
+
+ kernel = args[1]
+ initramfs = args[2]
+
+ if initramfs == "none":
+ initramfs = ''
+
+ if cmd == "add":
+ print "** Adding kernel '%s' and initramfs '%s'" % (kernel, initramfs,)
+ add_kernel(kernel, initramfs)
+ elif cmd == "remove":
+ print "** Removing kernel '%s' and initramfs '%s'" % (kernel,
+ initramfs,)
+ remove_kernel(kernel, initramfs)
+ raise SystemExit(0)