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