Skip to content
Snippets Groups Projects
bootloader.py 5.69 KiB
Newer Older
# -*- coding: utf-8 -*-
#
# Copyright (C) 2015 GRNET S.A.
#
# 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 3 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, see <http://www.gnu.org/licenses/>.

"""Module hosting code for determining the installed boot loader on an image.
The bootloader signatures are taken from the "Boot Info Script" project:
    https://github.com/arvidjaar/bootinfoscript
"""

# Master Boot Record Bootloaders
MBR_LDR = [
    # GRUB Legacy
    "grub1",            # 0
    # GRUB 2
    "grub2",
    # LILO (LInux LOader)
    "lilo",
    # SYSLINUX
    "syslinux",
    # Windows
    "Windows",
    # FreeBSD's boot0
    "freebsd",          # 5
    # NetBSD's Stage 0/SUSE generic MBR
    "netbsd",
    # OpenBSD's MBR
    "openbsd",
    # ReactOS
    "reactos",
    # FreeDOS
    "freedos",
    # MS-DOS
    "msdos",            # 10
    # Solaris
    "solaris",
    # ThinkPad MBR
    "thinkpad",
    # HP/Gateway
    "hp",
    # Plop Boot Manager
    "plop",
    # TrueCrypt boot loader
    "truecrypt",        # 15
    # Paragon Partition Manager
    "paragon",
    # Testdisk MBR
    "testdist",
    # GAG Graphical Boot Manager
    "gag",
    # BootIt Boot Manager
    "bootit",
    # DiskCryptor
    "diskcryptor",  # 20
    # xOSL (Extended Operating System Loader)
    "xosl",
    # Fbinst
    "fbinst",
    # Grub4Dos
    "grub4dos",
    # WEE boot manager
    "wee",
    # mbldr (Master Boot LoaDeR)
    "mbldr",   # 25
    # Libparted generic boot code
    "libparted",
    # ISOhybrid
    "isohybrid",
    # Acer PQservice MBR
    "pqservice",
]

# 2-bytes MBR signature
MBR_SIG2 = {
    "\x3b\x48": 0,
    "\xeb\x4c": 1,
    "\xeb\x63": 1,
    "\xeb\x04": 11,
    "\x0e\xbe": 12,
    "\x33\xed": 27,
    "\x33\xff": 13,
    "\xb8\x00": 14,
    "\xea\x1e": 15,
    "\xeb\x04":        11,
    "\xeb\x31": 16,
    "\xfa\x33": 10,
    "\xfa\xeb": 2,
    "\xfa\xfc": 8,
    "\xfc\x31": 17,
    "\xfc\x33": 18,
    "\xfc\xeb": 19,
}

# 3-bytes MBR signature
MBR_SIG3 = {
    "\x33\xc0\x8e": 4,
    "\x33\xc0\x90": 20,
    "\x33\xc0\xfa": 3,
    "\xea\x05\x00": 7,
    "\xea\x05\x01": 21,
    "\xeb\x5e\x00": 22,
    "\xeb\x5e\x80": 23,
    "\xeb\x5e\x90": 24,
    "\xfa\x31\xc0": 3,
    "\xfa\x31\xc9": 25,
    "\xfa\x31\xed": 27,
}

# 4-bytes MBR signature
MBR_SIG4 = {
    "\xfa\xb8\x00\x00": 9,
    "\xfa\xb8\x00\x10": 26,
}

# 8-bytes MBR signature
MBR_SIG8 = {
    "\x31\xc0\x8e\xd0\xbc\x00\x7c\x8e": 6,
    "\x31\xc0\x8e\xd0\xbc\x00\x7c\xfb": 28,
}


def mbr_bootinfo(mbr):
    """Inspect a Master Boot Record and return the installed bootloader"""
    if mbr[:2] == '\x00\x00':
        return "none"

    try:
        ret = MBR_SIG2[mbr[:2]]
    except KeyError:
        try:
            ret = MBR_SIG3[mbr[:3]]
        except KeyError:
            try:
                ret = MBR_SIG4[mbr[:4]]
            except KeyError:
                try:
                    ret = MBR_SIG8[mbr[:8]]
                except KeyError:
                    return "unknown"

    return MBR_LDR[ret]

