Commit a37dac5c authored by Ard Biesheuvel's avatar Ard Biesheuvel

arm64: efi: Limit allocations to 48-bit addressable physical region

The UEFI spec does not mention or reason about the configured size of
the virtual address space at all, but it does mention that all memory
should be identity mapped using a page size of 4 KiB.

This means that a LPA2 capable system that has any system memory outside
of the 48-bit addressable physical range and follows the spec to the
letter may serve page allocation requests from regions of memory that
the kernel cannot access unless it was built with LPA2 support and
enables it at runtime.

So let's ensure that all page allocations are limited to the 48-bit
range.
Signed-off-by: default avatarArd Biesheuvel <ardb@kernel.org>
parent d9f26ae7
...@@ -92,6 +92,7 @@ static inline unsigned long efi_get_kimg_min_align(void) ...@@ -92,6 +92,7 @@ static inline unsigned long efi_get_kimg_min_align(void)
} }
#define EFI_ALLOC_ALIGN SZ_64K #define EFI_ALLOC_ALIGN SZ_64K
#define EFI_ALLOC_LIMIT ((1UL << 48) - 1)
/* /*
* On ARM systems, virtually remapped UEFI runtime services are set up in two * On ARM systems, virtually remapped UEFI runtime services are set up in two
......
...@@ -29,6 +29,8 @@ efi_status_t efi_allocate_pages_aligned(unsigned long size, unsigned long *addr, ...@@ -29,6 +29,8 @@ efi_status_t efi_allocate_pages_aligned(unsigned long size, unsigned long *addr,
efi_status_t status; efi_status_t status;
int slack; int slack;
max = min(max, EFI_ALLOC_LIMIT);
if (align < EFI_ALLOC_ALIGN) if (align < EFI_ALLOC_ALIGN)
align = EFI_ALLOC_ALIGN; align = EFI_ALLOC_ALIGN;
......
...@@ -113,10 +113,11 @@ efi_status_t handle_kernel_image(unsigned long *image_addr, ...@@ -113,10 +113,11 @@ efi_status_t handle_kernel_image(unsigned long *image_addr,
if (status != EFI_SUCCESS) { if (status != EFI_SUCCESS) {
if (!check_image_region((u64)_text, kernel_memsize)) { if (!check_image_region((u64)_text, kernel_memsize)) {
efi_err("FIRMWARE BUG: Image BSS overlaps adjacent EFI memory region\n"); efi_err("FIRMWARE BUG: Image BSS overlaps adjacent EFI memory region\n");
} else if (IS_ALIGNED((u64)_text, min_kimg_align)) { } else if (IS_ALIGNED((u64)_text, min_kimg_align) &&
(u64)_end < EFI_ALLOC_LIMIT) {
/* /*
* Just execute from wherever we were loaded by the * Just execute from wherever we were loaded by the
* UEFI PE/COFF loader if the alignment is suitable. * UEFI PE/COFF loader if the placement is suitable.
*/ */
*image_addr = (u64)_text; *image_addr = (u64)_text;
*reserve_size = 0; *reserve_size = 0;
......
...@@ -29,6 +29,10 @@ ...@@ -29,6 +29,10 @@
#define EFI_ALLOC_ALIGN EFI_PAGE_SIZE #define EFI_ALLOC_ALIGN EFI_PAGE_SIZE
#endif #endif
#ifndef EFI_ALLOC_LIMIT
#define EFI_ALLOC_LIMIT ULONG_MAX
#endif
extern bool efi_nochunk; extern bool efi_nochunk;
extern bool efi_nokaslr; extern bool efi_nokaslr;
extern int efi_loglevel; extern int efi_loglevel;
......
...@@ -89,6 +89,8 @@ efi_status_t efi_allocate_pages(unsigned long size, unsigned long *addr, ...@@ -89,6 +89,8 @@ efi_status_t efi_allocate_pages(unsigned long size, unsigned long *addr,
efi_physical_addr_t alloc_addr; efi_physical_addr_t alloc_addr;
efi_status_t status; efi_status_t status;
max = min(max, EFI_ALLOC_LIMIT);
if (EFI_ALLOC_ALIGN > EFI_PAGE_SIZE) if (EFI_ALLOC_ALIGN > EFI_PAGE_SIZE)
return efi_allocate_pages_aligned(size, addr, max, return efi_allocate_pages_aligned(size, addr, max,
EFI_ALLOC_ALIGN, EFI_ALLOC_ALIGN,
......
...@@ -29,7 +29,7 @@ static unsigned long get_entry_num_slots(efi_memory_desc_t *md, ...@@ -29,7 +29,7 @@ static unsigned long get_entry_num_slots(efi_memory_desc_t *md,
return 0; return 0;
region_end = min(md->phys_addr + md->num_pages * EFI_PAGE_SIZE - 1, region_end = min(md->phys_addr + md->num_pages * EFI_PAGE_SIZE - 1,
(u64)ULONG_MAX); (u64)EFI_ALLOC_LIMIT);
if (region_end < size) if (region_end < size)
return 0; return 0;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment