Source-Changes-HG archive

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]

[src/trunk]: src/sys Consolidate entropy on RNDADDDATA and writes to /dev/ran...



details:   https://anonhg.NetBSD.org/src/rev/f0ce5513bff7
branches:  trunk
changeset: 1009922:f0ce5513bff7
user:      riastradh <riastradh%NetBSD.org@localhost>
date:      Thu May 07 19:05:51 2020 +0000

description:
Consolidate entropy on RNDADDDATA and writes to /dev/random.

The man page for some time has advertised:

  Writing to either /dev/random or /dev/urandom influences subsequent
  output of both devices, guaranteed to take effect at next open.

So let's make that true again.

It is a conscious choice _not_ to consolidate entropy frequently.
For example, if you have a _slow_ HWRNG, which provides 32 bits of
entropy every few seconds, and you reveal a hash that to the
adversary before any more comes in, the adversary can in principle
just keep guessing the intermediate state by a brute force search
over ~2^32 possibilities.

To mitigate this, the kernel generally tries to avoid consolidating
entropy from the per-CPU pools until doing so would bring us from
zero entropy to full entropy.

However, there are various _possible_ sources of entropy which are
just hard to give honest estimates for that are valid on ~all
machines -- like interrupt timings.  The time at which we read a seed
in, which usually happens via /etc/rc.d/random_seed early in
userland, is a reasonable time to gather this up.  An operator or
system engineer who knows another opportune moment can always issue
`sysctl -w kern.entropy.consolidate=1'.

Prompted by a suggestion from nia@ to consolidate entropy at the
first transition to userland.  I chose not to do that because it
would likely cause warning fatigue on systems that are perfectly fine
with a random seed -- doing it this way instead lets rndctl -L
trigger the consolidation automatically.  A subsequent commit will
reorder the operations in rndctl again to make it work out better.

diffstat:

 sys/dev/random.c        |  12 +++++-
 sys/kern/kern_entropy.c |  81 ++++++++++++++++++++++++++++++------------------
 sys/sys/entropy.h       |   3 +-
 3 files changed, 61 insertions(+), 35 deletions(-)

diffs (239 lines):

diff -r b7d803c01f40 -r f0ce5513bff7 sys/dev/random.c
--- a/sys/dev/random.c  Thu May 07 18:15:29 2020 +0000
+++ b/sys/dev/random.c  Thu May 07 19:05:51 2020 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: random.c,v 1.2 2020/04/30 04:26:29 riastradh Exp $     */
+/*     $NetBSD: random.c,v 1.3 2020/05/07 19:05:51 riastradh Exp $     */
 
 /*-
  * Copyright (c) 2019 The NetBSD Foundation, Inc.
@@ -47,7 +47,7 @@
  */
 
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: random.c,v 1.2 2020/04/30 04:26:29 riastradh Exp $");
+__KERNEL_RCSID(0, "$NetBSD: random.c,v 1.3 2020/05/07 19:05:51 riastradh Exp $");
 
 #include <sys/param.h>
 #include <sys/types.h>
@@ -384,7 +384,7 @@
 {
        kauth_cred_t cred = kauth_cred_get();
        uint8_t *buf;
-       bool privileged = false;
+       bool privileged = false, any = false;
        int error = 0;
 
        /* Verify user's authorization to affect the entropy pool.  */
@@ -429,10 +429,16 @@
                if (error)
                        break;
                rnd_add_data(&user_rndsource, buf, n, privileged ? n*NBBY : 0);
+               any = true;
        }
 
        /* Zero the buffer and return it to the pool cache.  */
        explicit_memset(buf, 0, RANDOM_BUFSIZE);
        pool_cache_put(random_buf_pc, buf);
+
+       /* If we added anything, consolidate entropy now.  */
+       if (any)
+               entropy_consolidate();
+
        return error;
 }
diff -r b7d803c01f40 -r f0ce5513bff7 sys/kern/kern_entropy.c
--- a/sys/kern/kern_entropy.c   Thu May 07 18:15:29 2020 +0000
+++ b/sys/kern/kern_entropy.c   Thu May 07 19:05:51 2020 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: kern_entropy.c,v 1.12 2020/05/07 00:55:13 riastradh Exp $      */
+/*     $NetBSD: kern_entropy.c,v 1.13 2020/05/07 19:05:51 riastradh Exp $      */
 
 /*-
  * Copyright (c) 2019 The NetBSD Foundation, Inc.
@@ -60,9 +60,7 @@
  *       transition from partial entropy to full entropy, so that
  *       users can easily determine when to reseed.  This also
  *       facilitates an operator explicitly causing everything to
- *       reseed by sysctl -w kern.entropy.consolidate=1, e.g. if they
- *       just flipped a coin 256 times and wrote `echo tthhhhhthh... >
- *       /dev/random'.
+ *       reseed by sysctl -w kern.entropy.consolidate=1.
  *
  *     * No entropy estimation based on the sample values, which is a
  *       contradiction in terms and a potential source of side
@@ -77,7 +75,7 @@
  */
 
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: kern_entropy.c,v 1.12 2020/05/07 00:55:13 riastradh Exp $");
+__KERNEL_RCSID(0, "$NetBSD: kern_entropy.c,v 1.13 2020/05/07 19:05:51 riastradh Exp $");
 
 #include <sys/param.h>
 #include <sys/types.h>
@@ -241,8 +239,8 @@
 static void    entropy_thread(void *);
 static uint32_t        entropy_pending(void);
 static void    entropy_pending_cpu(void *, void *, struct cpu_info *);
