Commit e73aaae2 authored by Jason A. Donenfeld's avatar Jason A. Donenfeld

siphash: use one source of truth for siphash permutations

The SipHash family of permutations is currently used in three places:

- siphash.c itself, used in the ordinary way it was intended.
- random32.c, in a construction from an anonymous contributor.
- random.c, as part of its fast_mix function.

Each one of these places reinvents the wheel with the same C code, same
rotation constants, and same symmetry-breaking constants.

This commit tidies things up a bit by placing macros for the
permutations and constants into siphash.h, where each of the three .c
users can access them. It also leaves a note dissuading more users of
them from emerging.
Signed-off-by: default avatarJason A. Donenfeld <Jason@zx2c4.com>
parent 791332b3
...@@ -52,6 +52,7 @@ ...@@ -52,6 +52,7 @@
#include <linux/uuid.h> #include <linux/uuid.h>
#include <linux/uaccess.h> #include <linux/uaccess.h>
#include <linux/suspend.h> #include <linux/suspend.h>
#include <linux/siphash.h>
#include <crypto/chacha.h> #include <crypto/chacha.h>
#include <crypto/blake2s.h> #include <crypto/blake2s.h>
#include <asm/processor.h> #include <asm/processor.h>
...@@ -1086,12 +1087,11 @@ struct fast_pool { ...@@ -1086,12 +1087,11 @@ struct fast_pool {
static DEFINE_PER_CPU(struct fast_pool, irq_randomness) = { static DEFINE_PER_CPU(struct fast_pool, irq_randomness) = {
#ifdef CONFIG_64BIT #ifdef CONFIG_64BIT
/* SipHash constants */ #define FASTMIX_PERM SIPHASH_PERMUTATION
.pool = { 0x736f6d6570736575UL, 0x646f72616e646f6dUL, .pool = { SIPHASH_CONST_0, SIPHASH_CONST_1, SIPHASH_CONST_2, SIPHASH_CONST_3 }
0x6c7967656e657261UL, 0x7465646279746573UL }
#else #else
/* HalfSipHash constants */ #define FASTMIX_PERM HSIPHASH_PERMUTATION
.pool = { 0, 0, 0x6c796765U, 0x74656462U } .pool = { HSIPHASH_CONST_0, HSIPHASH_CONST_1, HSIPHASH_CONST_2, HSIPHASH_CONST_3 }
#endif #endif
}; };
...@@ -1103,27 +1103,11 @@ static DEFINE_PER_CPU(struct fast_pool, irq_randomness) = { ...@@ -1103,27 +1103,11 @@ static DEFINE_PER_CPU(struct fast_pool, irq_randomness) = {
*/ */
static void fast_mix(unsigned long s[4], unsigned long v1, unsigned long v2) static void fast_mix(unsigned long s[4], unsigned long v1, unsigned long v2)
{ {
#ifdef CONFIG_64BIT
#define PERM() do { \
s[0] += s[1]; s[1] = rol64(s[1], 13); s[1] ^= s[0]; s[0] = rol64(s[0], 32); \
s[2] += s[3]; s[3] = rol64(s[3], 16); s[3] ^= s[2]; \
s[0] += s[3]; s[3] = rol64(s[3], 21); s[3] ^= s[0]; \
s[2] += s[1]; s[1] = rol64(s[1], 17); s[1] ^= s[2]; s[2] = rol64(s[2], 32); \
} while (0)
#else
#define PERM() do { \
s[0] += s[1]; s[1] = rol32(s[1], 5); s[1] ^= s[0]; s[0] = rol32(s[0], 16); \
s[2] += s[3]; s[3] = rol32(s[3], 8); s[3] ^= s[2]; \
s[0] += s[3]; s[3] = rol32(s[3], 7); s[3] ^= s[0]; \
s[2] += s[1]; s[1] = rol32(s[1], 13); s[1] ^= s[2]; s[2] = rol32(s[2], 16); \
} while (0)
#endif
s[3] ^= v1; s[3] ^= v1;
PERM(); FASTMIX_PERM(s[0], s[1], s[2], s[3]);
s[0] ^= v1; s[0] ^= v1;
s[3] ^= v2; s[3] ^= v2;
PERM(); FASTMIX_PERM(s[0], s[1], s[2], s[3]);
s[0] ^= v2; s[0] ^= v2;
} }
......
...@@ -10,6 +10,7 @@ ...@@ -10,6 +10,7 @@
#include <linux/types.h> #include <linux/types.h>
#include <linux/percpu.h> #include <linux/percpu.h>
#include <linux/siphash.h>
u32 prandom_u32(void); u32 prandom_u32(void);
void prandom_bytes(void *buf, size_t nbytes); void prandom_bytes(void *buf, size_t nbytes);
...@@ -27,15 +28,10 @@ DECLARE_PER_CPU(unsigned long, net_rand_noise); ...@@ -27,15 +28,10 @@ DECLARE_PER_CPU(unsigned long, net_rand_noise);
* The core SipHash round function. Each line can be executed in * The core SipHash round function. Each line can be executed in
* parallel given enough CPU resources. * parallel given enough CPU resources.
*/ */
#define PRND_SIPROUND(v0, v1, v2, v3) ( \ #define PRND_SIPROUND(v0, v1, v2, v3) SIPHASH_PERMUTATION(v0, v1, v2, v3)
v0 += v1, v1 = rol64(v1, 13), v2 += v3, v3 = rol64(v3, 16), \
v1 ^= v0, v0 = rol64(v0, 32), v3 ^= v2, \
v0 += v3, v3 = rol64(v3, 21), v2 += v1, v1 = rol64(v1, 17), \
v3 ^= v0, v1 ^= v2, v2 = rol64(v2, 32) \
)
#define PRND_K0 (0x736f6d6570736575 ^ 0x6c7967656e657261) #define PRND_K0 (SIPHASH_CONST_0 ^ SIPHASH_CONST_2)
#define PRND_K1 (0x646f72616e646f6d ^ 0x7465646279746573) #define PRND_K1 (SIPHASH_CONST_1 ^ SIPHASH_CONST_3)
#elif BITS_PER_LONG == 32 #elif BITS_PER_LONG == 32
/* /*
...@@ -43,14 +39,9 @@ DECLARE_PER_CPU(unsigned long, net_rand_noise); ...@@ -43,14 +39,9 @@ DECLARE_PER_CPU(unsigned long, net_rand_noise);
* This is weaker, but 32-bit machines are not used for high-traffic * This is weaker, but 32-bit machines are not used for high-traffic
* applications, so there is less output for an attacker to analyze. * applications, so there is less output for an attacker to analyze.
*/ */
#define PRND_SIPROUND(v0, v1, v2, v3) ( \ #define PRND_SIPROUND(v0, v1, v2, v3) HSIPHASH_PERMUTATION(v0, v1, v2, v3)
v0 += v1, v1 = rol32(v1, 5), v2 += v3, v3 = rol32(v3, 8), \ #define PRND_K0 (HSIPHASH_CONST_0 ^ HSIPHASH_CONST_2)
v1 ^= v0, v0 = rol32(v0, 16), v3 ^= v2, \ #define PRND_K1 (HSIPHASH_CONST_1 ^ HSIPHASH_CONST_3)
v0 += v3, v3 = rol32(v3, 7), v2 += v1, v1 = rol32(v1, 13), \
v3 ^= v0, v1 ^= v2, v2 = rol32(v2, 16) \
)
#define PRND_K0 0x6c796765
#define PRND_K1 0x74656462
#else #else
#error Unsupported BITS_PER_LONG #error Unsupported BITS_PER_LONG
......
...@@ -138,4 +138,32 @@ static inline u32 hsiphash(const void *data, size_t len, ...@@ -138,4 +138,32 @@ static inline u32 hsiphash(const void *data, size_t len,
return ___hsiphash_aligned(data, len, key); return ___hsiphash_aligned(data, len, key);
} }
/*
* These macros expose the raw SipHash and HalfSipHash permutations.
* Do not use them directly! If you think you have a use for them,
* be sure to CC the maintainer of this file explaining why.
*/
#define SIPHASH_PERMUTATION(a, b, c, d) ( \
(a) += (b), (b) = rol64((b), 13), (b) ^= (a), (a) = rol64((a), 32), \
(c) += (d), (d) = rol64((d), 16), (d) ^= (c), \
(a) += (d), (d) = rol64((d), 21), (d) ^= (a), \
(c) += (b), (b) = rol64((b), 17), (b) ^= (c), (c) = rol64((c), 32))
#define SIPHASH_CONST_0 0x736f6d6570736575ULL
#define SIPHASH_CONST_1 0x646f72616e646f6dULL
#define SIPHASH_CONST_2 0x6c7967656e657261ULL
#define SIPHASH_CONST_3 0x7465646279746573ULL
#define HSIPHASH_PERMUTATION(a, b, c, d) ( \
(a) += (b), (b) = rol32((b), 5), (b) ^= (a), (a) = rol32((a), 16), \
(c) += (d), (d) = rol32((d), 8), (d) ^= (c), \
(a) += (d), (d) = rol32((d), 7), (d) ^= (a), \
(c) += (b), (b) = rol32((b), 13), (b) ^= (c), (c) = rol32((c), 16))
#define HSIPHASH_CONST_0 0U
#define HSIPHASH_CONST_1 0U
#define HSIPHASH_CONST_2 0x6c796765U
#define HSIPHASH_CONST_3 0x74656462U
#endif /* _LINUX_SIPHASH_H */ #endif /* _LINUX_SIPHASH_H */
...@@ -18,19 +18,13 @@ ...@@ -18,19 +18,13 @@
#include <asm/word-at-a-time.h> #include <asm/word-at-a-time.h>
#endif #endif
#define SIPROUND \ #define SIPROUND SIPHASH_PERMUTATION(v0, v1, v2, v3)
do { \
v0 += v1; v1 = rol64(v1, 13); v1 ^= v0; v0 = rol64(v0, 32); \
v2 += v3; v3 = rol64(v3, 16); v3 ^= v2; \
v0 += v3; v3 = rol64(v3, 21); v3 ^= v0; \
v2 += v1; v1 = rol64(v1, 17); v1 ^= v2; v2 = rol64(v2, 32); \
} while (0)
#define PREAMBLE(len) \ #define PREAMBLE(len) \
u64 v0 = 0x736f6d6570736575ULL; \ u64 v0 = SIPHASH_CONST_0; \
u64 v1 = 0x646f72616e646f6dULL; \ u64 v1 = SIPHASH_CONST_1; \
u64 v2 = 0x6c7967656e657261ULL; \ u64 v2 = SIPHASH_CONST_2; \
u64 v3 = 0x7465646279746573ULL; \ u64 v3 = SIPHASH_CONST_3; \
u64 b = ((u64)(len)) << 56; \ u64 b = ((u64)(len)) << 56; \
v3 ^= key->key[1]; \ v3 ^= key->key[1]; \
v2 ^= key->key[0]; \ v2 ^= key->key[0]; \
...@@ -389,19 +383,13 @@ u32 hsiphash_4u32(const u32 first, const u32 second, const u32 third, ...@@ -389,19 +383,13 @@ u32 hsiphash_4u32(const u32 first, const u32 second, const u32 third,
} }
EXPORT_SYMBOL(hsiphash_4u32); EXPORT_SYMBOL(hsiphash_4u32);
#else #else
#define HSIPROUND \ #define HSIPROUND HSIPHASH_PERMUTATION(v0, v1, v2, v3)
do { \
v0 += v1; v1 = rol32(v1, 5); v1 ^= v0; v0 = rol32(v0, 16); \
v2 += v3; v3 = rol32(v3, 8); v3 ^= v2; \
v0 += v3; v3 = rol32(v3, 7); v3 ^= v0; \
v2 += v1; v1 = rol32(v1, 13); v1 ^= v2; v2 = rol32(v2, 16); \
} while (0)
#define HPREAMBLE(len) \ #define HPREAMBLE(len) \
u32 v0 = 0; \ u32 v0 = HSIPHASH_CONST_0; \
u32 v1 = 0; \ u32 v1 = HSIPHASH_CONST_1; \
u32 v2 = 0x6c796765U; \ u32 v2 = HSIPHASH_CONST_2; \
u32 v3 = 0x74656462U; \ u32 v3 = HSIPHASH_CONST_3; \
u32 b = ((u32)(len)) << 24; \ u32 b = ((u32)(len)) << 24; \
v3 ^= key->key[1]; \ v3 ^= key->key[1]; \
v2 ^= key->key[0]; \ v2 ^= key->key[0]; \
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment