]> git.dujemihanovic.xyz Git - u-boot.git/commitdiff
efi_loader: Get rid of kaslr-seed if EFI_RNG_PROTOCOL is installed
authorIlias Apalodimas <ilias.apalodimas@linaro.org>
Mon, 3 Jan 2022 12:07:37 +0000 (14:07 +0200)
committerHeinrich Schuchardt <heinrich.schuchardt@canonical.com>
Sat, 15 Jan 2022 09:57:22 +0000 (10:57 +0100)
U-Boot, in some occasions, injects a 'kaslr-seed' property on the /chosen
node. That would be problematic in case we want to measure the DTB we
install in the configuration table, since it would change across reboots.

The Linux kernel EFI-stub completely ignores it and only relies on
EFI_RNG_PROTOCOL for it's own randomness needs (i.e the randomization
of the physical placement of the kernel). In fact it (blindly) overwrites
the existing seed if the protocol is installed. However it still uses it
for randomizing it's virtual placement.
So let's get rid of it in the presence of the RNG protocol.

It's worth noting that TPMs also provide an RNG.  So if we tweak our
EFI_RNG_PROTOCOL slightly and install the protocol when a TPM device
is present the 'kaslr-seed' property will always be removed, allowing
us to reliably measure our DTB.

Acked-by: Ard Biesheuvel <ardb@kernel.org>
Signed-off-by: Ilias Apalodimas <ilias.apalodimas@linaro.org>
Reviewed-by: Mark Kettenis <kettenis@openbsd.org>
cmd/bootefi.c
include/efi_loader.h
lib/efi_loader/efi_dt_fixup.c

index 83eab0bd7f180a5dd93ba0ae2ecb974e91f692a7..3a8b2b6061848ceaf64d7348b699e0f35723b48c 100644 (file)
@@ -310,6 +310,8 @@ efi_status_t efi_install_fdt(void *fdt)
        /* Create memory reservations as indicated by the device tree */
        efi_carve_out_dt_rsv(fdt);
 
+       efi_try_purge_kaslr_seed(fdt);
+
        /* Install device tree as UEFI table */
        ret = efi_install_configuration_table(&efi_guid_fdt, fdt);
        if (ret != EFI_SUCCESS) {
index f4860e87fc1f118a76c40ff199ab75f6e2b9fb6f..f20d3618765f8f36ec77d4d102cc8503602c459a 100644 (file)
@@ -517,6 +517,8 @@ efi_status_t EFIAPI efi_convert_pointer(efi_uintn_t debug_disposition,
                                        void **address);
 /* Carve out DT reserved memory ranges */
 void efi_carve_out_dt_rsv(void *fdt);
+/* Purge unused kaslr-seed */
+void efi_try_purge_kaslr_seed(void *fdt);
 /* Called by bootefi to make console interface available */
 efi_status_t efi_console_register(void);
 /* Called by bootefi to make all disk storage accessible as EFI objects */
index b6fe5d2e5a34d49ce00e6dbe7a6311749cb0dcc5..d3923e5dba1b5e0f7f7f7347a901169004234915 100644 (file)
@@ -8,6 +8,7 @@
 #include <common.h>
 #include <efi_dt_fixup.h>
 #include <efi_loader.h>
+#include <efi_rng.h>
 #include <fdtdec.h>
 #include <mapmem.h>
 
@@ -40,6 +41,38 @@ static void efi_reserve_memory(u64 addr, u64 size, bool nomap)
                        addr, size);
 }
 
+/**
+ * efi_try_purge_kaslr_seed() - Remove unused kaslr-seed
+ *
+ * Kernel's EFI STUB only relies on EFI_RNG_PROTOCOL for randomization
+ * and completely ignores the kaslr-seed for its own randomness needs
+ * (i.e the randomization of the physical placement of the kernel).
+ * Weed it out from the DTB we hand over, which would mess up our DTB
+ * TPM measurements as well.
+ *
+ * @fdt: Pointer to device tree
+ */
+void efi_try_purge_kaslr_seed(void *fdt)
+{
+       const efi_guid_t efi_guid_rng_protocol = EFI_RNG_PROTOCOL_GUID;
+       struct efi_handler *handler;
+       efi_status_t ret;
+       int nodeoff = 0;
+       int err = 0;
+
+       ret = efi_search_protocol(efi_root, &efi_guid_rng_protocol, &handler);
+       if (ret != EFI_SUCCESS)
+               return;
+
+       nodeoff = fdt_path_offset(fdt, "/chosen");
+       if (nodeoff < 0)
+               return;
+
+       err = fdt_delprop(fdt, nodeoff, "kaslr-seed");
+       if (err < 0 && err != -FDT_ERR_NOTFOUND)
+               log_err("Error deleting kaslr-seed\n");
+}
+
 /**
  * efi_carve_out_dt_rsv() - Carve out DT reserved memory ranges
  *