Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
stenzek
GitHub Repository: stenzek/duckstation
Path: blob/master/src/common/md5_digest.cpp
4212 views
1
// SPDX-FileCopyrightText: 2019-2024 Connor McLaughlin <[email protected]>
2
// SPDX-License-Identifier: CC-BY-NC-ND-4.0
3
4
#include "md5_digest.h"
5
6
// based heavily on this public-domain implementation:
7
// http://www.fourmilab.ch/md5/
8
9
/* The four core functions - F1 is optimized somewhat */
10
11
/* #define F1(x, y, z) (x & y | ~x & z) */
12
#define F1(x, y, z) (z ^ (x & (y ^ z)))
13
#define F2(x, y, z) F1(z, x, y)
14
#define F3(x, y, z) (x ^ y ^ z)
15
#define F4(x, y, z) (y ^ (x | ~z))
16
17
/* This is the central step in the MD5 algorithm. */
18
#define MD5STEP(f, w, x, y, z, data, s) (w += f(x, y, z) + data, w = w << s | w >> (32 - s), w += x)
19
20
/*
21
* The core of the MD5 algorithm, this alters an existing MD5 hash to
22
* reflect the addition of 16 longwords of new data. MD5Update blocks
23
* the data and converts bytes into longwords for this routine.
24
*/
25
static void MD5Transform(u32 buf[4], u32 in[16])
26
{
27
// register u32 a, b, c, d;
28
u32 a, b, c, d;
29
30
a = buf[0];
31
b = buf[1];
32
c = buf[2];
33
d = buf[3];
34
35
MD5STEP(F1, a, b, c, d, in[0] + 0xd76aa478, 7);
36
MD5STEP(F1, d, a, b, c, in[1] + 0xe8c7b756, 12);
37
MD5STEP(F1, c, d, a, b, in[2] + 0x242070db, 17);
38
MD5STEP(F1, b, c, d, a, in[3] + 0xc1bdceee, 22);
39
MD5STEP(F1, a, b, c, d, in[4] + 0xf57c0faf, 7);
40
MD5STEP(F1, d, a, b, c, in[5] + 0x4787c62a, 12);
41
MD5STEP(F1, c, d, a, b, in[6] + 0xa8304613, 17);
42
MD5STEP(F1, b, c, d, a, in[7] + 0xfd469501, 22);
43
MD5STEP(F1, a, b, c, d, in[8] + 0x698098d8, 7);
44
MD5STEP(F1, d, a, b, c, in[9] + 0x8b44f7af, 12);
45
MD5STEP(F1, c, d, a, b, in[10] + 0xffff5bb1, 17);
46
MD5STEP(F1, b, c, d, a, in[11] + 0x895cd7be, 22);
47
MD5STEP(F1, a, b, c, d, in[12] + 0x6b901122, 7);
48
MD5STEP(F1, d, a, b, c, in[13] + 0xfd987193, 12);
49
MD5STEP(F1, c, d, a, b, in[14] + 0xa679438e, 17);
50
MD5STEP(F1, b, c, d, a, in[15] + 0x49b40821, 22);
51
52
MD5STEP(F2, a, b, c, d, in[1] + 0xf61e2562, 5);
53
MD5STEP(F2, d, a, b, c, in[6] + 0xc040b340, 9);
54
MD5STEP(F2, c, d, a, b, in[11] + 0x265e5a51, 14);
55
MD5STEP(F2, b, c, d, a, in[0] + 0xe9b6c7aa, 20);
56
MD5STEP(F2, a, b, c, d, in[5] + 0xd62f105d, 5);
57
MD5STEP(F2, d, a, b, c, in[10] + 0x02441453, 9);
58
MD5STEP(F2, c, d, a, b, in[15] + 0xd8a1e681, 14);
59
MD5STEP(F2, b, c, d, a, in[4] + 0xe7d3fbc8, 20);
60
MD5STEP(F2, a, b, c, d, in[9] + 0x21e1cde6, 5);
61
MD5STEP(F2, d, a, b, c, in[14] + 0xc33707d6, 9);
62
MD5STEP(F2, c, d, a, b, in[3] + 0xf4d50d87, 14);
63
MD5STEP(F2, b, c, d, a, in[8] + 0x455a14ed, 20);
64
MD5STEP(F2, a, b, c, d, in[13] + 0xa9e3e905, 5);
65
MD5STEP(F2, d, a, b, c, in[2] + 0xfcefa3f8, 9);
66
MD5STEP(F2, c, d, a, b, in[7] + 0x676f02d9, 14);
67
MD5STEP(F2, b, c, d, a, in[12] + 0x8d2a4c8a, 20);
68
69
MD5STEP(F3, a, b, c, d, in[5] + 0xfffa3942, 4);
70
MD5STEP(F3, d, a, b, c, in[8] + 0x8771f681, 11);
71
MD5STEP(F3, c, d, a, b, in[11] + 0x6d9d6122, 16);
72
MD5STEP(F3, b, c, d, a, in[14] + 0xfde5380c, 23);
73
MD5STEP(F3, a, b, c, d, in[1] + 0xa4beea44, 4);
74
MD5STEP(F3, d, a, b, c, in[4] + 0x4bdecfa9, 11);
75
MD5STEP(F3, c, d, a, b, in[7] + 0xf6bb4b60, 16);
76
MD5STEP(F3, b, c, d, a, in[10] + 0xbebfbc70, 23);
77
MD5STEP(F3, a, b, c, d, in[13] + 0x289b7ec6, 4);
78
MD5STEP(F3, d, a, b, c, in[0] + 0xeaa127fa, 11);
79
MD5STEP(F3, c, d, a, b, in[3] + 0xd4ef3085, 16);
80
MD5STEP(F3, b, c, d, a, in[6] + 0x04881d05, 23);
81
MD5STEP(F3, a, b, c, d, in[9] + 0xd9d4d039, 4);
82
MD5STEP(F3, d, a, b, c, in[12] + 0xe6db99e5, 11);
83
MD5STEP(F3, c, d, a, b, in[15] + 0x1fa27cf8, 16);
84
MD5STEP(F3, b, c, d, a, in[2] + 0xc4ac5665, 23);
85
86
MD5STEP(F4, a, b, c, d, in[0] + 0xf4292244, 6);
87
MD5STEP(F4, d, a, b, c, in[7] + 0x432aff97, 10);
88
MD5STEP(F4, c, d, a, b, in[14] + 0xab9423a7, 15);
89
MD5STEP(F4, b, c, d, a, in[5] + 0xfc93a039, 21);
90
MD5STEP(F4, a, b, c, d, in[12] + 0x655b59c3, 6);
91
MD5STEP(F4, d, a, b, c, in[3] + 0x8f0ccc92, 10);
92
MD5STEP(F4, c, d, a, b, in[10] + 0xffeff47d, 15);
93
MD5STEP(F4, b, c, d, a, in[1] + 0x85845dd1, 21);
94
MD5STEP(F4, a, b, c, d, in[8] + 0x6fa87e4f, 6);
95
MD5STEP(F4, d, a, b, c, in[15] + 0xfe2ce6e0, 10);
96
MD5STEP(F4, c, d, a, b, in[6] + 0xa3014314, 15);
97
MD5STEP(F4, b, c, d, a, in[13] + 0x4e0811a1, 21);
98
MD5STEP(F4, a, b, c, d, in[4] + 0xf7537e82, 6);
99
MD5STEP(F4, d, a, b, c, in[11] + 0xbd3af235, 10);
100
MD5STEP(F4, c, d, a, b, in[2] + 0x2ad7d2bb, 15);
101
MD5STEP(F4, b, c, d, a, in[9] + 0xeb86d391, 21);
102
103
buf[0] += a;
104
buf[1] += b;
105
buf[2] += c;
106
buf[3] += d;
107
}
108
109
MD5Digest::MD5Digest()
110
{
111
Reset();
112
}
113
114
void MD5Digest::Reset()
115
{
116
this->buf[0] = 0x67452301;
117
this->buf[1] = 0xefcdab89;
118
this->buf[2] = 0x98badcfe;
119
this->buf[3] = 0x10325476;
120
121
this->bits[0] = 0;
122
this->bits[1] = 0;
123
}
124
125
std::array<u8, MD5Digest::DIGEST_SIZE> MD5Digest::HashData(std::span<const u8> data)
126
{
127
std::array<u8, DIGEST_SIZE> ret;
128
129
MD5Digest digest;
130
digest.Update(data);
131
digest.Final(ret);
132
return ret;
133
}
134
135
void MD5Digest::Update(const void* pData, u32 cbData)
136
{
137
u32 t;
138
const u8* pByteData = reinterpret_cast<const u8*>(pData);
139
140
/* Update bitcount */
141
142
t = this->bits[0];
143
if ((this->bits[0] = t + ((u32)cbData << 3)) < t)
144
this->bits[1]++; /* Carry from low to high */
145
this->bits[1] += cbData >> 29;
146
147
t = (t >> 3) & 0x3f; /* Bytes already in shsInfo->data */
148
149
/* Handle any leading odd-sized chunks */
150
151
if (t)
152
{
153
u8* p = (u8*)this->in + t;
154
155
t = 64 - t;
156
if (cbData < t)
157
{
158
std::memcpy(p, pByteData, cbData);
159
return;
160
}
161
std::memcpy(p, pByteData, t);
162
MD5Transform(this->buf, (u32*)this->in);
163
pByteData += t;
164
cbData -= t;
165
}
166
/* Process data in 64-byte chunks */
167
168
while (cbData >= 64)
169
{
170
std::memcpy(this->in, pByteData, 64);
171
MD5Transform(this->buf, (u32*)this->in);
172
pByteData += 64;
173
cbData -= 64;
174
}
175
176
/* Handle any remaining bytes of data. */
177
178
std::memcpy(this->in, pByteData, cbData);
179
}
180
181
void MD5Digest::Update(std::span<const u8> data)
182
{
183
if (!data.empty())
184
Update(data.data(), static_cast<u32>(data.size_bytes()));
185
}
186
187
void MD5Digest::Final(std::span<u8, DIGEST_SIZE> digest)
188
{
189
u32 count;
190
u8* p;
191
192
/* Compute number of bytes mod 64 */
193
count = (this->bits[0] >> 3) & 0x3F;
194
195
/* Set the first char of padding to 0x80. This is safe since there is
196
always at least one byte free */
197
p = this->in + count;
198
*p++ = 0x80;
199
200
/* Bytes of padding needed to make 64 bytes */
201
count = 64 - 1 - count;
202
203
/* Pad out to 56 mod 64 */
204
if (count < 8)
205
{
206
/* Two lots of padding: Pad the first block to 64 bytes */
207
std::memset(p, 0, count);
208
MD5Transform(this->buf, (u32*)this->in);
209
210
/* Now fill the next block with 56 bytes */
211
std::memset(this->in, 0, 56);
212
}
213
else
214
{
215
/* Pad block to 56 bytes */
216
std::memset(p, 0, count - 8);
217
}
218
219
/* Append length in bits and transform */
220
((u32*)this->in)[14] = this->bits[0];
221
((u32*)this->in)[15] = this->bits[1];
222
223
MD5Transform(this->buf, (u32*)this->in);
224
std::memcpy(digest.data(), this->buf, 16);
225
}
226
227