-
-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathsha0
247 lines (205 loc) · 9.9 KB
/
sha0
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
/*
Copyright (C) 2018-2024 Geoffrey Daniels. https://gpdaniels.com/
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, version 3 of the License only.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
#pragma once
#ifndef GTL_HASH_SHA0_HPP
#define GTL_HASH_SHA0_HPP
// Summary: An implementation of the sha0 hashing function.
#ifndef NDEBUG
# if defined(_MSC_VER)
# define __builtin_trap() __debugbreak()
# endif
/// @brief A simple assert macro to break the program if the sha0 is misused.
# define GTL_SHA0_ASSERT(ASSERTION, MESSAGE) static_cast<void>((ASSERTION) || (__builtin_trap(), 0))
#else
/// @brief At release time the assert macro is implemented as a nop.
# define GTL_SHA0_ASSERT(ASSERTION, MESSAGE) static_cast<void>(0)
#endif
namespace gtl {
/// @brief The sha0 class computes and SHA0 hash of a sequence of data.
class sha0 final {
public:
/// @brief Handy variable for accessing the size in bits.
constexpr static const unsigned long long int size = 160;
/// @brief Size of the hash in bytes.
constexpr static const unsigned long long int hash_size = sha0::size / 8;
/// @brief Internal size of each processed block of data in bytes.
constexpr static const unsigned long long int block_size = 64;
/// @brief Number of rounds used to process each block of data.
constexpr static const unsigned long long int processing_rounds = 80;
public:
/// @brief Simple type to hold the hash data.
struct hash_type final {
unsigned int data[sha0::hash_size / sizeof(unsigned int)] = {
0x67452301, 0xEFCDAB89, 0x98BADCFE, 0x10325476, 0xC3D2E1F0
};
};
private:
/// @brief Internal type used to construct blocks of data for processing.
struct block_type final {
unsigned long long int length_bytes = 0;
unsigned int data[sha0::block_size / sizeof(unsigned int)] = {};
};
private:
/// @brief The number of bytes of data that have been hashed.
unsigned long long int hashed_length = 0;
/// @brief The current block of data being filled.
block_type block = {};
/// @brief The current hash.
hash_type hash = {};
/// @brief Flag showing whether the hash has been finalised.
bool finalised = false;
private:
/// @brief Rotate a 32 bit value left by a shift amount.
/// @param value The value to rotate.
/// @param shift The number of bits to rotate by.
/// @return The value after it has been rotated left by shift bits.
constexpr static unsigned int rotate_left(unsigned int value, unsigned int shift) {
constexpr const unsigned int mask = (8 * sizeof(unsigned int) - 1);
shift &= mask;
return (value << shift) | (value >> ((static_cast<unsigned int>(-static_cast<signed int>(shift))) & mask));
}
private:
/// @brief Perform the hashing algorithm on a block of data.
constexpr void hash_block() {
unsigned int A = this->hash.data[0];
unsigned int B = this->hash.data[1];
unsigned int C = this->hash.data[2];
unsigned int D = this->hash.data[3];
unsigned int E = this->hash.data[4];
// Fill the start of the working block from the current block.
unsigned int working_block[80] = {};
for (unsigned int index = 0; index < sha0::block_size / sizeof(unsigned int); ++index) {
working_block[index] = this->block.data[index];
}
// Mix to fill the remainder of the working block.
for (unsigned int index = sha0::block_size / sizeof(unsigned int); index < 80; ++index) {
// This line is the only change from sha1, which includes the addition of a 1 bit left rotate.
working_block[index] = (working_block[index - 3] ^ working_block[index - 8] ^ working_block[index - 14] ^ working_block[index - 16]);
}
// Do the hash iterations.
for (unsigned int iteration = 0; iteration < this->processing_rounds; ++iteration) {
unsigned int F = 0;
unsigned int K = 0;
if (iteration < 20) {
F = (B & C) | (~B & D);
K = 0x5A827999;
}
else if (iteration < 40) {
F = B ^ C ^ D;
K = 0x6ED9EBA1;
}
else if (iteration < 60) {
F = (B & C) | (B & D) | (C & D);
K = 0x8F1BBCDC;
}
else {
F = B ^ C ^ D;
K = 0xCA62C1D6;
}
unsigned int next_A = rotate_left(A, 5) + F + E + K + working_block[iteration];
E = D;
D = C;
C = rotate_left(B, 30);
B = A;
A = next_A;
}
this->hash.data[0] += A;
this->hash.data[1] += B;
this->hash.data[2] += C;
this->hash.data[3] += D;
this->hash.data[4] += E;
}
public:
/// @brief Reset the internal state of the hash.
constexpr void reset() {
this->hashed_length = 0;
this->block = {};
this->hash = {};
this->finalised = false;
}
/// @brief Consume data, constructing blocks and hashing as needed.
/// @param data Input data pointer.
/// @param length Valid length of the input data pointer.
constexpr void consume(const char* data, unsigned long long int length) {
GTL_SHA0_ASSERT(this->finalised == false, "Finalise must not be called before consume().");
// Count how much data has been consumed.
this->hashed_length += length;
// Block construction and processing.
while (length--) {
// Append data to the current block.
this->block.data[this->block.length_bytes / sizeof(unsigned int)] |= static_cast<unsigned int>(static_cast<unsigned char>(*data++)) << (((sizeof(unsigned int) - 1) - (this->block.length_bytes % sizeof(unsigned int))) * 8);
++this->block.length_bytes;
// If full, hash.
if (this->block.length_bytes == sha0::block_size) {
this->hash_block();
this->block = {};
}
}
}
/// @brief Finishes processing any remaining buffered data.
constexpr void finalise() {
GTL_SHA0_ASSERT(this->finalised == false, "Finalise must not be called before finalise().");
// Pad the data with a single 0x80 on the end.
this->block.data[this->block.length_bytes / sizeof(unsigned int)] |= static_cast<unsigned int>(static_cast<unsigned char>(0x80)) << (((sizeof(unsigned int) - 1) - (this->block.length_bytes % sizeof(unsigned int))) * 8);
++this->block.length_bytes;
// Check if there is enough space in the final chunk for the data length.
if (this->block.length_bytes > (sha0::block_size - sizeof(unsigned long long int))) {
this->hash_block();
this->block = {};
}
// Add the data length to the chunk.
const unsigned long long int hashed_length_bits = this->hashed_length * 8;
this->block.data[14] = (hashed_length_bits & 0xFFFFFFFF00000000ull) >> 32u;
this->block.data[15] = (hashed_length_bits & 0x00000000FFFFFFFFull) >> 0u;
// Hash the block.
this->hash_block();
this->finalised = true;
}
/// @brief Calculate the final hash from the state and return it.
/// @return The resultant hash.
constexpr const hash_type& get_hash() const {
GTL_SHA0_ASSERT(this->finalised == true, "Finalise must be called before get_hash().");
return this->hash;
}
public:
/// @brief Simple type to hold the string representation of the hash.
struct hash_string_type final {
char hash[(sha0::hash_size * 2) + 1] = {};
};
/// @brief Convert a hash into its string representation.
/// @param hash The hash to convert.
/// @return The string representation of the hash, with null termination.
constexpr static hash_string_type hash_to_string(const hash_type& hash) {
hash_string_type output;
constexpr const char* hex_characters = "0123456789ABCDEF";
for (unsigned int index = 0u; index < (sha0::hash_size * 2); ++index) {
output.hash[index] = hex_characters[(hash.data[index / (sizeof(unsigned int) * 2)] >> ((((sizeof(unsigned int) * 2) - 1) - (index % (sizeof(unsigned int) * 2))) * 4)) & 0xF];
}
output.hash[sha0::hash_size * 2] = 0;
return output;
}
public:
/// @brief Hash a buffer of data.
/// @param data Input data pointer.
/// @param length Valid length of the input data pointer.
static hash_type hash_buffer(const char* data, const unsigned long long int length) {
sha0 hash_function;
hash_function.reset();
hash_function.consume(data, length);
hash_function.finalise();
return hash_function.get_hash();
}
};
}
#undef GTL_SHA0_ASSERT
#endif // GTL_HASH_SHA0_HPP