Patch queue for efi - 2018-07-25

Highlights this time:
 
   - Many small fixes to improve spec compatibility (found by SCT)
   - Almost enough to run with sandbox target
   - GetTime() improvements
   - Enable EFI_LOADER and HYP entry on ARMv7 with NONSEC=y
 -----BEGIN PGP SIGNATURE-----
 Version: GnuPG v2
 
 iQIcBAABAgAGBQJbWHUwAAoJECszeR4D/txgKlUP/3Lu6pxScwzL7TsSD6OFZQ6M
 qQq2FH89XOA6/3r04RXrJZoHwOKIH5uj7ea6FlitpS1sQ4UOCQhp/lJJuJennHHj
 +veeuzI1sKTCX8Kd9ptrZDEF3G8lbF/zSyCFa1MOd1ONDVsTgSO9fOGmiqcC9FBF
 UrUH7dzXlE2CWs/mv/UikCBI7rYF+NOFJNuwHVXfsW4PyQ/7uaNsa7Rl3mXYb/Lr
 gjdcJkeHZAmFv/r84tGS9AFv+m0So9AGEYD7MeQDt02hSOuH9/nu4HgPmiwln3Fn
 3sFA3+daMrlFNi6kFw10S0sjKz94nN+Arm6cIXlvGaoc/wnPM2wEcKOSiXhzVM8d
 CoP/26N/ETRoI9P01C2WyTrKjo0O0aFwp3ubfmmbdcaKr/pyjAJgb3BnqXBfAr7T
 OjsE086jtHzdyKYKXDIz8+ZxSo4VsiDjBzDK7LVA0L5KtqAFFa+OYjlgEMJnQCk3
 YJfj+rhxfpjzFI7x5BAgq0q3XQRvAJS8QcUq+V2todQ3JkUlCIaVUNQLWAfNJN0q
 ze/WR8l4nwj5YFo8XiEbFHpQi/1bkR6cSzyjlBKUqrtHUUEu32vWlzZpNi6HzPMq
 cP7gehboFlcCSr8T2UMjBTE1LWI35eWNQQoiNRrS2UVcTH9h1vXMFV9PT+I82BBl
 ivJ+YwF9nU1JdS8CG3n7
 =ePTT
 -----END PGP SIGNATURE-----

Merge tag 'signed-efi-next' of git://github.com/agraf/u-boot

Patch queue for efi - 2018-07-25

Highlights this time:

  - Many small fixes to improve spec compatibility (found by SCT)
  - Almost enough to run with sandbox target
  - GetTime() improvements
  - Enable EFI_LOADER and HYP entry on ARMv7 with NONSEC=y
This commit is contained in:
Tom Rini 2018-07-30 16:02:29 -04:00
commit 406fd7e207
70 changed files with 1171 additions and 401 deletions

View file

@ -368,6 +368,7 @@ F: doc/README.iscsi
F: include/efi* F: include/efi*
F: include/pe.h F: include/pe.h
F: include/asm-generic/pe.h F: include/asm-generic/pe.h
F: lib/charset.c
F: lib/efi*/ F: lib/efi*/
F: test/py/tests/test_efi* F: test/py/tests/test_efi*
F: cmd/bootefi.c F: cmd/bootefi.c

View file

@ -134,11 +134,11 @@ endif
ifdef CONFIG_ARM64 ifdef CONFIG_ARM64
OBJCOPYFLAGS += -j .text -j .secure_text -j .secure_data -j .rodata -j .data \ OBJCOPYFLAGS += -j .text -j .secure_text -j .secure_data -j .rodata -j .data \
-j .u_boot_list -j .rela.dyn -j .got -j .got.plt \ -j .u_boot_list -j .rela.dyn -j .got -j .got.plt \
-j .binman_sym_table -j .binman_sym_table -j .text_rest
else else
OBJCOPYFLAGS += -j .text -j .secure_text -j .secure_data -j .rodata -j .hash \ OBJCOPYFLAGS += -j .text -j .secure_text -j .secure_data -j .rodata -j .hash \
-j .data -j .got -j .got.plt -j .u_boot_list -j .rel.dyn \ -j .data -j .got -j .got.plt -j .u_boot_list -j .rel.dyn \
-j .binman_sym_table -j .binman_sym_table -j .text_rest
endif endif
# if a dtb section exists we always have to include it # if a dtb section exists we always have to include it

View file

@ -53,7 +53,7 @@ config ARMV7_PSCI_NR_CPUS
config ARMV7_LPAE config ARMV7_LPAE
bool "Use LPAE page table format" if EXPERT bool "Use LPAE page table format" if EXPERT
depends on CPU_V7A depends on CPU_V7A
default n default y if ARMV7_VIRT
---help--- ---help---
Say Y here to use the long descriptor page table format. This is Say Y here to use the long descriptor page table format. This is
required if U-Boot runs in HYP mode. required if U-Boot runs in HYP mode.

View file

@ -80,6 +80,8 @@ _secure_monitor:
#ifdef CONFIG_ARMV7_VIRT #ifdef CONFIG_ARMV7_VIRT
orreq r5, r5, #0x100 @ allow HVC instruction orreq r5, r5, #0x100 @ allow HVC instruction
moveq r6, #HYP_MODE @ Enter the kernel as HYP moveq r6, #HYP_MODE @ Enter the kernel as HYP
mrseq r3, sp_svc
msreq sp_hyp, r3 @ migrate SP
#endif #endif
mcr p15, 0, r5, c1, c1, 0 @ write SCR (with NS bit set) mcr p15, 0, r5, c1, c1, 0 @ write SCR (with NS bit set)

View file

@ -25,6 +25,19 @@ SECTIONS
{ {
*(.__image_copy_start) *(.__image_copy_start)
CPUDIR/start.o (.text*) CPUDIR/start.o (.text*)
}
/* This needs to come before *(.text*) */
.efi_runtime : {
__efi_runtime_start = .;
*(.text.efi_runtime*)
*(.rodata.efi_runtime*)
*(.data.efi_runtime*)
__efi_runtime_stop = .;
}
.text_rest :
{
*(.text*) *(.text*)
} }
@ -98,17 +111,10 @@ SECTIONS
. = ALIGN(8); . = ALIGN(8);
.efi_runtime : {
__efi_runtime_start = .;
*(efi_runtime_text)
*(efi_runtime_data)
__efi_runtime_stop = .;
}
.efi_runtime_rel : { .efi_runtime_rel : {
__efi_runtime_rel_start = .; __efi_runtime_rel_start = .;
*(.relaefi_runtime_text) *(.rel*.efi_runtime)
*(.relaefi_runtime_data) *(.rel*.efi_runtime.*)
__efi_runtime_rel_stop = .; __efi_runtime_rel_stop = .;
} }

View file

@ -43,6 +43,25 @@ SECTIONS
*(.__image_copy_start) *(.__image_copy_start)
*(.vectors) *(.vectors)
CPUDIR/start.o (.text*) CPUDIR/start.o (.text*)
}
/* This needs to come before *(.text*) */
.__efi_runtime_start : {
*(.__efi_runtime_start)
}
.efi_runtime : {
*(.text.efi_runtime*)
*(.rodata.efi_runtime*)
*(.data.efi_runtime*)
}
.__efi_runtime_stop : {
*(.__efi_runtime_stop)
}
.text_rest :
{
*(.text*) *(.text*)
} }
@ -136,27 +155,14 @@ SECTIONS
. = ALIGN(4); . = ALIGN(4);
.__efi_runtime_start : {
*(.__efi_runtime_start)
}
.efi_runtime : {
*(efi_runtime_text)
*(efi_runtime_data)
}
.__efi_runtime_stop : {
*(.__efi_runtime_stop)
}
.efi_runtime_rel_start : .efi_runtime_rel_start :
{ {
*(.__efi_runtime_rel_start) *(.__efi_runtime_rel_start)
} }
.efi_runtime_rel : { .efi_runtime_rel : {
*(.relefi_runtime_text) *(.rel*.efi_runtime)
*(.relefi_runtime_data) *(.rel*.efi_runtime.*)
} }
.efi_runtime_rel_stop : .efi_runtime_rel_stop :

View file

@ -19,6 +19,25 @@ SECTIONS
*(.__image_copy_start) *(.__image_copy_start)
*(.vectors) *(.vectors)
CPUDIR/start.o (.text*) CPUDIR/start.o (.text*)
}
/* This needs to come before *(.text*) */
.__efi_runtime_start : {
*(.__efi_runtime_start)
}
.efi_runtime : {
*(.text.efi_runtime*)
*(.rodata.efi_runtime*)
*(.data.efi_runtime*)
}
.__efi_runtime_stop : {
*(.__efi_runtime_stop)
}
.text_rest :
{
*(.text*) *(.text*)
} }
@ -41,27 +60,14 @@ SECTIONS
. = ALIGN(4); . = ALIGN(4);
.__efi_runtime_start : {
*(.__efi_runtime_start)
}
.efi_runtime : {
*(efi_runtime_text)
*(efi_runtime_data)
}
.__efi_runtime_stop : {
*(.__efi_runtime_stop)
}
.efi_runtime_rel_start : .efi_runtime_rel_start :
{ {
*(.__efi_runtime_rel_start) *(.__efi_runtime_rel_start)
} }
.efi_runtime_rel : { .efi_runtime_rel : {
*(.relefi_runtime_text) *(.rel*.efi_runtime)
*(.relefi_runtime_data) *(.rel*.efi_runtime.*)
} }
.efi_runtime_rel_stop : .efi_runtime_rel_stop :

View file

@ -12,7 +12,20 @@ SECTIONS
.text : .text :
{ {
arch/riscv/cpu/ax25/start.o (.text) arch/riscv/cpu/ax25/start.o (.text)
*(.text) }
/* This needs to come before *(.text*) */
.efi_runtime : {
__efi_runtime_start = .;
*(.text.efi_runtime*)
*(.rodata.efi_runtime*)
*(.data.efi_runtime*)
__efi_runtime_stop = .;
}
.text_rest :
{
*(.text*)
} }
. = ALIGN(4); . = ALIGN(4);
@ -39,17 +52,10 @@ SECTIONS
. = ALIGN(4); . = ALIGN(4);
.efi_runtime : {
__efi_runtime_start = .;
*(efi_runtime_text)
*(efi_runtime_data)
__efi_runtime_stop = .;
}
.efi_runtime_rel : { .efi_runtime_rel : {
__efi_runtime_rel_start = .; __efi_runtime_rel_start = .;
*(.relaefi_runtime_text) *(.rel*.efi_runtime)
*(.relaefi_runtime_data) *(.rel*.efi_runtime.*)
__efi_runtime_rel_stop = .; __efi_runtime_rel_stop = .;
} }

View file

@ -5,6 +5,9 @@ PLATFORM_CPPFLAGS += -D__SANDBOX__ -U_FORTIFY_SOURCE
PLATFORM_CPPFLAGS += -DCONFIG_ARCH_MAP_SYSMEM PLATFORM_CPPFLAGS += -DCONFIG_ARCH_MAP_SYSMEM
PLATFORM_LIBS += -lrt PLATFORM_LIBS += -lrt
LDFLAGS_FINAL += --gc-sections
PLATFORM_RELFLAGS += -ffunction-sections -fdata-sections
# Define this to avoid linking with SDL, which requires SDL libraries # Define this to avoid linking with SDL, which requires SDL libraries
# This can solve 'sdl-config: Command not found' errors # This can solve 'sdl-config: Command not found' errors
ifneq ($(NO_SDL),) ifneq ($(NO_SDL),)

View file

@ -24,8 +24,9 @@ SECTIONS
} }
.efi_runtime : { .efi_runtime : {
*(efi_runtime_text) *(.text.efi_runtime*)
*(efi_runtime_data) *(.rodata.efi_runtime*)
*(.data.efi_runtime*)
} }
.__efi_runtime_stop : { .__efi_runtime_stop : {
@ -38,8 +39,8 @@ SECTIONS
} }
.efi_runtime_rel : { .efi_runtime_rel : {
*(.relefi_runtime_text) *(.rel*.efi_runtime)
*(.relefi_runtime_data) *(.rel*.efi_runtime.*)
} }
.efi_runtime_rel_stop : .efi_runtime_rel_stop :

View file

@ -23,6 +23,8 @@ endif
ifeq ($(IS_32BIT),y) ifeq ($(IS_32BIT),y)
PLATFORM_CPPFLAGS += -march=i386 -m32 PLATFORM_CPPFLAGS += -march=i386 -m32
# TODO: These break on x86_64; need to debug further
PLATFORM_RELFLAGS += -fdata-sections
else else
PLATFORM_CPPFLAGS += $(if $(CONFIG_SPL_BUILD),,-fpic) -fno-common -m64 PLATFORM_CPPFLAGS += $(if $(CONFIG_SPL_BUILD),,-fpic) -fno-common -m64
endif endif

View file

@ -17,7 +17,7 @@
#include <generated/generic-asm-offsets.h> #include <generated/generic-asm-offsets.h>
#include <generated/asm-offsets.h> #include <generated/asm-offsets.h>
.section .text .section .text.start
.code32 .code32
.globl _start .globl _start
.type _start, @function .type _start, @function

View file

@ -8,7 +8,7 @@
#include <config.h> #include <config.h>
.section .text .section .text.start
.code64 .code64
.globl _start .globl _start
.type _start, @function .type _start, @function

View file

@ -17,6 +17,23 @@ SECTIONS
. = CONFIG_SYS_TEXT_BASE; /* Location of bootcode in flash */ . = CONFIG_SYS_TEXT_BASE; /* Location of bootcode in flash */
__text_start = .; __text_start = .;
.text.start : { *(.text.start); }
.__efi_runtime_start : {
*(.__efi_runtime_start)
}
.efi_runtime : {
*(.text.efi_runtime*)
*(.rodata.efi_runtime*)
*(.data.efi_runtime*)
}
.__efi_runtime_stop : {
*(.__efi_runtime_stop)
}
.text : { *(.text*); } .text : { *(.text*); }
. = ALIGN(4); . = ALIGN(4);
@ -27,7 +44,10 @@ SECTIONS
} }
. = ALIGN(4); . = ALIGN(4);
.rodata : { *(SORT_BY_ALIGNMENT(SORT_BY_NAME(.rodata*))) } .rodata : {
*(SORT_BY_ALIGNMENT(SORT_BY_NAME(.rodata*)))
KEEP(*(.rodata.efi.init));
}
. = ALIGN(4); . = ALIGN(4);
.data : { *(.data*) } .data : { *(.data*) }
@ -38,6 +58,21 @@ SECTIONS
. = ALIGN(4); . = ALIGN(4);
.got : { *(.got*) } .got : { *(.got*) }
.efi_runtime_rel_start :
{
*(.__efi_runtime_rel_start)
}
.efi_runtime_rel : {
*(.rel*.efi_runtime)
*(.rel*.efi_runtime.*)
}
.efi_runtime_rel_stop :
{
*(.__efi_runtime_rel_stop)
}
. = ALIGN(4); . = ALIGN(4);
__data_end = .; __data_end = .;
__init_end = .; __init_end = .;

View file

