M5Utility 0.0.12 git rev:13bfbd6
Loading...
Searching...
No Matches
sha1.hpp
Go to the documentation of this file.
1/*
2 * SPDX-FileCopyrightText: 2025 M5Stack Technology CO LTD
3 *
4 * SPDX-License-Identifier: MIT
5 */
10#ifndef M5_UTILITY_SHA1_HPP
11#define M5_UTILITY_SHA1_HPP
12
13#include <cstdint>
14#include <cstring>
15#include "../stl/byteswap.hpp"
16
17namespace m5 {
18namespace utility {
19
25class SHA1 {
26public:
27 SHA1()
28 {
29 }
30
31 void init()
32 {
33 _state[0] = 0x67452301;
34 _state[1] = 0xEFCDAB89;
35 _state[2] = 0x98BADCFE;
36 _state[3] = 0x10325476;
37 _state[4] = 0xC3D2E1F0;
38 _count = 0;
39 _bufferLen = 0;
40 memset(_buffer, 0, sizeof(_buffer));
41 }
42
43 void update(const uint8_t* data, size_t len)
44 {
45 _count += static_cast<uint64_t>(len) << 3;
46 size_t fill{64 - _bufferLen};
47
48 if (len >= fill) {
49 memcpy(_buffer + _bufferLen, data, fill);
50 transform(_state, _buffer);
51 data += fill;
52 len -= fill;
53 _bufferLen = 0;
54
55 while (len >= 64) {
56 transform(_state, data);
57 data += 64;
58 len -= 64;
59 }
60 }
61
62 memcpy(_buffer + _bufferLen, data, len);
63 _bufferLen += len;
64 }
65
66 void finalize(uint8_t digest[20])
67 {
68 uint8_t padding[64]{0x80};
69 uint8_t length[8]{};
70 uint64_t count_be = m5::stl::byteswap(_count);
71 memcpy(length, &count_be, 8);
72
73 size_t padLen = (_bufferLen < 56) ? (56 - _bufferLen) : (120 - _bufferLen);
74 update(padding, padLen);
75 update(length, 8);
76
77 for (size_t i = 0; i < 5; ++i) {
78 uint32_t be = m5::stl::byteswap(_state[i]);
79 memcpy(digest + i * 4, &be, 4);
80 }
81 }
82
83 static void sha1(uint8_t digest[20], const uint8_t* data, size_t len)
84 {
85 SHA1 ctx;
86 ctx.update(data, len);
87 ctx.finalize(digest);
88 }
89
90private:
91 void transform(uint32_t state[5], const uint8_t buffer[64])
92 {
93 uint32_t a = state[0], b = state[1], c = state[2], d = state[3], e = state[4];
94 uint32_t block[80];
95
96 for (int i = 0; i < 16; ++i) {
97 block[i] = (uint32_t(buffer[i * 4 + 0]) << 24) | (uint32_t(buffer[i * 4 + 1]) << 16) |
98 (uint32_t(buffer[i * 4 + 2]) << 8) | (uint32_t(buffer[i * 4 + 3]));
99 }
100 for (int i = 16; i < 80; ++i) {
101 block[i] = rol(block[i - 3] ^ block[i - 8] ^ block[i - 14] ^ block[i - 16], 1);
102 }
103
104 for (int i = 0; i < 80; ++i) {
105 uint32_t f, k;
106 if (i < 20) {
107 f = (b & c) | ((~b) & d);
108 k = 0x5A827999;
109 } else if (i < 40) {
110 f = b ^ c ^ d;
111 k = 0x6ED9EBA1;
112 } else if (i < 60) {
113 f = (b & c) | (b & d) | (c & d);
114 k = 0x8F1BBCDC;
115 } else {
116 f = b ^ c ^ d;
117 k = 0xCA62C1D6;
118 }
119
120 uint32_t temp = rol(a, 5) + f + e + k + block[i];
121 e = d;
122 d = c;
123 c = rol(b, 30);
124 b = a;
125 a = temp;
126 }
127
128 state[0] += a;
129 state[1] += b;
130 state[2] += c;
131 state[3] += d;
132 state[4] += e;
133 }
134
135 inline uint32_t rol(uint32_t value, uint32_t bits)
136 {
137 return (value << bits) | (value >> (32 - bits));
138 }
139
140 uint32_t _state[5]{0x67452301, 0xEFCDAB89, 0x98BADCFE, 0x10325476, 0xC3D2E1F0};
141 uint64_t _count{};
142 uint8_t _buffer[64]{};
143 size_t _bufferLen{};
144};
145
146} // namespace utility
147} // namespace m5
148#endif
std::byteswap for less than C++23
constexpr T byteswap(T v) noexcept
byteswap for integral type
Definition byteswap.hpp:193
Top level namespace of M5.
Definition base64.cpp:39
For utilities.