summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorDamien Miller <djm@cvs.openbsd.org>2007-10-15 01:01:48 +0000
committerDamien Miller <djm@cvs.openbsd.org>2007-10-15 01:01:48 +0000
commit8a85eaf9c4bb1ab831fbbccb617d6e7696d5a9ac (patch)
tree5eb7a5a323526f0b75cdfa9efbe98b9034eb3aa3
parentb67e3138f8c23e1627e4c1fe9f1fbcb82eeb2fc9 (diff)
Switch arc4random to using routines from crypto/arc4.[ch], and rework
spl swizzling so we go to splhigh and back once per request instead of once per byte. Service large requests for arc4random_bytes using an independently keyed generator rather than hogging the main one (at splhigh). feedback from henric@; ok deraadt@
-rw-r--r--sys/dev/rnd.c152
1 files changed, 70 insertions, 82 deletions
diff --git a/sys/dev/rnd.c b/sys/dev/rnd.c
index badb616ff78..e042cbab25c 100644
--- a/sys/dev/rnd.c
+++ b/sys/dev/rnd.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: rnd.c,v 1.83 2007/10/09 17:05:19 gilles Exp $ */
+/* $OpenBSD: rnd.c,v 1.84 2007/10/15 01:01:47 djm Exp $ */
/*
* rnd.c -- A strong random number generator
@@ -237,6 +237,7 @@
#undef RNDEBUG
#include <sys/param.h>
+#include <sys/endian.h>
#include <sys/systm.h>
#include <sys/conf.h>
#include <sys/disk.h>
@@ -249,6 +250,7 @@
#include <sys/poll.h>
#include <crypto/md5.h>
+#include <crypto/arc4.h>
#include <dev/rndvar.h>
#include <dev/rndioctl.h>
@@ -261,6 +263,19 @@ int rnd_debug = 0x0000;
#endif
/*
+ * Maximum number of bytes to serve directly from the main arc4random
+ * pool. Larger requests are served from discrete arc4 instances keyed
+ * from the main pool.
+ */
+#define ARC4_MAIN_MAX_BYTES 2048
+
+/*
+ * Key size (in bytes) for arc4 instances setup to serve requests larger
+ * than ARC4_MAIN_MAX_BYTES.
+ */
+#define ARC4_SUB_KEY_BYTES (256 / 8)
+
+/*
* The pool is stirred with a primitive polynomial of degree 128
* over GF(2), namely x^128 + x^99 + x^59 + x^31 + x^9 + x^7 + 1.
* For a pool of size 64, try x^64+x^62+x^38+x^10+x^6+x+1.
@@ -380,13 +395,6 @@ struct timer_rand_state {
u_int max_entropy : 1;
};
-struct arc4_stream {
- u_int8_t s[256];
- u_int cnt;
- u_int8_t i;
- u_int8_t j;
-};
-
struct rand_event {
struct timer_rand_state *re_state;
u_int re_nbits;
@@ -396,7 +404,8 @@ struct rand_event {
struct timeout rnd_timeout, arc4_timeout;
struct random_bucket random_state;
-struct arc4_stream arc4random_state;
+struct rc4_ctx arc4random_state;
+u_long arc4random_count = 0;
struct timer_rand_state rnd_states[RND_SRC_NUM];
struct rand_event rnd_event_space[QEVLEN];
struct rand_event *rnd_event_head = rnd_event_space;
@@ -474,56 +483,15 @@ void dequeue_randomness(void *);
static void add_entropy_words(const u_int32_t *, u_int n);
void extract_entropy(u_int8_t *, int);
-static u_int8_t arc4_getbyte(void);
void arc4_stir(void);
void arc4_reinit(void *v);
void arc4maybeinit(void);
-/* Arcfour random stream generator. This code is derived from section
- * 17.1 of Applied Cryptography, second edition, which describes a
- * stream cipher allegedly compatible with RSA Labs "RC4" cipher (the
- * actual description of which is a trade secret). The same algorithm
- * is used as a stream cipher called "arcfour" in Tatu Ylonen's ssh
- * package.
- *
- * The initialization function here has been modified to not discard
- * the old state, and its input always includes the time of day in
- * microseconds. Moreover, bytes from the stream may at any point be
- * diverted to multiple processes or even kernel functions desiring
- * random numbers. This increases the strength of the random stream,
- * but makes it impossible to use this code for encryption, since there
- * is no way to ever reproduce the same stream of random bytes.
- *
- * RC4 is a registered trademark of RSA Laboratories.
- */
-
-static u_int8_t
-arc4_getbyte(void)
-{
- u_int8_t si, sj, ret;
- int s;
-
- s = splhigh();
- rndstats.arc4_reads++;
- arc4random_state.cnt++;
- arc4random_state.i++;
- si = arc4random_state.s[arc4random_state.i];
- arc4random_state.j += si;
- sj = arc4random_state.s[arc4random_state.j];
- arc4random_state.s[arc4random_state.i] = sj;
- arc4random_state.s[arc4random_state.j] = si;
- ret = arc4random_state.s[(si + sj) & 0xff];
- splx(s);
- return (ret);
-}
-
void
arc4_stir(void)
{
u_int8_t buf[256];
- u_int8_t si;
- int n, s;
- int len;
+ int s, len;
nanotime((struct timespec *) buf);
len = random_state.entropy_count / 8; /* XXX maybe a half? */
@@ -533,28 +501,21 @@ arc4_stir(void)
len += sizeof(struct timeval);
s = splhigh();
- arc4random_state.i--;
- for (n = 0; n < 256; n++) {
- arc4random_state.i++;
- si = arc4random_state.s[arc4random_state.i];
- arc4random_state.j += si + buf[n % len];
- arc4random_state.s[arc4random_state.i] =
- arc4random_state.s[arc4random_state.j];
- arc4random_state.s[arc4random_state.j] = si;
- }
- arc4random_state.j = arc4random_state.i;
- arc4random_state.cnt = 0;
+ if (rndstats.arc4_nstirs > 0)
+ rc4_crypt(&arc4random_state, buf, buf, sizeof(buf));
+
+ rc4_keysetup(&arc4random_state, buf, sizeof(buf));
+ arc4random_count = 0;
rndstats.arc4_stirs += len;
rndstats.arc4_nstirs++;
- splx(s);
/*
* Throw away the first N words of output, as suggested in the
* paper "Weaknesses in the Key Scheduling Algorithm of RC4"
* by Fluher, Mantin, and Shamir. (N = 256 in our case.)
*/
- for (n = 0; n < 256 * 4; n++)
- arc4_getbyte();
+ rc4_skip(&arc4random_state, 256 * 4);
+ splx(s);
}
void
@@ -587,27 +548,61 @@ arc4_reinit(void *v)
u_int32_t
arc4random(void)
{
+ int s;
+ u_int32_t ret;
+
arc4maybeinit();
- return ((arc4_getbyte() << 24) | (arc4_getbyte() << 16)
- | (arc4_getbyte() << 8) | arc4_getbyte());
+ s = splhigh();
+ rc4_getbytes(&arc4random_state, (u_char*)&ret, sizeof(ret));
+ rndstats.arc4_reads += sizeof(ret);
+ arc4random_count += sizeof(ret);
+ splx(s);
+ return ret;
+}
+
+static void
+arc4random_bytes_large(void *buf, size_t n)
+{
+ int s;
+ u_char lbuf[ARC4_SUB_KEY_BYTES];
+ struct rc4_ctx lctx;
+
+ s = splhigh();
+ rc4_getbytes(&arc4random_state, lbuf, sizeof(lbuf));
+ rndstats.arc4_reads += n;
+ arc4random_count += sizeof(lbuf);
+ splx(s);
+
+ rc4_keysetup(&lctx, lbuf, sizeof(lbuf));
+ rc4_skip(&lctx, 256 * 4);
+ rc4_getbytes(&lctx, (u_char*)buf, n);
+ bzero(lbuf, sizeof(lbuf));
+ bzero(&lctx, sizeof(lctx));
}
void
arc4random_bytes(void *buf, size_t n)
{
- u_int8_t *cp = buf;
- u_int8_t *end = cp + n;
+ int s;
arc4maybeinit();
- while (cp < end)
- *cp++ = arc4_getbyte();
+
+ /* Satisfy large requests via an independent ARC4 instance */
+ if (n > ARC4_MAIN_MAX_BYTES) {
+ arc4random_bytes_large(buf, n);
+ return;
+ }
+
+ s = splhigh();
+ rc4_getbytes(&arc4random_state, (u_char*)buf, n);
+ rndstats.arc4_reads += n;
+ arc4random_count += n;
+ splx(s);
}
void
randomattach(void)
{
- int i;
-
if (rnd_attached) {
#ifdef RNDEBUG
printf("random: second attach\n");
@@ -627,8 +622,7 @@ randomattach(void)
bzero(&rndstats, sizeof(rndstats));
bzero(&rnd_event_space, sizeof(rnd_event_space));
- for (i = 0; i < 256; i++)
- arc4random_state.s[i] = i;
+ bzero(&arc4random_state, sizeof(arc4random_state));
arc4_reinit(NULL);
rnd_attached = 1;
@@ -998,14 +992,8 @@ randomread(dev_t dev, struct uio *uio, int ioflag)
buf[i] = random() << 16 | (random() & 0xFFFF);
break;
case RND_ARND:
- {
- u_int8_t *cp = (u_int8_t *) buf;
- u_int8_t *end = cp + n;
- arc4maybeinit();
- while (cp < end)
- *cp++ = arc4_getbyte();
+ arc4random_bytes(buf, n);
break;
- }
default:
ret = ENXIO;
}