@ -17,6 +17,23 @@ SECTIONS
. = CONFIG_SYS_TEXT_BASE; /* Location of bootcode in flash */ . = CONFIG_SYS_TEXT_BASE; /* Location of bootcode in flash */
__text_start = .; __text_start = .;
.text.start : { *(.text.start); }
.__efi_runtime_start : {
*(.__efi_runtime_start)
}
.efi_runtime : {
*(.text.efi_runtime*)
*(.rodata.efi_runtime*)
*(.data.efi_runtime*)
}
.__efi_runtime_stop : {
*(.__efi_runtime_stop)
}
.text : { *(.text*); } .text : { *(.text*); }
. = ALIGN(4); . = ALIGN(4);
@ -43,27 +60,14 @@ SECTIONS
. = ALIGN(4); . = ALIGN(4);
.__efi_runtime_start : {
*(.__efi_runtime_start)
}
.efi_runtime : {
*(efi_runtime_text)
*(efi_runtime_data)
}
.__efi_runtime_stop : {
*(.__efi_runtime_stop)
}
.efi_runtime_rel_start : .efi_runtime_rel_start :
{ {
*(.__efi_runtime_rel_start) *(.__efi_runtime_rel_start)
} }
.efi_runtime_rel : { .efi_runtime_rel : {
*(.relefi_runtime_text) *(.rel*.efi_runtime)
*(.relefi_runtime_data) *(.rel*.efi_runtime.*)
} }
.efi_runtime_rel_stop : .efi_runtime_rel_stop :

View file

@ -1,45 +0,0 @@
/* SPDX-License-Identifier: GPL-2.0+ */
/*
* Brought in from Linux 4.1, removed things not useful to U-Boot.
* The definitions perhaps came from the GNU Library which is GPL.
*/
#ifndef _ASM_X86_ELF_H
#define _ASM_X86_ELF_H
/* ELF register definitions */
#define R_386_NONE 0
#define R_386_32 1
#define R_386_PC32 2
#define R_386_GOT32 3
#define R_386_PLT32 4
#define R_386_COPY 5
#define R_386_GLOB_DAT 6
#define R_386_JMP_SLOT 7
#define R_386_RELATIVE 8
#define R_386_GOTOFF 9
#define R_386_GOTPC 10
#define R_386_NUM 11
/* x86-64 relocation types */
#define R_X86_64_NONE 0 /* No reloc */
#define R_X86_64_64 1 /* Direct 64 bit */
#define R_X86_64_PC32 2 /* PC relative 32 bit signed */
#define R_X86_64_GOT32 3 /* 32 bit GOT entry */
#define R_X86_64_PLT32 4 /* 32 bit PLT address */
#define R_X86_64_COPY 5 /* Copy symbol at runtime */
#define R_X86_64_GLOB_DAT 6 /* Create GOT entry */
#define R_X86_64_JUMP_SLOT 7 /* Create PLT entry */
#define R_X86_64_RELATIVE 8 /* Adjust by program base */
/* 32 bit signed pc relative offset to GOT */
#define R_X86_64_GOTPCREL 9
#define R_X86_64_32 10 /* Direct 32 bit zero extended */
#define R_X86_64_32S 11 /* Direct 32 bit sign extended */
#define R_X86_64_16 12 /* Direct 16 bit zero extended */
#define R_X86_64_PC16 13 /* 16 bit sign extended pc relative */
#define R_X86_64_8 14 /* Direct 8 bit sign extended */
#define R_X86_64_PC8 15 /* 8 bit sign extended pc relative */
#define R_X86_64_NUM 16
#endif

View file

@ -10,7 +10,6 @@
#include <common.h> #include <common.h>
#include <efi.h> #include <efi.h>
#include <elf.h> #include <elf.h>
#include <asm/elf.h>
efi_status_t EFIAPI _relocate(long ldbase, Elf32_Dyn *dyn) efi_status_t EFIAPI _relocate(long ldbase, Elf32_Dyn *dyn)
{ {

View file

@ -12,7 +12,6 @@
#include <common.h> #include <common.h>
#include <efi.h> #include <efi.h>
#include <elf.h> #include <elf.h>
#include <asm/elf.h>
efi_status_t EFIAPI _relocate(long ldbase, Elf64_Dyn *dyn) efi_status_t EFIAPI _relocate(long ldbase, Elf64_Dyn *dyn)
{ {

View file

@ -20,6 +20,19 @@ SECTIONS
*(.__image_copy_start) *(.__image_copy_start)
board/qualcomm/dragonboard410c/head.o (.text*) board/qualcomm/dragonboard410c/head.o (.text*)
CPUDIR/start.o (.text*) CPUDIR/start.o (.text*)
}
/* This needs to come before *(.text*) */
.efi_runtime : {
__efi_runtime_start = .;
*(.text.efi_runtime*)
*(.rodata.efi_runtime*)
*(.data.efi_runtime*)
__efi_runtime_stop = .;
}
.text_rest :
{
*(.text*) *(.text*)
} }
@ -51,8 +64,8 @@ SECTIONS
.efi_runtime_rel : { .efi_runtime_rel : {
__efi_runtime_rel_start = .; __efi_runtime_rel_start = .;
*(.relaefi_runtime_text) *(.rel*.efi_runtime)
*(.relaefi_runtime_data) *(.rel*.efi_runtime.*)
__efi_runtime_rel_stop = .; __efi_runtime_rel_stop = .;
} }

View file

@ -20,6 +20,19 @@ SECTIONS
*(.__image_copy_start) *(.__image_copy_start)
board/qualcomm/dragonboard820c/head.o (.text*) board/qualcomm/dragonboard820c/head.o (.text*)
CPUDIR/start.o (.text*) CPUDIR/start.o (.text*)
}
/* This needs to come before *(.text*) */
.efi_runtime : {
__efi_runtime_start = .;
*(.text.efi_runtime*)
*(.rodata.efi_runtime*)
*(.data.efi_runtime*)
__efi_runtime_stop = .;
}
.text_rest :
{
*(.text*) *(.text*)
} }
@ -42,17 +55,10 @@ SECTIONS
. = ALIGN(8); . = ALIGN(8);
.efi_runtime : {
__efi_runtime_start = .;
*(efi_runtime_text)
*(efi_runtime_data)
__efi_runtime_stop = .;
}
.efi_runtime_rel : { .efi_runtime_rel : {
__efi_runtime_rel_start = .; __efi_runtime_rel_start = .;
*(.relaefi_runtime_text) *(.rel*.efi_runtime)
*(.relaefi_runtime_data) *(.rel*.efi_runtime.*)
__efi_runtime_rel_stop = .; __efi_runtime_rel_stop = .;
} }

View file

@ -37,6 +37,25 @@ SECTIONS
*(.vectors) *(.vectors)
CPUDIR/start.o (.text*) CPUDIR/start.o (.text*)
board/ti/am335x/built-in.o (.text*) board/ti/am335x/built-in.o (.text*)
}
/* This needs to come before *(.text*) */
.__efi_runtime_start : {
*(.__efi_runtime_start)
}
.efi_runtime : {
*(.text.efi_runtime*)
*(.rodata.efi_runtime*)
*(.data.efi_runtime*)
}
.__efi_runtime_stop : {
*(.__efi_runtime_stop)
}
.text_rest :
{
*(.text*) *(.text*)
} }
@ -59,27 +78,14 @@ SECTIONS
. = ALIGN(4); . = ALIGN(4);
.__efi_runtime_start : {
*(.__efi_runtime_start)
}
.efi_runtime : {
*(efi_runtime_text)
*(efi_runtime_data)
}
.__efi_runtime_stop : {
*(.__efi_runtime_stop)
}
.efi_runtime_rel_start : .efi_runtime_rel_start :
{ {
*(.__efi_runtime_rel_start) *(.__efi_runtime_rel_start)
} }
.efi_runtime_rel : { .efi_runtime_rel : {
*(.relefi_runtime_text) *(.rel*.efi_runtime)
*(.relefi_runtime_data) *(.rel*.efi_runtime.*)
} }
.efi_runtime_rel_stop : .efi_runtime_rel_stop :

View file

@ -14,12 +14,18 @@
#include <errno.h> #include <errno.h>
#include <linux/libfdt.h> #include <linux/libfdt.h>
#include <linux/libfdt_env.h> #include <linux/libfdt_env.h>
#include <mapmem.h>
#include <memalign.h> #include <memalign.h>
#include <asm/global_data.h> #include <asm/global_data.h>
#include <asm-generic/sections.h> #include <asm-generic/sections.h>
#include <asm-generic/unaligned.h> #include <asm-generic/unaligned.h>
#include <linux/linkage.h> #include <linux/linkage.h>
#ifdef CONFIG_ARMV7_NONSEC
#include <asm/armv7.h>
#include <asm/secure.h>
#endif
DECLARE_GLOBAL_DATA_PTR; DECLARE_GLOBAL_DATA_PTR;
#define OBJ_LIST_NOT_INITIALIZED 1 #define OBJ_LIST_NOT_INITIALIZED 1
@ -38,6 +44,11 @@ efi_status_t efi_init_obj_list(void)
if (efi_obj_list_initialized != OBJ_LIST_NOT_INITIALIZED) if (efi_obj_list_initialized != OBJ_LIST_NOT_INITIALIZED)
return efi_obj_list_initialized; return efi_obj_list_initialized;
/* Initialize system table */
ret = efi_initialize_system_table();
if (ret != EFI_SUCCESS)
goto out;
/* Initialize EFI driver uclass */ /* Initialize EFI driver uclass */
ret = efi_driver_init(); ret = efi_driver_init();
if (ret != EFI_SUCCESS) if (ret != EFI_SUCCESS)
@ -79,9 +90,6 @@ efi_status_t efi_init_obj_list(void)
ret = efi_reset_system_init(); ret = efi_reset_system_init();
if (ret != EFI_SUCCESS) if (ret != EFI_SUCCESS)
goto out; goto out;
ret = efi_get_time_init();
if (ret != EFI_SUCCESS)
goto out;
out: out:
efi_obj_list_initialized = ret; efi_obj_list_initialized = ret;
@ -142,8 +150,12 @@ static void *copy_fdt(void *fdt)
fdt_ram_start = ram_start; fdt_ram_start = ram_start;
} }
/* Give us at least 4kb breathing room */ /*
fdt_size = ALIGN(fdt_size + 4096, EFI_PAGE_SIZE); * Give us at least 4KB of breathing room in case the device tree needs
* to be expanded later. Round up to the nearest EFI page boundary.
*/
fdt_size += 4096;
fdt_size = ALIGN(fdt_size + EFI_PAGE_SIZE - 1, EFI_PAGE_SIZE);
fdt_pages = fdt_size >> EFI_PAGE_SHIFT; fdt_pages = fdt_size >> EFI_PAGE_SHIFT;
/* Safe fdt location is at 128MB */ /* Safe fdt location is at 128MB */
@ -194,8 +206,32 @@ static efi_status_t efi_run_in_el2(EFIAPI efi_status_t (*entry)(
} }
#endif #endif
/* Carve out DT reserved memory ranges */ #ifdef CONFIG_ARMV7_NONSEC
static efi_status_t efi_carve_out_dt_rsv(void *fdt) static bool is_nonsec;
static efi_status_t efi_run_in_hyp(EFIAPI efi_status_t (*entry)(
efi_handle_t image_handle, struct efi_system_table *st),
efi_handle_t image_handle, struct efi_system_table *st)
{
/* Enable caches again */
dcache_enable();
is_nonsec = true;
return efi_do_enter(image_handle, st, entry);
}
#endif
/*
* efi_carve_out_dt_rsv() - Carve out DT reserved memory ranges
*
* The mem_rsv entries of the FDT are added to the memory map. Any failures are
* ignored because this is not critical and we would rather continue to try to
* boot.
*
* @fdt: Pointer to device tree
*/
static void efi_carve_out_dt_rsv(void *fdt)
{ {
int nr_rsv, i; int nr_rsv, i;
uint64_t addr, size, pages; uint64_t addr, size, pages;
@ -208,11 +244,10 @@ static efi_status_t efi_carve_out_dt_rsv(void *fdt)
continue; continue;
pages = ALIGN(size, EFI_PAGE_SIZE) >> EFI_PAGE_SHIFT; pages = ALIGN(size, EFI_PAGE_SIZE) >> EFI_PAGE_SHIFT;
efi_add_memory_map(addr, pages, EFI_RESERVED_MEMORY_TYPE, if (!efi_add_memory_map(addr, pages, EFI_RESERVED_MEMORY_TYPE,
false); false))
printf("FDT memrsv map %d: Failed to add to map\n", i);
} }
return EFI_SUCCESS;
} }
static efi_status_t efi_install_fdt(void *fdt) static efi_status_t efi_install_fdt(void *fdt)
@ -236,10 +271,7 @@ static efi_status_t efi_install_fdt(void *fdt)
return EFI_LOAD_ERROR; return EFI_LOAD_ERROR;
} }
if (efi_carve_out_dt_rsv(fdt) != EFI_SUCCESS) { efi_carve_out_dt_rsv(fdt);
printf("ERROR: failed to carve out memory\n");
return EFI_LOAD_ERROR;
}
/* Link to it in the efi tables */ /* Link to it in the efi tables */
ret = efi_install_configuration_table(&efi_guid_fdt, fdt); ret = efi_install_configuration_table(&efi_guid_fdt, fdt);
@ -350,6 +382,22 @@ static efi_status_t do_bootefi_exec(void *efi,
} }
#endif #endif
#ifdef CONFIG_ARMV7_NONSEC
if (armv7_boot_nonsec() && !is_nonsec) {
dcache_disable(); /* flush cache before switch to HYP */
armv7_init_nonsec();
secure_ram_addr(_do_nonsec_entry)(
efi_run_in_hyp,
(uintptr_t)entry,
(uintptr_t)loaded_image_info_obj.handle,
(uintptr_t)&systab);
/* Should never reach here, efi exits with longjmp */
while (1) { }
}
#endif
ret = efi_do_enter(loaded_image_info_obj.handle, &systab, entry); ret = efi_do_enter(loaded_image_info_obj.handle, &systab, entry);
exit: exit:
@ -394,7 +442,8 @@ static int do_bootefi(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
unsigned long addr; unsigned long addr;
char *saddr; char *saddr;
efi_status_t r; efi_status_t r;
void *fdt_addr; unsigned long fdt_addr;
void *fdt;
/* Allow unaligned memory access */ /* Allow unaligned memory access */
allow_unaligned(); allow_unaligned();
@ -411,11 +460,12 @@ static int do_bootefi(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
return CMD_RET_USAGE; return CMD_RET_USAGE;
if (argc > 2) { if (argc > 2) {
fdt_addr = (void *)simple_strtoul(argv[2], NULL, 16); fdt_addr = simple_strtoul(argv[2], NULL, 16);
if (!fdt_addr && *argv[2] != '0') if (!fdt_addr && *argv[2] != '0')
return CMD_RET_USAGE; return CMD_RET_USAGE;
/* Install device tree */ /* Install device tree */
r = efi_install_fdt(fdt_addr); fdt = map_sysmem(fdt_addr, 0);
r = efi_install_fdt(fdt);
if (r != EFI_SUCCESS) { if (r != EFI_SUCCESS) {
printf("ERROR: failed to install device tree\n"); printf("ERROR: failed to install device tree\n");
return CMD_RET_FAILURE; return CMD_RET_FAILURE;
@ -434,7 +484,7 @@ static int do_bootefi(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
addr = simple_strtoul(saddr, NULL, 16); addr = simple_strtoul(saddr, NULL, 16);
else else
addr = CONFIG_SYS_LOAD_ADDR; addr = CONFIG_SYS_LOAD_ADDR;
memcpy((char *)addr, __efi_helloworld_begin, size); memcpy(map_sysmem(addr, size), __efi_helloworld_begin, size);
} else } else
#endif #endif
#ifdef CONFIG_CMD_BOOTEFI_SELFTEST #ifdef CONFIG_CMD_BOOTEFI_SELFTEST
@ -480,7 +530,7 @@ static int do_bootefi(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
} }
printf("## Starting EFI application at %08lx ...\n", addr); printf("## Starting EFI application at %08lx ...\n", addr);
r = do_bootefi_exec((void *)addr, bootefi_device_path, r = do_bootefi_exec(map_sysmem(addr, 0), bootefi_device_path,
bootefi_image_path); bootefi_image_path);
printf("## Application terminated, r = %lu\n", printf("## Application terminated, r = %lu\n",
r & ~EFI_ERROR_MASK); r & ~EFI_ERROR_MASK);

View file

@ -329,8 +329,6 @@ This driver is only available if U-Boot is configured with
* persistence * persistence
* runtime support * runtime support
* support bootefi booting ARMv7 in non-secure mode (CONFIG_ARMV7_NONSEC=y)
## Links ## Links
* [1](http://uefi.org/specifications) * [1](http://uefi.org/specifications)

View file

@ -27,8 +27,6 @@
#include <asm/arch/at91_rtt.h> #include <asm/arch/at91_rtt.h>
#include <asm/arch/at91_gpbr.h> #include <asm/arch/at91_gpbr.h>
#if defined(CONFIG_CMD_DATE)
int rtc_get (struct rtc_time *tmp) int rtc_get (struct rtc_time *tmp)
{ {
at91_rtt_t *rtt = (at91_rtt_t *) ATMEL_BASE_RTT; at91_rtt_t *rtt = (at91_rtt_t *) ATMEL_BASE_RTT;
@ -78,5 +76,3 @@ void rtc_reset (void)
while (readl(&rtt->vr) != 0) while (readl(&rtt->vr) != 0)
; ;
} }
#endif

View file

@ -9,7 +9,6 @@
#include <asm/io.h> #include <asm/io.h>
#include <asm/davinci_rtc.h> #include <asm/davinci_rtc.h>
#if defined(CONFIG_CMD_DATE)
int rtc_get(struct rtc_time *tmp) int rtc_get(struct rtc_time *tmp)
{ {
struct davinci_rtc *rtc = (struct davinci_rtc *)DAVINCI_RTC_BASE; struct davinci_rtc *rtc = (struct davinci_rtc *)DAVINCI_RTC_BASE;
@ -79,4 +78,3 @@ void rtc_reset(void)
/* run RTC counter */ /* run RTC counter */
writel(0x01, &rtc->ctrl); writel(0x01, &rtc->ctrl);
} }
#endif

View file

@ -9,8 +9,6 @@
#include <command.h> #include <command.h>
#include <rtc.h> #include <rtc.h>
#if defined(CONFIG_CMD_DATE)
/* GPP Pins */ /* GPP Pins */
#define DATA 0x200 #define DATA 0x200
#define SCLK 0x400 #define SCLK 0x400
@ -328,5 +326,3 @@ int rtc_set(struct rtc_time *tmp)
return 0; return 0;
} }
#endif

View file

@ -19,8 +19,6 @@
#include <rtc.h> #include <rtc.h>
#include <spi.h> #include <spi.h>
#if defined(CONFIG_CMD_DATE)
#define RTC_SECONDS 0x00 #define RTC_SECONDS 0x00
#define RTC_MINUTES 0x01 #define RTC_MINUTES 0x01
#define RTC_HOURS 0x02 #define RTC_HOURS 0x02
@ -437,5 +435,3 @@ static void rtc_write (unsigned char reg, unsigned char val)
} }
#endif /* end of code exclusion (see #ifdef CONFIG_SXNI855T above) */ #endif /* end of code exclusion (see #ifdef CONFIG_SXNI855T above) */
#endif

View file

@ -51,8 +51,6 @@ enum ds_type {
#ifndef CONFIG_DM_RTC #ifndef CONFIG_DM_RTC
#if defined(CONFIG_CMD_DATE)
/*---------------------------------------------------------------------*/ /*---------------------------------------------------------------------*/
#undef DEBUG_RTC #undef DEBUG_RTC
@ -204,8 +202,6 @@ static void rtc_write (uchar reg, uchar val)
i2c_reg_write (CONFIG_SYS_I2C_RTC_ADDR, reg, val); i2c_reg_write (CONFIG_SYS_I2C_RTC_ADDR, reg, val);
} }
#endif /* CONFIG_CMD_DATE*/
#endif /* !CONFIG_DM_RTC */ #endif /* !CONFIG_DM_RTC */
#ifdef CONFIG_DM_RTC #ifdef CONFIG_DM_RTC

