Adeon
Loading...
Searching...
No Matches
MD5.cpp
Go to the documentation of this file.
1
9#include "MD5.h"
10
11MD5::MD5()
12{
13 //nothing
14 return;
15}
16
17char* MD5::make_digest(const unsigned char *digest, int len) /* {{{ */
18{
19 char * md5str = (char*) malloc(sizeof(char)*(len*2+1));
20 static const char hexits[17] = "0123456789abcdef";
21 int i;
22
23 for (i = 0; i < len; i++) {
24 md5str[i * 2] = hexits[digest[i] >> 4];
25 md5str[(i * 2) + 1] = hexits[digest[i] & 0x0F];
26 }
27 md5str[len * 2] = '\0';
28 return md5str;
29}
30
31/*
32 * The basic MD5 functions.
33 *
34 * E and G are optimized compared to their RFC 1321 definitions for
35 * architectures that lack an AND-NOT instruction, just like in Colin Plumb's
36 * implementation.
37 * E() has been used instead of F() because F() is already defined in the Arduino core
38 */
39#define E(x, y, z) ((z) ^ ((x) & ((y) ^ (z))))
40#define G(x, y, z) ((y) ^ ((z) & ((x) ^ (y))))
41#define H(x, y, z) ((x) ^ (y) ^ (z))
42#define I(x, y, z) ((y) ^ ((x) | ~(z)))
43
44/*
45 * The MD5 transformation for all four rounds.
46 */
47#define STEP(f, a, b, c, d, x, t, s) \
48 (a) += f((b), (c), (d)) + (x) + (t); \
49 (a) = (((a) << (s)) | (((a) & 0xffffffff) >> (32 - (s)))); \
50 (a) += (b);
51
52/*
53 * SET reads 4 input bytes in little-endian byte order and stores them
54 * in a properly aligned word in host byte order.
55 *
56 * The check for little-endian architectures that tolerate unaligned
57 * memory accesses is just an optimization. Nothing will break if it
58 * doesn't work.
59 */
60#if defined(__i386__) || defined(__x86_64__) || defined(__vax__)
61# define SET(n) \
62 (*(MD5_u32plus *)&ptr[(n) * 4])
63# define GET(n) \
64 SET(n)
65#else
66# define SET(n) \
67 (ctx->block[(n)] = \
68 (MD5_u32plus)ptr[(n) * 4] | \
69 ((MD5_u32plus)ptr[(n) * 4 + 1] << 8) | \
70 ((MD5_u32plus)ptr[(n) * 4 + 2] << 16) | \
71 ((MD5_u32plus)ptr[(n) * 4 + 3] << 24))
72# define GET(n) \
73 (ctx->block[(n)])
74#endif
75
76/*
77 * This processes one or more 64-byte data blocks, but does NOT update
78 * the bit counters. There are no alignment requirements.
79 */
80const void *MD5::body(void *ctxBuf, const void *data, size_t size)
81{
82 MD5_CTX *ctx = (MD5_CTX*)ctxBuf;
83 const unsigned char *ptr;
84 MD5_u32plus a, b, c, d;
85 MD5_u32plus saved_a, saved_b, saved_c, saved_d;
86
87 ptr = (unsigned char*)data;
88
89 a = ctx->a;
90 b = ctx->b;
91 c = ctx->c;
92 d = ctx->d;
93
94 do {
95 saved_a = a;
96 saved_b = b;
97 saved_c = c;
98 saved_d = d;
99
100/* Round 1
101 * E() has been used instead of F() because F() is already defined in the Arduino core
102 */
103 STEP(E, a, b, c, d, SET(0), 0xd76aa478, 7)
104 STEP(E, d, a, b, c, SET(1), 0xe8c7b756, 12)
105 STEP(E, c, d, a, b, SET(2), 0x242070db, 17)
106 STEP(E, b, c, d, a, SET(3), 0xc1bdceee, 22)
107 STEP(E, a, b, c, d, SET(4), 0xf57c0faf, 7)
108 STEP(E, d, a, b, c, SET(5), 0x4787c62a, 12)
109 STEP(E, c, d, a, b, SET(6), 0xa8304613, 17)
110 STEP(E, b, c, d, a, SET(7), 0xfd469501, 22)
111 STEP(E, a, b, c, d, SET(8), 0x698098d8, 7)
112 STEP(E, d, a, b, c, SET(9), 0x8b44f7af, 12)
113 STEP(E, c, d, a, b, SET(10), 0xffff5bb1, 17)
114 STEP(E, b, c, d, a, SET(11), 0x895cd7be, 22)
115 STEP(E, a, b, c, d, SET(12), 0x6b901122, 7)
116 STEP(E, d, a, b, c, SET(13), 0xfd987193, 12)
117 STEP(E, c, d, a, b, SET(14), 0xa679438e, 17)
118 STEP(E, b, c, d, a, SET(15), 0x49b40821, 22)
119
120/* Round 2 */
121 STEP(G, a, b, c, d, GET(1), 0xf61e2562, 5)
122 STEP(G, d, a, b, c, GET(6), 0xc040b340, 9)
123 STEP(G, c, d, a, b, GET(11), 0x265e5a51, 14)
124 STEP(G, b, c, d, a, GET(0), 0xe9b6c7aa, 20)
125 STEP(G, a, b, c, d, GET(5), 0xd62f105d, 5)
126 STEP(G, d, a, b, c, GET(10), 0x02441453, 9)
127 STEP(G, c, d, a, b, GET(15), 0xd8a1e681, 14)
128 STEP(G, b, c, d, a, GET(4), 0xe7d3fbc8, 20)
129 STEP(G, a, b, c, d, GET(9), 0x21e1cde6, 5)
130 STEP(G, d, a, b, c, GET(14), 0xc33707d6, 9)
131 STEP(G, c, d, a, b, GET(3), 0xf4d50d87, 14)
132 STEP(G, b, c, d, a, GET(8), 0x455a14ed, 20)
133 STEP(G, a, b, c, d, GET(13), 0xa9e3e905, 5)
134 STEP(G, d, a, b, c, GET(2), 0xfcefa3f8, 9)
135 STEP(G, c, d, a, b, GET(7), 0x676f02d9, 14)
136 STEP(G, b, c, d, a, GET(12), 0x8d2a4c8a, 20)
137
138/* Round 3 */
139 STEP(H, a, b, c, d, GET(5), 0xfffa3942, 4)
140 STEP(H, d, a, b, c, GET(8), 0x8771f681, 11)
141 STEP(H, c, d, a, b, GET(11), 0x6d9d6122, 16)
142 STEP(H, b, c, d, a, GET(14), 0xfde5380c, 23)
143 STEP(H, a, b, c, d, GET(1), 0xa4beea44, 4)
144 STEP(H, d, a, b, c, GET(4), 0x4bdecfa9, 11)
145 STEP(H, c, d, a, b, GET(7), 0xf6bb4b60, 16)
146 STEP(H, b, c, d, a, GET(10), 0xbebfbc70, 23)
147 STEP(H, a, b, c, d, GET(13), 0x289b7ec6, 4)
148 STEP(H, d, a, b, c, GET(0), 0xeaa127fa, 11)
149 STEP(H, c, d, a, b, GET(3), 0xd4ef3085, 16)
150 STEP(H, b, c, d, a, GET(6), 0x04881d05, 23)
151 STEP(H, a, b, c, d, GET(9), 0xd9d4d039, 4)
152 STEP(H, d, a, b, c, GET(12), 0xe6db99e5, 11)
153 STEP(H, c, d, a, b, GET(15), 0x1fa27cf8, 16)
154 STEP(H, b, c, d, a, GET(2), 0xc4ac5665, 23)
155
156/* Round 4 */
157 STEP(I, a, b, c, d, GET(0), 0xf4292244, 6)
158 STEP(I, d, a, b, c, GET(7), 0x432aff97, 10)
159 STEP(I, c, d, a, b, GET(14), 0xab9423a7, 15)
160 STEP(I, b, c, d, a, GET(5), 0xfc93a039, 21)
161 STEP(I, a, b, c, d, GET(12), 0x655b59c3, 6)
162 STEP(I, d, a, b, c, GET(3), 0x8f0ccc92, 10)
163 STEP(I, c, d, a, b, GET(10), 0xffeff47d, 15)
164 STEP(I, b, c, d, a, GET(1), 0x85845dd1, 21)
165 STEP(I, a, b, c, d, GET(8), 0x6fa87e4f, 6)
166 STEP(I, d, a, b, c, GET(15), 0xfe2ce6e0, 10)
167 STEP(I, c, d, a, b, GET(6), 0xa3014314, 15)
168 STEP(I, b, c, d, a, GET(13), 0x4e0811a1, 21)
169 STEP(I, a, b, c, d, GET(4), 0xf7537e82, 6)
170 STEP(I, d, a, b, c, GET(11), 0xbd3af235, 10)
171 STEP(I, c, d, a, b, GET(2), 0x2ad7d2bb, 15)
172 STEP(I, b, c, d, a, GET(9), 0xeb86d391, 21)
173
174 a += saved_a;
175 b += saved_b;
176 c += saved_c;
177 d += saved_d;
178
179 ptr += 64;
180 } while (size -= 64);
181
182 ctx->a = a;
183 ctx->b = b;
184 ctx->c = c;
185 ctx->d = d;
186
187 return ptr;
188}
189
190void MD5::MD5Init(void *ctxBuf)
191{
192 MD5_CTX *ctx = (MD5_CTX*)ctxBuf;
193 ctx->a = 0x67452301;
194 ctx->b = 0xefcdab89;
195 ctx->c = 0x98badcfe;
196 ctx->d = 0x10325476;
197
198 ctx->lo = 0;
199 ctx->hi = 0;
200
201 memset(ctx->block, 0, sizeof(ctx->block));
202 memset(ctx->buffer, 0, sizeof(ctx->buffer));
203}
204
205void MD5::MD5Update(void *ctxBuf, const void *data, size_t size)
206{
207 MD5_CTX *ctx = (MD5_CTX*)ctxBuf;
208 MD5_u32plus saved_lo;
209 MD5_u32plus used, free;
210
211 saved_lo = ctx->lo;
212 if ((ctx->lo = (saved_lo + size) & 0x1fffffff) < saved_lo) {
213 ctx->hi++;
214 }
215 ctx->hi += size >> 29;
216
217 used = saved_lo & 0x3f;
218
219 if (used) {
220 free = 64 - used;
221
222 if (size < free) {
223 memcpy(&ctx->buffer[used], data, size);
224 return;
225 }
226
227 memcpy(&ctx->buffer[used], data, free);
228 data = (unsigned char *)data + free;
229 size -= free;
230 body(ctx, ctx->buffer, 64);
231 }
232
233 if (size >= 64) {
234 data = body(ctx, data, size & ~(size_t)0x3f);
235 size &= 0x3f;
236 }
237
238 memcpy(ctx->buffer, data, size);
239}
240
241void MD5::MD5Final(unsigned char *result, void *ctxBuf)
242{
243 MD5_CTX *ctx = (MD5_CTX*)ctxBuf;
244 MD5_u32plus used, free;
245
246 used = ctx->lo & 0x3f;
247
248 ctx->buffer[used++] = 0x80;
249
250 free = 64 - used;
251
252 if (free < 8) {
253 memset(&ctx->buffer[used], 0, free);
254 body(ctx, ctx->buffer, 64);
255 used = 0;
256 free = 64;
257 }
258
259 memset(&ctx->buffer[used], 0, free - 8);
260
261 ctx->lo <<= 3;
262 ctx->buffer[56] = ctx->lo;
263 ctx->buffer[57] = ctx->lo >> 8;
264 ctx->buffer[58] = ctx->lo >> 16;
265 ctx->buffer[59] = ctx->lo >> 24;
266 ctx->buffer[60] = ctx->hi;
267 ctx->buffer[61] = ctx->hi >> 8;
268 ctx->buffer[62] = ctx->hi >> 16;
269 ctx->buffer[63] = ctx->hi >> 24;
270
271 body(ctx, ctx->buffer, 64);
272
273 result[0] = ctx->a;
274 result[1] = ctx->a >> 8;
275 result[2] = ctx->a >> 16;
276 result[3] = ctx->a >> 24;
277 result[4] = ctx->b;
278 result[5] = ctx->b >> 8;
279 result[6] = ctx->b >> 16;
280 result[7] = ctx->b >> 24;
281 result[8] = ctx->c;
282 result[9] = ctx->c >> 8;
283 result[10] = ctx->c >> 16;
284 result[11] = ctx->c >> 24;
285 result[12] = ctx->d;
286 result[13] = ctx->d >> 8;
287 result[14] = ctx->d >> 16;
288 result[15] = ctx->d >> 24;
289
290 memset(ctx, 0, sizeof(*ctx));
291}
292unsigned char* MD5::make_hash(char *arg)
293{
294 MD5_CTX context;
295 unsigned char * hash = (unsigned char *) malloc(16);
296 MD5Init(&context);
297 MD5Update(&context, arg, strlen(arg));
298 MD5Final(hash, &context);
299 return hash;
300}
301unsigned char* MD5::make_hash(char *arg,size_t size)
302{
303 MD5_CTX context;
304 unsigned char * hash = (unsigned char *) malloc(16);
305 MD5Init(&context);
306 MD5Update(&context, arg, size);
307 MD5Final(hash, &context);
308 return hash;
309}
310
Definition: MD5.h:39
Definition: MD5.h:31