vaseboot/util/VasEBoot-install.c

2052 lines
60 KiB
C

/*
* VAS_EBOOT -- GRand Unified Bootloader
* Copyright (C) 1999,2000,2001,2002,2003,2004,2005,2006,2007,2008,2009,2010,2011,2012,2013 Free Software Foundation, Inc.
*
* VAS_EBOOT 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.
*
* VAS_EBOOT 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 VAS_EBOOT. If not, see <http://www.gnu.org/licenses/>.
*/
#include <config.h>
#include <VasEBoot/types.h>
#include <VasEBoot/emu/misc.h>
#include <VasEBoot/util/misc.h>
#include <VasEBoot/misc.h>
#include <VasEBoot/device.h>
#include <VasEBoot/disk.h>
#include <VasEBoot/file.h>
#include <VasEBoot/fs.h>
#include <VasEBoot/env.h>
#include <VasEBoot/term.h>
#include <VasEBoot/mm.h>
#include <VasEBoot/lib/hexdump.h>
#include <VasEBoot/crypto.h>
#include <VasEBoot/command.h>
#include <VasEBoot/i18n.h>
#include <VasEBoot/zfs/zfs.h>
#include <VasEBoot/util/install.h>
#include <VasEBoot/emu/getroot.h>
#include <VasEBoot/diskfilter.h>
#include <VasEBoot/cryptodisk.h>
#include <VasEBoot/legacy_parse.h>
#include <VasEBoot/gpt_partition.h>
#include <VasEBoot/emu/config.h>
#include <VasEBoot/util/ofpath.h>
#include <VasEBoot/hfsplus.h>
#include <string.h>
#pragma GCC diagnostic ignored "-Wmissing-prototypes"
#pragma GCC diagnostic ignored "-Wmissing-declarations"
#include <argp.h>
#pragma GCC diagnostic error "-Wmissing-prototypes"
#pragma GCC diagnostic error "-Wmissing-declarations"
#include "progname.h"
static char *target;
static int removable = 0;
static int recheck = 0;
static int update_nvram = 1;
static char *install_device = NULL;
static char *debug_image = NULL;
static char *rootdir = NULL;
static char *bootdir = NULL;
static int allow_floppy = 0;
static int force_file_id = 0;
static char *disk_module = NULL;
static char *efidir = NULL;
static char *macppcdir = NULL;
static int force = 0;
static int have_abstractions = 0;
static int have_cryptodisk = 0;
static char * bootloader_id;
static int have_load_cfg = 0;
static FILE * load_cfg_f = NULL;
static char *load_cfg;
static int install_bootsector = 1;
static char *label_font;
static char *label_color;
static char *label_bgcolor;
static char *product_version;
static int add_rs_codes = 1;
enum
{
OPTION_BOOT_DIRECTORY = 0x301,
OPTION_ROOT_DIRECTORY,
OPTION_TARGET,
OPTION_SETUP,
OPTION_MKRELPATH,
OPTION_MKDEVICEMAP,
OPTION_PROBE,
OPTION_EDITENV,
OPTION_ALLOW_FLOPPY,
OPTION_RECHECK,
OPTION_FORCE,
OPTION_FORCE_FILE_ID,
OPTION_NO_NVRAM,
OPTION_REMOVABLE,
OPTION_BOOTLOADER_ID,
OPTION_EFI_DIRECTORY,
OPTION_FONT,
OPTION_DEBUG,
OPTION_DEBUG_IMAGE,
OPTION_NO_FLOPPY,
OPTION_DISK_MODULE,
OPTION_NO_BOOTSECTOR,
OPTION_NO_RS_CODES,
OPTION_MACPPC_DIRECTORY,
OPTION_LABEL_FONT,
OPTION_LABEL_COLOR,
OPTION_LABEL_BGCOLOR,
OPTION_PRODUCT_VERSION
};
static int fs_probe = 1;
static error_t
argp_parser (int key, char *arg, struct argp_state *state)
{
if (VasEBoot_install_parse (key, arg))
return 0;
switch (key)
{
case OPTION_FORCE_FILE_ID:
force_file_id = 1;
return 0;
case 's':
fs_probe = 0;
return 0;
case OPTION_SETUP:
if (!VasEBoot_strstr (arg, "setup"))
install_bootsector = 0;
return 0;
case OPTION_PRODUCT_VERSION:
free (product_version);
product_version = xstrdup (arg);
return 0;
case OPTION_LABEL_FONT:
free (label_font);
label_font = xstrdup (arg);
return 0;
case OPTION_LABEL_COLOR:
free (label_color);
label_color = xstrdup (arg);
return 0;
case OPTION_LABEL_BGCOLOR:
free (label_bgcolor);
label_bgcolor = xstrdup (arg);
return 0;
/* Accept and ignore for compatibility. */
case OPTION_FONT:
case OPTION_MKRELPATH:
case OPTION_PROBE:
case OPTION_EDITENV:
case OPTION_MKDEVICEMAP:
case OPTION_NO_FLOPPY:
return 0;
case OPTION_ROOT_DIRECTORY:
/* Accept for compatibility. */
free (rootdir);
rootdir = xstrdup (arg);
return 0;
case OPTION_BOOT_DIRECTORY:
free (bootdir);
bootdir = xstrdup (arg);
return 0;
case OPTION_MACPPC_DIRECTORY:
free (macppcdir);
macppcdir = xstrdup (arg);
return 0;
case OPTION_EFI_DIRECTORY:
free (efidir);
efidir = xstrdup (arg);
return 0;
case OPTION_DISK_MODULE:
free (disk_module);
disk_module = xstrdup (arg);
return 0;
case OPTION_TARGET:
free (target);
target = xstrdup (arg);
return 0;
case OPTION_DEBUG_IMAGE:
free (debug_image);
debug_image = xstrdup (arg);
return 0;
case OPTION_NO_NVRAM:
update_nvram = 0;
return 0;
case OPTION_FORCE:
force = 1;
return 0;
case OPTION_RECHECK:
recheck = 1;
return 0;
case OPTION_REMOVABLE:
removable = 1;
return 0;
case OPTION_ALLOW_FLOPPY:
allow_floppy = 1;
return 0;
case OPTION_NO_BOOTSECTOR:
install_bootsector = 0;
return 0;
case OPTION_NO_RS_CODES:
add_rs_codes = 0;
return 0;
case OPTION_DEBUG:
verbosity++;
return 0;
case OPTION_BOOTLOADER_ID:
free (bootloader_id);
bootloader_id = xstrdup (arg);
return 0;
case ARGP_KEY_ARG:
if (install_device)
VasEBoot_util_error ("%s", _("More than one install device?"));
install_device = xstrdup (arg);
return 0;
default:
return ARGP_ERR_UNKNOWN;
}
}
static struct argp_option options[] = {
VAS_EBOOT_INSTALL_OPTIONS,
{"boot-directory", OPTION_BOOT_DIRECTORY, N_("DIR"),
0, N_("install VAS_EBOOT images under the directory DIR/%s instead of the %s directory"), 2},
{"root-directory", OPTION_ROOT_DIRECTORY, N_("DIR"),
OPTION_HIDDEN, 0, 2},
{"font", OPTION_FONT, N_("FILE"),
OPTION_HIDDEN, 0, 2},
{"target", OPTION_TARGET, N_("TARGET"),
/* TRANSLATORS: "TARGET" as in "target platform". */
0, N_("install VAS_EBOOT for TARGET platform [default=%s]; available targets: %s"), 2},
{"VasEBoot-setup", OPTION_SETUP, "FILE", OPTION_HIDDEN, 0, 2},
{"VasEBoot-mkrelpath", OPTION_MKRELPATH, "FILE", OPTION_HIDDEN, 0, 2},
{"VasEBoot-mkdevicemap", OPTION_MKDEVICEMAP, "FILE", OPTION_HIDDEN, 0, 2},
{"VasEBoot-probe", OPTION_PROBE, "FILE", OPTION_HIDDEN, 0, 2},
{"VasEBoot-editenv", OPTION_EDITENV, "FILE", OPTION_HIDDEN, 0, 2},
{"allow-floppy", OPTION_ALLOW_FLOPPY, 0, 0,
/* TRANSLATORS: "may break" doesn't just mean that option wouldn't have any
effect but that it will make the resulting install unbootable from HDD. */
N_("make the drive also bootable as floppy (default for fdX devices)."
" May break on some BIOSes."), 2},
{"recheck", OPTION_RECHECK, 0, 0,
N_("delete device map if it already exists"), 2},
{"force", OPTION_FORCE, 0, 0,
N_("install even if problems are detected"), 2},
{"force-file-id", OPTION_FORCE_FILE_ID, 0, 0,
N_("use identifier file even if UUID is available"), 2},
{"disk-module", OPTION_DISK_MODULE, N_("MODULE"), 0,
N_("disk module to use (biosdisk or native). "
"This option is only available on BIOS target."), 2},
{"no-nvram", OPTION_NO_NVRAM, 0, 0,
N_("don't update the `boot-device'/`Boot*' NVRAM variables. "
"This option is only available on EFI and IEEE1275 targets."), 2},
{"skip-fs-probe",'s',0, 0,
N_("do not probe for filesystems in DEVICE"), 0},
{"no-bootsector", OPTION_NO_BOOTSECTOR, 0, 0,
N_("do not install bootsector"), 0},
{"no-rs-codes", OPTION_NO_RS_CODES, 0, 0,
N_("Do not apply any reed-solomon codes when embedding core.img. "
"This option is only available on x86 BIOS targets."), 0},
{"debug", OPTION_DEBUG, 0, OPTION_HIDDEN, 0, 2},
{"no-floppy", OPTION_NO_FLOPPY, 0, OPTION_HIDDEN, 0, 2},
{"debug-image", OPTION_DEBUG_IMAGE, N_("STRING"), OPTION_HIDDEN, 0, 2},
{"removable", OPTION_REMOVABLE, 0, 0,
N_("the installation device is removable. "
"This option is only available on EFI."), 2},
{"bootloader-id", OPTION_BOOTLOADER_ID, N_("ID"), 0,
N_("the ID of bootloader. This option is only available on EFI and Macs."), 2},
{"efi-directory", OPTION_EFI_DIRECTORY, N_("DIR"), 0,
N_("use DIR as the EFI System Partition root."), 2},
{"macppc-directory", OPTION_MACPPC_DIRECTORY, N_("DIR"), 0,
N_("use DIR for PPC MAC install."), 2},
{"label-font", OPTION_LABEL_FONT, N_("FILE"), 0, N_("use FILE as font for label"), 2},
{"label-color", OPTION_LABEL_COLOR, N_("COLOR"), 0, N_("use COLOR for label"), 2},
{"label-bgcolor", OPTION_LABEL_BGCOLOR, N_("COLOR"), 0, N_("use COLOR for label background"), 2},
{"product-version", OPTION_PRODUCT_VERSION, N_("STRING"), 0, N_("use STRING as product version"), 2},
{0, 0, 0, 0, 0, 0}
};
static const char *
get_default_platform (void)
{
#ifdef __powerpc__
return "powerpc-ieee1275";
#elif defined (__sparc__) || defined (__sparc64__)
return "sparc64-ieee1275";
#elif defined (__MIPSEL__)
return "mipsel-loongson";
#elif defined (__MIPSEB__)
return "mips-arc";
#elif defined (__ia64__)
return "ia64-efi";
#elif defined (__arm__)
return VasEBoot_install_get_default_arm_platform ();
#elif defined (__aarch64__)
return "arm64-efi";
#elif defined (__amd64__) || defined (__x86_64__) || defined (__i386__)
return VasEBoot_install_get_default_x86_platform ();
#elif defined (__loongarch_lp64)
return "loongarch64-efi";
#elif defined (__riscv)
#if __riscv_xlen == 32
return "riscv32-efi";
#elif __riscv_xlen == 64
return "riscv64-efi";
#else
return NULL;
#endif
#else
return NULL;
#endif
}
#pragma GCC diagnostic ignored "-Wformat-nonliteral"
static char *
help_filter (int key, const char *text, void *input __attribute__ ((unused)))
{
switch (key)
{
case OPTION_BOOT_DIRECTORY:
return xasprintf (text, VAS_EBOOT_DIR_NAME, VAS_EBOOT_BOOT_DIR_NAME "/" VAS_EBOOT_DIR_NAME);
case OPTION_TARGET:
{
char *plats = VasEBoot_install_get_platforms_string ();
char *ret;
ret = xasprintf (text, get_default_platform (), plats);
free (plats);
return ret;
}
case ARGP_KEY_HELP_POST_DOC:
return xasprintf (text, program_name, VAS_EBOOT_BOOT_DIR_NAME "/" VAS_EBOOT_DIR_NAME);
default:
return VasEBoot_install_help_filter (key, text, input);
}
}
#pragma GCC diagnostic error "-Wformat-nonliteral"
/* TRANSLATORS: INSTALL_DEVICE isn't an identifier and is the DEVICE you
install to. */
struct argp argp = {
options, argp_parser, N_("[OPTION] [INSTALL_DEVICE]"),
N_("Install VAS_EBOOT on your drive.")"\v"
N_("INSTALL_DEVICE must be system device filename.\n"
"%s copies VAS_EBOOT images into %s. On some platforms, it"
" may also install VAS_EBOOT into the boot sector."),
NULL, help_filter, NULL
};
static int
probe_raid_level (VasEBoot_disk_t disk)
{
/* disk might be NULL in the case of a LVM physical volume with no LVM
signature. Ignore such cases here. */
if (!disk)
return -1;
if (disk->dev->id != VAS_EBOOT_DISK_DEVICE_DISKFILTER_ID)
return -1;
if (disk->name[0] != 'm' || disk->name[1] != 'd')
return -1;
if (!((struct VasEBoot_diskfilter_lv *) disk->data)->segments)
return -1;
return ((struct VasEBoot_diskfilter_lv *) disk->data)->segments->type;
}
static void
push_partmap_module (const char *map, void *data __attribute__ ((unused)))
{
char buf[50];
if (strcmp (map, "openbsd") == 0 || strcmp (map, "netbsd") == 0)
{
VasEBoot_install_push_module ("part_bsd");
return;
}
snprintf (buf, sizeof (buf), "part_%s", map);
VasEBoot_install_push_module (buf);
}
static void
push_cryptodisk_module (const char *mod, void *data __attribute__ ((unused)))
{
VasEBoot_install_push_module (mod);
}
static void
probe_mods (VasEBoot_disk_t disk)
{
VasEBoot_partition_t part;
VasEBoot_disk_memberlist_t list = NULL, tmp;
int raid_level;
if (disk->partition == NULL)
VasEBoot_util_info ("no partition map found for %s", disk->name);
for (part = disk->partition; part; part = part->parent)
push_partmap_module (part->partmap->name, NULL);
if (disk->dev->id == VAS_EBOOT_DISK_DEVICE_DISKFILTER_ID)
{
VasEBoot_diskfilter_get_partmap (disk, push_partmap_module, NULL);
have_abstractions = 1;
}
if (disk->dev->id == VAS_EBOOT_DISK_DEVICE_DISKFILTER_ID
&& (VasEBoot_memcmp (disk->name, "lvm/", sizeof ("lvm/") - 1) == 0 ||
VasEBoot_memcmp (disk->name, "lvmid/", sizeof ("lvmid/") - 1) == 0))
VasEBoot_install_push_module ("lvm");
if (disk->dev->id == VAS_EBOOT_DISK_DEVICE_DISKFILTER_ID
&& VasEBoot_memcmp (disk->name, "ldm/", sizeof ("ldm/") - 1) == 0)
VasEBoot_install_push_module ("ldm");
if (disk->dev->id == VAS_EBOOT_DISK_DEVICE_CRYPTODISK_ID)
{
VasEBoot_util_cryptodisk_get_abstraction (disk,
push_cryptodisk_module, NULL);
have_abstractions = 1;
have_cryptodisk = 1;
}
raid_level = probe_raid_level (disk);
if (raid_level >= 0)
{
VasEBoot_install_push_module ("diskfilter");
if (disk->dev->disk_raidname)
VasEBoot_install_push_module (disk->dev->disk_raidname (disk));
}
if (raid_level == 4 || raid_level == 5)
VasEBoot_install_push_module ("raid5rec");
if (raid_level == 6)
VasEBoot_install_push_module ("raid6rec");
/* In case of LVM/RAID, check the member devices as well. */
if (disk->dev->disk_memberlist)
list = disk->dev->disk_memberlist (disk);
while (list)
{
probe_mods (list->disk);
tmp = list->next;
free (list);
list = tmp;
}
}
static int
have_bootdev (enum VasEBoot_install_plat pl)
{
switch (pl)
{
case VAS_EBOOT_INSTALL_PLATFORM_I386_PC:
case VAS_EBOOT_INSTALL_PLATFORM_I386_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_X86_64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_IA64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_ARM_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_ARM64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_LOONGARCH64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_RISCV32_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_RISCV64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_I386_IEEE1275:
case VAS_EBOOT_INSTALL_PLATFORM_SPARC64_IEEE1275:
case VAS_EBOOT_INSTALL_PLATFORM_POWERPC_IEEE1275:
case VAS_EBOOT_INSTALL_PLATFORM_MIPSEL_ARC:
case VAS_EBOOT_INSTALL_PLATFORM_MIPS_ARC:
return 1;
case VAS_EBOOT_INSTALL_PLATFORM_I386_QEMU:
case VAS_EBOOT_INSTALL_PLATFORM_I386_COREBOOT:
case VAS_EBOOT_INSTALL_PLATFORM_ARM_COREBOOT:
case VAS_EBOOT_INSTALL_PLATFORM_I386_MULTIBOOT:
case VAS_EBOOT_INSTALL_PLATFORM_MIPSEL_QEMU_MIPS:
case VAS_EBOOT_INSTALL_PLATFORM_MIPS_QEMU_MIPS:
case VAS_EBOOT_INSTALL_PLATFORM_MIPSEL_LOONGSON:
case VAS_EBOOT_INSTALL_PLATFORM_ARM_UBOOT:
case VAS_EBOOT_INSTALL_PLATFORM_I386_XEN:
case VAS_EBOOT_INSTALL_PLATFORM_X86_64_XEN:
case VAS_EBOOT_INSTALL_PLATFORM_I386_XEN_PVH:
return 0;
/* pacify warning. */
case VAS_EBOOT_INSTALL_PLATFORM_MAX:
return 0;
}
return 0;
}
static void
probe_cryptodisk_uuid (VasEBoot_disk_t disk)
{
VasEBoot_disk_memberlist_t list = NULL, tmp;
/* In case of LVM/RAID, check the member devices as well. */
if (disk->dev->disk_memberlist)
{
list = disk->dev->disk_memberlist (disk);
}
while (list)
{
probe_cryptodisk_uuid (list->disk);
tmp = list->next;
free (list);
list = tmp;
}
if (disk->dev->id == VAS_EBOOT_DISK_DEVICE_CRYPTODISK_ID)
{
const char *uuid = VasEBoot_util_cryptodisk_get_uuid (disk);
if (!load_cfg_f)
load_cfg_f = VasEBoot_util_fopen (load_cfg, "wb");
have_load_cfg = 1;
fprintf (load_cfg_f, "cryptomount -u %s\n",
uuid);
}
}
static int
is_same_disk (const char *a, const char *b)
{
while (1)
{
if ((*a == ',' || *a == '\0') && (*b == ',' || *b == '\0'))
return 1;
if (*a != *b)
return 0;
if (*a == '\\')
{
if (a[1] != b[1])
return 0;
a += 2;
b += 2;
continue;
}
a++;
b++;
}
}
static char *
get_rndstr (void)
{
VasEBoot_uint8_t rnd[15];
const size_t sz = sizeof (rnd) * VAS_EBOOT_CHAR_BIT / 5;
char * ret = xmalloc (sz + 1);
size_t i;
if (VasEBoot_get_random (rnd, sizeof (rnd)))
VasEBoot_util_error ("%s", _("couldn't retrieve random data"));
for (i = 0; i < sz; i++)
{
VasEBoot_size_t b = i * 5;
VasEBoot_uint8_t r;
VasEBoot_size_t f1 = VAS_EBOOT_CHAR_BIT - b % VAS_EBOOT_CHAR_BIT;
VasEBoot_size_t f2;
if (f1 > 5)
f1 = 5;
f2 = 5 - f1;
r = (rnd[b / VAS_EBOOT_CHAR_BIT] >> (b % VAS_EBOOT_CHAR_BIT)) & ((1 << f1) - 1);
if (f2)
r |= (rnd[b / VAS_EBOOT_CHAR_BIT + 1] & ((1 << f2) - 1)) << f1;
if (r < 10)
ret[i] = '0' + r;
else
ret[i] = 'a' + (r - 10);
}
ret[sz] = '\0';
return ret;
}
static char *
escape (const char *in)
{
char *ptr;
char *ret;
int overhead = 0;
for (ptr = (char*)in; *ptr; ptr++)
if (*ptr == '\'')
overhead += 3;
ret = VasEBoot_malloc (ptr - in + overhead + 1);
if (!ret)
return NULL;
VasEBoot_strchrsub (ret, in, '\'', "'\\''");
return ret;
}
static struct VasEBoot_util_config config;
static void
device_map_check_duplicates (const char *dev_map)
{
FILE *fp;
char buf[1024]; /* XXX */
size_t alloced = 8;
size_t filled = 0;
char **d;
size_t i;
if (dev_map[0] == '\0')
return;
fp = VasEBoot_util_fopen (dev_map, "r");
if (! fp)
return;
d = xcalloc (alloced, sizeof (d[0]));
while (fgets (buf, sizeof (buf), fp))
{
char *p = buf;
char *e;
/* Skip leading spaces. */
while (*p && VasEBoot_isspace (*p))
p++;
/* If the first character is `#' or NUL, skip this line. */
if (*p == '\0' || *p == '#')
continue;
if (*p != '(')
continue;
p++;
e = p;
p = strchr (p, ')');
if (! p)
continue;
if (filled >= alloced)
{
alloced *= 2;
d = xrealloc (d, alloced * sizeof (d[0]));
}
*p = '\0';
d[filled++] = xstrdup (e);
}
fclose (fp);
qsort (d, filled, sizeof (d[0]), VasEBoot_qsort_strcmp);
for (i = 0; i + 1 < filled; i++)
if (strcmp (d[i], d[i+1]) == 0)
{
VasEBoot_util_error (_("the drive %s is defined multiple times in the device map %s"),
d[i], dev_map);
}
for (i = 0; i < filled; i++)
free (d[i]);
free (d);
}
static VasEBoot_err_t
write_to_disk (VasEBoot_device_t dev, const char *fn)
{
char *core_img;
size_t core_size;
VasEBoot_err_t err;
core_size = VasEBoot_util_get_image_size (fn);
core_img = VasEBoot_util_read_image (fn);
VasEBoot_util_info ("writing `%s' to `%s'", fn, dev->disk->name);
err = VasEBoot_disk_write (dev->disk, 0, 0,
core_size, core_img);
free (core_img);
return err;
}
static int
is_prep_partition (VasEBoot_device_t dev)
{
if (!dev->disk)
return 0;
if (!dev->disk->partition)
return 0;
if (strcmp(dev->disk->partition->partmap->name, "msdos") == 0)
return (dev->disk->partition->msdostype == 0x41);
if (strcmp (dev->disk->partition->partmap->name, "gpt") == 0)
{
struct VasEBoot_gpt_partentry gptdata;
VasEBoot_partition_t p = dev->disk->partition;
int ret = 0;
dev->disk->partition = dev->disk->partition->parent;
if (VasEBoot_disk_read (dev->disk, p->offset, p->index,
sizeof (gptdata), &gptdata) == 0)
{
const VasEBoot_guid_t template = {
VasEBoot_cpu_to_le32_compile_time (0x9e1a2d38),
VasEBoot_cpu_to_le16_compile_time (0xc612),
VasEBoot_cpu_to_le16_compile_time (0x4316),
{ 0xaa, 0x26, 0x8b, 0x49, 0x52, 0x1e, 0x5a, 0x8b }
};
ret = VasEBoot_memcmp (&template, &gptdata.type,
sizeof (template)) == 0;
}
dev->disk->partition = p;
return ret;
}
return 0;
}
static int
is_prep_empty (VasEBoot_device_t dev)
{
VasEBoot_disk_addr_t dsize, addr;
VasEBoot_uint32_t buffer[32768];
dsize = VasEBoot_disk_native_sectors (dev->disk);
for (addr = 0; addr < dsize;
addr += sizeof (buffer) / VAS_EBOOT_DISK_SECTOR_SIZE)
{
VasEBoot_size_t sz = sizeof (buffer);
VasEBoot_uint32_t *ptr;
if (sizeof (buffer) / VAS_EBOOT_DISK_SECTOR_SIZE > dsize - addr)
sz = (dsize - addr) * VAS_EBOOT_DISK_SECTOR_SIZE;
VasEBoot_disk_read (dev->disk, addr, 0, sz, buffer);
if (addr == 0 && VasEBoot_memcmp (buffer, ELFMAG, SELFMAG) == 0)
return 1;
for (ptr = buffer; ptr < buffer + sz / sizeof (*buffer); ptr++)
if (*ptr)
return 0;
}
return 1;
}
static void
bless (VasEBoot_device_t dev, const char *path, int x86)
{
struct stat st;
VasEBoot_err_t err;
VasEBoot_util_info ("blessing %s", path);
if (stat (path, &st) < 0)
VasEBoot_util_error (N_("cannot stat `%s': %s"),
path, strerror (errno));
err = VasEBoot_mac_bless_inode (dev, st.st_ino, S_ISDIR (st.st_mode), x86);
if (err)
VasEBoot_util_error ("%s", VasEBoot_errmsg);
VasEBoot_util_info ("blessed");
}
static void
fill_core_services (const char *core_services)
{
char *label;
FILE *f;
char *label_text;
char *label_string = xasprintf ("%s %s", bootloader_id, product_version);
char *sysv_plist;
label = VasEBoot_util_path_concat (2, core_services, ".disk_label");
VasEBoot_util_info ("rendering label %s", label_string);
VasEBoot_util_render_label (label_font, label_bgcolor ? : "white",
label_color ? : "black", label_string, label);
VasEBoot_util_info ("label rendered");
free (label);
label_text = VasEBoot_util_path_concat (2, core_services, ".disk_label.contentDetails");
f = VasEBoot_util_fopen (label_text, "wb");
fprintf (f, "%s\n", label_string);
fclose (f);
free (label_string);
free (label_text);
sysv_plist = VasEBoot_util_path_concat (2, core_services, "SystemVersion.plist");
f = VasEBoot_util_fopen (sysv_plist, "wb");
fprintf (f,
"<plist version=\"1.0\">\n"
"<dict>\n"
" <key>ProductBuildVersion</key>\n"
" <string></string>\n"
" <key>ProductName</key>\n"
" <string>%s</string>\n"
" <key>ProductVersion</key>\n"
" <string>%s</string>\n"
"</dict>\n"
"</plist>\n", bootloader_id, product_version);
fclose (f);
free (sysv_plist);
}
#ifdef __linux__
static void
try_open (const char *path)
{
FILE *f;
f = VasEBoot_util_fopen (path, "r+");
if (f == NULL)
VasEBoot_util_error (_("Unable to open %s: %s"), path, strerror (errno));
fclose (f);
}
#endif
int
main (int argc, char *argv[])
{
int is_efi = 0;
const char *efi_distributor = NULL;
const char *efi_file = NULL;
char **VasEBoot_devices;
VasEBoot_fs_t VasEBoot_fs;
VasEBoot_device_t VasEBoot_dev = NULL;
enum VasEBoot_install_plat platform;
char *VasEBootdir, *device_map;
char **curdev, **curdrive;
char **VasEBoot_drives;
char *relative_VasEBootdir;
char **efidir_device_names = NULL;
VasEBoot_device_t efidir_VasEBoot_dev = NULL;
char *efidir_VasEBoot_devname;
int efidir_is_mac = 0;
int is_prep = 0;
const char *pkgdatadir;
VasEBoot_util_host_init (&argc, &argv);
product_version = xstrdup (PACKAGE_VERSION);
pkgdatadir = VasEBoot_util_get_pkgdatadir ();
label_font = VasEBoot_util_path_concat (2, pkgdatadir, "unicode.pf2");
argp_parse (&argp, argc, argv, 0, 0, 0);
if (verbosity > 1)
VasEBoot_env_set ("debug", "all");
VasEBoot_util_load_config (&config);
if (!bootloader_id && config.VasEBoot_distributor)
{
char *ptr;
bootloader_id = xstrdup (config.VasEBoot_distributor);
for (ptr = bootloader_id; *ptr && *ptr != ' '; ptr++)
if (*ptr >= 'A' && *ptr <= 'Z')
*ptr = *ptr - 'A' + 'a';
*ptr = '\0';
}
if (!bootloader_id || bootloader_id[0] == '\0')
{
free (bootloader_id);
bootloader_id = xstrdup ("VasEBoot");
}
if (!VasEBoot_install_source_directory)
{
if (!target)
{
const char * t;
t = get_default_platform ();
if (!t)
VasEBoot_util_error ("%s",
_("Unable to determine your platform."
" Use --target.")
);
target = xstrdup (t);
}
VasEBoot_install_source_directory
= VasEBoot_util_path_concat (2, VasEBoot_util_get_pkglibdir (), target);
}
platform = VasEBoot_install_get_target (VasEBoot_install_source_directory);
{
char *platname = VasEBoot_install_get_platform_name (platform);
fprintf (stderr, _("Installing for %s platform.\n"), platname);
free (platname);
}
switch (platform)
{
case VAS_EBOOT_INSTALL_PLATFORM_I386_PC:
if (!disk_module)
disk_module = xstrdup ("biosdisk");
break;
case VAS_EBOOT_INSTALL_PLATFORM_I386_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_X86_64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_ARM_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_ARM64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_LOONGARCH64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_RISCV32_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_RISCV64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_IA64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_I386_IEEE1275:
case VAS_EBOOT_INSTALL_PLATFORM_SPARC64_IEEE1275:
case VAS_EBOOT_INSTALL_PLATFORM_POWERPC_IEEE1275:
case VAS_EBOOT_INSTALL_PLATFORM_MIPSEL_ARC:
case VAS_EBOOT_INSTALL_PLATFORM_MIPS_ARC:
case VAS_EBOOT_INSTALL_PLATFORM_ARM_UBOOT:
case VAS_EBOOT_INSTALL_PLATFORM_I386_XEN:
case VAS_EBOOT_INSTALL_PLATFORM_X86_64_XEN:
case VAS_EBOOT_INSTALL_PLATFORM_I386_XEN_PVH:
break;
case VAS_EBOOT_INSTALL_PLATFORM_I386_QEMU:
case VAS_EBOOT_INSTALL_PLATFORM_I386_COREBOOT:
case VAS_EBOOT_INSTALL_PLATFORM_ARM_COREBOOT:
case VAS_EBOOT_INSTALL_PLATFORM_I386_MULTIBOOT:
case VAS_EBOOT_INSTALL_PLATFORM_MIPSEL_LOONGSON:
case VAS_EBOOT_INSTALL_PLATFORM_MIPSEL_QEMU_MIPS:
case VAS_EBOOT_INSTALL_PLATFORM_MIPS_QEMU_MIPS:
disk_module = xstrdup ("native");
break;
/* pacify warning. */
case VAS_EBOOT_INSTALL_PLATFORM_MAX:
break;
}
switch (platform)
{
case VAS_EBOOT_INSTALL_PLATFORM_I386_PC:
case VAS_EBOOT_INSTALL_PLATFORM_SPARC64_IEEE1275:
if (!install_device)
VasEBoot_util_error ("%s", _("install device isn't specified"));
break;
case VAS_EBOOT_INSTALL_PLATFORM_POWERPC_IEEE1275:
if (install_device)
is_prep = 1;
break;
case VAS_EBOOT_INSTALL_PLATFORM_MIPS_ARC:
case VAS_EBOOT_INSTALL_PLATFORM_MIPSEL_ARC:
break;
case VAS_EBOOT_INSTALL_PLATFORM_I386_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_X86_64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_ARM_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_ARM64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_LOONGARCH64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_RISCV32_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_RISCV64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_IA64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_I386_IEEE1275:
case VAS_EBOOT_INSTALL_PLATFORM_ARM_UBOOT:
case VAS_EBOOT_INSTALL_PLATFORM_I386_QEMU:
case VAS_EBOOT_INSTALL_PLATFORM_I386_COREBOOT:
case VAS_EBOOT_INSTALL_PLATFORM_ARM_COREBOOT:
case VAS_EBOOT_INSTALL_PLATFORM_I386_MULTIBOOT:
case VAS_EBOOT_INSTALL_PLATFORM_MIPSEL_LOONGSON:
case VAS_EBOOT_INSTALL_PLATFORM_MIPSEL_QEMU_MIPS:
case VAS_EBOOT_INSTALL_PLATFORM_MIPS_QEMU_MIPS:
case VAS_EBOOT_INSTALL_PLATFORM_I386_XEN:
case VAS_EBOOT_INSTALL_PLATFORM_X86_64_XEN:
case VAS_EBOOT_INSTALL_PLATFORM_I386_XEN_PVH:
free (install_device);
install_device = NULL;
break;
/* pacify warning. */
case VAS_EBOOT_INSTALL_PLATFORM_MAX:
break;
}
if (!bootdir)
bootdir = VasEBoot_util_path_concat (3, "/", rootdir, VAS_EBOOT_BOOT_DIR_NAME);
{
char * t = VasEBoot_util_path_concat (2, bootdir, VAS_EBOOT_DIR_NAME);
VasEBoot_install_mkdir_p (t);
VasEBootdir = VasEBoot_canonicalize_file_name (t);
if (!VasEBootdir)
VasEBoot_util_error (_("failed to get canonical path of `%s'"), t);
free (t);
}
device_map = VasEBoot_util_path_concat (2, VasEBootdir, "device.map");
if (recheck)
VasEBoot_util_unlink (device_map);
device_map_check_duplicates (device_map);
VasEBoot_util_biosdisk_init (device_map);
/* Initialize all modules. */
VasEBoot_init_all ();
VasEBoot_gcry_init_all ();
VasEBoot_hostfs_init ();
VasEBoot_host_init ();
switch (platform)
{
case VAS_EBOOT_INSTALL_PLATFORM_I386_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_X86_64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_ARM_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_ARM64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_LOONGARCH64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_RISCV32_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_RISCV64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_IA64_EFI:
is_efi = 1;
break;
default:
is_efi = 0;
break;
/* pacify warning. */
case VAS_EBOOT_INSTALL_PLATFORM_MAX:
break;
}
switch (platform)
{
case VAS_EBOOT_INSTALL_PLATFORM_I386_IEEE1275:
case VAS_EBOOT_INSTALL_PLATFORM_POWERPC_IEEE1275:
#ifdef __linux__
/* On Linux, ensure /dev/nvram is _functional_. */
if (update_nvram)
try_open ("/dev/nvram");
#endif
break;
default:
break;
}
/* Find the EFI System Partition. */
if (is_efi)
{
VasEBoot_fs_t fs;
free (install_device);
install_device = NULL;
if (!efidir)
{
char *d = VasEBoot_util_path_concat (2, bootdir, "efi");
char *dr = NULL;
if (!VasEBoot_util_is_directory (d))
{
free (d);
d = VasEBoot_util_path_concat (2, bootdir, "EFI");
}
/*
The EFI System Partition may have been given directly using
--root-directory.
*/
if (!VasEBoot_util_is_directory (d)
&& rootdir && VasEBoot_strcmp (rootdir, "/") != 0)
{
free (d);
d = xstrdup (rootdir);
}
if (VasEBoot_util_is_directory (d))
dr = VasEBoot_make_system_path_relative_to_its_root (d);
/* Is it a mount point? */
if (dr && dr[0] == '\0')
efidir = d;
else
free (d);
free (dr);
}
if (!efidir)
VasEBoot_util_error ("%s", _("cannot find EFI directory"));
efidir_device_names = VasEBoot_guess_root_devices (efidir);
if (!efidir_device_names || !efidir_device_names[0])
VasEBoot_util_error (_("cannot find a device for %s (is /dev mounted?)"),
efidir);
install_device = efidir_device_names[0];
for (curdev = efidir_device_names; *curdev; curdev++)
VasEBoot_util_pull_device (*curdev);
efidir_VasEBoot_devname = VasEBoot_util_get_VasEBoot_dev (efidir_device_names[0]);
if (!efidir_VasEBoot_devname)
VasEBoot_util_error (_("cannot find a VAS_EBOOT drive for %s. Check your device.map"),
efidir_device_names[0]);
efidir_VasEBoot_dev = VasEBoot_device_open (efidir_VasEBoot_devname);
if (! efidir_VasEBoot_dev)
VasEBoot_util_error ("%s", VasEBoot_errmsg);
fs = VasEBoot_fs_probe (efidir_VasEBoot_dev);
if (! fs)
VasEBoot_util_error ("%s", VasEBoot_errmsg);
efidir_is_mac = 0;
if (VasEBoot_strcmp (fs->name, "hfs") == 0
|| VasEBoot_strcmp (fs->name, "hfsplus") == 0)
efidir_is_mac = 1;
if (!efidir_is_mac && VasEBoot_strcmp (fs->name, "fat") != 0)
{
if (force)
VasEBoot_util_warn (_("%s doesn't look like an EFI partition, system may not boot"), efidir);
else
VasEBoot_util_error (_("%s doesn't look like an EFI partition"), efidir);
}
/* The EFI specification requires that an EFI System Partition must
contain an "EFI" subdirectory, and that OS loaders are stored in
subdirectories below EFI. Vendors are expected to pick names that do
not collide with other vendors. To minimise collisions, we use the
name of our distributor if possible.
*/
char *t;
efi_distributor = bootloader_id;
if (removable)
{
/* The specification makes stricter requirements of removable
devices, in order that only one image can be automatically loaded
from them. The image must always reside under /EFI/BOOT, and it
must have a specific file name depending on the architecture.
*/
efi_distributor = "BOOT";
switch (platform)
{
case VAS_EBOOT_INSTALL_PLATFORM_I386_EFI:
efi_file = "BOOTIA32.EFI";
break;
case VAS_EBOOT_INSTALL_PLATFORM_X86_64_EFI:
efi_file = "BOOTX64.EFI";
break;
case VAS_EBOOT_INSTALL_PLATFORM_IA64_EFI:
efi_file = "BOOTIA64.EFI";
break;
case VAS_EBOOT_INSTALL_PLATFORM_ARM_EFI:
efi_file = "BOOTARM.EFI";
break;
case VAS_EBOOT_INSTALL_PLATFORM_ARM64_EFI:
efi_file = "BOOTAA64.EFI";
break;
case VAS_EBOOT_INSTALL_PLATFORM_LOONGARCH64_EFI:
efi_file = "BOOTLOONGARCH64.EFI";
break;
case VAS_EBOOT_INSTALL_PLATFORM_RISCV32_EFI:
efi_file = "BOOTRISCV32.EFI";
break;
case VAS_EBOOT_INSTALL_PLATFORM_RISCV64_EFI:
efi_file = "BOOTRISCV64.EFI";
break;
default:
VasEBoot_util_error ("%s", _("You've found a bug"));
break;
}
}
else
{
/* It is convenient for each architecture to have a different
efi_file, so that different versions can be installed in parallel.
*/
switch (platform)
{
case VAS_EBOOT_INSTALL_PLATFORM_I386_EFI:
efi_file = "VasEBootia32.efi";
break;
case VAS_EBOOT_INSTALL_PLATFORM_X86_64_EFI:
efi_file = "VasEBootx64.efi";
break;
case VAS_EBOOT_INSTALL_PLATFORM_IA64_EFI:
efi_file = "VasEBootia64.efi";
break;
case VAS_EBOOT_INSTALL_PLATFORM_ARM_EFI:
efi_file = "VasEBootarm.efi";
break;
case VAS_EBOOT_INSTALL_PLATFORM_ARM64_EFI:
efi_file = "VasEBootaa64.efi";
break;
case VAS_EBOOT_INSTALL_PLATFORM_LOONGARCH64_EFI:
efi_file = "VasEBootloongarch64.efi";
break;
case VAS_EBOOT_INSTALL_PLATFORM_RISCV32_EFI:
efi_file = "VasEBootriscv32.efi";
break;
case VAS_EBOOT_INSTALL_PLATFORM_RISCV64_EFI:
efi_file = "VasEBootriscv64.efi";
break;
default:
efi_file = "VasEBoot.efi";
break;
}
}
t = VasEBoot_util_path_concat (3, efidir, "EFI", efi_distributor);
free (efidir);
efidir = t;
VasEBoot_install_mkdir_p (efidir);
}
if (platform == VAS_EBOOT_INSTALL_PLATFORM_POWERPC_IEEE1275)
{
int is_guess = 0;
if (!macppcdir)
{
char *d;
is_guess = 1;
d = VasEBoot_util_path_concat (2, bootdir, "macppc");
if (!VasEBoot_util_is_directory (d))
{
free (d);
d = VasEBoot_util_path_concat (2, bootdir, "efi");
}
/* Find the Mac HFS(+) System Partition. */
if (!VasEBoot_util_is_directory (d))
{
free (d);
d = VasEBoot_util_path_concat (2, bootdir, "EFI");
}
if (!VasEBoot_util_is_directory (d))
{
free (d);
d = 0;
}
if (d)
macppcdir = d;
}
if (macppcdir)
{
char **macppcdir_device_names = NULL;
VasEBoot_device_t macppcdir_VasEBoot_dev = NULL;
char *macppcdir_VasEBoot_devname;
VasEBoot_fs_t fs;
macppcdir_device_names = VasEBoot_guess_root_devices (macppcdir);
if (!macppcdir_device_names || !macppcdir_device_names[0])
VasEBoot_util_error (_("cannot find a device for %s (is /dev mounted?)"),
macppcdir);
for (curdev = macppcdir_device_names; *curdev; curdev++)
VasEBoot_util_pull_device (*curdev);
macppcdir_VasEBoot_devname = VasEBoot_util_get_VasEBoot_dev (macppcdir_device_names[0]);
if (!macppcdir_VasEBoot_devname)
VasEBoot_util_error (_("cannot find a VAS_EBOOT drive for %s. Check your device.map"),
macppcdir_device_names[0]);
macppcdir_VasEBoot_dev = VasEBoot_device_open (macppcdir_VasEBoot_devname);
if (! macppcdir_VasEBoot_dev)
VasEBoot_util_error ("%s", VasEBoot_errmsg);
fs = VasEBoot_fs_probe (macppcdir_VasEBoot_dev);
if (! fs)
VasEBoot_util_error ("%s", VasEBoot_errmsg);
if (VasEBoot_strcmp (fs->name, "hfs") != 0
&& VasEBoot_strcmp (fs->name, "hfsplus") != 0
&& !is_guess)
VasEBoot_util_error (_("filesystem on %s is neither HFS nor HFS+"),
macppcdir);
if (VasEBoot_strcmp (fs->name, "hfs") == 0
|| VasEBoot_strcmp (fs->name, "hfsplus") == 0)
{
install_device = macppcdir_device_names[0];
is_prep = 0;
}
}
}
size_t ndev = 0;
/* Write device to a variable so we don't have to traverse /dev every time. */
VasEBoot_devices = VasEBoot_guess_root_devices (VasEBootdir);
if (!VasEBoot_devices || !VasEBoot_devices[0])
VasEBoot_util_error (_("cannot find a device for %s (is /dev mounted?)"),
VasEBootdir);
for (curdev = VasEBoot_devices; *curdev; curdev++)
{
VasEBoot_util_pull_device (*curdev);
ndev++;
}
VasEBoot_drives = xcalloc (ndev + 1, sizeof (VasEBoot_drives[0]));
for (curdev = VasEBoot_devices, curdrive = VasEBoot_drives; *curdev; curdev++,
curdrive++)
{
*curdrive = VasEBoot_util_get_VasEBoot_dev (*curdev);
if (! *curdrive)
VasEBoot_util_error (_("cannot find a VAS_EBOOT drive for %s. Check your device.map"),
*curdev);
}
*curdrive = 0;
VasEBoot_dev = VasEBoot_device_open (VasEBoot_drives[0]);
if (! VasEBoot_dev)
VasEBoot_util_error ("%s", VasEBoot_errmsg);
VasEBoot_fs = VasEBoot_fs_probe (VasEBoot_dev);
if (! VasEBoot_fs)
VasEBoot_util_error ("%s", VasEBoot_errmsg);
VasEBoot_install_push_module (VasEBoot_fs->name);
if (VasEBoot_dev->disk)
probe_mods (VasEBoot_dev->disk);
for (curdrive = VasEBoot_drives + 1; *curdrive; curdrive++)
{
VasEBoot_device_t dev = VasEBoot_device_open (*curdrive);
if (!dev)
continue;
if (dev->disk)
probe_mods (dev->disk);
VasEBoot_device_close (dev);
}
if (!config.is_cryptodisk_enabled && have_cryptodisk)
VasEBoot_util_error (_("attempt to install to encrypted disk without cryptodisk enabled. "
"Set `%s' in file `%s'"), "VAS_EBOOT_ENABLE_CRYPTODISK=y",
VasEBoot_util_get_config_filename ());
if (disk_module && VasEBoot_strcmp (disk_module, "ata") == 0)
VasEBoot_install_push_module ("pata");
else if (disk_module && VasEBoot_strcmp (disk_module, "native") == 0)
{
VasEBoot_install_push_module ("pata");
VasEBoot_install_push_module ("ahci");
VasEBoot_install_push_module ("ohci");
VasEBoot_install_push_module ("uhci");
VasEBoot_install_push_module ("ehci");
VasEBoot_install_push_module ("usbms");
}
else if (disk_module && disk_module[0])
VasEBoot_install_push_module (disk_module);
relative_VasEBootdir = VasEBoot_make_system_path_relative_to_its_root (VasEBootdir);
if (relative_VasEBootdir[0] == '\0')
{
free (relative_VasEBootdir);
relative_VasEBootdir = xstrdup ("/");
}
char *prefix_drive = NULL;
char *install_drive = NULL;
if (install_device)
{
if (install_device[0] == '('
&& install_device[VasEBoot_strlen (install_device) - 1] == ')')
{
size_t len = VasEBoot_strlen (install_device) - 2;
install_drive = xmalloc (len + 1);
memcpy (install_drive, install_device + 1, len);
install_drive[len] = '\0';
}
else
{
VasEBoot_util_pull_device (install_device);
install_drive = VasEBoot_util_get_VasEBoot_dev (install_device);
if (!install_drive)
VasEBoot_util_error (_("cannot find a VAS_EBOOT drive for %s. Check your device.map"),
install_device);
}
}
VasEBoot_install_copy_files (VasEBoot_install_source_directory,
VasEBootdir, platform);
char *envfile = VasEBoot_util_path_concat (2, VasEBootdir, "VasEBootenv");
if (!VasEBoot_util_is_regular (envfile))
VasEBoot_util_create_envblk_file (envfile);
char *platname = VasEBoot_install_get_platform_name (platform);
char *platdir;
{
char *t = VasEBoot_util_path_concat (2, VasEBootdir,
platname);
platdir = VasEBoot_canonicalize_file_name (t);
if (!platdir)
VasEBoot_util_error (_("failed to get canonical path of `%s'"),
t);
free (t);
}
load_cfg = VasEBoot_util_path_concat (2, platdir,
"load.cfg");
VasEBoot_util_unlink (load_cfg);
if (debug_image && debug_image[0])
{
load_cfg_f = VasEBoot_util_fopen (load_cfg, "wb");
have_load_cfg = 1;
fprintf (load_cfg_f, "set debug='%s'\n",
debug_image);
}
if (!have_abstractions)
{
if ((disk_module && VasEBoot_strcmp (disk_module, "biosdisk") != 0)
|| VasEBoot_drives[1]
|| (!install_drive
&& platform != VAS_EBOOT_INSTALL_PLATFORM_POWERPC_IEEE1275)
|| (install_drive && !is_same_disk (VasEBoot_drives[0], install_drive))
|| !have_bootdev (platform))
{
char *uuid = NULL;
/* generic method (used on coreboot and ata mod). */
if (!force_file_id
&& VasEBoot_fs->fs_uuid && VasEBoot_fs->fs_uuid (VasEBoot_dev, &uuid))
{
VasEBoot_print_error ();
VasEBoot_errno = 0;
uuid = NULL;
}
if (!load_cfg_f)
load_cfg_f = VasEBoot_util_fopen (load_cfg, "wb");
have_load_cfg = 1;
if (uuid)
{
fprintf (load_cfg_f, "search.fs_uuid %s root ",
uuid);
VasEBoot_install_push_module ("search_fs_uuid");
}
else
{
char *rndstr = get_rndstr ();
char *fl = VasEBoot_util_path_concat (3, VasEBootdir,
"uuid", rndstr);
char *fldir = VasEBoot_util_path_concat (2, VasEBootdir,
"uuid");
char *relfl;
FILE *flf;
VasEBoot_install_mkdir_p (fldir);
flf = VasEBoot_util_fopen (fl, "w");
if (!flf)
VasEBoot_util_error (_("Can't create file: %s"), strerror (errno));
fclose (flf);
relfl = VasEBoot_make_system_path_relative_to_its_root (fl);
fprintf (load_cfg_f, "search.file %s root ",
relfl);
VasEBoot_install_push_module ("search_fs_file");
}
for (curdev = VasEBoot_devices, curdrive = VasEBoot_drives; *curdev; curdev++,
curdrive++)
{
const char *map;
char *g = NULL;
VasEBoot_device_t dev;
if (curdrive == VasEBoot_drives)
dev = VasEBoot_dev;
else
dev = VasEBoot_device_open (*curdrive);
if (!dev)
continue;
if (dev->disk->dev->id != VAS_EBOOT_DISK_DEVICE_HOSTDISK_ID)
{
VasEBoot_util_fprint_full_disk_name (load_cfg_f,
dev->disk->name,
dev);
fprintf (load_cfg_f, " ");
if (dev != VasEBoot_dev)
VasEBoot_device_close (dev);
continue;
}
map = VasEBoot_util_biosdisk_get_compatibility_hint (dev->disk);
if (map)
{
VasEBoot_util_fprint_full_disk_name (load_cfg_f, map, dev);
fprintf (load_cfg_f, " ");
}
if (disk_module && disk_module[0]
&& VasEBoot_strcmp (disk_module, "biosdisk") != 0)
g = VasEBoot_util_guess_baremetal_drive (*curdev);
else
switch (platform)
{
case VAS_EBOOT_INSTALL_PLATFORM_I386_PC:
g = VasEBoot_util_guess_bios_drive (*curdev);
break;
case VAS_EBOOT_INSTALL_PLATFORM_I386_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_X86_64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_ARM_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_ARM64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_LOONGARCH64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_RISCV32_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_RISCV64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_IA64_EFI:
g = VasEBoot_util_guess_efi_drive (*curdev);
break;
case VAS_EBOOT_INSTALL_PLATFORM_SPARC64_IEEE1275:
case VAS_EBOOT_INSTALL_PLATFORM_POWERPC_IEEE1275:
case VAS_EBOOT_INSTALL_PLATFORM_I386_IEEE1275:
{
const char * ofpath = VasEBoot_util_devname_to_ofpath (*curdev);
g = xasprintf ("ieee1275/%s", ofpath);
break;
}
case VAS_EBOOT_INSTALL_PLATFORM_MIPSEL_LOONGSON:
case VAS_EBOOT_INSTALL_PLATFORM_I386_QEMU:
case VAS_EBOOT_INSTALL_PLATFORM_I386_COREBOOT:
case VAS_EBOOT_INSTALL_PLATFORM_ARM_COREBOOT:
case VAS_EBOOT_INSTALL_PLATFORM_I386_MULTIBOOT:
case VAS_EBOOT_INSTALL_PLATFORM_MIPSEL_QEMU_MIPS:
case VAS_EBOOT_INSTALL_PLATFORM_MIPS_QEMU_MIPS:
g = VasEBoot_util_guess_baremetal_drive (*curdev);
break;
case VAS_EBOOT_INSTALL_PLATFORM_MIPS_ARC:
case VAS_EBOOT_INSTALL_PLATFORM_MIPSEL_ARC:
case VAS_EBOOT_INSTALL_PLATFORM_ARM_UBOOT:
case VAS_EBOOT_INSTALL_PLATFORM_I386_XEN:
case VAS_EBOOT_INSTALL_PLATFORM_X86_64_XEN:
case VAS_EBOOT_INSTALL_PLATFORM_I386_XEN_PVH:
VasEBoot_util_warn ("%s", _("no hints available for your platform. Expect reduced performance"));
break;
/* pacify warning. */
case VAS_EBOOT_INSTALL_PLATFORM_MAX:
break;
}
if (g)
{
VasEBoot_util_fprint_full_disk_name (load_cfg_f, g, dev);
fprintf (load_cfg_f, " ");
free (g);
}
if (dev != VasEBoot_dev)
VasEBoot_device_close (dev);
}
fprintf (load_cfg_f, "\n");
char *escaped_relpath = escape (relative_VasEBootdir);
fprintf (load_cfg_f, "set prefix=($root)'%s'\n",
escaped_relpath);
}
else
{
/* We need to hardcode the partition number in the core image's prefix. */
char *p;
for (p = VasEBoot_drives[0]; *p; )
{
if (*p == '\\' && p[1])
{
p += 2;
continue;
}
if (*p == ',' || *p == '\0')
break;
p++;
}
prefix_drive = xasprintf ("(%s)", p);
}
}
else
{
if (config.is_cryptodisk_enabled)
{
if (VasEBoot_dev->disk)
probe_cryptodisk_uuid (VasEBoot_dev->disk);
for (curdrive = VasEBoot_drives + 1; *curdrive; curdrive++)
{
VasEBoot_device_t dev = VasEBoot_device_open (*curdrive);
if (!dev)
continue;
if (dev->disk)
probe_cryptodisk_uuid (dev->disk);
VasEBoot_device_close (dev);
}
}
prefix_drive = xasprintf ("(%s)", VasEBoot_drives[0]);
}
char mkimage_target[200];
const char *core_name = NULL;
switch (platform)
{
case VAS_EBOOT_INSTALL_PLATFORM_I386_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_X86_64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_ARM_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_ARM64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_LOONGARCH64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_RISCV32_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_RISCV64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_IA64_EFI:
core_name = "core.efi";
snprintf (mkimage_target, sizeof (mkimage_target),
"%s-%s",
VasEBoot_install_get_platform_cpu (platform),
VasEBoot_install_get_platform_platform (platform));
break;
case VAS_EBOOT_INSTALL_PLATFORM_MIPSEL_LOONGSON:
case VAS_EBOOT_INSTALL_PLATFORM_MIPSEL_QEMU_MIPS:
case VAS_EBOOT_INSTALL_PLATFORM_MIPS_QEMU_MIPS:
core_name = "core.elf";
snprintf (mkimage_target, sizeof (mkimage_target),
"%s-%s-elf",
VasEBoot_install_get_platform_cpu (platform),
VasEBoot_install_get_platform_platform (platform));
break;
case VAS_EBOOT_INSTALL_PLATFORM_I386_COREBOOT:
case VAS_EBOOT_INSTALL_PLATFORM_ARM_COREBOOT:
case VAS_EBOOT_INSTALL_PLATFORM_I386_MULTIBOOT:
case VAS_EBOOT_INSTALL_PLATFORM_I386_IEEE1275:
case VAS_EBOOT_INSTALL_PLATFORM_POWERPC_IEEE1275:
case VAS_EBOOT_INSTALL_PLATFORM_I386_XEN:
case VAS_EBOOT_INSTALL_PLATFORM_X86_64_XEN:
case VAS_EBOOT_INSTALL_PLATFORM_I386_XEN_PVH:
core_name = "core.elf";
snprintf (mkimage_target, sizeof (mkimage_target),
"%s-%s",
VasEBoot_install_get_platform_cpu (platform),
VasEBoot_install_get_platform_platform (platform));
break;
case VAS_EBOOT_INSTALL_PLATFORM_I386_PC:
case VAS_EBOOT_INSTALL_PLATFORM_MIPSEL_ARC:
case VAS_EBOOT_INSTALL_PLATFORM_MIPS_ARC:
case VAS_EBOOT_INSTALL_PLATFORM_ARM_UBOOT:
case VAS_EBOOT_INSTALL_PLATFORM_I386_QEMU:
snprintf (mkimage_target, sizeof (mkimage_target),
"%s-%s",
VasEBoot_install_get_platform_cpu (platform),
VasEBoot_install_get_platform_platform (platform));
core_name = "core.img";
break;
case VAS_EBOOT_INSTALL_PLATFORM_SPARC64_IEEE1275:
strcpy (mkimage_target, "sparc64-ieee1275-raw");
core_name = "core.img";
break;
/* pacify warning. */
case VAS_EBOOT_INSTALL_PLATFORM_MAX:
break;
}
if (!core_name)
VasEBoot_util_error ("%s", _("You've found a bug"));
if (load_cfg_f)
fclose (load_cfg_f);
char *imgfile = VasEBoot_util_path_concat (2, platdir,
core_name);
char *prefix = xasprintf ("%s%s", prefix_drive ? : "",
relative_VasEBootdir);
VasEBoot_install_make_image_wrap (/* source dir */ VasEBoot_install_source_directory,
/*prefix */ prefix,
/* output */ imgfile,
/* memdisk */ NULL,
have_load_cfg ? load_cfg : NULL,
/* image target */ mkimage_target, 0);
/* Backward-compatibility kludges. */
switch (platform)
{
case VAS_EBOOT_INSTALL_PLATFORM_MIPSEL_LOONGSON:
{
char *dst = VasEBoot_util_path_concat (2, bootdir, "VasEBoot.elf");
VasEBoot_install_copy_file (imgfile, dst, 1);
free (dst);
}
break;
case VAS_EBOOT_INSTALL_PLATFORM_I386_IEEE1275:
case VAS_EBOOT_INSTALL_PLATFORM_POWERPC_IEEE1275:
{
char *dst = VasEBoot_util_path_concat (2, VasEBootdir, "VasEBoot");
VasEBoot_install_copy_file (imgfile, dst, 1);
free (dst);
}
break;
case VAS_EBOOT_INSTALL_PLATFORM_I386_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_X86_64_EFI:
{
char *dst = VasEBoot_util_path_concat (2, platdir, "VasEBoot.efi");
VasEBoot_install_make_image_wrap (/* source dir */ VasEBoot_install_source_directory,
/* prefix */ "",
/* output */ dst,
/* memdisk */ NULL,
have_load_cfg ? load_cfg : NULL,
/* image target */ mkimage_target, 0);
}
break;
case VAS_EBOOT_INSTALL_PLATFORM_ARM_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_ARM64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_LOONGARCH64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_RISCV32_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_RISCV64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_IA64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_MIPSEL_QEMU_MIPS:
case VAS_EBOOT_INSTALL_PLATFORM_MIPS_QEMU_MIPS:
case VAS_EBOOT_INSTALL_PLATFORM_I386_COREBOOT:
case VAS_EBOOT_INSTALL_PLATFORM_ARM_COREBOOT:
case VAS_EBOOT_INSTALL_PLATFORM_I386_MULTIBOOT:
case VAS_EBOOT_INSTALL_PLATFORM_I386_PC:
case VAS_EBOOT_INSTALL_PLATFORM_MIPSEL_ARC:
case VAS_EBOOT_INSTALL_PLATFORM_MIPS_ARC:
case VAS_EBOOT_INSTALL_PLATFORM_ARM_UBOOT:
case VAS_EBOOT_INSTALL_PLATFORM_I386_QEMU:
case VAS_EBOOT_INSTALL_PLATFORM_SPARC64_IEEE1275:
case VAS_EBOOT_INSTALL_PLATFORM_I386_XEN:
case VAS_EBOOT_INSTALL_PLATFORM_X86_64_XEN:
case VAS_EBOOT_INSTALL_PLATFORM_I386_XEN_PVH:
break;
/* pacify warning. */
case VAS_EBOOT_INSTALL_PLATFORM_MAX:
break;
}
/* Perform the platform-dependent install */
switch (platform)
{
case VAS_EBOOT_INSTALL_PLATFORM_I386_PC:
{
char *boot_img_src = VasEBoot_util_path_concat (2,
VasEBoot_install_source_directory,
"boot.img");
char *boot_img = VasEBoot_util_path_concat (2, platdir,
"boot.img");
VasEBoot_install_copy_file (boot_img_src, boot_img, 1);
VasEBoot_util_info ("%sVasEBoot-bios-setup %s %s %s %s %s --directory='%s' --device-map='%s' '%s'",
/* TRANSLATORS: This is a prefix in the log to indicate that usually
a command would be executed but due to an option was skipped. */
install_bootsector ? "" : _("NOT RUNNING: "),
allow_floppy ? "--allow-floppy " : "",
verbosity ? "--verbose " : "",
force ? "--force " : "",
!fs_probe ? "--skip-fs-probe" : "",
!add_rs_codes ? "--no-rs-codes" : "",
platdir,
device_map,
install_device);
/* Now perform the installation. */
if (install_bootsector)
{
VasEBoot_util_bios_setup (platdir, "boot.img", "core.img",
install_drive, force,
fs_probe, allow_floppy, add_rs_codes,
!VasEBoot_install_is_short_mbrgap_supported ());
VasEBoot_set_install_backup_ponr ();
}
break;
}
case VAS_EBOOT_INSTALL_PLATFORM_SPARC64_IEEE1275:
{
char *boot_img_src = VasEBoot_util_path_concat (2,
VasEBoot_install_source_directory,
"boot.img");
char *boot_img = VasEBoot_util_path_concat (2, platdir,
"boot.img");
VasEBoot_install_copy_file (boot_img_src, boot_img, 1);
VasEBoot_util_info ("%sVasEBoot-sparc64-setup %s %s %s %s --directory='%s' --device-map='%s' '%s'",
install_bootsector ? "" : "NOT RUNNING: ",
allow_floppy ? "--allow-floppy " : "",
verbosity ? "--verbose " : "",
force ? "--force " : "",
!fs_probe ? "--skip-fs-probe" : "",
platdir,
device_map,
install_drive);
/* Now perform the installation. */
if (install_bootsector)
{
VasEBoot_util_sparc_setup (platdir, "boot.img", "core.img",
install_drive, force,
fs_probe, allow_floppy,
0 /* unused */, 0 /* unused */ );
VasEBoot_set_install_backup_ponr ();
}
break;
}
case VAS_EBOOT_INSTALL_PLATFORM_POWERPC_IEEE1275:
if (macppcdir)
{
char *core_services = VasEBoot_util_path_concat (4, macppcdir,
"System", "Library",
"CoreServices");
char *mach_kernel = VasEBoot_util_path_concat (2, macppcdir,
"mach_kernel");
char *VasEBoot_elf, *bootx;
FILE *f;
VasEBoot_device_t ins_dev;
char *VasEBoot_chrp = VasEBoot_util_path_concat (2,
VasEBoot_install_source_directory,
"VasEBoot.chrp");
VasEBoot_install_mkdir_p (core_services);
bootx = VasEBoot_util_path_concat (2, core_services, "BootX");
VasEBoot_install_copy_file (VasEBoot_chrp, bootx, 1);
VasEBoot_elf = VasEBoot_util_path_concat (2, core_services, "VasEBoot.elf");
VasEBoot_install_copy_file (imgfile, VasEBoot_elf, 1);
VasEBoot_set_install_backup_ponr ();
f = VasEBoot_util_fopen (mach_kernel, "a+");
if (!f)
VasEBoot_util_error (_("Can't create file: %s"), strerror (errno));
fclose (f);
fill_core_services (core_services);
ins_dev = VasEBoot_device_open (install_drive);
if (ins_dev == NULL)
VasEBoot_util_error ("%s", VasEBoot_errmsg);
bless (ins_dev, core_services, 0);
if (update_nvram)
{
const char *dev;
int partno;
partno = ins_dev->disk->partition
? ins_dev->disk->partition->number + 1 : 0;
dev = VasEBoot_util_get_os_disk (install_device);
VasEBoot_install_register_ieee1275 (0, dev, partno,
"\\\\BootX");
}
VasEBoot_device_close (ins_dev);
free (VasEBoot_elf);
free (bootx);
free (mach_kernel);
free (VasEBoot_chrp);
break;
}
/* If a install device is defined, copy the core.elf to PReP partition. */
if (is_prep && install_device && install_device[0])
{
VasEBoot_device_t ins_dev;
ins_dev = VasEBoot_device_open (install_drive);
if (!ins_dev || !is_prep_partition (ins_dev))
{
VasEBoot_util_error ("%s", _("the chosen partition is not a PReP partition"));
}
if (is_prep_empty (ins_dev))
{
if (write_to_disk (ins_dev, imgfile))
VasEBoot_util_error ("%s", _("failed to copy VasEBoot to the PReP partition"));
VasEBoot_set_install_backup_ponr ();
}
else
{
char *s = xasprintf ("dd if=/dev/zero of=%s", install_device);
VasEBoot_util_error (_("the PReP partition is not empty. If you are sure you want to use it, run dd to clear it: `%s'"),
s);
}
VasEBoot_device_close (ins_dev);
if (update_nvram)
VasEBoot_install_register_ieee1275 (1, VasEBoot_util_get_os_disk (install_device),
0, NULL);
break;
}
/* fallthrough. */
case VAS_EBOOT_INSTALL_PLATFORM_I386_IEEE1275:
if (update_nvram)
{
const char *dev;
char *relpath;
int partno;
relpath = VasEBoot_make_system_path_relative_to_its_root (imgfile);
partno = VasEBoot_dev->disk->partition
? VasEBoot_dev->disk->partition->number + 1 : 0;
dev = VasEBoot_util_get_os_disk (VasEBoot_devices[0]);
VasEBoot_install_register_ieee1275 (0, dev,
partno, relpath);
}
break;
case VAS_EBOOT_INSTALL_PLATFORM_MIPS_ARC:
VasEBoot_install_sgi_setup (install_device, imgfile, "VasEBoot");
break;
case VAS_EBOOT_INSTALL_PLATFORM_I386_EFI:
if (!efidir_is_mac)
{
char *dst = VasEBoot_util_path_concat (2, efidir, "VasEBoot.efi");
/* For old macs. Suggested by Peter Jones. */
VasEBoot_install_copy_file (imgfile, dst, 1);
free (dst);
}
/* Fallthrough. */
case VAS_EBOOT_INSTALL_PLATFORM_X86_64_EFI:
if (efidir_is_mac)
{
char *boot_efi;
char *core_services = VasEBoot_util_path_concat (4, efidir,
"System", "Library",
"CoreServices");
char *mach_kernel = VasEBoot_util_path_concat (2, efidir,
"mach_kernel");
FILE *f;
VasEBoot_device_t ins_dev;
VasEBoot_install_mkdir_p (core_services);
boot_efi = VasEBoot_util_path_concat (2, core_services, "boot.efi");
VasEBoot_install_copy_file (imgfile, boot_efi, 1);
VasEBoot_set_install_backup_ponr ();
f = VasEBoot_util_fopen (mach_kernel, "r+");
if (!f)
VasEBoot_util_error (_("Can't create file: %s"), strerror (errno));
fclose (f);
fill_core_services(core_services);
ins_dev = VasEBoot_device_open (install_drive);
if (ins_dev == NULL)
VasEBoot_util_error ("%s", VasEBoot_errmsg);
bless (ins_dev, boot_efi, 1);
if (!removable && update_nvram)
{
/* Try to make this image bootable using the EFI Boot Manager, if available. */
int ret;
ret = VasEBoot_install_register_efi (efidir_VasEBoot_dev,
"\\System\\Library\\CoreServices",
efi_distributor);
if (ret)
VasEBoot_util_error (_("efibootmgr failed to register the boot entry: %s"),
strerror (ret));
}
VasEBoot_device_close (ins_dev);
free (boot_efi);
free (mach_kernel);
break;
}
/* FALLTHROUGH */
case VAS_EBOOT_INSTALL_PLATFORM_ARM_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_ARM64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_LOONGARCH64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_RISCV32_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_RISCV64_EFI:
case VAS_EBOOT_INSTALL_PLATFORM_IA64_EFI:
{
char *dst = VasEBoot_util_path_concat (2, efidir, efi_file);
VasEBoot_install_copy_file (imgfile, dst, 1);
VasEBoot_set_install_backup_ponr ();
free (dst);
}
if (!removable && update_nvram)
{
char * efifile_path;
char * part;
int ret;
/* Try to make this image bootable using the EFI Boot Manager, if available. */
if (!efi_distributor || efi_distributor[0] == '\0')
VasEBoot_util_error ("%s", _("EFI bootloader id isn't specified."));
efifile_path = xasprintf ("\\EFI\\%s\\%s",
efi_distributor,
efi_file);
part = (efidir_VasEBoot_dev->disk->partition
? VasEBoot_partition_get_name (efidir_VasEBoot_dev->disk->partition)
: 0);
VasEBoot_util_info ("Registering with EFI: distributor = `%s',"
" path = `%s', ESP at %s%s%s",
efi_distributor, efifile_path,
efidir_VasEBoot_dev->disk->name,
(part ? ",": ""), (part ? : ""));
VasEBoot_free (part);
ret = VasEBoot_install_register_efi (efidir_VasEBoot_dev,
efifile_path, efi_distributor);
if (ret)
VasEBoot_util_error (_("efibootmgr failed to register the boot entry: %s"),
strerror (ret));
}
break;
case VAS_EBOOT_INSTALL_PLATFORM_MIPSEL_LOONGSON:
case VAS_EBOOT_INSTALL_PLATFORM_MIPSEL_QEMU_MIPS:
case VAS_EBOOT_INSTALL_PLATFORM_MIPS_QEMU_MIPS:
case VAS_EBOOT_INSTALL_PLATFORM_I386_COREBOOT:
case VAS_EBOOT_INSTALL_PLATFORM_ARM_COREBOOT:
case VAS_EBOOT_INSTALL_PLATFORM_I386_MULTIBOOT:
case VAS_EBOOT_INSTALL_PLATFORM_MIPSEL_ARC:
case VAS_EBOOT_INSTALL_PLATFORM_ARM_UBOOT:
case VAS_EBOOT_INSTALL_PLATFORM_I386_QEMU:
case VAS_EBOOT_INSTALL_PLATFORM_I386_XEN:
case VAS_EBOOT_INSTALL_PLATFORM_X86_64_XEN:
case VAS_EBOOT_INSTALL_PLATFORM_I386_XEN_PVH:
VasEBoot_util_warn ("%s",
_("WARNING: no platform-specific install was performed"));
break;
/* pacify warning. */
case VAS_EBOOT_INSTALL_PLATFORM_MAX:
break;
}
/*
* Either there are no platform specific code, or it didn't raise
* ponr. Raise it here, because usually this is already past point
* of no return. If we leave this flag false, at exit all the modules
* will be removed from the prefix which would be very confusing.
*/
VasEBoot_set_install_backup_ponr ();
fprintf (stderr, "%s\n", _("Installation finished. No error reported."));
/* Free resources. */
VasEBoot_gcry_fini_all ();
VasEBoot_fini_all ();
return 0;
}