View file

@ -15,8 +15,6 @@
#include <rtc.h> #include <rtc.h>
#include <i2c.h> #include <i2c.h>
#if defined(CONFIG_CMD_DATE)
/* /*
* RTC register addresses * RTC register addresses
*/ */
@ -190,5 +188,3 @@ static void rtc_write (uchar reg, uchar val)
{ {
i2c_reg_write (CONFIG_SYS_I2C_RTC_ADDR, reg, val); i2c_reg_write (CONFIG_SYS_I2C_RTC_ADDR, reg, val);
} }
#endif

View file

@ -18,8 +18,6 @@
#include <rtc.h> #include <rtc.h>
#include <i2c.h> #include <i2c.h>
#if defined(CONFIG_CMD_DATE)
/*---------------------------------------------------------------------*/ /*---------------------------------------------------------------------*/
#undef DEBUG_RTC #undef DEBUG_RTC
#define DEBUG_RTC #define DEBUG_RTC
@ -214,4 +212,3 @@ static void rtc_write_raw (uchar reg, uchar val)
{ {
i2c_reg_write (CONFIG_SYS_I2C_RTC_ADDR, reg, val); i2c_reg_write (CONFIG_SYS_I2C_RTC_ADDR, reg, val);
} }
#endif

View file

@ -20,8 +20,6 @@
#include <rtc.h> #include <rtc.h>
#if defined(CONFIG_CMD_DATE)
static uchar rtc_read(unsigned int addr ); static uchar rtc_read(unsigned int addr );
static void rtc_write(unsigned int addr, uchar val); static void rtc_write(unsigned int addr, uchar val);
@ -171,5 +169,3 @@ static void rtc_write( unsigned int addr, uchar val )
#endif #endif
*(volatile unsigned char*)(addr) = val; *(volatile unsigned char*)(addr) = val;
} }
#endif

View file

@ -16,8 +16,6 @@
#include <command.h> #include <command.h>
#include <rtc.h> #include <rtc.h>
#if defined(CONFIG_CMD_DATE)
static uchar rtc_read( unsigned int addr ); static uchar rtc_read( unsigned int addr );
static void rtc_write( unsigned int addr, uchar val); static void rtc_write( unsigned int addr, uchar val);
@ -172,5 +170,3 @@ static void rtc_write( unsigned int addr, uchar val )
#endif #endif
out8( addr, val ); out8( addr, val );
} }
#endif

View file

@ -16,8 +16,6 @@
#include <rtc.h> #include <rtc.h>
#include <i2c.h> #include <i2c.h>
#if defined(CONFIG_CMD_DATE)
/* /*
* RTC register addresses * RTC register addresses
*/ */
@ -166,5 +164,3 @@ static void rtc_write (uchar reg, uchar val)
{ {
i2c_reg_write (CONFIG_SYS_I2C_RTC_ADDR, reg, val); i2c_reg_write (CONFIG_SYS_I2C_RTC_ADDR, reg, val);
} }
#endif

View file

@ -17,8 +17,6 @@
#include <linux/compat.h> #include <linux/compat.h>
#include <rtc.h> #include <rtc.h>
#if defined(CONFIG_CMD_DATE)
#include <asm/io.h> #include <asm/io.h>
#include <asm/arch/imx-regs.h> #include <asm/arch/imx-regs.h>
@ -222,5 +220,3 @@ void rtc_reset(void)
{ {
di_init(); di_init();
} }
#endif

View file

@ -29,8 +29,6 @@
#endif #endif
*/ */
#if defined(CONFIG_SYS_I2C_RTC_ADDR) && defined(CONFIG_CMD_DATE)
/* ------------------------------------------------------------------------- */ /* ------------------------------------------------------------------------- */
/* /*
these are simple defines for the chip local to here so they aren't too these are simple defines for the chip local to here so they aren't too
@ -167,4 +165,3 @@ void rtc_reset (void)
val = val & 0x3F;/*turn off freq test keep calibration*/ val = val & 0x3F;/*turn off freq test keep calibration*/
i2c_write(CONFIG_SYS_I2C_RTC_ADDR, RTC_CONTROL_ADDR, 1, &val, 1); i2c_write(CONFIG_SYS_I2C_RTC_ADDR, RTC_CONTROL_ADDR, 1, &val, 1);
} }
#endif

View file

@ -20,8 +20,6 @@
#include <rtc.h> #include <rtc.h>
#include <i2c.h> #include <i2c.h>
#if defined(CONFIG_SYS_I2C_RTC_ADDR) && defined(CONFIG_CMD_DATE)
/* /*
* Convert between century and "century bits" (CB1 and CB0). These routines * Convert between century and "century bits" (CB1 and CB0). These routines
* assume years are in the range 1900 - 2299. * assume years are in the range 1900 - 2299.
@ -237,4 +235,3 @@ void rtc_reset(void)
} }
rtc_dump("end reset"); rtc_dump("end reset");
} }
#endif /* CONFIG_RTC_M41T60 && CONFIG_SYS_I2C_RTC_ADDR && CONFIG_CMD_DATE */

View file

@ -18,8 +18,6 @@
#include <rtc.h> #include <rtc.h>
#include <i2c.h> #include <i2c.h>
#if defined(CONFIG_CMD_DATE)
#define M41T62_REG_SSEC 0 #define M41T62_REG_SSEC 0
#define M41T62_REG_SEC 1 #define M41T62_REG_SEC 1
#define M41T62_REG_MIN 2 #define M41T62_REG_MIN 2
@ -130,5 +128,3 @@ void rtc_reset(void)
val &= ~M41T80_ALHOUR_HT; val &= ~M41T80_ALHOUR_HT;
i2c_write(CONFIG_SYS_I2C_RTC_ADDR, M41T62_REG_ALARM_HOUR, 1, &val, 1); i2c_write(CONFIG_SYS_I2C_RTC_ADDR, M41T62_REG_ALARM_HOUR, 1, &val, 1);
} }
#endif

View file

@ -16,8 +16,6 @@
#include <rtc.h> #include <rtc.h>
#include <config.h> #include <config.h>
#if defined(CONFIG_CMD_DATE)
static uchar rtc_read (uchar reg); static uchar rtc_read (uchar reg);
static void rtc_write (uchar reg, uchar val); static void rtc_write (uchar reg, uchar val);
@ -135,5 +133,3 @@ static void rtc_write (uchar reg, uchar val)
*(unsigned char *) *(unsigned char *)
((CONFIG_SYS_NVRAM_BASE_ADDR + CONFIG_SYS_NVRAM_SIZE - 8) + reg) = val; ((CONFIG_SYS_NVRAM_BASE_ADDR + CONFIG_SYS_NVRAM_SIZE - 8) + reg) = val;
} }
#endif

View file

@ -15,8 +15,6 @@
#include <rtc.h> #include <rtc.h>
#include <i2c.h> #include <i2c.h>
#if defined(CONFIG_CMD_DATE)
#ifndef CONFIG_SYS_I2C_RTC_ADDR #ifndef CONFIG_SYS_I2C_RTC_ADDR
#define CONFIG_SYS_I2C_RTC_ADDR 0x50 #define CONFIG_SYS_I2C_RTC_ADDR 0x50
#endif #endif
@ -104,5 +102,3 @@ int rtc_set (struct rtc_time *tmp)
void rtc_reset (void) void rtc_reset (void)
{ {
} }
#endif

View file

@ -19,8 +19,6 @@
#define out8(p, v) outb(v, p) #define out8(p, v) outb(v, p)
#endif #endif
#if defined(CONFIG_CMD_DATE)
/* Set this to 1 to clear the CMOS RAM */ /* Set this to 1 to clear the CMOS RAM */
#define CLEAR_CMOS 0 #define CLEAR_CMOS 0
@ -196,7 +194,6 @@ static void mc146818_init(void)
/* Clear any pending interrupts */ /* Clear any pending interrupts */
mc146818_read8(RTC_CONFIG_C); mc146818_read8(RTC_CONFIG_C);
} }
#endif /* CONFIG_CMD_DATE */
#ifdef CONFIG_DM_RTC #ifdef CONFIG_DM_RTC

View file

@ -6,8 +6,6 @@
#include <common.h> #include <common.h>
#if defined(CONFIG_CMD_DATE)
#include <command.h> #include <command.h>
#include <rtc.h> #include <rtc.h>
#include <asm/immap.h> #include <asm/immap.h>
@ -104,5 +102,3 @@ void rtc_reset(void)
rtc->cr |= RTC_CR_SWR; rtc->cr |= RTC_CR_SWR;
} }
#endif /* CONFIG_MCFRTC && CONFIG_CMD_DATE */

View file

@ -70,8 +70,6 @@ void nvram_write(short dest, const void *src, size_t count)
rtc_write(d++, *s++); rtc_write(d++, *s++);
} }
#if defined(CONFIG_CMD_DATE)
/* ------------------------------------------------------------------------- */ /* ------------------------------------------------------------------------- */
int rtc_get (struct rtc_time *tmp) int rtc_get (struct rtc_time *tmp)
@ -175,5 +173,3 @@ void rtc_set_watchdog(short multi, short res)
wd_value = RTC_WDS | ((multi & 0x1F) << 2) | (res & 0x3); wd_value = RTC_WDS | ((multi & 0x1F) << 2) | (res & 0x3);
rtc_write(RTC_WATCHDOG, wd_value); rtc_write(RTC_WATCHDOG, wd_value);
} }
#endif

View file

@ -15,8 +15,6 @@
#include <rtc.h> #include <rtc.h>
#include <i2c.h> #include <i2c.h>
#if defined(CONFIG_CMD_DATE)
static uchar rtc_read (uchar reg); static uchar rtc_read (uchar reg);
static void rtc_write (uchar reg, uchar val); static void rtc_write (uchar reg, uchar val);
@ -117,5 +115,3 @@ static void rtc_write (uchar reg, uchar val)
{ {
i2c_reg_write (CONFIG_SYS_I2C_RTC_ADDR, reg, val); i2c_reg_write (CONFIG_SYS_I2C_RTC_ADDR, reg, val);
} }
#endif

View file

@ -24,7 +24,6 @@
#include <rtc.h> #include <rtc.h>
#include <i2c.h> #include <i2c.h>
#if defined(CONFIG_CMD_DATE)
/* /*
* Reads are always done starting with register 15, which requires some * Reads are always done starting with register 15, which requires some
* jumping-through-hoops to access the data correctly. * jumping-through-hoops to access the data correctly.
@ -255,5 +254,3 @@ rtc_reset (void)
if (!setup_done) if (!setup_done)
rs5c372_enable(); rs5c372_enable();
} }
#endif

View file

@ -13,8 +13,6 @@
#include <rtc.h> #include <rtc.h>
#include <i2c.h> #include <i2c.h>
#if defined(CONFIG_CMD_DATE)
/*---------------------------------------------------------------------*/ /*---------------------------------------------------------------------*/
#undef DEBUG_RTC #undef DEBUG_RTC
@ -190,5 +188,3 @@ static void rtc_write (uchar reg, uchar val)
printf("Error writing to RTC\n"); printf("Error writing to RTC\n");
} }
#endif /* CONFIG_RTC_RX8025 && CONFIG_CMD_DATE */

View file

