Source-Changes-HG archive

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

[src/trunk]: src/sbin/rndctl Teach `rndctl -L' to update the seed file, not j...



details:   https://anonhg.NetBSD.org/src/rev/ebf2d87204bc
branches:  trunk
changeset: 1005388:ebf2d87204bc
user:      riastradh <riastradh%NetBSD.org@localhost>
date:      Fri Dec 06 14:43:18 2019 +0000

description:
Teach `rndctl -L' to update the seed file, not just delete it.

The seed file is updated by entering the old seed into the system and
then hashing the old seed together with data from /dev/urandom, and
writing it atomically with write-to-temporary/rename-to-permanent.

This way, interruption by crash or power loss does not obliterate
your persistent entropy (unless it causes file system corruption).

diffstat:

 sbin/rndctl/Makefile    |   14 ++-
 sbin/rndctl/namespace.h |    1 +
 sbin/rndctl/rndctl.8    |   12 +-
 sbin/rndctl/rndctl.c    |  260 +++++++++++++++++++++++++++++++++++------------
 4 files changed, 212 insertions(+), 75 deletions(-)

diffs (truncated from 396 to 300 lines):

diff -r 58e2057cb2e1 -r ebf2d87204bc sbin/rndctl/Makefile
--- a/sbin/rndctl/Makefile      Fri Dec 06 12:46:06 2019 +0000
+++ b/sbin/rndctl/Makefile      Fri Dec 06 14:43:18 2019 +0000
@@ -1,8 +1,20 @@
-#      $NetBSD: Makefile,v 1.3 2019/10/13 07:28:13 mrg Exp $
+#      $NetBSD: Makefile,v 1.4 2019/12/06 14:43:18 riastradh Exp $
 
 PROG=  rndctl
 MAN=   rndctl.8
 
 COPTS.rndctl.c+=       ${GCC_NO_STRINGOP_TRUNCATION}
 
+SRCS+= rndctl.c
+
+# Hack: libc does not export public SHA-3 symbols, so we'll just copy
+# them here statically.
+.PATH: ${NETBSDSRCDIR}/common/lib/libc/hash/sha3
+
+# Hack for namespace.h in sha3.c.
+CPPFLAGS+=     -I${.CURDIR}
+
+SRCS+= sha3.c
+SRCS+= keccak.c
+
 .include <bsd.prog.mk>
diff -r 58e2057cb2e1 -r ebf2d87204bc sbin/rndctl/namespace.h
--- /dev/null   Thu Jan 01 00:00:00 1970 +0000
+++ b/sbin/rndctl/namespace.h   Fri Dec 06 14:43:18 2019 +0000
@@ -0,0 +1,1 @@
+/* dummy for sha3.c */
diff -r 58e2057cb2e1 -r ebf2d87204bc sbin/rndctl/rndctl.8
--- a/sbin/rndctl/rndctl.8      Fri Dec 06 12:46:06 2019 +0000
+++ b/sbin/rndctl/rndctl.8      Fri Dec 06 14:43:18 2019 +0000
@@ -1,4 +1,4 @@
-.\"    $NetBSD: rndctl.8,v 1.22 2014/08/10 17:13:02 wiz Exp $
+.\"    $NetBSD: rndctl.8,v 1.23 2019/12/06 14:43:18 riastradh Exp $
 .\"
 .\" Copyright (c) 1997 Michael Graff
 .\" All rights reserved.
@@ -78,9 +78,13 @@
 for the given device name or device type.
 .It Fl L
 Load saved entropy from file
-.Ar save-file ,
-which will be overwritten and deleted before the entropy is loaded into
-the kernel.
+.Ar save-file
+and overwrite it with a seed derived by hashing it together with output
+from
+.Pa /dev/urandom
+so that the new seed has at least as much entropy as either the old
+seed had or the system already has.
+If interrupted, either the old seed or the new seed will be in place.
 .It Fl l
 List all sources, or, if the
 .Fl t