Nikos Skalkotos's avatar
Nikos Skalkotos committed

# Volume Boot Record Bootloaders
VBR_LDR = [
    # GRUB Legacy
    "grub1",        # 0
    # GRUB 2
    "grub2",
    # LILO (LInux LOader)
    "lilo"
    # SYSLINUX
    "syslinux",
    # Windows
    "windows",
    # FreeBSD's boot1 (First sector of FreeBSD slice)
    "freebsd",      # 5
    # NetBSD's PBR
    "netbsd",
    # OpenBSD
    "openbsd",
    # ReactOS
    "reactos",
    # FreeDOS
    "freedos",
    # MS-DOS
    "msdos",        # 10
    # HP Recovery
    "hp",
    # Dell Utility/Recovery
    "dell",
    # ISOhybrid
    "isohybrid",
    # Grub4Dos
    "grub4dos",
    # BootIt Boot Manager
    "bootit",       # 15
    # Acronis Secure Zone
    "acronis_sz",
]

# 2-byte VBR "no bootloader" signature
VBR_SIG2_NONE = (
    "\x00\x00",
    "\x55\xcd",
    "\x69\x6e",
    "\x6f\x6e",
)

# 2-byte VBR signature
VBR_SIG2 = {
    "\x00\x20": 1,
    "\x00\x69": 13,
    "\x01\x0f": 11,
    "\x02\x11": 12,
    "\x04\x88": 1,
    "\x06\x89": 3,
    "\x07\x34": 10,
    "\x07\x45": 4,
    "\x08\x9e": 10,
    "\x08\xcd": 4,
    "\x08\xc7": 7,
    "\x0b\x60": 12,
    "\x0b\xd0": 4,
    "\x0e\x00": 12,
    "\x0f\xb6": 13,
    "\x2a\x00": 8,
    "\x2d\x5e": 10,
    "\x31\xc0": 3,
    "\x3c\x9a": 6,
    "\x40\x7c": 3,
    "\x42\x16": 14,
    "\x44\x45": 12,
    "\x48\xb4": 1,
    "\x52\x72": 0,
    "\x56\x26": 14,
    "\x63\x8b": 9,
    "\x66\x16": 4,
    "\x69\x74": 15,
    "\x6a\x02": 5,
    "\x6f\x65": 15,
    "\x74\x05": 4,
    "\x7c\x3c": 1,
    "\x7c\xc6": 4,
    "\x7e\x1e": 14,
    "\x80\x53": 2,
    "\x8a\x56": 16,
    "\x83\xe1": 13,
    "\x8e\xc0": 4,
    "\x8e\xd0": 12,
    "\xaa\x75": 0,
    "\xb1\x06": 3,
    "\xb6\x00": 12,
    "\xb6\xc6": 13,
    "\xb6\xd1": 4,
    "\xe8\x79": 13,
    "\xe9\xd8": 4,
    "\xf6\xc1": 4,
    "\xfa\x33": 4,
}

# 4-byte VBR signature
VBR_SIG4 = {
    "\x55\xaa\x75\x0a": 14,
    "\x55\xaa\x75\x06": 4,
    "\x78\x15\xb1\x06": 3,
}


def vbr_bootinfo(vbr):
    """Inspect a Volume Boot Record and return the installed bootloader"""
    if vbr[0x80:0x82] in VBR_SIG2_NONE:
        return "none"

    try:
        ret = VBR_SIG2[vbr[0x80:0x82]]
    except KeyError:
        try:
            ret = VBR_SIG4[vbr[0x80:0x84]]
        except KeyError:
            return "unknown"

    return MBR_LDR[ret]

# vim: set sta sts=4 shiftwidth=4 sw=4 et ai :