@ -11,8 +11,6 @@
#include <common.h> #include <common.h>
#include <command.h> #include <command.h>
#if (defined(CONFIG_CMD_DATE))
#include <asm/arch/s3c24x0_cpu.h> #include <asm/arch/s3c24x0_cpu.h>
#include <rtc.h> #include <rtc.h>
@ -149,5 +147,3 @@ void rtc_reset(void)
writeb((readb(&rtc->rtccon) & ~0x06) | 0x08, &rtc->rtccon); writeb((readb(&rtc->rtccon) & ~0x06) | 0x08, &rtc->rtccon);
writeb(readb(&rtc->rtccon) & ~(0x08 | 0x01), &rtc->rtccon); writeb(readb(&rtc->rtccon) & ~(0x08 | 0x01), &rtc->rtccon);
} }
#endif

View file

@ -22,8 +22,6 @@
#include <rtc.h> #include <rtc.h>
#include <i2c.h> #include <i2c.h>
#if defined(CONFIG_CMD_DATE)
#define CCR_SEC 0 #define CCR_SEC 0
#define CCR_MIN 1 #define CCR_MIN 1
#define CCR_HOUR 2 #define CCR_HOUR 2
@ -160,5 +158,3 @@ void rtc_reset(void)
* Nothing to do * Nothing to do
*/ */
} }
#endif

View file

@ -909,9 +909,11 @@ static int do_fat_write(const char *filename, void *buffer, loff_t size,
volume_info volinfo; volume_info volinfo;
fsdata datablock; fsdata datablock;
fsdata *mydata = &datablock; fsdata *mydata = &datablock;
int cursect; int cursect, i;
int ret = -1, name_len; int ret = -1, name_len;
char l_filename[VFAT_MAXLEN_BYTES]; char l_filename[VFAT_MAXLEN_BYTES];
char bad[2] = " ";
const char illegal[] = "<>:\"/\\|?*";
*actwrite = size; *actwrite = size;
dir_curclust = 0; dir_curclust = 0;
@ -971,6 +973,18 @@ static int do_fat_write(const char *filename, void *buffer, loff_t size,
} }
dentptr = (dir_entry *) do_fat_read_at_block; dentptr = (dir_entry *) do_fat_read_at_block;
/* Strip leading (back-)slashes */
while ISDIRDELIM(*filename)
++filename;
/* Check that the filename is valid */
for (i = 0; i < strlen(illegal); ++i) {
*bad = illegal[i];
if (strstr(filename, bad)) {
printf("FAT: illegal filename (%s)\n", filename);
return -1;
}
}
name_len = strlen(filename); name_len = strlen(filename);
if (name_len >= VFAT_MAXLEN_BYTES) if (name_len >= VFAT_MAXLEN_BYTES)
name_len = VFAT_MAXLEN_BYTES - 1; name_len = VFAT_MAXLEN_BYTES - 1;

View file

@ -29,8 +29,16 @@
*/ */
#ifdef __x86_64__ #ifdef __x86_64__
#define EFIAPI __attribute__((ms_abi)) #define EFIAPI __attribute__((ms_abi))
#define efi_va_list __builtin_ms_va_list
#define efi_va_start __builtin_ms_va_start
#define efi_va_arg __builtin_va_arg
#define efi_va_end __builtin_ms_va_end
#else #else
#define EFIAPI asmlinkage #define EFIAPI asmlinkage
#define efi_va_list va_list
#define efi_va_start va_start
#define efi_va_arg va_arg
#define efi_va_end va_end
#endif /* __x86_64__ */ #endif /* __x86_64__ */
struct efi_device_path; struct efi_device_path;

View file