diff -r 58e2057cb2e1 -r ebf2d87204bc sbin/rndctl/rndctl.c
--- a/sbin/rndctl/rndctl.c      Fri Dec 06 12:46:06 2019 +0000
+++ b/sbin/rndctl/rndctl.c      Fri Dec 06 14:43:18 2019 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: rndctl.c,v 1.30 2015/04/13 22:18:50 riastradh Exp $    */
+/*     $NetBSD: rndctl.c,v 1.31 2019/12/06 14:43:18 riastradh Exp $    */
 
 /*-
  * Copyright (c) 1997 Michael Graff.
@@ -33,7 +33,7 @@
 #include <sha1.h>
 
 #ifndef lint
-__RCSID("$NetBSD: rndctl.c,v 1.30 2015/04/13 22:18:50 riastradh Exp $");
+__RCSID("$NetBSD: rndctl.c,v 1.31 2019/12/06 14:43:18 riastradh Exp $");
 #endif
 
 
@@ -41,6 +41,7 @@
 #include <sys/ioctl.h>
 #include <sys/param.h>
 #include <sys/rndio.h>
+#include <sys/sha3.h>
 
 #include <stdio.h>
 #include <stdlib.h>
@@ -127,111 +128,227 @@
 }
 
 static void
-do_save(const char *const filename)
+do_save(const char *filename, const void *extra, size_t nextra,
+    uint32_t extraentropy)
 {
-       int est1, est2;
-       rndpoolstat_t rp;
+       char tmp[PATH_MAX];
+       uint32_t systementropy;
+       uint8_t buf[32];
+       SHAKE128_CTX shake128;
        rndsave_t rs;
        SHA1_CTX s;
-
+       ssize_t nread, nwrit;
        int fd;
 
-       fd = open(_PATH_URANDOM, O_RDONLY, 0644);
-       if (fd < 0) {
+       /* Paranoia: Avoid stack memory disclosure.  */
+       memset(&rs, 0, sizeof rs);
+
+       /* Format the temporary file name.  */
+       if (snprintf(tmp, sizeof tmp, "%s.tmp", filename) >= PATH_MAX)
+               errx(1, "path too long");
+
+       /* Open /dev/urandom.  */
+       if ((fd = open(_PATH_URANDOM, O_RDONLY)) == -1)
                err(1, "device open");
-       }
-
-       if (ioctl(fd, RNDGETPOOLSTAT, &rp) < 0) {
-               err(1, "ioctl(RNDGETPOOLSTAT)");
-       }
 
-       est1 = rp.curentropy;
+       /* Find how much entropy is in the pool.  */
+       if (ioctl(fd, RNDGETENTCNT, &systementropy) == -1)
+               err(1, "ioctl(RNDGETENTCNT)");
 
