[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-Id: <20220210160925.156697-3-Jason@zx2c4.com>
Date: Thu, 10 Feb 2022 17:09:25 +0100
From: "Jason A. Donenfeld" <Jason@...c4.com>
To: linux@...inikbrodowski.net, linux-kernel@...r.kernel.org
Cc: "Jason A. Donenfeld" <Jason@...c4.com>,
Theodore Ts'o <tytso@....edu>
Subject: [PATCH 2/2] random: deobfuscate irq u32/u64 contributions
In the irq handler, we fill out 16 bytes differently on 32-bit and
64-bit platforms. Whether or not you like that, it is a matter of fact.
But it might not be a fact you well realized until now, because the code
that loaded the irq info into 4 32-bit words was quite confusing.
Instead, this commit makes everything explicit by having separate
(compile-time) branches for 32-bit and 64-bit machines. In the process,
it exposed a shortcoming in in mix_interrupt_randomness() which we
rectify.
Cc: Theodore Ts'o <tytso@....edu>
Cc: Dominik Brodowski <linux@...inikbrodowski.net>
Signed-off-by: Jason A. Donenfeld <Jason@...c4.com>
---
drivers/char/random.c | 54 ++++++++++++++++++++++++-------------------
1 file changed, 30 insertions(+), 24 deletions(-)
diff --git a/drivers/char/random.c b/drivers/char/random.c
index 923a8f861437..1eb3c059025b 100644
--- a/drivers/char/random.c
+++ b/drivers/char/random.c
@@ -731,9 +731,12 @@ EXPORT_SYMBOL_GPL(add_disk_randomness);
#endif
struct fast_pool {
+ union {
+ u64 pool64[2];
+ u32 pool32[4];
+ };
struct work_struct mix;
unsigned long last;
- u32 pool[4];
unsigned int count;
u16 reg_idx;
};
@@ -744,10 +747,10 @@ struct fast_pool {
* collector. It's hardcoded for an 128 bit pool and assumes that any
* locks that might be needed are taken by the caller.
*/
-static void fast_mix(struct fast_pool *f)
+static void fast_mix(u32 pool[4])
{
- u32 a = f->pool[0], b = f->pool[1];
- u32 c = f->pool[2], d = f->pool[3];
+ u32 a = pool[0], b = pool[1];
+ u32 c = pool[2], d = pool[3];
a += b; c += d;
b = rol32(b, 6); d = rol32(d, 27);
@@ -765,11 +768,10 @@ static void fast_mix(struct fast_pool *f)
b = rol32(b, 16); d = rol32(d, 14);
d ^= a; b ^= c;
- f->pool[0] = a; f->pool[1] = b;
- f->pool[2] = c; f->pool[3] = d;
+ pool[0] = a; pool[1] = b;
+ pool[2] = c; pool[3] = d;
}
-
static DEFINE_PER_CPU(struct fast_pool, irq_randomness);
static u32 get_reg(struct fast_pool *f, struct pt_regs *regs)
@@ -790,15 +792,19 @@ static u32 get_reg(struct fast_pool *f, struct pt_regs *regs)
static void mix_interrupt_randomness(struct work_struct *work)
{
struct fast_pool *fast_pool = container_of(work, struct fast_pool, mix);
- u32 pool[ARRAY_SIZE(fast_pool->pool)];
+ u32 pool[ARRAY_SIZE(fast_pool->pool32)];
/*
* Since this is the result of a trip through the scheduler, xor in
* a cycle counter. It can't hurt, and might help.
*/
- fast_pool->pool[3] ^= random_get_entropy();
+ if (sizeof(unsigned long) == 8)
+ fast_pool->pool64[1] ^= random_get_entropy();
+ else
+ fast_pool->pool32[3] ^= random_get_entropy();
+
/* Copy the pool to the stack so that the mixer always has a consistent view. */
- memcpy(pool, fast_pool->pool, sizeof(pool));
+ memcpy(pool, fast_pool->pool32, sizeof(pool));
/* We take care to zero out the count only after we're done reading the pool. */
WRITE_ONCE(fast_pool->count, 0);
fast_pool->last = jiffies;
@@ -815,26 +821,26 @@ void add_interrupt_randomness(int irq)
unsigned long now = jiffies;
cycles_t cycles = random_get_entropy();
unsigned int new_count;
- u32 c_high, j_high;
- u64 ip;
if (cycles == 0)
cycles = get_reg(fast_pool, regs);
- c_high = (sizeof(cycles) > 4) ? cycles >> 32 : 0;
- j_high = (sizeof(now) > 4) ? now >> 32 : 0;
- fast_pool->pool[0] ^= cycles ^ j_high ^ irq;
- fast_pool->pool[1] ^= now ^ c_high;
- ip = regs ? instruction_pointer(regs) : _RET_IP_;
- fast_pool->pool[2] ^= ip;
- fast_pool->pool[3] ^=
- (sizeof(ip) > 4) ? ip >> 32 : get_reg(fast_pool, regs);
-
- fast_mix(fast_pool);
+
+ if (sizeof(unsigned long) == 8) {
+ fast_pool->pool64[0] ^= cycles ^ rol64(now, 32) ^ irq;
+ fast_pool->pool64[1] ^= regs ? instruction_pointer(regs) : _RET_IP_;
+ } else {
+ fast_pool->pool32[0] ^= cycles ^ irq;
+ fast_pool->pool32[1] ^= now;
+ fast_pool->pool32[2] ^= regs ? instruction_pointer(regs) : _RET_IP_;
+ fast_pool->pool32[3] ^= get_reg(fast_pool, regs);
+ }
+
+ fast_mix(fast_pool->pool32);
new_count = ++fast_pool->count;
if (unlikely(crng_init == 0)) {
if (new_count >= 64 &&
- crng_fast_load(fast_pool->pool, sizeof(fast_pool->pool)) > 0) {
+ crng_fast_load(fast_pool->pool32, sizeof(fast_pool->pool32)) > 0) {
fast_pool->count = 0;
fast_pool->last = now;
@@ -843,7 +849,7 @@ void add_interrupt_randomness(int irq)
* However, this only happens during boot, and then never
* again, so we live with it.
*/
- mix_pool_bytes(&fast_pool->pool, sizeof(fast_pool->pool));
+ mix_pool_bytes(&fast_pool->pool32, sizeof(fast_pool->pool32));
}
return;
}
--
2.35.0
Powered by blists - more mailing lists