@ -21,6 +21,9 @@
#include <asm/setjmp.h> #include <asm/setjmp.h>
#endif #endif
/* UEFI spec version 2.7 */
#define EFI_SPECIFICATION_VERSION (2 << 16 | 70)
/* Types and defines for EFI CreateEvent */ /* Types and defines for EFI CreateEvent */
enum efi_timer_delay { enum efi_timer_delay {
EFI_TIMER_STOP = 0, EFI_TIMER_STOP = 0,
@ -46,6 +49,7 @@ typedef uint16_t *efi_string_t;
struct efi_event; struct efi_event;
/* EFI Boot Services table */ /* EFI Boot Services table */
#define EFI_BOOT_SERVICES_SIGNATURE 0x56524553544f4f42
struct efi_boot_services { struct efi_boot_services {
struct efi_table_hdr hdr; struct efi_table_hdr hdr;
efi_status_t (EFIAPI *raise_tpl)(efi_uintn_t new_tpl); efi_status_t (EFIAPI *raise_tpl)(efi_uintn_t new_tpl);
@ -161,8 +165,9 @@ struct efi_boot_services {
void **handle, ...); void **handle, ...);
efi_status_t (EFIAPI *uninstall_multiple_protocol_interfaces)( efi_status_t (EFIAPI *uninstall_multiple_protocol_interfaces)(
void *handle, ...); void *handle, ...);
efi_status_t (EFIAPI *calculate_crc32)(void *data, efi_status_t (EFIAPI *calculate_crc32)(const void *data,
unsigned long data_size, uint32_t *crc32); efi_uintn_t data_size,
u32 *crc32);
void (EFIAPI *copy_mem)(void *destination, const void *source, void (EFIAPI *copy_mem)(void *destination, const void *source,
size_t length); size_t length);
void (EFIAPI *set_mem)(void *buffer, size_t size, uint8_t value); void (EFIAPI *set_mem)(void *buffer, size_t size, uint8_t value);
@ -185,8 +190,7 @@ enum efi_reset_type {
}; };
/* EFI Runtime Services table */ /* EFI Runtime Services table */
#define EFI_RUNTIME_SERVICES_SIGNATURE 0x5652453544e5552ULL #define EFI_RUNTIME_SERVICES_SIGNATURE 0x56524553544e5552ULL
#define EFI_RUNTIME_SERVICES_REVISION 0x00010000
#define CAPSULE_FLAGS_PERSIST_ACROSS_RESET 0x00010000 #define CAPSULE_FLAGS_PERSIST_ACROSS_RESET 0x00010000
#define CAPSULE_FLAGS_POPULATE_SYSTEM_TABLE 0x00020000 #define CAPSULE_FLAGS_POPULATE_SYSTEM_TABLE 0x00020000
@ -300,7 +304,7 @@ struct efi_configuration_table
struct efi_system_table { struct efi_system_table {
struct efi_table_hdr hdr; struct efi_table_hdr hdr;
unsigned long fw_vendor; /* physical addr of wchar_t vendor string */ u16 *fw_vendor; /* physical addr of wchar_t vendor string */
u32 fw_revision; u32 fw_revision;
efi_handle_t con_in_handle; efi_handle_t con_in_handle;
struct efi_simple_input_interface *con_in; struct efi_simple_input_interface *con_in;
@ -318,6 +322,8 @@ struct efi_system_table {
EFI_GUID(0x5b1b31a1, 0x9562, 0x11d2, \ EFI_GUID(0x5b1b31a1, 0x9562, 0x11d2, \
0x8e, 0x3f, 0x00, 0xa0, 0xc9, 0x69, 0x72, 0x3b) 0x8e, 0x3f, 0x00, 0xa0, 0xc9, 0x69, 0x72, 0x3b)
#define EFI_LOADED_IMAGE_PROTOCOL_REVISION 0x1000
struct efi_loaded_image { struct efi_loaded_image {
u32 revision; u32 revision;
void *parent_handle; void *parent_handle;

View file

@ -17,6 +17,9 @@
#include <linux/list.h> #include <linux/list.h>
/* Maximum number of configuration tables */
#define EFI_MAX_CONFIGURATION_TABLES 16
int __efi_entry_check(void); int __efi_entry_check(void);
int __efi_exit_check(void); int __efi_exit_check(void);
const char *__efi_nesting(void); const char *__efi_nesting(void);
@ -82,6 +85,9 @@ const char *__efi_nesting_dec(void);
#define EFI_CACHELINE_SIZE 128 #define EFI_CACHELINE_SIZE 128
#endif #endif
/* Key identifying current memory map */
extern efi_uintn_t efi_memory_map_key;
extern struct efi_runtime_services efi_runtime_services; extern struct efi_runtime_services efi_runtime_services;
extern struct efi_system_table systab; extern struct efi_system_table systab;
@ -199,6 +205,8 @@ extern struct list_head efi_obj_list;
/* List of all events */ /* List of all events */
extern struct list_head efi_events; extern struct list_head efi_events;
/* Called by bootefi to initialize runtime */
efi_status_t efi_initialize_system_table(void);
/* Called by bootefi to make console interface available */ /* Called by bootefi to make console interface available */
int efi_console_register(void); int efi_console_register(void);
/* Called by bootefi to make all disk storage accessible as EFI objects */ /* Called by bootefi to make all disk storage accessible as EFI objects */
@ -406,8 +414,8 @@ static inline int guidcmp(const efi_guid_t *g1, const efi_guid_t *g2)
* Use these to indicate that your code / data should go into the EFI runtime * Use these to indicate that your code / data should go into the EFI runtime
* section and thus still be available when the OS is running * section and thus still be available when the OS is running
*/ */
#define __efi_runtime_data __attribute__ ((section ("efi_runtime_data"))) #define __efi_runtime_data __attribute__ ((section (".data.efi_runtime")))
#define __efi_runtime __attribute__ ((section ("efi_runtime_text"))) #define __efi_runtime __attribute__ ((section (".text.efi_runtime")))
/* Call this with mmio_ptr as the _pointer_ to a pointer to an MMIO region /* Call this with mmio_ptr as the _pointer_ to a pointer to an MMIO region
* to make it available at runtime */ * to make it available at runtime */
@ -426,7 +434,6 @@ efi_status_t efi_reset_system_init(void);
efi_status_t __efi_runtime EFIAPI efi_get_time( efi_status_t __efi_runtime EFIAPI efi_get_time(
struct efi_time *time, struct efi_time *time,
struct efi_time_cap *capabilities); struct efi_time_cap *capabilities);
efi_status_t efi_get_time_init(void);
#ifdef CONFIG_CMD_BOOTEFI_SELFTEST #ifdef CONFIG_CMD_BOOTEFI_SELFTEST
/* /*

View file

@ -550,6 +550,41 @@ unsigned long elf_hash(const unsigned char *name);
#endif /* __ASSEMBLER */ #endif /* __ASSEMBLER */
/* ELF register definitions */
#define R_386_NONE 0
#define R_386_32 1
#define R_386_PC32 2
#define R_386_GOT32 3
#define R_386_PLT32 4
#define R_386_COPY 5
#define R_386_GLOB_DAT 6
#define R_386_JMP_SLOT 7
#define R_386_RELATIVE 8
#define R_386_GOTOFF 9
#define R_386_GOTPC 10
#define R_386_NUM 11
/* x86-64 relocation types */
#define R_X86_64_NONE 0 /* No reloc */
#define R_X86_64_64 1 /* Direct 64 bit */
#define R_X86_64_PC32 2 /* PC relative 32 bit signed */
#define R_X86_64_GOT32 3 /* 32 bit GOT entry */
#define R_X86_64_PLT32 4 /* 32 bit PLT address */
#define R_X86_64_COPY 5 /* Copy symbol at runtime */
#define R_X86_64_GLOB_DAT 6 /* Create GOT entry */
#define R_X86_64_JUMP_SLOT 7 /* Create PLT entry */
#define R_X86_64_RELATIVE 8 /* Adjust by program base */
/* 32 bit signed pc relative offset to GOT */
#define R_X86_64_GOTPCREL 9
#define R_X86_64_32 10 /* Direct 32 bit zero extended */
#define R_X86_64_32S 11 /* Direct 32 bit sign extended */
#define R_X86_64_16 12 /* Direct 16 bit zero extended */
#define R_X86_64_PC16 13 /* 16 bit sign extended pc relative */
#define R_X86_64_8 14 /* Direct 8 bit sign extended */
#define R_X86_64_PC8 15 /* 8 bit sign extended pc relative */
#define R_X86_64_NUM 16
/* /*
* XXX - PowerPC defines really don't belong in here, * XXX - PowerPC defines really don't belong in here,
* but we'll put them in for simplicity. * but we'll put them in for simplicity.

View file

@ -161,6 +161,8 @@ static int efi_bl_bind(efi_handle_t handle, void *interface)
return ret; return ret;
if (!bdev) if (!bdev)
return -ENOENT; return -ENOENT;
/* Set the DM_FLAG_NAME_ALLOCED flag to avoid a memory leak */
device_set_name_alloced(bdev);
/* Allocate priv */ /* Allocate priv */
ret = device_probe(bdev); ret = device_probe(bdev);
if (ret) if (ret)

View file

@ -1,8 +1,6 @@
config EFI_LOADER config EFI_LOADER
bool "Support running EFI Applications in U-Boot" bool "Support running EFI Applications in U-Boot"
depends on (ARM || X86 || RISCV) && OF_LIBFDT depends on (ARM || X86 || RISCV) && OF_LIBFDT
# We do not support bootefi booting ARMv7 in non-secure mode
depends on !ARMV7_NONSEC
# We need EFI_STUB_64BIT to be set on x86_64 with EFI_STUB # We need EFI_STUB_64BIT to be set on x86_64 with EFI_STUB
depends on !EFI_STUB || !X86_64 || EFI_STUB_64BIT depends on !EFI_STUB || !X86_64 || EFI_STUB_64BIT
# We need EFI_STUB_32BIT to be set on x86_32 with EFI_STUB # We need EFI_STUB_32BIT to be set on x86_32 with EFI_STUB

View file

@ -6,6 +6,9 @@
# This file only gets included with CONFIG_EFI_LOADER set, so all # This file only gets included with CONFIG_EFI_LOADER set, so all
# object inclusion implicitly depends on it # object inclusion implicitly depends on it
CFLAGS_efi_boottime.o += \
-DFW_VERSION="0x$(VERSION)" \
-DFW_PATCHLEVEL="0x$(PATCHLEVEL)"
CFLAGS_helloworld.o := $(CFLAGS_EFI) -Os -ffreestanding CFLAGS_helloworld.o := $(CFLAGS_EFI) -Os -ffreestanding
CFLAGS_REMOVE_helloworld.o := $(CFLAGS_NON_EFI) -Os CFLAGS_REMOVE_helloworld.o := $(CFLAGS_NON_EFI) -Os

View file

@ -35,16 +35,6 @@ LIST_HEAD(efi_events);
*/ */
static bool efi_is_direct_boot = true; static bool efi_is_direct_boot = true;
/*
* EFI can pass arbitrary additional "tables" containing vendor specific
* information to the payload. One such table is the FDT table which contains
* a pointer to a flattened device tree blob.
*
* In most cases we want to pass an FDT to the payload, so reserve one slot of
* config table space for it. The pointer gets populated by do_bootefi_exec().
*/
static struct efi_configuration_table __efi_runtime_data efi_conf_table[16];
#ifdef CONFIG_ARM #ifdef CONFIG_ARM
/* /*
* The "gd" pointer lives in a register on ARM and AArch64 that we declare * The "gd" pointer lives in a register on ARM and AArch64 that we declare
@ -163,6 +153,18 @@ const char *__efi_nesting_dec(void)
return indent_string(--nesting_level); return indent_string(--nesting_level);
} }
/**
* efi_update_table_header_crc32() - Update CRC32 in table header
*
* @table: EFI table
*/
static void efi_update_table_header_crc32(struct efi_table_hdr *table)
{
table->crc32 = 0;
table->crc32 = crc32(0, (const unsigned char *)table,
table->headersize);
}
/** /**
* efi_queue_event() - queue an EFI event * efi_queue_event() - queue an EFI event
* @event: event to signal * @event: event to signal
@ -190,6 +192,25 @@ static void efi_queue_event(struct efi_event *event, bool check_tpl)
event->is_queued = false; event->is_queued = false;
} }
/**
* is_valid_tpl() - check if the task priority level is valid
*
* @tpl: TPL level to check
* ReturnValue: status code
*/
efi_status_t is_valid_tpl(efi_uintn_t tpl)
{
switch (tpl) {
case TPL_APPLICATION:
case TPL_CALLBACK:
case TPL_NOTIFY:
case TPL_HIGH_LEVEL:
return EFI_SUCCESS;
default:
return EFI_INVALID_PARAMETER;
}
}
/** /**
* efi_signal_event() - signal an EFI event * efi_signal_event() - signal an EFI event
* @event: event to signal * @event: event to signal
@ -592,11 +613,21 @@ efi_status_t efi_create_event(uint32_t type, efi_uintn_t notify_tpl,
if (event == NULL) if (event == NULL)
return EFI_INVALID_PARAMETER; return EFI_INVALID_PARAMETER;
if ((type & EVT_NOTIFY_SIGNAL) && (type & EVT_NOTIFY_WAIT)) switch (type) {
case 0:
case EVT_TIMER:
case EVT_NOTIFY_SIGNAL:
case EVT_TIMER | EVT_NOTIFY_SIGNAL:
case EVT_NOTIFY_WAIT:
case EVT_TIMER | EVT_NOTIFY_WAIT:
case EVT_SIGNAL_EXIT_BOOT_SERVICES:
case EVT_SIGNAL_VIRTUAL_ADDRESS_CHANGE:
break;
default:
return EFI_INVALID_PARAMETER; return EFI_INVALID_PARAMETER;
}
if ((type & (EVT_NOTIFY_SIGNAL | EVT_NOTIFY_WAIT)) && if (is_valid_tpl(notify_tpl) != EFI_SUCCESS)
notify_function == NULL)
return EFI_INVALID_PARAMETER; return EFI_INVALID_PARAMETER;
evt = calloc(1, sizeof(struct efi_event)); evt = calloc(1, sizeof(struct efi_event));
@ -1361,9 +1392,9 @@ static efi_status_t EFIAPI efi_locate_handle_ext(
*/ */
static void efi_remove_configuration_table(int i) static void efi_remove_configuration_table(int i)
{ {
struct efi_configuration_table *this = &efi_conf_table[i]; struct efi_configuration_table *this = &systab.tables[i];
struct efi_configuration_table *next = &efi_conf_table[i + 1]; struct efi_configuration_table *next = &systab.tables[i + 1];
struct efi_configuration_table *end = &efi_conf_table[systab.nr_tables]; struct efi_configuration_table *end = &systab.tables[systab.nr_tables];
memmove(this, next, (ulong)end - (ulong)next); memmove(this, next, (ulong)end - (ulong)next);
systab.nr_tables--; systab.nr_tables--;
@ -1391,9 +1422,9 @@ efi_status_t efi_install_configuration_table(const efi_guid_t *guid,
/* Check for guid override */ /* Check for guid override */
for (i = 0; i < systab.nr_tables; i++) { for (i = 0; i < systab.nr_tables; i++) {
if (!guidcmp(guid, &efi_conf_table[i].guid)) { if (!guidcmp(guid, &systab.tables[i].guid)) {
if (table) if (table)
efi_conf_table[i].table = table; systab.tables[i].table = table;
else else
efi_remove_configuration_table(i); efi_remove_configuration_table(i);
goto out; goto out;
@ -1404,15 +1435,18 @@ efi_status_t efi_install_configuration_table(const efi_guid_t *guid,
return EFI_NOT_FOUND; return EFI_NOT_FOUND;
/* No override, check for overflow */ /* No override, check for overflow */
if (i >= ARRAY_SIZE(efi_conf_table)) if (i >= EFI_MAX_CONFIGURATION_TABLES)
return EFI_OUT_OF_RESOURCES; return EFI_OUT_OF_RESOURCES;
/* Add a new entry */ /* Add a new entry */
memcpy(&efi_conf_table[i].guid, guid, sizeof(*guid)); memcpy(&systab.tables[i].guid, guid, sizeof(*guid));
efi_conf_table[i].table = table; systab.tables[i].table = table;
systab.nr_tables = i + 1; systab.nr_tables = i + 1;
out: out:
/* systab.nr_tables may have changed. So we need to update the crc32 */
efi_update_table_header_crc32(&systab.hdr);
/* Notify that the configuration table was changed */ /* Notify that the configuration table was changed */
list_for_each_entry(evt, &efi_events, link) { list_for_each_entry(evt, &efi_events, link) {
if (evt->group && !guidcmp(evt->group, guid)) { if (evt->group && !guidcmp(evt->group, guid)) {
@ -1468,6 +1502,7 @@ efi_status_t efi_setup_loaded_image(
/* efi_exit() assumes that the handle points to the info */ /* efi_exit() assumes that the handle points to the info */
obj->handle = info; obj->handle = info;
info->revision = EFI_LOADED_IMAGE_PROTOCOL_REVISION;
info->file_path = file_path; info->file_path = file_path;
if (device_path) { if (device_path) {
@ -1825,6 +1860,10 @@ static efi_status_t EFIAPI efi_exit_boot_services(efi_handle_t image_handle,
EFI_ENTRY("%p, %ld", image_handle, map_key); EFI_ENTRY("%p, %ld", image_handle, map_key);
/* Check that the caller has read the current memory map */
if (map_key != efi_memory_map_key)
return EFI_INVALID_PARAMETER;
/* Make sure that notification functions are not called anymore */ /* Make sure that notification functions are not called anymore */
efi_tpl = TPL_HIGH_LEVEL; efi_tpl = TPL_HIGH_LEVEL;
@ -1867,9 +1906,7 @@ static efi_status_t EFIAPI efi_exit_boot_services(efi_handle_t image_handle,
systab.boottime = NULL; systab.boottime = NULL;
/* Recalculate CRC32 */ /* Recalculate CRC32 */
systab.hdr.crc32 = 0; efi_update_table_header_crc32(&systab.hdr);
systab.hdr.crc32 = crc32(0, (const unsigned char *)&systab,
sizeof(struct efi_system_table));
/* Give the payload some time to boot */ /* Give the payload some time to boot */
efi_set_watchdog(0); efi_set_watchdog(0);
@ -2302,7 +2339,7 @@ static efi_status_t EFIAPI efi_install_multiple_protocol_interfaces(
{ {
EFI_ENTRY("%p", handle); EFI_ENTRY("%p", handle);
va_list argptr; efi_va_list argptr;
const efi_guid_t *protocol; const efi_guid_t *protocol;
void *protocol_interface; void *protocol_interface;
efi_status_t r = EFI_SUCCESS; efi_status_t r = EFI_SUCCESS;
@ -2311,12 +2348,12 @@ static efi_status_t EFIAPI efi_install_multiple_protocol_interfaces(
if (!handle) if (!handle)
return EFI_EXIT(EFI_INVALID_PARAMETER); return EFI_EXIT(EFI_INVALID_PARAMETER);
va_start(argptr, handle); efi_va_start(argptr, handle);
for (;;) { for (;;) {
protocol = va_arg(argptr, efi_guid_t*); protocol = efi_va_arg(argptr, efi_guid_t*);
if (!protocol) if (!protocol)
break; break;
protocol_interface = va_arg(argptr, void*); protocol_interface = efi_va_arg(argptr, void*);
r = EFI_CALL(efi_install_protocol_interface( r = EFI_CALL(efi_install_protocol_interface(
handle, protocol, handle, protocol,
EFI_NATIVE_INTERFACE, EFI_NATIVE_INTERFACE,
@ -2325,19 +2362,19 @@ static efi_status_t EFIAPI efi_install_multiple_protocol_interfaces(
break; break;
i++; i++;
} }
va_end(argptr); efi_va_end(argptr);
if (r == EFI_SUCCESS) if (r == EFI_SUCCESS)
return EFI_EXIT(r); return EFI_EXIT(r);
/* If an error occurred undo all changes. */ /* If an error occurred undo all changes. */
va_start(argptr, handle); efi_va_start(argptr, handle);
for (; i; --i) { for (; i; --i) {
protocol = va_arg(argptr, efi_guid_t*); protocol = efi_va_arg(argptr, efi_guid_t*);
protocol_interface = va_arg(argptr, void*); protocol_interface = efi_va_arg(argptr, void*);
EFI_CALL(efi_uninstall_protocol_interface(handle, protocol, EFI_CALL(efi_uninstall_protocol_interface(handle, protocol,
protocol_interface)); protocol_interface));
} }
va_end(argptr); efi_va_end(argptr);
return EFI_EXIT(r); return EFI_EXIT(r);
} }
@ -2361,7 +2398,7 @@ static efi_status_t EFIAPI efi_uninstall_multiple_protocol_interfaces(
{ {
EFI_ENTRY("%p", handle); EFI_ENTRY("%p", handle);
va_list argptr; efi_va_list argptr;
const efi_guid_t *protocol; const efi_guid_t *protocol;
void *protocol_interface; void *protocol_interface;
efi_status_t r = EFI_SUCCESS; efi_status_t r = EFI_SUCCESS;
@ -2370,12 +2407,12 @@ static efi_status_t EFIAPI efi_uninstall_multiple_protocol_interfaces(
if (!handle) if (!handle)
return EFI_EXIT(EFI_INVALID_PARAMETER); return EFI_EXIT(EFI_INVALID_PARAMETER);
va_start(argptr, handle); efi_va_start(argptr, handle);
for (;;) { for (;;) {
protocol = va_arg(argptr, efi_guid_t*); protocol = efi_va_arg(argptr, efi_guid_t*);
if (!protocol) if (!protocol)
break; break;
protocol_interface = va_arg(argptr, void*); protocol_interface = efi_va_arg(argptr, void*);
r = EFI_CALL(efi_uninstall_protocol_interface( r = EFI_CALL(efi_uninstall_protocol_interface(
handle, protocol, handle, protocol,
protocol_interface)); protocol_interface));
@ -2383,20 +2420,20 @@ static efi_status_t EFIAPI efi_uninstall_multiple_protocol_interfaces(
break; break;
i++; i++;
} }
va_end(argptr); efi_va_end(argptr);
if (r == EFI_SUCCESS) if (r == EFI_SUCCESS)
return EFI_EXIT(r); return EFI_EXIT(r);
/* If an error occurred undo all changes. */ /* If an error occurred undo all changes. */
va_start(argptr, handle); efi_va_start(argptr, handle);
for (; i; --i) { for (; i; --i) {
protocol = va_arg(argptr, efi_guid_t*); protocol = efi_va_arg(argptr, efi_guid_t*);
protocol_interface = va_arg(argptr, void*); protocol_interface = efi_va_arg(argptr, void*);
EFI_CALL(efi_install_protocol_interface(&handle, protocol, EFI_CALL(efi_install_protocol_interface(&handle, protocol,
EFI_NATIVE_INTERFACE, EFI_NATIVE_INTERFACE,
protocol_interface)); protocol_interface));
} }
va_end(argptr); efi_va_end(argptr);
return EFI_EXIT(r); return EFI_EXIT(r);
} }
@ -2414,11 +2451,11 @@ static efi_status_t EFIAPI efi_uninstall_multiple_protocol_interfaces(
* *
* Return: status code * Return: status code
*/ */
static efi_status_t EFIAPI efi_calculate_crc32(void *data, static efi_status_t EFIAPI efi_calculate_crc32(const void *data,
unsigned long data_size, efi_uintn_t data_size,
uint32_t *crc32_p) u32 *crc32_p)
{ {
EFI_ENTRY("%p, %ld", data, data_size); EFI_ENTRY("%p, %zu", data, data_size);
*crc32_p = crc32(0, data, data_size); *crc32_p = crc32(0, data, data_size);
return EFI_EXIT(EFI_SUCCESS); return EFI_EXIT(EFI_SUCCESS);
} }
@ -3022,9 +3059,11 @@ out:
return EFI_EXIT(r); return EFI_EXIT(r);
} }
static const struct efi_boot_services efi_boot_services = { static struct efi_boot_services efi_boot_services = {
.hdr = { .hdr = {
.headersize = sizeof(struct efi_table_hdr), .signature = EFI_BOOT_SERVICES_SIGNATURE,
.revision = EFI_SPECIFICATION_VERSION,
.headersize = sizeof(struct efi_boot_services),
}, },
.raise_tpl = efi_raise_tpl, .raise_tpl = efi_raise_tpl,
.restore_tpl = efi_restore_tpl, .restore_tpl = efi_restore_tpl,
@ -3074,20 +3113,44 @@ static const struct efi_boot_services efi_boot_services = {
.create_event_ex = efi_create_event_ex, .create_event_ex = efi_create_event_ex,
}; };
static uint16_t __efi_runtime_data firmware_vendor[] = L"Das U-Boot"; static u16 __efi_runtime_data firmware_vendor[] = L"Das U-Boot";
struct efi_system_table __efi_runtime_data systab = { struct efi_system_table __efi_runtime_data systab = {
.hdr = { .hdr = {
.signature = EFI_SYSTEM_TABLE_SIGNATURE, .signature = EFI_SYSTEM_TABLE_SIGNATURE,
.revision = 2 << 16 | 70, /* 2.7 */ .revision = EFI_SPECIFICATION_VERSION,
.headersize = sizeof(struct efi_table_hdr), .headersize = sizeof(struct efi_system_table),
}, },
.fw_vendor = (long)firmware_vendor, .fw_vendor = firmware_vendor,
.fw_revision = FW_VERSION << 16 | FW_PATCHLEVEL << 8,
.con_in = (void *)&efi_con_in, .con_in = (void *)&efi_con_in,
.con_out = (void *)&efi_con_out, .con_out = (void *)&efi_con_out,
.std_err = (void *)&efi_con_out, .std_err = (void *)&efi_con_out,
.runtime = (void *)&efi_runtime_services, .runtime = (void *)&efi_runtime_services,
.boottime = (void *)&efi_boot_services, .boottime = (void *)&efi_boot_services,
.nr_tables = 0, .nr_tables = 0,
.tables = (void *)efi_conf_table, .tables = NULL,
}; };
/**
* efi_initialize_system_table() - Initialize system table
*
* Return Value: status code
*/
efi_status_t efi_initialize_system_table(void)
{
efi_status_t ret;
/* Allocate configuration table array */
ret = efi_allocate_pool(EFI_RUNTIME_SERVICES_DATA,
EFI_MAX_CONFIGURATION_TABLES *
sizeof(struct efi_configuration_table),
(void **)&systab.tables);
/* Set crc32 field in table headers */
efi_update_table_header_crc32(&systab.hdr);
efi_update_table_header_crc32(&efi_runtime_services.hdr);
efi_update_table_header_crc32(&efi_boot_services.hdr);
return ret;
}

View file

@ -335,6 +335,8 @@ static efi_status_t EFIAPI efi_cout_clear_screen(
EFI_ENTRY("%p", this); EFI_ENTRY("%p", this);
printf(ESC"[2J"); printf(ESC"[2J");
efi_con_mode.cursor_column = 0;
efi_con_mode.cursor_row = 0;
return EFI_EXIT(EFI_SUCCESS); return EFI_EXIT(EFI_SUCCESS);
} }
@ -381,7 +383,12 @@ static efi_status_t EFIAPI efi_cin_reset(
bool extended_verification) bool extended_verification)
{ {
EFI_ENTRY("%p, %d", this, extended_verification); EFI_ENTRY("%p, %d", this, extended_verification);
return EFI_EXIT(EFI_UNSUPPORTED);
/* Empty input buffer */
while (tstc())
getc();
return EFI_EXIT(EFI_SUCCESS);
} }
/* /*

View file

@ -19,25 +19,25 @@ const efi_guid_t efi_simple_file_system_protocol_guid =
const efi_guid_t efi_file_info_guid = EFI_FILE_INFO_GUID; const efi_guid_t efi_file_info_guid = EFI_FILE_INFO_GUID;
static int machines[] = { static int machines[] = {
#if defined(CONFIG_ARM64) #if defined(__aarch64__)
IMAGE_FILE_MACHINE_ARM64, IMAGE_FILE_MACHINE_ARM64,
#elif defined(CONFIG_ARM) #elif defined(__arm__)
IMAGE_FILE_MACHINE_ARM, IMAGE_FILE_MACHINE_ARM,
IMAGE_FILE_MACHINE_THUMB, IMAGE_FILE_MACHINE_THUMB,
IMAGE_FILE_MACHINE_ARMNT, IMAGE_FILE_MACHINE_ARMNT,
#endif #endif
#if defined(CONFIG_X86_64) #if defined(__x86_64__)
IMAGE_FILE_MACHINE_AMD64, IMAGE_FILE_MACHINE_AMD64,
#elif defined(CONFIG_X86) #elif defined(__i386__)
IMAGE_FILE_MACHINE_I386, IMAGE_FILE_MACHINE_I386,
#endif #endif
#if defined(CONFIG_CPU_RISCV_32) #if defined(__riscv) && (__riscv_xlen == 32)
IMAGE_FILE_MACHINE_RISCV32, IMAGE_FILE_MACHINE_RISCV32,
#endif #endif
#if defined(CONFIG_CPU_RISCV_64) #if defined(__riscv) && (__riscv_xlen == 64)
IMAGE_FILE_MACHINE_RISCV64, IMAGE_FILE_MACHINE_RISCV64,
#endif #endif
0 }; 0 };

View file

@ -9,11 +9,14 @@
#include <efi_loader.h> #include <efi_loader.h>
#include <inttypes.h> #include <inttypes.h>
#include <malloc.h> #include <malloc.h>
#include <mapmem.h>
#include <watchdog.h> #include <watchdog.h>
#include <linux/list_sort.h> #include <linux/list_sort.h>
DECLARE_GLOBAL_DATA_PTR; DECLARE_GLOBAL_DATA_PTR;
efi_uintn_t efi_memory_map_key;
struct efi_mem_list { struct efi_mem_list {
struct list_head link; struct list_head link;
struct efi_mem_desc desc; struct efi_mem_desc desc;
@ -159,9 +162,13 @@ uint64_t efi_add_memory_map(uint64_t start, uint64_t pages, int memory_type,
debug("%s: 0x%" PRIx64 " 0x%" PRIx64 " %d %s\n", __func__, debug("%s: 0x%" PRIx64 " 0x%" PRIx64 " %d %s\n", __func__,
start, pages, memory_type, overlap_only_ram ? "yes" : "no"); start, pages, memory_type, overlap_only_ram ? "yes" : "no");
if (memory_type >= EFI_MAX_MEMORY_TYPE)
return EFI_INVALID_PARAMETER;
if (!pages) if (!pages)
return start; return start;
++efi_memory_map_key;
newlist = calloc(1, sizeof(*newlist)); newlist = calloc(1, sizeof(*newlist));
newlist->desc.type = memory_type; newlist->desc.type = memory_type;
newlist->desc.physical_start = start; newlist->desc.physical_start = start;
@ -292,10 +299,13 @@ efi_status_t efi_allocate_pages(int type, int memory_type,
efi_status_t r = EFI_SUCCESS; efi_status_t r = EFI_SUCCESS;
uint64_t addr; uint64_t addr;
if (!memory)
return EFI_INVALID_PARAMETER;
switch (type) { switch (type) {
case EFI_ALLOCATE_ANY_PAGES: case EFI_ALLOCATE_ANY_PAGES:
/* Any page */ /* Any page */
addr = efi_find_free_memory(len, gd->start_addr_sp); addr = efi_find_free_memory(len, -1ULL);
if (!addr) { if (!addr) {
r = EFI_NOT_FOUND; r = EFI_NOT_FOUND;
break; break;
@ -325,7 +335,7 @@ efi_status_t efi_allocate_pages(int type, int memory_type,
/* Reserve that map in our memory maps */ /* Reserve that map in our memory maps */
ret = efi_add_memory_map(addr, pages, memory_type, true); ret = efi_add_memory_map(addr, pages, memory_type, true);
if (ret == addr) { if (ret == addr) {
*memory = addr; *memory = (uintptr_t)map_sysmem(addr, len);
} else { } else {
/* Map would overlap, bail out */ /* Map would overlap, bail out */
r = EFI_OUT_OF_RESOURCES; r = EFI_OUT_OF_RESOURCES;
@ -359,11 +369,12 @@ void *efi_alloc(uint64_t len, int memory_type)
efi_status_t efi_free_pages(uint64_t memory, efi_uintn_t pages) efi_status_t efi_free_pages(uint64_t memory, efi_uintn_t pages)
{ {
uint64_t r = 0; uint64_t r = 0;
uint64_t addr = map_to_sysmem((void *)(uintptr_t)memory);
r = efi_add_memory_map(memory, pages, EFI_CONVENTIONAL_MEMORY, false); r = efi_add_memory_map(addr, pages, EFI_CONVENTIONAL_MEMORY, false);
/* Merging of adjacent free regions is missing */ /* Merging of adjacent free regions is missing */
if (r == memory) if (r == addr)
return EFI_SUCCESS; return EFI_SUCCESS;
return EFI_NOT_FOUND; return EFI_NOT_FOUND;
@ -380,20 +391,22 @@ efi_status_t efi_free_pages(uint64_t memory, efi_uintn_t pages)
efi_status_t efi_allocate_pool(int pool_type, efi_uintn_t size, void **buffer) efi_status_t efi_allocate_pool(int pool_type, efi_uintn_t size, void **buffer)
{ {
efi_status_t r; efi_status_t r;
efi_physical_addr_t t; struct efi_pool_allocation *alloc;
u64 num_pages = (size + sizeof(struct efi_pool_allocation) + u64 num_pages = (size + sizeof(struct efi_pool_allocation) +
EFI_PAGE_MASK) >> EFI_PAGE_SHIFT; EFI_PAGE_MASK) >> EFI_PAGE_SHIFT;
if (!buffer)
return EFI_INVALID_PARAMETER;
if (size == 0) { if (size == 0) {
*buffer = NULL; *buffer = NULL;
return EFI_SUCCESS; return EFI_SUCCESS;
} }
r = efi_allocate_pages(EFI_ALLOCATE_ANY_PAGES, pool_type, num_pages, r = efi_allocate_pages(EFI_ALLOCATE_ANY_PAGES, pool_type, num_pages,
&t); (uint64_t *)&alloc);
if (r == EFI_SUCCESS) { if (r == EFI_SUCCESS) {
struct efi_pool_allocation *alloc = (void *)(uintptr_t)t;
alloc->num_pages = num_pages; alloc->num_pages = num_pages;
*buffer = alloc->data; *buffer = alloc->data;
} }
@ -446,6 +459,9 @@ efi_status_t efi_get_memory_map(efi_uintn_t *memory_map_size,
struct list_head *lhandle; struct list_head *lhandle;
efi_uintn_t provided_map_size = *memory_map_size; efi_uintn_t provided_map_size = *memory_map_size;
if (!memory_map_size)
return EFI_INVALID_PARAMETER;
list_for_each(lhandle, &efi_mem) list_for_each(lhandle, &efi_mem)
map_entries++; map_entries++;
@ -456,6 +472,9 @@ efi_status_t efi_get_memory_map(efi_uintn_t *memory_map_size,
if (provided_map_size < map_size) if (provided_map_size < map_size)
return EFI_BUFFER_TOO_SMALL; return EFI_BUFFER_TOO_SMALL;
if (!memory_map)
return EFI_INVALID_PARAMETER;
if (descriptor_size) if (descriptor_size)
*descriptor_size = sizeof(struct efi_mem_desc); *descriptor_size = sizeof(struct efi_mem_desc);
@ -463,19 +482,18 @@ efi_status_t efi_get_memory_map(efi_uintn_t *memory_map_size,
*descriptor_version = EFI_MEMORY_DESCRIPTOR_VERSION; *descriptor_version = EFI_MEMORY_DESCRIPTOR_VERSION;
/* Copy list into array */ /* Copy list into array */
if (memory_map) { /* Return the list in ascending order */
/* Return the list in ascending order */ memory_map = &memory_map[map_entries - 1];
memory_map = &memory_map[map_entries - 1]; list_for_each(lhandle, &efi_mem) {
list_for_each(lhandle, &efi_mem) { struct efi_mem_list *lmem;
struct efi_mem_list *lmem;
lmem = list_entry(lhandle, struct efi_mem_list, link); lmem = list_entry(lhandle, struct efi_mem_list, link);
*memory_map = lmem->desc; *memory_map = lmem->desc;
memory_map--; memory_map--;
}
} }
*map_key = 0; if (map_key)
*map_key = efi_memory_map_key;
return EFI_SUCCESS; return EFI_SUCCESS;
} }
@ -496,14 +514,13 @@ __weak void efi_add_known_memory(void)
} }
} }
int efi_memory_init(void) /* Add memory regions for U-Boot's memory and for the runtime services code */
static void add_u_boot_and_runtime(void)
{ {
unsigned long runtime_start, runtime_end, runtime_pages; unsigned long runtime_start, runtime_end, runtime_pages;
unsigned long uboot_start, uboot_pages; unsigned long uboot_start, uboot_pages;
unsigned long uboot_stack_size = 16 * 1024 * 1024; unsigned long uboot_stack_size = 16 * 1024 * 1024;
efi_add_known_memory();
/* Add U-Boot */ /* Add U-Boot */
uboot_start = (gd->start_addr_sp - uboot_stack_size) & ~EFI_PAGE_MASK; uboot_start = (gd->start_addr_sp - uboot_stack_size) & ~EFI_PAGE_MASK;
uboot_pages = (gd->ram_top - uboot_start) >> EFI_PAGE_SHIFT; uboot_pages = (gd->ram_top - uboot_start) >> EFI_PAGE_SHIFT;
@ -516,6 +533,14 @@ int efi_memory_init(void)
runtime_pages = (runtime_end - runtime_start) >> EFI_PAGE_SHIFT; runtime_pages = (runtime_end - runtime_start) >> EFI_PAGE_SHIFT;
efi_add_memory_map(runtime_start, runtime_pages, efi_add_memory_map(runtime_start, runtime_pages,
EFI_RUNTIME_SERVICES_CODE, false); EFI_RUNTIME_SERVICES_CODE, false);
}
int efi_memory_init(void)
{
efi_add_known_memory();
if (!IS_ENABLED(CONFIG_SANDBOX))
add_u_boot_and_runtime();
#ifdef CONFIG_EFI_LOADER_BOUNCE_BUFFER #ifdef CONFIG_EFI_LOADER_BOUNCE_BUFFER
/* Request a 32bit 64MB bounce buffer region */ /* Request a 32bit 64MB bounce buffer region */

View file

@ -8,6 +8,7 @@
#include <common.h> #include <common.h>
#include <command.h> #include <command.h>
#include <dm.h> #include <dm.h>
#include <elf.h>
#include <efi_loader.h> #include <efi_loader.h>
#include <rtc.h> #include <rtc.h>
@ -32,19 +33,17 @@ static efi_status_t __efi_runtime EFIAPI efi_invalid_parameter(void);
* TODO(sjg@chromium.org): These defines and structs should come from the elf * TODO(sjg@chromium.org): These defines and structs should come from the elf
* header for each arch (or a generic header) rather than being repeated here. * header for each arch (or a generic header) rather than being repeated here.
*/ */
#if defined(CONFIG_ARM64) #if defined(__aarch64__)
#define R_RELATIVE 1027 #define R_RELATIVE R_AARCH64_RELATIVE
#define R_MASK 0xffffffffULL #define R_MASK 0xffffffffULL
#define IS_RELA 1 #define IS_RELA 1
#elif defined(CONFIG_ARM) #elif defined(__arm__)
#define R_RELATIVE 23 #define R_RELATIVE R_ARM_RELATIVE
#define R_MASK 0xffULL #define R_MASK 0xffULL
#elif defined(CONFIG_X86) #elif defined(__x86_64__) || defined(__i386__)
#include <asm/elf.h>
#define R_RELATIVE R_386_RELATIVE #define R_RELATIVE R_386_RELATIVE
#define R_MASK 0xffULL #define R_MASK 0xffULL
#elif defined(CONFIG_RISCV) #elif defined(__riscv)
#include <elf.h>
#define R_RELATIVE R_RISCV_RELATIVE #define R_RELATIVE R_RISCV_RELATIVE
#define R_MASK 0xffULL #define R_MASK 0xffULL
#define IS_RELA 1 #define IS_RELA 1
@ -55,12 +54,14 @@ struct dyn_sym {
u32 foo2; u32 foo2;
u32 foo3; u32 foo3;
}; };
#ifdef CONFIG_CPU_RISCV_32 #if (__riscv_xlen == 32)
#define R_ABSOLUTE R_RISCV_32 #define R_ABSOLUTE R_RISCV_32
#define SYM_INDEX 8 #define SYM_INDEX 8
#else #elif (__riscv_xlen == 64)
#define R_ABSOLUTE R_RISCV_64 #define R_ABSOLUTE R_RISCV_64
#define SYM_INDEX 32 #define SYM_INDEX 32
#else
#error unknown riscv target
#endif #endif
#else #else
#error Need to add relocation awareness #error Need to add relocation awareness
@ -116,24 +117,41 @@ static void EFIAPI efi_reset_system_boottime(
while (1) { } while (1) { }
} }
/**
* efi_get_time_boottime - get current time
*
* This function implements the GetTime runtime service.
* See the Unified Extensible Firmware Interface (UEFI) specification
* for details.
*
* @time: pointer to structure to receive current time
* @capabilities: pointer to structure to receive RTC properties
* Return Value: status code
*/
static efi_status_t EFIAPI efi_get_time_boottime( static efi_status_t EFIAPI efi_get_time_boottime(
struct efi_time *time, struct efi_time *time,
struct efi_time_cap *capabilities) struct efi_time_cap *capabilities)
{ {
#if defined(CONFIG_CMD_DATE) && defined(CONFIG_DM_RTC) #ifdef CONFIG_DM_RTC
struct rtc_time tm; efi_status_t ret = EFI_SUCCESS;
int r; int r;
struct rtc_time tm;
struct udevice *dev; struct udevice *dev;
EFI_ENTRY("%p %p", time, capabilities); EFI_ENTRY("%p %p", time, capabilities);
r = uclass_get_device(UCLASS_RTC, 0, &dev); if (!time) {
if (r) ret = EFI_INVALID_PARAMETER;
return EFI_EXIT(EFI_DEVICE_ERROR); goto out;
}
r = dm_rtc_get(dev, &tm); r = uclass_get_device(UCLASS_RTC, 0, &dev);
if (r) if (!r)
return EFI_EXIT(EFI_DEVICE_ERROR); r = dm_rtc_get(dev, &tm);
if (r) {
ret = EFI_DEVICE_ERROR;
goto out;
}
memset(time, 0, sizeof(*time)); memset(time, 0, sizeof(*time));
time->year = tm.tm_year; time->year = tm.tm_year;
@ -141,11 +159,23 @@ static efi_status_t EFIAPI efi_get_time_boottime(
time->day = tm.tm_mday; time->day = tm.tm_mday;
time->hour = tm.tm_hour; time->hour = tm.tm_hour;
time->minute = tm.tm_min; time->minute = tm.tm_min;
time->daylight = tm.tm_isdst; time->second = tm.tm_sec;
time->daylight = EFI_TIME_ADJUST_DAYLIGHT;
if (tm.tm_isdst > 0)
time->daylight |= EFI_TIME_IN_DAYLIGHT;
time->timezone = EFI_UNSPECIFIED_TIMEZONE;
return EFI_EXIT(EFI_SUCCESS); if (capabilities) {
/* Set reasonable dummy values */
capabilities->resolution = 1; /* 1 Hz */
capabilities->accuracy = 100000000; /* 100 ppm */
capabilities->sets_to_zero = false;
}
out:
return EFI_EXIT(ret);
#else #else
return EFI_DEVICE_ERROR; EFI_ENTRY("%p %p", time, capabilities);
return EFI_EXIT(EFI_DEVICE_ERROR);
#endif #endif
} }
@ -173,11 +203,6 @@ efi_status_t __weak __efi_runtime EFIAPI efi_get_time(
return EFI_DEVICE_ERROR; return EFI_DEVICE_ERROR;
} }
efi_status_t __weak efi_get_time_init(void)
{
return EFI_SUCCESS;
}
struct efi_runtime_detach_list_struct { struct efi_runtime_detach_list_struct {
void *ptr; void *ptr;
void *patchto; void *patchto;
@ -458,8 +483,8 @@ efi_status_t __efi_runtime EFIAPI efi_query_variable_info(
struct efi_runtime_services __efi_runtime_data efi_runtime_services = { struct efi_runtime_services __efi_runtime_data efi_runtime_services = {
.hdr = { .hdr = {
.signature = EFI_RUNTIME_SERVICES_SIGNATURE, .signature = EFI_RUNTIME_SERVICES_SIGNATURE,
.revision = EFI_RUNTIME_SERVICES_REVISION, .revision = EFI_SPECIFICATION_VERSION,
.headersize = sizeof(struct efi_table_hdr), .headersize = sizeof(struct efi_runtime_services),
}, },
.get_time = &efi_get_time_boottime, .get_time = &efi_get_time_boottime,
.set_time = (void *)&efi_device_error, .set_time = (void *)&efi_device_error,

View file

@ -26,8 +26,15 @@ efi_status_t efi_smbios_register(void)
/* Reserve 4kiB page for SMBIOS */ /* Reserve 4kiB page for SMBIOS */
ret = efi_allocate_pages(EFI_ALLOCATE_MAX_ADDRESS, ret = efi_allocate_pages(EFI_ALLOCATE_MAX_ADDRESS,
EFI_RUNTIME_SERVICES_DATA, 1, &dmi); EFI_RUNTIME_SERVICES_DATA, 1, &dmi);
if (ret != EFI_SUCCESS)
return ret; if (ret != EFI_SUCCESS) {
/* Could not find space in lowmem, use highmem instead */
ret = efi_allocate_pages(EFI_ALLOCATE_ANY_PAGES,
EFI_RUNTIME_SERVICES_DATA, 1, &dmi);
if (ret != EFI_SUCCESS)
return ret;
}
/* /*
* Generate SMBIOS tables - we know that efi_allocate_pages() returns * Generate SMBIOS tables - we know that efi_allocate_pages() returns

View file

@ -1,2 +1,4 @@
efi_miniapp_file_image.h efi_miniapp_file_image_exit.h
efi_miniapp_file_image_return.h
*.efi *.efi
*.so

View file

@ -13,8 +13,10 @@ CFLAGS_REMOVE_efi_selftest_miniapp_return.o := $(CFLAGS_NON_EFI) -Os
obj-$(CONFIG_CMD_BOOTEFI_SELFTEST) += \ obj-$(CONFIG_CMD_BOOTEFI_SELFTEST) += \
efi_selftest.o \ efi_selftest.o \
efi_selftest_bitblt.o \ efi_selftest_bitblt.o \
efi_selftest_config_table.o \
efi_selftest_controllers.o \ efi_selftest_controllers.o \
efi_selftest_console.o \ efi_selftest_console.o \
efi_selftest_crc32.o \
efi_selftest_devicepath.o \ efi_selftest_devicepath.o \
efi_selftest_devicepath_util.o \ efi_selftest_devicepath_util.o \
efi_selftest_events.o \ efi_selftest_events.o \
@ -23,6 +25,7 @@ efi_selftest_exitbootservices.o \
efi_selftest_fdt.o \ efi_selftest_fdt.o \
efi_selftest_gop.o \ efi_selftest_gop.o \
efi_selftest_manageprotocols.o \ efi_selftest_manageprotocols.o \
efi_selftest_rtc.o \
efi_selftest_snp.o \ efi_selftest_snp.o \
efi_selftest_textinput.o \ efi_selftest_textinput.o \
efi_selftest_textoutput.o \ efi_selftest_textoutput.o \
@ -41,7 +44,7 @@ endif
# TODO: As of v2018.01 the relocation code for the EFI application cannot # TODO: As of v2018.01 the relocation code for the EFI application cannot
# be built on x86_64. # be built on x86_64.
ifeq ($(CONFIG_X86_64),) ifeq ($(CONFIG_X86_64)$(CONFIG_SANDBOX),)
ifneq ($(CONFIG_CMD_BOOTEFI_SELFTEST),) ifneq ($(CONFIG_CMD_BOOTEFI_SELFTEST),)

View file

@ -8,9 +8,7 @@
#include <efi_selftest.h> #include <efi_selftest.h>
#include <vsprintf.h> #include <vsprintf.h>
/* /* Constants for test step bitmap */
* Constants for test step bitmap
*/
#define EFI_ST_SETUP 1 #define EFI_ST_SETUP 1
#define EFI_ST_EXECUTE 2 #define EFI_ST_EXECUTE 2
#define EFI_ST_TEARDOWN 4 #define EFI_ST_TEARDOWN 4
@ -26,7 +24,7 @@ static u16 reset_message[] = L"Selftest completed";
* *
* The size of the memory map is determined. * The size of the memory map is determined.
* Pool memory is allocated to copy the memory map. * Pool memory is allocated to copy the memory map.
* The memory amp is copied and the map key is obtained. * The memory map is copied and the map key is obtained.
* The map key is used to exit the boot services. * The map key is used to exit the boot services.
*/ */
void efi_st_exit_boot_services(void) void efi_st_exit_boot_services(void)
@ -146,7 +144,7 @@ static int teardown(struct efi_unit_test *test, unsigned int *failures)
* Check that a test exists. * Check that a test exists.
* *
* @testname: name of the test * @testname: name of the test
* @return: test * @return: test, or NULL if not found
*/ */
static struct efi_unit_test *find_test(const u16 *testname) static struct efi_unit_test *find_test(const u16 *testname)
{ {
@ -182,7 +180,7 @@ static void list_all_tests(void)
* *
* @testname name of a single selected test or NULL * @testname name of a single selected test or NULL
* @phase test phase * @phase test phase
* @steps steps to execute * @steps steps to execute (mask with bits from EFI_ST_...)
* failures returns EFI_ST_SUCCESS if all test steps succeeded * failures returns EFI_ST_SUCCESS if all test steps succeeded
*/ */
void efi_st_do_tests(const u16 *testname, unsigned int phase, void efi_st_do_tests(const u16 *testname, unsigned int phase,
@ -296,12 +294,12 @@ efi_status_t EFIAPI efi_selftest(efi_handle_t image_handle,
efi_st_printc(EFI_WHITE, "\nSummary: %u failures\n\n", failures); efi_st_printc(EFI_WHITE, "\nSummary: %u failures\n\n", failures);
/* Reset system */ /* Reset system */
efi_st_printf("Preparing for reset. Press any key.\n"); efi_st_printf("Preparing for reset. Press any key...\n");
efi_st_get_key(); efi_st_get_key();
runtime->reset_system(EFI_RESET_WARM, EFI_NOT_READY, runtime->reset_system(EFI_RESET_WARM, EFI_NOT_READY,
sizeof(reset_message), reset_message); sizeof(reset_message), reset_message);
efi_st_printf("\n"); efi_st_printf("\n");
efi_st_error("Reset failed.\n"); efi_st_error("Reset failed\n");
return EFI_UNSUPPORTED; return EFI_UNSUPPORTED;
} }

View file

@ -309,11 +309,14 @@ static int execute(void)
efi_uintn_t buf_size; efi_uintn_t buf_size;
char buf[16] __aligned(ARCH_DMA_MINALIGN); char buf[16] __aligned(ARCH_DMA_MINALIGN);
/* Connect controller to virtual disk */
ret = boottime->connect_controller(disk_handle, NULL, NULL, 1); ret = boottime->connect_controller(disk_handle, NULL, NULL, 1);
if (ret != EFI_SUCCESS) { if (ret != EFI_SUCCESS) {
efi_st_error("Failed to connect controller\n"); efi_st_error("Failed to connect controller\n");
return EFI_ST_FAILURE; return EFI_ST_FAILURE;
} }
/* Get the handle for the partition */
ret = boottime->locate_handle_buffer( ret = boottime->locate_handle_buffer(
BY_PROTOCOL, &guid_device_path, NULL, BY_PROTOCOL, &guid_device_path, NULL,
&no_handles, &handles); &no_handles, &handles);
@ -347,6 +350,8 @@ static int execute(void)
efi_st_error("Partition handle not found\n"); efi_st_error("Partition handle not found\n");
return EFI_ST_FAILURE; return EFI_ST_FAILURE;
} }
/* Open the simple file system protocol */
ret = boottime->open_protocol(handle_partition, ret = boottime->open_protocol(handle_partition,
&guid_simple_file_system_protocol, &guid_simple_file_system_protocol,
(void **)&file_system, NULL, NULL, (void **)&file_system, NULL, NULL,
@ -355,6 +360,8 @@ static int execute(void)
efi_st_error("Failed to open simple file system protocol\n"); efi_st_error("Failed to open simple file system protocol\n");
return EFI_ST_FAILURE; return EFI_ST_FAILURE;
} }
/* Open volume */
ret = file_system->open_volume(file_system, &root); ret = file_system->open_volume(file_system, &root);
if (ret != EFI_SUCCESS) { if (ret != EFI_SUCCESS) {
efi_st_error("Failed to open volume\n"); efi_st_error("Failed to open volume\n");
@ -377,6 +384,8 @@ static int execute(void)
"Wrong volume label '%ps', expected 'U-BOOT TEST'\n", "Wrong volume label '%ps', expected 'U-BOOT TEST'\n",
system_info.info.volume_label); system_info.info.volume_label);
} }
/* Read file */
ret = root->open(root, &file, (s16 *)L"hello.txt", EFI_FILE_MODE_READ, ret = root->open(root, &file, (s16 *)L"hello.txt", EFI_FILE_MODE_READ,
0); 0);
if (ret != EFI_SUCCESS) { if (ret != EFI_SUCCESS) {
@ -389,6 +398,11 @@ static int execute(void)
efi_st_error("Failed to read file\n"); efi_st_error("Failed to read file\n");
return EFI_ST_FAILURE; return EFI_ST_FAILURE;
} }
if (buf_size != 13) {
efi_st_error("Wrong number of bytes read: %u\n",
(unsigned int)buf_size);
return EFI_ST_FAILURE;
}
if (efi_st_memcmp(buf, "Hello world!", 12)) { if (efi_st_memcmp(buf, "Hello world!", 12)) {
efi_st_error("Unexpected file content\n"); efi_st_error("Unexpected file content\n");
return EFI_ST_FAILURE; return EFI_ST_FAILURE;
@ -398,6 +412,62 @@ static int execute(void)
efi_st_error("Failed to close file\n"); efi_st_error("Failed to close file\n");
return EFI_ST_FAILURE; return EFI_ST_FAILURE;
} }
#ifdef CONFIG_FAT_WRITE
/* Write file */
ret = root->open(root, &file, (s16 *)L"u-boot.txt",
EFI_FILE_MODE_WRITE | EFI_FILE_MODE_CREATE, 0);
if (ret != EFI_SUCCESS) {
efi_st_error("Failed to open file\n");
return EFI_ST_FAILURE;
}
buf_size = 7;
boottime->set_mem(buf, sizeof(buf), 0);
boottime->copy_mem(buf, "U-Boot", buf_size);
ret = file->write(file, &buf_size, buf);
if (ret != EFI_SUCCESS || buf_size != 7) {
efi_st_error("Failed to write file\n");
return EFI_ST_FAILURE;
}
ret = file->close(file);
if (ret != EFI_SUCCESS) {
efi_st_error("Failed to close file\n");
return EFI_ST_FAILURE;
}
/* Verify file */
boottime->set_mem(buf, sizeof(buf), 0);
ret = root->open(root, &file, (s16 *)L"u-boot.txt", EFI_FILE_MODE_READ,
0);
if (ret != EFI_SUCCESS) {
efi_st_error("Failed to open file\n");
return EFI_ST_FAILURE;
}
buf_size = sizeof(buf) - 1;
ret = file->read(file, &buf_size, buf);
if (ret != EFI_SUCCESS) {
efi_st_error("Failed to read file\n");
return EFI_ST_FAILURE;
}
if (buf_size != 7) {
efi_st_error("Wrong number of bytes read: %u\n",
(unsigned int)buf_size);
return EFI_ST_FAILURE;
}
if (efi_st_memcmp(buf, "U-Boot", 7)) {
efi_st_error("Unexpected file content %s\n", buf);
return EFI_ST_FAILURE;
}
ret = file->close(file);
if (ret != EFI_SUCCESS) {
efi_st_error("Failed to close file\n");
return EFI_ST_FAILURE;
}
#else
efi_st_todo("CONFIG_FAT_WRITE is not set\n");
#endif /* CONFIG_FAT_WRITE */
/* Close volume */
ret = root->close(root); ret = root->close(root);
if (ret != EFI_SUCCESS) { if (ret != EFI_SUCCESS) {
efi_st_error("Failed to close volume\n"); efi_st_error("Failed to close volume\n");

View file

@ -0,0 +1,266 @@
// SPDX-License-Identifier: GPL-2.0+
/*
* efi_selftest_config_tables
*
* Copyright (c) 2018 Heinrich Schuchardt <xypron.glpk@gmx.de>
*
* This test checks the following service:
* InstallConfigurationTable.
*/
#include <efi_selftest.h>
static const struct efi_system_table *sys_table;
static struct efi_boot_services *boottime;
static efi_guid_t table_guid =
EFI_GUID(0xff1c3f9e, 0x795b, 0x1529, 0xf1, 0x55,
0x17, 0x2e, 0x51, 0x6b, 0x49, 0x75);
/*
* Notification function, increments the notfication count if parameter
* context is provided.
*
* @event notified event
* @context pointer to the notification count
*/
static void EFIAPI notify(struct efi_event *event, void *context)
{
unsigned int *count = context;
if (count)
++*count;
}
/*
* Check crc32 of a table.
*/
static int check_table(const void *table)
{
efi_status_t ret;
u32 crc32, res;
/* Casting from const to not const */
struct efi_table_hdr *hdr = (struct efi_table_hdr *)table;
crc32 = hdr->crc32;
/*
* Setting the crc32 of the 'const' table to zero is easier than
* copying
*/
hdr->crc32 = 0;
ret = boottime->calculate_crc32(table, hdr->headersize, &res);
/* Reset table crc32 so it stays constant */
hdr->crc32 = crc32;
if (ret != EFI_ST_SUCCESS) {
efi_st_error("CalculateCrc32 failed\n");
return EFI_ST_FAILURE;
}
if (res != crc32) {
efi_st_error("Incorrect CRC32\n");
return EFI_ST_FAILURE;
}
return EFI_ST_SUCCESS;
}
/*
* Setup unit test.
*
* @handle: handle of the loaded image
* @systable: system table
* @return: EFI_ST_SUCCESS for success
*/
static int setup(const efi_handle_t handle,
const struct efi_system_table *systable)
{
sys_table = systable;
boottime = systable->boottime;
return EFI_ST_SUCCESS;
}
/*
* Execute unit test.
*
* A table is installed, updated, removed. The table entry and the
* triggering of events is checked.
*
* @return: EFI_ST_SUCCESS for success
*/
static int execute(void)
{
efi_status_t ret;
unsigned int counter = 0;
struct efi_event *event;
void *table;
const unsigned int tables[2];
efi_uintn_t i;
efi_uintn_t tabcnt;
efi_uintn_t table_count = sys_table->nr_tables;
ret = boottime->create_event_ex(0, TPL_NOTIFY,
notify, (void *)&counter,
&table_guid, &event);
if (ret != EFI_SUCCESS) {
efi_st_error("Failed to create event\n");
return EFI_ST_FAILURE;
}
/* Try to delete non-existent table */
ret = boottime->install_configuration_table(&table_guid, NULL);
if (ret != EFI_NOT_FOUND) {
efi_st_error("Failed to detect missing table\n");
return EFI_ST_FAILURE;
}
if (counter) {
efi_st_error("Notification function was called.\n");
return EFI_ST_FAILURE;
}
/* Check if the event was signaled */
ret = boottime->check_event(event);
if (ret == EFI_SUCCESS) {
efi_st_error("Event was signaled on EFI_NOT_FOUND\n");
return EFI_ST_FAILURE;
}
if (counter != 1) {
efi_st_error("Notification function was not called.\n");
return EFI_ST_FAILURE;
}
if (table_count != sys_table->nr_tables) {
efi_st_error("Incorrect table count %u, expected %u\n",
(unsigned int)sys_table->nr_tables,
(unsigned int)table_count);
return EFI_ST_FAILURE;
}
/* Install table */
ret = boottime->install_configuration_table(&table_guid,
(void *)&tables[0]);
if (ret != EFI_SUCCESS) {
efi_st_error("Failed to install table\n");
return EFI_ST_FAILURE;
}
/* Check signaled state */
ret = boottime->check_event(event);
if (ret != EFI_SUCCESS) {
efi_st_error("Event was not signaled on insert\n");
return EFI_ST_FAILURE;
}
if (++table_count != sys_table->nr_tables) {
efi_st_error("Incorrect table count %u, expected %u\n",
(unsigned int)sys_table->nr_tables,
(unsigned int)table_count);
return EFI_ST_FAILURE;
}
table = NULL;
for (i = 0; i < sys_table->nr_tables; ++i) {
if (!efi_st_memcmp(&sys_table->tables[i].guid, &table_guid,
sizeof(efi_guid_t)))
table = sys_table->tables[i].table;
}
if (!table) {
efi_st_error("Installed table not found\n");
return EFI_ST_FAILURE;
}
if (table != &tables[0]) {
efi_st_error("Incorrect table address\n");
return EFI_ST_FAILURE;
}
if (check_table(sys_table) != EFI_ST_SUCCESS) {
efi_st_error("Checking system table\n");
return EFI_ST_FAILURE;
}
/* Update table */
ret = boottime->install_configuration_table(&table_guid,
(void *)&tables[1]);
if (ret != EFI_SUCCESS) {
efi_st_error("Failed to update table\n");
return EFI_ST_FAILURE;
}
/* Check signaled state */
ret = boottime->check_event(event);
if (ret != EFI_SUCCESS) {
efi_st_error("Event was not signaled on update\n");
return EFI_ST_FAILURE;
}
if (table_count != sys_table->nr_tables) {
efi_st_error("Incorrect table count %u, expected %u\n",
(unsigned int)sys_table->nr_tables,
(unsigned int)table_count);
return EFI_ST_FAILURE;
}
table = NULL;
tabcnt = 0;
for (i = 0; i < sys_table->nr_tables; ++i) {
if (!efi_st_memcmp(&sys_table->tables[i].guid, &table_guid,
sizeof(efi_guid_t))) {
table = sys_table->tables[i].table;
++tabcnt;
}
}
if (!table) {
efi_st_error("Installed table not found\n");
return EFI_ST_FAILURE;
}
if (tabcnt > 1) {
efi_st_error("Duplicate table guid\n");
return EFI_ST_FAILURE;
}
if (table != &tables[1]) {
efi_st_error("Incorrect table address\n");
return EFI_ST_FAILURE;
}
if (check_table(sys_table) != EFI_ST_SUCCESS) {
efi_st_error("Checking system table\n");
return EFI_ST_FAILURE;
}
/* Delete table */
ret = boottime->install_configuration_table(&table_guid, NULL);
if (ret != EFI_SUCCESS) {
efi_st_error("Failed to delete table\n");
return EFI_ST_FAILURE;
}
/* Check signaled state */
ret = boottime->check_event(event);
if (ret != EFI_SUCCESS) {
efi_st_error("Event was not signaled on delete\n");
return EFI_ST_FAILURE;
}
if (--table_count != sys_table->nr_tables) {
efi_st_error("Incorrect table count %u, expected %u\n",
(unsigned int)sys_table->nr_tables,
(unsigned int)table_count);
return EFI_ST_FAILURE;
}
table = NULL;
for (i = 0; i < sys_table->nr_tables; ++i) {
if (!efi_st_memcmp(&sys_table->tables[i].guid, &table_guid,
sizeof(efi_guid_t))) {
table = sys_table->tables[i].table;
}
}
if (table) {
efi_st_error("Wrong table deleted\n");
return EFI_ST_FAILURE;
}
ret = boottime->close_event(event);
if (ret != EFI_SUCCESS) {
efi_st_error("Failed to close event\n");
return EFI_ST_FAILURE;
}
if (check_table(sys_table) != EFI_ST_SUCCESS) {
efi_st_error("Checking system table\n");
return EFI_ST_FAILURE;
}
return EFI_ST_SUCCESS;
}
EFI_UNIT_TEST(configtables) = {
.name = "configuration tables",
.phase = EFI_EXECUTE_BEFORE_BOOTTIME_EXIT,
.setup = setup,
.execute = execute,
};

View file

@ -70,11 +70,12 @@ static void pointer(void *pointer, u16 **buf)
/* /*
* Print an unsigned 32bit value as decimal number to an u16 string * Print an unsigned 32bit value as decimal number to an u16 string
* *
* @value: value to be printed * @value: value to be printed
* @buf: pointer to buffer address * @prec: minimum number of digits to display
* on return position of terminating zero word * @buf: pointer to buffer address
* on return position of terminating zero word
*/ */
static void uint2dec(u32 value, u16 **buf) static void uint2dec(u32 value, int prec, u16 **buf)
{ {
u16 *pos = *buf; u16 *pos = *buf;
int i; int i;
@ -93,7 +94,7 @@ static void uint2dec(u32 value, u16 **buf)
for (i = 0; i < 10; ++i) { for (i = 0; i < 10; ++i) {
/* Write current digit */ /* Write current digit */
c = f >> 60; c = f >> 60;
if (c || pos != *buf) if (c || pos != *buf || 10 - i <= prec)
*pos++ = c + '0'; *pos++ = c + '0';
/* Eliminate current digit */ /* Eliminate current digit */
f &= 0xfffffffffffffff; f &= 0xfffffffffffffff;
@ -109,11 +110,12 @@ static void uint2dec(u32 value, u16 **buf)
/* /*
* Print a signed 32bit value as decimal number to an u16 string * Print a signed 32bit value as decimal number to an u16 string
* *
* @value: value to be printed * @value: value to be printed
* @buf: pointer to buffer address * @prec: minimum number of digits to display
* @buf: pointer to buffer address
* on return position of terminating zero word * on return position of terminating zero word
*/ */
static void int2dec(s32 value, u16 **buf) static void int2dec(s32 value, int prec, u16 **buf)
{ {
u32 u; u32 u;
u16 *pos = *buf; u16 *pos = *buf;
@ -124,7 +126,7 @@ static void int2dec(s32 value, u16 **buf)
} else { } else {
u = value; u = value;
} }
uint2dec(u, &pos); uint2dec(u, prec, &pos);
*buf = pos; *buf = pos;
} }
@ -143,6 +145,7 @@ void efi_st_printc(int color, const char *fmt, ...)
u16 *pos = buf; u16 *pos = buf;
const char *s; const char *s;
u16 *u; u16 *u;
int prec;
va_start(args, fmt); va_start(args, fmt);
@ -172,12 +175,20 @@ void efi_st_printc(int color, const char *fmt, ...)
break; break;
case '%': case '%':
++c; ++c;
/* Parse precision */
if (*c == '.') {
++c;
prec = *c - '0';
++c;
} else {
prec = 0;
}
switch (*c) { switch (*c) {
case '\0': case '\0':
--c; --c;
break; break;
case 'd': case 'd':
int2dec(va_arg(args, s32), &pos); int2dec(va_arg(args, s32), prec, &pos);
break; break;
case 'p': case 'p':
++c; ++c;
@ -209,7 +220,7 @@ void efi_st_printc(int color, const char *fmt, ...)
*pos++ = *s; *pos++ = *s;
break; break;
case 'u': case 'u':
uint2dec(va_arg(args, u32), &pos); uint2dec(va_arg(args, u32), prec, &pos);
break; break;
default: default:
break; break;

View file

@ -0,0 +1,141 @@
// SPDX-License-Identifier: GPL-2.0+
/*
* efi_selftest_crc32
*
* Copyright (c) 2018 Heinrich Schuchardt <xypron.glpk@gmx.de>
*
* This unit test checks the CalculateCrc32 bootservice and checks the
* headers of the system table, the boot services tablle, and the runtime
* services table before and after ExitBootServices().
*/
#include <efi_selftest.h>
const struct efi_system_table *st;
efi_status_t (EFIAPI *bs_crc32)(const void *data, efi_uintn_t data_size,
u32 *crc32);
static int check_table(const void *table)
{
efi_status_t ret;
u32 crc32, res;
/* Casting from const to not const */
struct efi_table_hdr *hdr = (struct efi_table_hdr *)table;
if (!hdr->signature) {
efi_st_error("Missing header signature\n");
return EFI_ST_FAILURE;
}
if (!hdr->revision) {
efi_st_error("Missing header revision\n");
return EFI_ST_FAILURE;
}
if (hdr->headersize <= sizeof(struct efi_table_hdr)) {
efi_st_error("Incorrect headersize value\n");
return EFI_ST_FAILURE;
}
if (hdr->reserved) {
efi_st_error("Reserved header field is not zero\n");
return EFI_ST_FAILURE;
}
crc32 = hdr->crc32;
/*
* Setting the crc32 of the 'const' table to zero is easier than
* copying
*/
hdr->crc32 = 0;
ret = bs_crc32(table, hdr->headersize, &res);
/* Reset table crc32 so it stays constant */
hdr->crc32 = crc32;
if (ret != EFI_ST_SUCCESS) {
efi_st_error("CalculateCrc32 failed\n");
return EFI_ST_FAILURE;
}
if (res != crc32) {
efi_st_error("Incorrect CRC32\n");
// return EFI_ST_FAILURE;
}
return EFI_ST_SUCCESS;
}
/*
* Setup unit test.
*
* Check that CalculateCrc32 is working correctly.
* Check tables before ExitBootServices().
*
* @handle: handle of the loaded image
* @systable: system table
* @return: EFI_ST_SUCCESS for success
*/
static int setup(const efi_handle_t handle,
const struct efi_system_table *systable)
{
efi_status_t ret;
u32 res;
st = systable;
bs_crc32 = systable->boottime->calculate_crc32;
/* Check that CalculateCrc32 is working */
ret = bs_crc32("U-Boot", 6, &res);
if (ret != EFI_ST_SUCCESS) {
efi_st_error("CalculateCrc32 failed\n");
return EFI_ST_FAILURE;
}
if (res != 0x134b0db4) {
efi_st_error("Incorrect CRC32\n");
return EFI_ST_FAILURE;
}
/* Check tables before ExitBootServices() */
if (check_table(st) != EFI_ST_SUCCESS) {
efi_st_error("Checking system table\n");
return EFI_ST_FAILURE;
}
if (check_table(st->boottime) != EFI_ST_SUCCESS) {
efi_st_error("Checking boottime table\n");
return EFI_ST_FAILURE;
}
if (check_table(st->runtime) != EFI_ST_SUCCESS) {
efi_st_error("Checking runtime table\n");
return EFI_ST_FAILURE;
}
return EFI_ST_SUCCESS;
}
/*
* Execute unit test
*
* Check tables after ExitBootServices()
*
* @return: EFI_ST_SUCCESS for success
*/
static int execute(void)
{
if (check_table(st) != EFI_ST_SUCCESS) {
efi_st_error("Checking system table\n");
return EFI_ST_FAILURE;
}
if (check_table(st->runtime) != EFI_ST_SUCCESS) {
efi_st_error("Checking runtime table\n");
return EFI_ST_FAILURE;
}
/*
* We cannot call SetVirtualAddressMap() and recheck the runtime
* table afterwards because this would invalidate the addresses of the
* unit tests.
*/
return EFI_ST_SUCCESS;
}
EFI_UNIT_TEST(crc32) = {
.name = "crc32",
.phase = EFI_SETUP_BEFORE_BOOTTIME_EXIT,
.setup = setup,
.execute = execute,
};

View file

@ -0,0 +1,67 @@
// SPDX-License-Identifier: GPL-2.0+
/*
* efi_selftest_rtc
*
* Copyright (c) 2018 Heinrich Schuchardt <xypron.glpk@gmx.de>
*
* Test the real time clock runtime services.
*/
#include <efi_selftest.h>
#define EFI_ST_NO_RTC "Could not read real time clock\n"
static struct efi_runtime_services *runtime;
/*
* Setup unit test.
*
* @handle: handle of the loaded image
* @systable: system table
* @return: EFI_ST_SUCCESS for success
*/
static int setup(const efi_handle_t handle,
const struct efi_system_table *systable)
{
runtime = systable->runtime;
return EFI_ST_SUCCESS;
}
/*
* Execute unit test.
*
* Display current time.
*
* @return: EFI_ST_SUCCESS for success
*/
static int execute(void)
{
efi_status_t ret;
struct efi_time tm;
/* Display current time */
ret = runtime->get_time(&tm, NULL);
if (ret != EFI_SUCCESS) {
#ifdef CONFIG_CMD_DATE
efi_st_error(EFI_ST_NO_RTC);
return EFI_ST_FAILURE;
#else
efi_st_todo(EFI_ST_NO_RTC);
return EFI_ST_SUCCESS;
#endif
} else {
efi_st_printf("Time according to real time clock: "
"%.4u-%.2u-%.2u %.2u:%.2u:%.2u\n",
tm.year, tm.month, tm.day,
tm.hour, tm.minute, tm.second);
}
return EFI_ST_SUCCESS;
}
EFI_UNIT_TEST(rtc) = {
.name = "real time clock",
.phase = EFI_EXECUTE_BEFORE_BOOTTIME_EXIT,
.setup = setup,
.execute = execute,
};

View file

@ -407,7 +407,10 @@ static char *uuid_string(char *buf, char *end, u8 *addr, int field_width,
break; break;
} }
uuid_bin_to_str(addr, uuid, str_format); if (addr)
uuid_bin_to_str(addr, uuid, str_format);
else
strcpy(uuid, "<NULL>");
return string(buf, end, uuid, field_width, precision, flags); return string(buf, end, uuid, field_width, precision, flags);
} }