-       if (read(fd, rs.data, sizeof(rs.data)) != sizeof(rs.data)) {
-               err(1, "entropy read");
-       }
-
-       if (ioctl(fd, RNDGETPOOLSTAT, &rp) < 0) {
-               err(1, "ioctl(RNDGETPOOLSTAT)");
+       /* Read some data from /dev/urandom.  */
+       if ((size_t)(nread = read(fd, buf, sizeof buf)) != sizeof buf) {
+               if (nread == -1)
+                       err(1, "read");
+               else
+                       errx(1, "truncated read");
        }
 
-       est2 = rp.curentropy;
+       /* Close /dev/urandom; we're done with it.  */
+       if (close(fd) == -1)
+               warn("close");
+       fd = -1;                /* paranoia */
+
+       /*
+        * Hash what we read together with the extra input to generate
+        * the seed data.
+        */
+       SHAKE128_Init(&shake128);
+       SHAKE128_Update(&shake128, buf, sizeof buf);
+       SHAKE128_Update(&shake128, extra, nextra);
+       SHAKE128_Final(rs.data, sizeof(rs.data), &shake128);
+       explicit_memset(&shake128, 0, sizeof shake128); /* paranoia */
 
-       if (est1 - est2 < 0) {
-               rs.entropy = 0;
-       } else {
-               rs.entropy = est1 - est2;
-       }
+       /*
+        * Report an upper bound on the min-entropy of the seed data.
+        * We take the larger of the system entropy and the extra
+        * entropy -- the system state and the extra input may or may
+        * not be independent, so we can't add them -- and clamp to the
+        * size of the data.
+        */
+       systementropy = MIN(systementropy,
+           MIN(sizeof(buf), UINT32_MAX/NBBY)*NBBY);
+       extraentropy = MIN(extraentropy, MIN(nextra, UINT32_MAX/NBBY)*NBBY);
+       rs.entropy = MIN(MAX(systementropy, extraentropy),
+           MIN(sizeof(rs.data), UINT32_MAX/NBBY)*NBBY);
 
+       /*
+        * Compute the checksum on the 32-bit entropy count, in host
+        * byte order (XXX this means it is not portable across
+        * different-endian platforms!), followed by the seed data.
+        */
        SHA1Init(&s);
-       SHA1Update(&s, (uint8_t *)&rs.entropy, sizeof(rs.entropy));
+       SHA1Update(&s, (const uint8_t *)&rs.entropy, sizeof(rs.entropy));
        SHA1Update(&s, rs.data, sizeof(rs.data));
        SHA1Final(rs.digest, &s);
-
-       close(fd);
-       unlink(filename);
-       fd = open(filename, O_CREAT|O_EXCL|O_WRONLY, 0600);
-       if (fd < 0) {
-               err(1, "output open");
-       }
+       explicit_memset(&s, 0, sizeof s); /* paranoia */
 
-       if (write(fd, &rs, sizeof(rs)) != sizeof(rs)) {
-               unlink(filename);
-               fsync_range(fd, FDATASYNC|FDISKSYNC, (off_t)0, (off_t)0);
-               err(1, "write");
+       /*
+        * Write it to a temporary file and sync it before we commit.
+        * This way either the old seed or the new seed is completely
+        * written in the expected location on disk even if the system
+        * crashes as long as the file system doesn't get corrupted too
+        * badly.
+        *
+        * If interrupted after this point and the temporary file is
+        * disclosed, no big deal -- either the pool was predictable to
+        * begin with in which case we're hosed either way, or we've
+        * just revealed some output which is not a problem.
+        */
+       if ((fd = open(tmp, O_CREAT|O_TRUNC|O_WRONLY, 0600)) == -1)
+               err(1, "open seed file to save");
+       if ((size_t)(nwrit = write(fd, &rs, sizeof rs)) != sizeof rs) {
+               int error = errno;
+               if (unlink(tmp) == -1)
+                       warn("unlink");
+               if (nwrit == -1)
+                       errc(1, error, "write");
+               else
+                       errx(1, "truncated write");
        }
-       fsync_range(fd, FDATASYNC|FDISKSYNC, (off_t)0, (off_t)0);
-       close(fd);
+       explicit_memset(&rs, 0, sizeof rs); /* paranoia */
+       if (fsync_range(fd, FDATASYNC|FDISKSYNC, 0, 0) == -1) {
+               int error = errno;
+               if (unlink(tmp) == -1)
+                       warn("unlink");
+               errc(1, error, "fsync_range");
+       }
+       if (close(fd) == -1)
+               warn("close");
+
+       /* Rename it over the original file to commit.  */
+       if (rename(tmp, filename) == -1)
+               err(1, "rename");
 }
 
 static void
-do_load(const char *const filename)
+do_load(const char *filename)
 {
-       int fd;
-       rndsave_t rs, rszero;
+       char tmp[PATH_MAX];
+       int fd_seed, fd_random;
+       rndsave_t rs;
        rnddata_t rd;
+       ssize_t nread, nwrit;
        SHA1_CTX s;
        uint8_t digest[SHA1_DIGEST_LENGTH];
 
-       fd = open(filename, O_RDWR, 0600);
-       if (fd < 0) {
-               err(1, "input open");
-       }
+       /*
+        * The order of operations is important here:
+        *
+        * 1. Load the old seed.
+        * 2. Feed the old seed into the kernel.
+        * 3. Generate and write a new seed.
+        * 4. Erase the old seed.
+        *
+        * This follows the procedure in
+        *
+        *      Niels Ferguson, Bruce Schneier, and Tadayoshi Kohno,
+        *      _Cryptography Engineering_, Wiley, 2010, Sec. 9.6.2
+        *      `Update Seed File'.
+        *
+        * There is a race condition: If another process generates a
+        * key from /dev/urandom after step (2) but before step (3),
+        * and if the machine crashes before step (3), an adversary who
+        * can read the disk after the crash can probably guess the
+        * complete state of the entropy pool and thereby predict the
+        * key.
+        *
+        * There's not much we can do here without some kind of
+        * systemwide lock on /dev/urandom and without introducing an
+        * opportunity for a crash to wipe out the entropy altogether.
+        * To avoid this race, you should ensure that any key
+        * generation happens _after_ `rndctl -L' has completed.
+        */
 
-       unlink(filename);
+       /* Format the temporary file name.  */
+       if (snprintf(tmp, sizeof tmp, "%s.tmp", filename) >= PATH_MAX)
+               errx(1, "path too long");
 
-       if (read(fd, &rs, sizeof(rs)) != sizeof(rs)) {
-               err(1, "read");
+       /* 1. Load the old seed.  */
+       if ((fd_seed = open(filename, O_RDWR)) == -1)
+               err(1, "open seed file to load");
+       if ((size_t)(nread = read(fd_seed, &rs, sizeof rs)) != sizeof rs) {
+               if (nread == -1)
+                       err(1, "read seed");
+               else
+                       errx(1, "seed too short");
        }
 
-       memset(&rszero, 0, sizeof(rszero));



Home | Main Index | Thread Index | Old Index