diff options
author | Jarod Wilson <jarod@redhat.com> | 2012-11-06 10:42:42 -0500 |
---|---|---|
committer | Theodore Ts'o <tytso@mit.edu> | 2012-11-08 07:19:18 -0500 |
commit | ec8f02da9ea500474417d1d31fa3d46a562ab366 (patch) | |
tree | 9a7bb64cdbc436f59345288181768eede207bcf2 | |
parent | 8eb2ffbf7be94c546a873540ff952140465125e5 (diff) |
random: prime last_data value per fips requirements
The value stored in last_data must be primed for FIPS 140-2 purposes. Upon
first use, either on system startup or after an RNDCLEARPOOL ioctl, we
need to take an initial random sample, store it internally in last_data,
then pass along the value after that to the requester, so that consistency
checks aren't being run against stale and possibly known data.
CC: Herbert Xu <herbert@gondor.apana.org.au>
CC: "David S. Miller" <davem@davemloft.net>
CC: Matt Mackall <mpm@selenic.com>
CC: linux-crypto@vger.kernel.org
Acked-by: Neil Horman <nhorman@tuxdriver.com>
Signed-off-by: Jarod Wilson <jarod@redhat.com>
Signed-off-by: "Theodore Ts'o" <tytso@mit.edu>
-rw-r--r-- | drivers/char/random.c | 17 |
1 files changed, 17 insertions, 0 deletions
diff --git a/drivers/char/random.c b/drivers/char/random.c index a1af1839576..85e81ec1451 100644 --- a/drivers/char/random.c +++ b/drivers/char/random.c | |||
@@ -433,6 +433,7 @@ struct entropy_store { | |||
433 | int entropy_count; | 433 | int entropy_count; |
434 | int entropy_total; | 434 | int entropy_total; |
435 | unsigned int initialized:1; | 435 | unsigned int initialized:1; |
436 | bool last_data_init; | ||
436 | __u8 last_data[EXTRACT_SIZE]; | 437 | __u8 last_data[EXTRACT_SIZE]; |
437 | }; | 438 | }; |
438 | 439 | ||
@@ -953,6 +954,10 @@ static ssize_t extract_entropy(struct entropy_store *r, void *buf, | |||
953 | ssize_t ret = 0, i; | 954 | ssize_t ret = 0, i; |
954 | __u8 tmp[EXTRACT_SIZE]; | 955 | __u8 tmp[EXTRACT_SIZE]; |
955 | 956 | ||
957 | /* if last_data isn't primed, we need EXTRACT_SIZE extra bytes */ | ||
958 | if (fips_enabled && !r->last_data_init) | ||
959 | nbytes += EXTRACT_SIZE; | ||
960 | |||
956 | trace_extract_entropy(r->name, nbytes, r->entropy_count, _RET_IP_); | 961 | trace_extract_entropy(r->name, nbytes, r->entropy_count, _RET_IP_); |
957 | xfer_secondary_pool(r, nbytes); | 962 | xfer_secondary_pool(r, nbytes); |
958 | nbytes = account(r, nbytes, min, reserved); | 963 | nbytes = account(r, nbytes, min, reserved); |
@@ -963,6 +968,17 @@ static ssize_t extract_entropy(struct entropy_store *r, void *buf, | |||
963 | if (fips_enabled) { | 968 | if (fips_enabled) { |
964 | unsigned long flags; | 969 | unsigned long flags; |
965 | 970 | ||
971 | |||
972 | /* prime last_data value if need be, per fips 140-2 */ | ||
973 | if (!r->last_data_init) { | ||
974 | spin_lock_irqsave(&r->lock, flags); | ||
975 | memcpy(r->last_data, tmp, EXTRACT_SIZE); | ||
976 | r->last_data_init = true; | ||
977 | nbytes -= EXTRACT_SIZE; | ||
978 | spin_unlock_irqrestore(&r->lock, flags); | ||
979 | extract_buf(r, tmp); | ||
980 | } | ||
981 | |||
966 | spin_lock_irqsave(&r->lock, flags); | 982 | spin_lock_irqsave(&r->lock, flags); |
967 | if (!memcmp(tmp, r->last_data, EXTRACT_SIZE)) | 983 | if (!memcmp(tmp, r->last_data, EXTRACT_SIZE)) |
968 | panic("Hardware RNG duplicated output!\n"); | 984 | panic("Hardware RNG duplicated output!\n"); |
@@ -1082,6 +1098,7 @@ static void init_std_data(struct entropy_store *r) | |||
1082 | 1098 | ||
1083 | r->entropy_count = 0; | 1099 | r->entropy_count = 0; |
1084 | r->entropy_total = 0; | 1100 | r->entropy_total = 0; |
1101 | r->last_data_init = false; | ||
1085 | mix_pool_bytes(r, &now, sizeof(now), NULL); | 1102 | mix_pool_bytes(r, &now, sizeof(now), NULL); |
1086 | for (i = r->poolinfo->POOLBYTES; i > 0; i -= sizeof(rv)) { | 1103 | for (i = r->poolinfo->POOLBYTES; i > 0; i -= sizeof(rv)) { |
1087 | if (!arch_get_random_long(&rv)) | 1104 | if (!arch_get_random_long(&rv)) |