Line data Source code
1 : // Copyright (c) 2016-2018 The Bitcoin Core developers 2 : // Distributed under the MIT software license, see the accompanying 3 : // file COPYING or http://www.opensource.org/licenses/mit-license.php. 4 : 5 : #include <crypto/siphash.h> 6 : 7 : #define ROTL(x, b) (uint64_t)(((x) << (b)) | ((x) >> (64 - (b)))) 8 : 9 : #define SIPROUND do { \ 10 : v0 += v1; v1 = ROTL(v1, 13); v1 ^= v0; \ 11 : v0 = ROTL(v0, 32); \ 12 : v2 += v3; v3 = ROTL(v3, 16); v3 ^= v2; \ 13 : v0 += v3; v3 = ROTL(v3, 21); v3 ^= v0; \ 14 : v2 += v1; v1 = ROTL(v1, 17); v1 ^= v2; \ 15 : v2 = ROTL(v2, 32); \ 16 : } while (0) 17 : 18 331022 : CSipHasher::CSipHasher(uint64_t k0, uint64_t k1) 19 165511 : { 20 165511 : v[0] = 0x736f6d6570736575ULL ^ k0; 21 165511 : v[1] = 0x646f72616e646f6dULL ^ k1; 22 165511 : v[2] = 0x6c7967656e657261ULL ^ k0; 23 165511 : v[3] = 0x7465646279746573ULL ^ k1; 24 165511 : count = 0; 25 165511 : tmp = 0; 26 331022 : } 27 : 28 2200 : CSipHasher& CSipHasher::Write(uint64_t data) 29 : { 30 2200 : uint64_t v0 = v[0], v1 = v[1], v2 = v[2], v3 = v[3]; 31 : 32 2200 : assert(count % 8 == 0); 33 : 34 2200 : v3 ^= data; 35 2200 : SIPROUND; 36 2200 : SIPROUND; 37 2200 : v0 ^= data; 38 : 39 2200 : v[0] = v0; 40 2200 : v[1] = v1; 41 2200 : v[2] = v2; 42 2200 : v[3] = v3; 43 : 44 2200 : count += 8; 45 2200 : return *this; 46 : } 47 : 48 164870 : CSipHasher& CSipHasher::Write(const unsigned char* data, size_t size) 49 : { 50 164870 : uint64_t v0 = v[0], v1 = v[1], v2 = v[2], v3 = v[3]; 51 164870 : uint64_t t = tmp; 52 164870 : uint8_t c = count; 53 : 54 5328781 : while (size--) { 55 5163911 : t |= ((uint64_t)(*(data++))) << (8 * (c % 8)); 56 5163911 : c++; 57 5163911 : if ((c & 7) == 0) { 58 643513 : v3 ^= t; 59 643513 : SIPROUND; 60 643513 : SIPROUND; 61 643513 : v0 ^= t; 62 : t = 0; 63 643513 : } 64 : } 65 : 66 164870 : v[0] = v0; 67 164870 : v[1] = v1; 68 164870 : v[2] = v2; 69 164870 : v[3] = v3; 70 164870 : count = c; 71 164870 : tmp = t; 72 : 73 164870 : return *this; 74 : } 75 : 76 165615 : uint64_t CSipHasher::Finalize() const 77 : { 78 165615 : uint64_t v0 = v[0], v1 = v[1], v2 = v[2], v3 = v[3]; 79 : 80 165615 : uint64_t t = tmp | (((uint64_t)count) << 56); 81 : 82 165615 : v3 ^= t; 83 165615 : SIPROUND; 84 165615 : SIPROUND; 85 165615 : v0 ^= t; 86 165615 : v2 ^= 0xFF; 87 165615 : SIPROUND; 88 165615 : SIPROUND; 89 165615 : SIPROUND; 90 165615 : SIPROUND; 91 165615 : return v0 ^ v1 ^ v2 ^ v3; 92 : } 93 : 94 30406858 : uint64_t SipHashUint256(uint64_t k0, uint64_t k1, const uint256& val) 95 : { 96 : /* Specialized implementation for efficiency */ 97 30406858 : uint64_t d = val.GetUint64(0); 98 : 99 30406858 : uint64_t v0 = 0x736f6d6570736575ULL ^ k0; 100 30406858 : uint64_t v1 = 0x646f72616e646f6dULL ^ k1; 101 30406858 : uint64_t v2 = 0x6c7967656e657261ULL ^ k0; 102 30406858 : uint64_t v3 = 0x7465646279746573ULL ^ k1 ^ d; 103 : 104 30406858 : SIPROUND; 105 30406858 : SIPROUND; 106 30406858 : v0 ^= d; 107 30406858 : d = val.GetUint64(1); 108 30406858 : v3 ^= d; 109 30406858 : SIPROUND; 110 30406858 : SIPROUND; 111 30406858 : v0 ^= d; 112 30406858 : d = val.GetUint64(2); 113 30406858 : v3 ^= d; 114 30406858 : SIPROUND; 115 30406858 : SIPROUND; 116 30406858 : v0 ^= d; 117 30406858 : d = val.GetUint64(3); 118 30406858 : v3 ^= d; 119 30406858 : SIPROUND; 120 30406858 : SIPROUND; 121 30406858 : v0 ^= d; 122 30406858 : v3 ^= ((uint64_t)4) << 59; 123 30406858 : SIPROUND; 124 30406858 : SIPROUND; 125 30406858 : v0 ^= ((uint64_t)4) << 59; 126 30406858 : v2 ^= 0xFF; 127 30406858 : SIPROUND; 128 30406858 : SIPROUND; 129 30406858 : SIPROUND; 130 30406858 : SIPROUND; 131 30406858 : return v0 ^ v1 ^ v2 ^ v3; 132 : } 133 : 134 104860894 : uint64_t SipHashUint256Extra(uint64_t k0, uint64_t k1, const uint256& val, uint32_t extra) 135 : { 136 : /* Specialized implementation for efficiency */ 137 104860894 : uint64_t d = val.GetUint64(0); 138 : 139 104860894 : uint64_t v0 = 0x736f6d6570736575ULL ^ k0; 140 104860894 : uint64_t v1 = 0x646f72616e646f6dULL ^ k1; 141 104860894 : uint64_t v2 = 0x6c7967656e657261ULL ^ k0; 142 104860894 : uint64_t v3 = 0x7465646279746573ULL ^ k1 ^ d; 143 : 144 104860894 : SIPROUND; 145 104860894 : SIPROUND; 146 104860894 : v0 ^= d; 147 104860894 : d = val.GetUint64(1); 148 104860894 : v3 ^= d; 149 104860894 : SIPROUND; 150 104860894 : SIPROUND; 151 104860894 : v0 ^= d; 152 104860894 : d = val.GetUint64(2); 153 104860894 : v3 ^= d; 154 104860894 : SIPROUND; 155 104860894 : SIPROUND; 156 104860894 : v0 ^= d; 157 104860894 : d = val.GetUint64(3); 158 104860894 : v3 ^= d; 159 104860894 : SIPROUND; 160 104860894 : SIPROUND; 161 104860894 : v0 ^= d; 162 104860894 : d = (((uint64_t)36) << 56) | extra; 163 104860894 : v3 ^= d; 164 104860894 : SIPROUND; 165 104860894 : SIPROUND; 166 104860894 : v0 ^= d; 167 104860894 : v2 ^= 0xFF; 168 104860894 : SIPROUND; 169 104860894 : SIPROUND; 170 104860894 : SIPROUND; 171 104860894 : SIPROUND; 172 104860894 : return v0 ^ v1 ^ v2 ^ v3; 173 : }