43 std::array<uint32_t, 8> hash = {0x6A09E667, 0xBB67AE85, 0x3C6EF372,
44 0xA54FF53A, 0x510E527F, 0x9B05688C,
45 0x1F83D9AB, 0x5BE0CD19};
48 void update(
const std::array<uint32_t, 64> &blocks);
70 const std::array<uint32_t, 64> round_constants = {
71 0x428A2F98, 0x71374491, 0xB5C0FBCF, 0xE9B5DBA5, 0x3956C25B, 0x59F111F1,
72 0x923F82A4, 0xAB1C5ED5, 0xD807AA98, 0x12835B01, 0x243185BE, 0x550C7DC3,
73 0x72BE5D74, 0x80DEB1FE, 0x9BDC06A7, 0xC19BF174, 0xE49B69C1, 0xEFBE4786,
74 0x0FC19DC6, 0x240CA1CC, 0x2DE92C6F, 0x4A7484AA, 0x5CB0A9DC, 0x76F988DA,
75 0x983E5152, 0xA831C66D, 0xB00327C8, 0xBF597FC7, 0xC6E00BF3, 0xD5A79147,
76 0x06CA6351, 0x14292967, 0x27B70A85, 0x2E1B2138, 0x4D2C6DFC, 0x53380D13,
77 0x650A7354, 0x766A0ABB, 0x81C2C92E, 0x92722C85, 0xA2BFE8A1, 0xA81A664B,
78 0xC24B8B70, 0xC76C51A3, 0xD192E819, 0xD6990624, 0xF40E3585, 0x106AA070,
79 0x19A4C116, 0x1E376C08, 0x2748774C, 0x34B0BCB5, 0x391C0CB3, 0x4ED8AA4A,
80 0x5B9CCA4F, 0x682E6FF3, 0x748F82EE, 0x78A5636F, 0x84C87814, 0x8CC70208,
81 0x90BEFFFA, 0xA4506CEB, 0xBEF9A3F7, 0xC67178F2};
94 for (
size_t block_num = 0; block_num < 64; ++block_num) {
97 const auto ch = (e & f) ^ (~e & g);
99 h + s1 + ch + round_constants[block_num] + blocks[block_num];
102 const auto maj = (a & b) ^ (a & c) ^ (b & c);
103 const auto temp2 = s0 + maj;
170char get_char(
const std::string &input, std::size_t pos) {
171 const auto input_size = input.length();
172 if (pos < input_size) {
175 if (pos == input_size) {
179 if (pos < padded_input_size - 8) {
182 if (padded_input_size <= pos) {
183 throw std::out_of_range(
"pos is out of range");
185 return static_cast<char>(
186 extract_byte<size_t>(input_size * 8, padded_input_size - pos - 1));
196 const size_t byte_num) {
197 std::array<uint32_t, 64> blocks{};
200 for (
size_t block_num = 0; block_num < 16; ++block_num) {
202 (
static_cast<uint8_t
>(
get_char(input, byte_num + block_num * 4))
204 (
static_cast<uint8_t
>(
get_char(input, byte_num + block_num * 4 + 1))
206 (
static_cast<uint8_t
>(
get_char(input, byte_num + block_num * 4 + 2))
208 static_cast<uint8_t
>(
get_char(input, byte_num + block_num * 4 + 3));
213 for (
size_t block_num = 16; block_num < 64; ++block_num) {
214 const auto s0 =
right_rotate(blocks[block_num - 15], 7) ^
216 (blocks[block_num - 15] >> 3);
217 const auto s1 =
right_rotate(blocks[block_num - 2], 17) ^
219 (blocks[block_num - 2] >> 10);
221 blocks[block_num - 16] + s0 + blocks[block_num - 7] + s1;