Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src-draft/trunk]: src/sys/arch/evbarm/fdt Teach arm fdt kernel to use /chose...
details:   https://anonhg.NetBSD.org/src-all/rev/08f59c9070cf
branches:  trunk
changeset: 932724:08f59c9070cf
user:      Taylor R Campbell <riastradh%NetBSD.org@localhost>
date:      Mon May 11 02:12:09 2020 +0000
description:
Teach arm fdt kernel to use /chosen/netbsd,efirng data.
Feed it in as a separate random source, with zero entropy since this
is a best-effort fallback for devices we really don't know anything
about.
diffstat:
 sys/arch/evbarm/fdt/fdt_machdep.c |  76 +++++++++++++++++++++++++++++++++++++-
 1 files changed, 74 insertions(+), 2 deletions(-)
diffs (122 lines):
diff -r 996ce8f9ccff -r 08f59c9070cf sys/arch/evbarm/fdt/fdt_machdep.c
--- a/sys/arch/evbarm/fdt/fdt_machdep.c Mon May 11 02:11:09 2020 +0000
+++ b/sys/arch/evbarm/fdt/fdt_machdep.c Mon May 11 02:12:09 2020 +0000
@@ -65,6 +65,7 @@
 #include <sys/md5.h>
 #include <sys/pserialize.h>
 #include <sys/rnd.h>
+#include <sys/rndsource.h>
 
 #include <net/if.h>
 #include <net/if_dl.h>
@@ -119,7 +120,8 @@
 const uint8_t *fdt_addr_r __attribute__((__section__(".data")));
 
 static uint64_t initrd_start, initrd_end;
-static uint64_t rndseed_start, rndseed_end;
+static uint64_t rndseed_start, rndseed_end; /* our on-disk seed */
+static uint64_t efirng_start, efirng_end;   /* firmware's EFI RNG output */
 
 #include <libfdt.h>
 #include <dev/fdt/fdtvar.h>
@@ -313,6 +315,10 @@
        if (rndseed_size > 0)
                fdt_memory_remove_range(rndseed_start, rndseed_size);
 
+       const uint64_t efirng_size = efirng_end - efirng_start;
+       if (efirng_size > 0)
+               fdt_memory_remove_range(efirng_start, efirng_size);
+
        const int framebuffer = OF_finddevice("/chosen/framebuffer");
        if (framebuffer >= 0) {
                for (index = 0;
@@ -451,6 +457,70 @@
        rnd_seed(rndseed, rndseed_size);
 }
 
+static void
+fdt_probe_efirng(uint64_t *pstart, uint64_t *pend)
+{
+       int chosen, len;
+       const void *start_data, *end_data;
+
+       *pstart = *pend = 0;
+       chosen = OF_finddevice("/chosen");
+       if (chosen < 0)
+               return;
+
+       start_data = fdtbus_get_prop(chosen, "netbsd,efirng-start", &len);
+       end_data = fdtbus_get_prop(chosen, "netbsd,efirng-end", NULL);
+       if (start_data == NULL || end_data == NULL)
+               return;
+
+       switch (len) {
+       case 4:
+               *pstart = be32dec(start_data);
+               *pend = be32dec(end_data);
+               break;
+       case 8:
+               *pstart = be64dec(start_data);
+               *pend = be64dec(end_data);
+               break;
+       default:
+               printf("Unsupported len %d for /chosen/efirng-start\n", len);
+               return;
+       }
+}
+
+static struct krndsource efirng_source;
+
+static void
+fdt_setup_efirng(void)
+{
+       const uint64_t efirng_size = efirng_end - efirng_start;
+       const paddr_t startpa = trunc_page(efirng_start);
+       const paddr_t endpa = round_page(efirng_end);
+       paddr_t pa;
+       vaddr_t va;
+       void *efirng;
+
+       if (efirng_size == 0)
+               return;
+
+       va = uvm_km_alloc(kernel_map, endpa - startpa, 0,
+           UVM_KMF_VAONLY | UVM_KMF_NOWAIT);
+       if (va == 0) {
+               printf("Failed to allocate VA for efirng\n");
+               return;
+       }
+       efirng = (void *)va;
+
+       for (pa = startpa; pa < endpa; pa += PAGE_SIZE, va += PAGE_SIZE)
+               pmap_kenter_pa(va, pa, VM_PROT_READ|VM_PROT_WRITE, 0);
+       pmap_update(pmap_kernel());
+
+       rnd_attach_source(&efirng_source, "efirng", RND_TYPE_RNG,
+           RND_FLAG_DEFAULT);
+       rnd_add_data(&efirng_source, efirng, efirng_size, 0);
+       explicit_memset(efirng, 0, efirng_size);
+}
+
 #ifdef EFI_RUNTIME
 static void
 fdt_map_efi_runtime(const char *prop, enum arm_efirt_mem_type type)
@@ -579,8 +649,9 @@
        /* Parse ramdisk info */
        fdt_probe_initrd(&initrd_start, &initrd_end);
 
-       /* Parse rndseed */
+       /* Parse our on-disk rndseed and the firmware's RNG from EFI */
        fdt_probe_rndseed(&rndseed_start, &rndseed_end);
+       fdt_probe_efirng(&efirng_start, &efirng_end);
 
        /*
         * Populate bootconfig structure for the benefit of
@@ -699,6 +770,7 @@
        fdtbus_intr_init();
 
        fdt_setup_rndseed();
+       fdt_setup_efirng();
 }
 
 void
Home |
Main Index |
Thread Index |
Old Index