Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys/stand/efiboot Add initrd support.
details: https://anonhg.NetBSD.org/src/rev/bb3c5e93e74e
branches: trunk
changeset: 993377:bb3c5e93e74e
user: jmcneill <jmcneill%NetBSD.org@localhost>
date: Fri Sep 07 17:30:58 2018 +0000
description:
Add initrd support.
diffstat:
sys/stand/efiboot/boot.c | 28 +++++++++++++-
sys/stand/efiboot/efiboot.h | 4 +-
sys/stand/efiboot/efifdt.c | 42 ++++++++++++++++++++-
sys/stand/efiboot/efifdt.h | 5 ++-
sys/stand/efiboot/exec.c | 88 +++++++++++++++++++++++++++++++++++++++++++-
5 files changed, 159 insertions(+), 8 deletions(-)
diffs (290 lines):
diff -r c898eacd4fc2 -r bb3c5e93e74e sys/stand/efiboot/boot.c
--- a/sys/stand/efiboot/boot.c Fri Sep 07 17:30:32 2018 +0000
+++ b/sys/stand/efiboot/boot.c Fri Sep 07 17:30:58 2018 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: boot.c,v 1.5 2018/09/03 00:17:00 jmcneill Exp $ */
+/* $NetBSD: boot.c,v 1.6 2018/09/07 17:30:58 jmcneill Exp $ */
/*-
* Copyright (c) 2016 Kimihiro Nonaka <nonaka%netbsd.org@localhost>
@@ -53,17 +53,20 @@
#define DEFTIMEOUT 5
static char default_device[32];
+static char initrd_path[255];
void command_boot(char *);
void command_dev(char *);
+void command_initrd(char *);
void command_ls(char *);
void command_reset(char *);
void command_version(char *);
void command_quit(char *);
const struct boot_command commands[] = {
- { "boot", command_boot, "boot [fsN:][filename] [args]\n (ex. \"fs0:\\netbsd.old -s\"" },
+ { "boot", command_boot, "boot [dev:][filename] [args]\n (ex. \"hd0a:\\netbsd.old -s\"" },
{ "dev", command_dev, "dev" },
+ { "initrd", command_initrd, "initrd [dev:][filename]" },
{ "ls", command_ls, "ls [hdNn:/path]" },
{ "version", command_version, "version" },
{ "help", command_help, "help|?" },
@@ -111,6 +114,12 @@
}
void
+command_initrd(char *arg)
+{
+ set_initrd_path(arg);
+}
+
+void
command_ls(char *arg)
{
ls(arg);
@@ -157,6 +166,21 @@
return default_device;
}
+int
+set_initrd_path(char *arg)
+{
+ if (strlen(arg) + 1 > sizeof(initrd_path))
+ return ERANGE;
+ strcpy(initrd_path, arg);
+ return 0;
+}
+
+char *
+get_initrd_path(void)
+{
+ return initrd_path;
+}
+
void
print_banner(void)
{
diff -r c898eacd4fc2 -r bb3c5e93e74e sys/stand/efiboot/efiboot.h
--- a/sys/stand/efiboot/efiboot.h Fri Sep 07 17:30:32 2018 +0000
+++ b/sys/stand/efiboot/efiboot.h Fri Sep 07 17:30:58 2018 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: efiboot.h,v 1.3 2018/09/03 00:04:02 jmcneill Exp $ */
+/* $NetBSD: efiboot.h,v 1.4 2018/09/07 17:30:58 jmcneill Exp $ */
/*-
* Copyright (c) 2016 Kimihiro Nonaka <nonaka%netbsd.org@localhost>
@@ -52,6 +52,8 @@
void command_help(char *);
int set_default_device(char *);
char *get_default_device(void);
+int set_initrd_path(char *);
+char *get_initrd_path(void);
/* console.c */
int ischar(void);
diff -r c898eacd4fc2 -r bb3c5e93e74e sys/stand/efiboot/efifdt.c
--- a/sys/stand/efiboot/efifdt.c Fri Sep 07 17:30:32 2018 +0000
+++ b/sys/stand/efiboot/efifdt.c Fri Sep 07 17:30:58 2018 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: efifdt.c,v 1.7 2018/09/03 00:17:00 jmcneill Exp $ */
+/* $NetBSD: efifdt.c,v 1.8 2018/09/07 17:30:58 jmcneill Exp $ */
/*-
* Copyright (c) 2018 Jared McNeill <jmcneill%invisible.ca@localhost>
@@ -78,6 +78,28 @@
}
void
+efi_fdt_init(u_long addr, u_long len)
+{
+ int error;
+
+ error = fdt_open_into(fdt_data, (void *)addr, len);
+ if (error < 0)
+ panic("fdt_open_into failed: %d", error);
+
+ fdt_data = (void *)addr;
+}
+
+void
+efi_fdt_fini(void)
+{
+ int error;
+
+ error = fdt_pack(fdt_data);
+ if (error < 0)
+ panic("fdt_pack failed: %d", error);
+}
+
+void
efi_fdt_show(void)
{
const char *model, *compat;
@@ -194,3 +216,21 @@
}
}
}
+
+void
+efi_fdt_initrd(u_long initrd_addr, u_long initrd_size)
+{
+ int chosen;
+
+ if (initrd_size == 0)
+ return;
+
+ chosen = fdt_path_offset(fdt_data, FDT_CHOSEN_NODE_PATH);
+ if (chosen < 0)
+ chosen = fdt_add_subnode(fdt_data, fdt_path_offset(fdt_data, "/"), FDT_CHOSEN_NODE_NAME);
+ if (chosen < 0)
+ panic("FDT: Failed to create " FDT_CHOSEN_NODE_PATH " node");
+
+ fdt_setprop_u64(fdt_data, chosen, "linux,initrd-start", initrd_addr);
+ fdt_setprop_u64(fdt_data, chosen, "linux,initrd-end", initrd_addr + initrd_size);
+}
diff -r c898eacd4fc2 -r bb3c5e93e74e sys/stand/efiboot/efifdt.h
--- a/sys/stand/efiboot/efifdt.h Fri Sep 07 17:30:32 2018 +0000
+++ b/sys/stand/efiboot/efifdt.h Fri Sep 07 17:30:58 2018 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: efifdt.h,v 1.2 2018/09/03 00:17:00 jmcneill Exp $ */
+/* $NetBSD: efifdt.h,v 1.3 2018/09/07 17:30:58 jmcneill Exp $ */
/*-
* Copyright (c) 2018 Jared McNeill <jmcneill%invisible.ca@localhost>
@@ -32,3 +32,6 @@
int efi_fdt_size(void);
void efi_fdt_show(void);
void efi_fdt_bootargs(const char *);
+void efi_fdt_initrd(u_long, u_long);
+void efi_fdt_init(u_long, u_long);
+void efi_fdt_fini(void);
diff -r c898eacd4fc2 -r bb3c5e93e74e sys/stand/efiboot/exec.c
--- a/sys/stand/efiboot/exec.c Fri Sep 07 17:30:32 2018 +0000
+++ b/sys/stand/efiboot/exec.c Fri Sep 07 17:30:58 2018 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: exec.c,v 1.3 2018/09/02 23:50:23 jmcneill Exp $ */
+/* $NetBSD: exec.c,v 1.4 2018/09/07 17:30:58 jmcneill Exp $ */
/*-
* Copyright (c) 2018 Jared McNeill <jmcneill%invisible.ca@localhost>
@@ -33,6 +33,78 @@
u_long load_offset = 0;
+#define FDT_SPACE (4 * 1024 * 1024)
+#define FDT_ALIGN ((2 * 1024 * 1024) - 1)
+
+static EFI_PHYSICAL_ADDRESS initrd_addr;
+static u_long initrd_size = 0;
+
+static int
+load_initrd(void)
+{
+ EFI_STATUS status;
+ struct stat st;
+ ssize_t len;
+ char *path;
+ int fd;
+
+ path = get_initrd_path();
+ if (strlen(path) == 0)
+ return 0;
+
+ fd = open(path, 0);
+ if (fd < 0) {
+ printf("boot: failed to open %s: %s\n", path, strerror(errno));
+ return errno;
+ }
+ if (fstat(fd, &st) < 0) {
+ printf("boot: failed to fstat %s: %s\n", path, strerror(errno));
+ close(fd);
+ return errno;
+ }
+ if (st.st_size == 0) {
+ printf("boot: empty initrd %s\n", path);
+ close(fd);
+ return EINVAL;
+ }
+
+ initrd_size = st.st_size;
+
+#ifdef EFIBOOT_ALLOCATE_MAX_ADDRESS
+ initrd_addr = EFIBOOT_ALLOCATE_MAX_ADDRESS;
+ status = uefi_call_wrapper(BS->AllocatePages, 4, AllocateMaxAddress, EfiLoaderData,
+ EFI_SIZE_TO_PAGES(initrd_size), &initrd_addr);
+#else
+ initrd_addr = 0;
+ status = uefi_call_wrapper(BS->AllocatePages, 4, AllocateAnyPages, EfiLoaderData,
+ EFI_SIZE_TO_PAGES(initrd_size), &initrd_addr);
+#endif
+ if (EFI_ERROR(status)) {
+ printf("Failed to allocate %lu bytes for initrd image (error %lu)\n",
+ initrd_size, status);
+ close(fd);
+ return ENOMEM;
+ }
+
+ printf("boot: loading %s ", path);
+ len = read(fd, (void *)initrd_addr, initrd_size);
+ close(fd);
+
+ if (len != initrd_size) {
+ if (len < 0)
+ printf(": %s\n", strerror(errno));
+ else
+ printf(": returned %ld (expected %ld)\n", len, initrd_size);
+ return EIO;
+ }
+
+ printf("done.\n");
+
+ efi_dcache_flush(initrd_addr, initrd_size);
+
+ return 0;
+}
+
int
exec_netbsd(const char *fname, const char *args)
{
@@ -41,6 +113,8 @@
EFI_STATUS status;
int fd;
+ load_initrd();
+
memset(marks, 0, sizeof(marks));
fd = loadfile(fname, marks, COUNT_KERNEL | LOAD_NOTE);
if (fd < 0) {
@@ -49,7 +123,7 @@
}
close(fd);
marks[MARK_END] = (((u_long) marks[MARK_END] + sizeof(int) - 1)) & (-sizeof(int));
- alloc_size = marks[MARK_END] - marks[MARK_START] + EFIBOOT_ALIGN;
+ alloc_size = marks[MARK_END] - marks[MARK_START] + FDT_SPACE + EFIBOOT_ALIGN;
#ifdef EFIBOOT_ALLOCATE_MAX_ADDRESS
addr = EFIBOOT_ALLOCATE_MAX_ADDRESS;
@@ -77,8 +151,11 @@
load_offset = 0;
if (efi_fdt_size() > 0) {
+ efi_fdt_init((marks[MARK_END] + FDT_ALIGN) & ~FDT_ALIGN, FDT_ALIGN + 1);
+ efi_fdt_initrd(initrd_addr, initrd_size);
efi_fdt_bootargs(args);
- efi_fdt_memory_map();
+ efi_fdt_memory_map();
+ efi_fdt_fini();
}
efi_cleanup();
@@ -89,5 +166,10 @@
cleanup:
uefi_call_wrapper(BS->FreePages, 2, addr, EFI_SIZE_TO_PAGES(alloc_size));
+ if (initrd_addr) {
+ uefi_call_wrapper(BS->FreePages, 2, initrd_addr, EFI_SIZE_TO_PAGES(initrd_size));
+ initrd_addr = 0;
+ initrd_size = 0;
+ }
return EIO;
}
Home |
Main Index |
Thread Index |
Old Index