|
1 | | -/* |
2 | | - SipHash reference C implementation |
3 | | -
|
4 | | - Copyright (c) 2012-2022 Jean-Philippe Aumasson |
5 | | - |
6 | | - Copyright (c) 2012-2014 Daniel J. Bernstein <[email protected]> |
7 | | -
|
8 | | - To the extent possible under law, the author(s) have dedicated all copyright |
9 | | - and related and neighboring rights to this software to the public domain |
10 | | - worldwide. This software is distributed without any warranty. |
11 | | -
|
12 | | - You should have received a copy of the CC0 Public Domain Dedication along |
13 | | - with |
14 | | - this software. If not, see |
15 | | - <http://creativecommons.org/publicdomain/zero/1.0/>. |
16 | | - */ |
17 | | - |
18 | | -#include "siphash.h" |
19 | | -#include <assert.h> |
20 | | -#include <stddef.h> |
21 | | -#include <stdint.h> |
22 | | - |
23 | | -/* default: SipHash-2-4 */ |
24 | | -#ifndef cROUNDS |
25 | | -#define cROUNDS 2 |
26 | | -#endif |
27 | | -#ifndef dROUNDS |
28 | | -#define dROUNDS 4 |
29 | | -#endif |
| 1 | +//===--- SipHash.cpp - An ABI-stable string hash --------------------------===// |
| 2 | +// |
| 3 | +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
| 4 | +// See https://llvm.org/LICENSE.txt for license information. |
| 5 | +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
| 6 | +// |
| 7 | +//===----------------------------------------------------------------------===// |
| 8 | + |
| 9 | +#include "llvm/Support/SipHash.h" |
| 10 | +#include "llvm/ADT/ArrayRef.h" |
| 11 | +#include "llvm/Support/Compiler.h" |
| 12 | +#include "llvm/Support/Endian.h" |
| 13 | +#include <cstdint> |
| 14 | + |
| 15 | +using namespace llvm; |
| 16 | +using namespace support; |
| 17 | + |
| 18 | +// Lightly adapted from the SipHash reference C implementation: |
| 19 | +// https://github.com/veorq/SipHash |
| 20 | +// by Jean-Philippe Aumasson and Daniel J. Bernstein |
30 | 21 |
|
31 | 22 | #define ROTL(x, b) (uint64_t)(((x) << (b)) | ((x) >> (64 - (b)))) |
32 | 23 |
|
33 | | -#define U32TO8_LE(p, v) \ |
34 | | - (p)[0] = (uint8_t)((v)); \ |
35 | | - (p)[1] = (uint8_t)((v) >> 8); \ |
36 | | - (p)[2] = (uint8_t)((v) >> 16); \ |
37 | | - (p)[3] = (uint8_t)((v) >> 24); |
38 | | - |
39 | | -#define U64TO8_LE(p, v) \ |
40 | | - U32TO8_LE((p), (uint32_t)((v))); \ |
41 | | - U32TO8_LE((p) + 4, (uint32_t)((v) >> 32)); |
42 | | - |
43 | | -#define U8TO64_LE(p) \ |
44 | | - (((uint64_t)((p)[0])) | ((uint64_t)((p)[1]) << 8) | \ |
45 | | - ((uint64_t)((p)[2]) << 16) | ((uint64_t)((p)[3]) << 24) | \ |
46 | | - ((uint64_t)((p)[4]) << 32) | ((uint64_t)((p)[5]) << 40) | \ |
47 | | - ((uint64_t)((p)[6]) << 48) | ((uint64_t)((p)[7]) << 56)) |
48 | | - |
49 | 24 | #define SIPROUND \ |
50 | | - do { \ |
51 | | - v0 += v1; \ |
52 | | - v1 = ROTL(v1, 13); \ |
53 | | - v1 ^= v0; \ |
54 | | - v0 = ROTL(v0, 32); \ |
55 | | - v2 += v3; \ |
56 | | - v3 = ROTL(v3, 16); \ |
57 | | - v3 ^= v2; \ |
58 | | - v0 += v3; \ |
59 | | - v3 = ROTL(v3, 21); \ |
60 | | - v3 ^= v0; \ |
61 | | - v2 += v1; \ |
62 | | - v1 = ROTL(v1, 17); \ |
63 | | - v1 ^= v2; \ |
64 | | - v2 = ROTL(v2, 32); \ |
65 | | - } while (0) |
66 | | - |
67 | | -#ifdef DEBUG_SIPHASH |
68 | | -#include <stdio.h> |
69 | | - |
70 | | -#define TRACE \ |
71 | | - do { \ |
72 | | - printf("(%3zu) v0 %016" PRIx64 "\n", inlen, v0); \ |
73 | | - printf("(%3zu) v1 %016" PRIx64 "\n", inlen, v1); \ |
74 | | - printf("(%3zu) v2 %016" PRIx64 "\n", inlen, v2); \ |
75 | | - printf("(%3zu) v3 %016" PRIx64 "\n", inlen, v3); \ |
76 | | - } while (0) |
77 | | -#else |
78 | | -#define TRACE |
79 | | -#endif |
80 | | - |
81 | | -/* |
82 | | - Computes a SipHash value |
83 | | - *in: pointer to input data (read-only) |
84 | | - inlen: input data length in bytes (any size_t value) |
85 | | - *k: pointer to the key data (read-only), must be 16 bytes |
86 | | - *out: pointer to output data (write-only), outlen bytes must be allocated |
87 | | - outlen: length of the output in bytes, must be 8 or 16 |
88 | | -*/ |
89 | | -int siphash(const void *in, const size_t inlen, const void *k, uint8_t *out, |
90 | | - const size_t outlen) { |
91 | | - |
92 | | - const unsigned char *ni = (const unsigned char *)in; |
93 | | - const unsigned char *kk = (const unsigned char *)k; |
94 | | - |
95 | | - assert((outlen == 8) || (outlen == 16)); |
96 | | - uint64_t v0 = UINT64_C(0x736f6d6570736575); |
97 | | - uint64_t v1 = UINT64_C(0x646f72616e646f6d); |
98 | | - uint64_t v2 = UINT64_C(0x6c7967656e657261); |
99 | | - uint64_t v3 = UINT64_C(0x7465646279746573); |
100 | | - uint64_t k0 = U8TO64_LE(kk); |
101 | | - uint64_t k1 = U8TO64_LE(kk + 8); |
102 | | - uint64_t m; |
103 | | - int i; |
104 | | - const unsigned char *end = ni + inlen - (inlen % sizeof(uint64_t)); |
105 | | - const int left = inlen & 7; |
106 | | - uint64_t b = ((uint64_t)inlen) << 56; |
107 | | - v3 ^= k1; |
108 | | - v2 ^= k0; |
109 | | - v1 ^= k1; |
110 | | - v0 ^= k0; |
111 | | - |
112 | | - if (outlen == 16) |
113 | | - v1 ^= 0xee; |
114 | | - |
115 | | - for (; ni != end; ni += 8) { |
116 | | - m = U8TO64_LE(ni); |
117 | | - v3 ^= m; |
118 | | - |
119 | | - TRACE; |
120 | | - for (i = 0; i < cROUNDS; ++i) |
121 | | - SIPROUND; |
122 | | - |
123 | | - v0 ^= m; |
124 | | - } |
125 | | - |
126 | | - switch (left) { |
127 | | - case 7: |
128 | | - b |= ((uint64_t)ni[6]) << 48; |
129 | | - /* FALLTHRU */ |
130 | | - case 6: |
131 | | - b |= ((uint64_t)ni[5]) << 40; |
132 | | - /* FALLTHRU */ |
133 | | - case 5: |
134 | | - b |= ((uint64_t)ni[4]) << 32; |
135 | | - /* FALLTHRU */ |
136 | | - case 4: |
137 | | - b |= ((uint64_t)ni[3]) << 24; |
138 | | - /* FALLTHRU */ |
139 | | - case 3: |
140 | | - b |= ((uint64_t)ni[2]) << 16; |
141 | | - /* FALLTHRU */ |
142 | | - case 2: |
143 | | - b |= ((uint64_t)ni[1]) << 8; |
144 | | - /* FALLTHRU */ |
145 | | - case 1: |
146 | | - b |= ((uint64_t)ni[0]); |
147 | | - break; |
148 | | - case 0: |
149 | | - break; |
150 | | - } |
151 | | - |
152 | | - v3 ^= b; |
153 | | - |
154 | | - TRACE; |
155 | | - for (i = 0; i < cROUNDS; ++i) |
156 | | - SIPROUND; |
157 | | - |
158 | | - v0 ^= b; |
159 | | - |
160 | | - if (outlen == 16) |
161 | | - v2 ^= 0xee; |
162 | | - else |
163 | | - v2 ^= 0xff; |
| 25 | + do { \ |
| 26 | + v0 += v1; \ |
| 27 | + v1 = ROTL(v1, 13); \ |
| 28 | + v1 ^= v0; \ |
| 29 | + v0 = ROTL(v0, 32); \ |
| 30 | + v2 += v3; \ |
| 31 | + v3 = ROTL(v3, 16); \ |
| 32 | + v3 ^= v2; \ |
| 33 | + v0 += v3; \ |
| 34 | + v3 = ROTL(v3, 21); \ |
| 35 | + v3 ^= v0; \ |
| 36 | + v2 += v1; \ |
| 37 | + v1 = ROTL(v1, 17); \ |
| 38 | + v1 ^= v2; \ |
| 39 | + v2 = ROTL(v2, 32); \ |
| 40 | + } while (0) |
| 41 | + |
| 42 | +namespace { |
| 43 | + |
| 44 | +/// Computes a SipHash value |
| 45 | +/// |
| 46 | +/// \param in: pointer to input data (read-only) |
| 47 | +/// \param inlen: input data length in bytes (any size_t value) |
| 48 | +/// \param k: reference to the key data 16-byte array (read-only) |
| 49 | +/// \returns output data, must be 8 or 16 bytes |
| 50 | +/// |
| 51 | +template <int cROUNDS, int dROUNDS, size_t outlen> |
| 52 | +void siphash(const unsigned char *in, uint64_t inlen, |
| 53 | + const unsigned char (&k)[16], unsigned char (&out)[outlen]) { |
| 54 | + |
| 55 | + const unsigned char *ni = (const unsigned char *)in; |
| 56 | + const unsigned char *kk = (const unsigned char *)k; |
| 57 | + |
| 58 | + static_assert(outlen == 8 || outlen == 16, "result should be 8 or 16 bytes"); |
| 59 | + |
| 60 | + uint64_t v0 = UINT64_C(0x736f6d6570736575); |
| 61 | + uint64_t v1 = UINT64_C(0x646f72616e646f6d); |
| 62 | + uint64_t v2 = UINT64_C(0x6c7967656e657261); |
| 63 | + uint64_t v3 = UINT64_C(0x7465646279746573); |
| 64 | + uint64_t k0 = endian::read64le(kk); |
| 65 | + uint64_t k1 = endian::read64le(kk + 8); |
| 66 | + uint64_t m; |
| 67 | + int i; |
| 68 | + const unsigned char *end = ni + inlen - (inlen % sizeof(uint64_t)); |
| 69 | + const int left = inlen & 7; |
| 70 | + uint64_t b = ((uint64_t)inlen) << 56; |
| 71 | + v3 ^= k1; |
| 72 | + v2 ^= k0; |
| 73 | + v1 ^= k1; |
| 74 | + v0 ^= k0; |
| 75 | + |
| 76 | + if (outlen == 16) |
| 77 | + v1 ^= 0xee; |
| 78 | + |
| 79 | + for (; ni != end; ni += 8) { |
| 80 | + m = endian::read64le(ni); |
| 81 | + v3 ^= m; |
164 | 82 |
|
165 | | - TRACE; |
166 | | - for (i = 0; i < dROUNDS; ++i) |
167 | | - SIPROUND; |
168 | | - |
169 | | - b = v0 ^ v1 ^ v2 ^ v3; |
170 | | - U64TO8_LE(out, b); |
171 | | - |
172 | | - if (outlen == 8) |
173 | | - return 0; |
174 | | - |
175 | | - v1 ^= 0xdd; |
| 83 | + for (i = 0; i < cROUNDS; ++i) |
| 84 | + SIPROUND; |
| 85 | + |
| 86 | + v0 ^= m; |
| 87 | + } |
| 88 | + |
| 89 | + switch (left) { |
| 90 | + case 7: |
| 91 | + b |= ((uint64_t)ni[6]) << 48; |
| 92 | + LLVM_FALLTHROUGH; |
| 93 | + case 6: |
| 94 | + b |= ((uint64_t)ni[5]) << 40; |
| 95 | + LLVM_FALLTHROUGH; |
| 96 | + case 5: |
| 97 | + b |= ((uint64_t)ni[4]) << 32; |
| 98 | + LLVM_FALLTHROUGH; |
| 99 | + case 4: |
| 100 | + b |= ((uint64_t)ni[3]) << 24; |
| 101 | + LLVM_FALLTHROUGH; |
| 102 | + case 3: |
| 103 | + b |= ((uint64_t)ni[2]) << 16; |
| 104 | + LLVM_FALLTHROUGH; |
| 105 | + case 2: |
| 106 | + b |= ((uint64_t)ni[1]) << 8; |
| 107 | + LLVM_FALLTHROUGH; |
| 108 | + case 1: |
| 109 | + b |= ((uint64_t)ni[0]); |
| 110 | + break; |
| 111 | + case 0: |
| 112 | + break; |
| 113 | + } |
| 114 | + |
| 115 | + v3 ^= b; |
| 116 | + |
| 117 | + for (i = 0; i < cROUNDS; ++i) |
| 118 | + SIPROUND; |
| 119 | + |
| 120 | + v0 ^= b; |
| 121 | + |
| 122 | + if (outlen == 16) |
| 123 | + v2 ^= 0xee; |
| 124 | + else |
| 125 | + v2 ^= 0xff; |
| 126 | + |
| 127 | + for (i = 0; i < dROUNDS; ++i) |
| 128 | + SIPROUND; |
| 129 | + |
| 130 | + b = v0 ^ v1 ^ v2 ^ v3; |
| 131 | + endian::write64le(out, b); |
| 132 | + |
| 133 | + if (outlen == 8) |
| 134 | + return; |
| 135 | + |
| 136 | + v1 ^= 0xdd; |
| 137 | + |
| 138 | + for (i = 0; i < dROUNDS; ++i) |
| 139 | + SIPROUND; |
| 140 | + |
| 141 | + b = v0 ^ v1 ^ v2 ^ v3; |
| 142 | + endian::write64le(out + 8, b); |
| 143 | +} |
176 | 144 |
|
177 | | - TRACE; |
178 | | - for (i = 0; i < dROUNDS; ++i) |
179 | | - SIPROUND; |
| 145 | +} // end anonymous namespace |
180 | 146 |
|
181 | | - b = v0 ^ v1 ^ v2 ^ v3; |
182 | | - U64TO8_LE(out + 8, b); |
| 147 | +void llvm::getSipHash_2_4_64(ArrayRef<uint8_t> In, const uint8_t (&K)[16], |
| 148 | + uint8_t (&Out)[8]) { |
| 149 | + siphash<2, 4>(In.data(), In.size(), K, Out); |
| 150 | +} |
183 | 151 |
|
184 | | - return 0; |
| 152 | +void llvm::getSipHash_2_4_128(ArrayRef<uint8_t> In, const uint8_t (&K)[16], |
| 153 | + uint8_t (&Out)[16]) { |
| 154 | + siphash<2, 4>(In.data(), In.size(), K, Out); |
185 | 155 | } |
0 commit comments