-static void    entropy_consolidate(void);
-static void    entropy_gather_xc(void *, void *);
+static void    entropy_do_consolidate(void);
+static void    entropy_consolidate_xc(void *, void *);
 static void    entropy_notify(void);
 static int     sysctl_entropy_consolidate(SYSCTLFN_ARGS);
 static int     sysctl_entropy_gather(SYSCTLFN_ARGS);
@@ -959,7 +957,7 @@
 
                if (consolidate) {
                        /* Do it.  */
-                       entropy_consolidate();
+                       entropy_do_consolidate();
 
                        /* Mitigate abuse.  */
                        kpause("entropy", false, hz, NULL);
@@ -993,13 +991,13 @@
 }
 
 /*
- * entropy_consolidate()
+ * entropy_do_consolidate()
  *
  *     Issue a cross-call to gather entropy on all CPUs and advance
  *     the entropy epoch.
  */
 static void
-entropy_consolidate(void)
+entropy_do_consolidate(void)
 {
        static const struct timeval interval = {.tv_sec = 60, .tv_usec = 0};
        static struct timeval lasttime; /* serialized by E->lock */
@@ -1007,7 +1005,7 @@
        uint64_t ticket;
 
        /* Gather entropy on all CPUs.  */
-       ticket = xc_broadcast(0, &entropy_gather_xc, NULL, NULL);
+       ticket = xc_broadcast(0, &entropy_consolidate_xc, NULL, NULL);
        xc_wait(ticket);
 
        /* Acquire the lock to notify waiters.  */
@@ -1037,13 +1035,13 @@
 }
 
 /*
- * entropy_gather_xc(arg1, arg2)
+ * entropy_consolidate_xc(arg1, arg2)
  *
  *     Extract output from the local CPU's input pool and enter it
  *     into the global pool.
  */
 static void
-entropy_gather_xc(void *arg1 __unused, void *arg2 __unused)
+entropy_consolidate_xc(void *arg1 __unused, void *arg2 __unused)
 {
        struct entropy_cpu *ec;
        uint8_t buf[ENTPOOL_CAPACITY];
@@ -1144,19 +1142,49 @@
 }
 
 /*
+ * entropy_consolidate()
+ *
+ *     Trigger entropy consolidation and wait for it to complete.
+ *
+ *     This should be used sparingly, not periodically -- requiring
+ *     conscious intervention by the operator or a clear policy
+ *     decision.  Otherwise, the kernel will automatically consolidate
+ *     when enough entropy has been gathered into per-CPU pools to
+ *     transition to full entropy.
+ */
+void
+entropy_consolidate(void)
+{
+       uint64_t ticket;
+       int error;
+
+       KASSERT(E->stage == ENTROPY_HOT);
+
+       mutex_enter(&E->lock);
+       ticket = entropy_consolidate_evcnt.ev_count;
+       E->consolidate = true;
+       cv_broadcast(&E->cv);
+       while (ticket == entropy_consolidate_evcnt.ev_count) {
+               error = cv_wait_sig(&E->cv, &E->lock);
+               if (error)
+                       break;
+       }
+       mutex_exit(&E->lock);
+}
+
+/*
  * sysctl -w kern.entropy.consolidate=1
  *
  *     Trigger entropy consolidation and wait for it to complete.
- *     Writable only by superuser.  This is the only way for the
- *     system to consolidate entropy if the operator knows something
- *     the kernel doesn't about how unpredictable the pending entropy
- *     pools are.
+ *     Writable only by superuser.  This, writing to /dev/random, and
+ *     ioctl(RNDADDDATA) are the only ways for the system to
+ *     consolidate entropy if the operator knows something the kernel
+ *     doesn't about how unpredictable the pending entropy pools are.
  */
 static int
 sysctl_entropy_consolidate(SYSCTLFN_ARGS)
 {
        struct sysctlnode node = *rnode;
-       uint64_t ticket;
        int arg;
        int error;
 
@@ -1166,18 +1194,8 @@
        error = sysctl_lookup(SYSCTLFN_CALL(&node));
        if (error || newp == NULL)
                return error;
-       if (arg) {
-               mutex_enter(&E->lock);
-               ticket = entropy_consolidate_evcnt.ev_count;
-               E->consolidate = true;
-               cv_broadcast(&E->cv);
-               while (ticket == entropy_consolidate_evcnt.ev_count) {
-                       error = cv_wait_sig(&E->cv, &E->lock);
-                       if (error)
-                               break;
-               }
-               mutex_exit(&E->lock);
-       }
+       if (arg)
+               entropy_consolidate();
 
        return error;
 }
@@ -2214,9 +2232,10 @@
                        mutex_exit(&E->lock);
                }
 
-               /* Enter the data.  */
+               /* Enter the data and consolidate entropy.  */
                rnd_add_data(&seed_rndsource, rdata->data, rdata->len,
                    entropybits);
+               entropy_consolidate();
                break;
        }
        default:
diff -r b7d803c01f40 -r f0ce5513bff7 sys/sys/entropy.h
--- a/sys/sys/entropy.h Thu May 07 18:15:29 2020 +0000
+++ b/sys/sys/entropy.h Thu May 07 19:05:51 2020 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: entropy.h,v 1.1 2020/04/30 03:28:19 riastradh Exp $    */
+/*     $NetBSD: entropy.h,v 1.2 2020/05/07 19:05:51 riastradh Exp $    */
 
 /*-
  * Copyright (c) 2019 The NetBSD Foundation, Inc.
@@ -48,6 +48,7 @@
 #define        ENTROPY_SIG     0x02
 
 void   entropy_bootrequest(void);
+void   entropy_consolidate(void);
 unsigned entropy_epoch(void);
 int    entropy_extract(void *, size_t, int);
 int    entropy_poll(int);



Home | Main Index | Thread Index | Old Index