2010-06-21 Atsushi Enomoto <atsushi@ximian.com>
[mcs.git] / class / corlib / System.Security.Cryptography / MD5CryptoServiceProvider.cs
blob2ea2a222423446f93f00b0c5935e77809325f7ef
1 //
2 // System.Security.Cryptography.MD5CryptoServiceProvider.cs
3 //
4 // Authors:
5 // Matthew S. Ford (Matthew.S.Ford@Rose-Hulman.Edu)
6 // Sebastien Pouliot (sebastien@ximian.com)
7 //
8 // Copyright 2001 by Matthew S. Ford.
9 // Copyright (C) 2004-2005 Novell, Inc (http://www.novell.com)
11 // Permission is hereby granted, free of charge, to any person obtaining
12 // a copy of this software and associated documentation files (the
13 // "Software"), to deal in the Software without restriction, including
14 // without limitation the rights to use, copy, modify, merge, publish,
15 // distribute, sublicense, and/or sell copies of the Software, and to
16 // permit persons to whom the Software is furnished to do so, subject to
17 // the following conditions:
18 //
19 // The above copyright notice and this permission notice shall be
20 // included in all copies or substantial portions of the Software.
21 //
22 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
23 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
24 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
25 // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
26 // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
27 // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
28 // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
31 using System.Runtime.InteropServices;
33 namespace System.Security.Cryptography {
35 [ComVisible (true)]
36 public sealed class MD5CryptoServiceProvider : MD5 {
37 private const int BLOCK_SIZE_BYTES = 64;
38 private uint[] _H;
39 private uint[] buff;
40 private ulong count;
41 private byte[] _ProcessingBuffer; // Used to start data when passed less than a block worth.
42 private int _ProcessingBufferCount; // Counts how much data we have stored that still needs processed.
44 public MD5CryptoServiceProvider ()
46 _H = new uint[4];
47 buff = new uint[16];
48 _ProcessingBuffer = new byte [BLOCK_SIZE_BYTES];
50 Initialize();
53 ~MD5CryptoServiceProvider ()
55 Dispose (false);
58 protected override void Dispose (bool disposing)
60 if (_ProcessingBuffer != null) {
61 Array.Clear (_ProcessingBuffer, 0, _ProcessingBuffer.Length);
62 _ProcessingBuffer = null;
64 if (_H != null) {
65 Array.Clear (_H, 0, _H.Length);
66 _H = null;
68 if (buff != null) {
69 Array.Clear (buff, 0, buff.Length);
70 buff = null;
74 protected override void HashCore (byte[] rgb, int ibStart, int cbSize)
76 int i;
77 State = 1;
79 if (_ProcessingBufferCount != 0) {
80 if (cbSize < (BLOCK_SIZE_BYTES - _ProcessingBufferCount)) {
81 System.Buffer.BlockCopy (rgb, ibStart, _ProcessingBuffer, _ProcessingBufferCount, cbSize);
82 _ProcessingBufferCount += cbSize;
83 return;
85 else {
86 i = (BLOCK_SIZE_BYTES - _ProcessingBufferCount);
87 System.Buffer.BlockCopy (rgb, ibStart, _ProcessingBuffer, _ProcessingBufferCount, i);
88 ProcessBlock (_ProcessingBuffer, 0);
89 _ProcessingBufferCount = 0;
90 ibStart += i;
91 cbSize -= i;
95 for (i = 0; i < cbSize - cbSize % BLOCK_SIZE_BYTES; i += BLOCK_SIZE_BYTES) {
96 ProcessBlock (rgb, ibStart + i);
99 if (cbSize % BLOCK_SIZE_BYTES != 0) {
100 System.Buffer.BlockCopy (rgb, cbSize - cbSize % BLOCK_SIZE_BYTES + ibStart, _ProcessingBuffer, 0, cbSize % BLOCK_SIZE_BYTES);
101 _ProcessingBufferCount = cbSize % BLOCK_SIZE_BYTES;
105 protected override byte[] HashFinal ()
107 byte[] hash = new byte[16];
108 int i, j;
110 ProcessFinalBlock (_ProcessingBuffer, 0, _ProcessingBufferCount);
112 for (i=0; i<4; i++) {
113 for (j=0; j<4; j++) {
114 hash[i*4+j] = (byte)(_H[i] >> j*8);
118 return hash;
121 public override void Initialize ()
123 count = 0;
124 _ProcessingBufferCount = 0;
126 _H[0] = 0x67452301;
127 _H[1] = 0xefcdab89;
128 _H[2] = 0x98badcfe;
129 _H[3] = 0x10325476;
132 private void ProcessBlock (byte[] inputBuffer, int inputOffset)
134 uint a, b, c, d;
135 int i;
137 count += BLOCK_SIZE_BYTES;
139 for (i=0; i<16; i++) {
140 buff[i] = (uint)(inputBuffer[inputOffset+4*i])
141 | (((uint)(inputBuffer[inputOffset+4*i+1])) << 8)
142 | (((uint)(inputBuffer[inputOffset+4*i+2])) << 16)
143 | (((uint)(inputBuffer[inputOffset+4*i+3])) << 24);
146 a = _H[0];
147 b = _H[1];
148 c = _H[2];
149 d = _H[3];
151 // This function was unrolled because it seems to be doubling our performance with current compiler/VM.
152 // Possibly roll up if this changes.
154 // ---- Round 1 --------
156 a += (((c ^ d) & b) ^ d) + (uint) K [0] + buff [0];
157 a = (a << 7) | (a >> 25);
158 a += b;
160 d += (((b ^ c) & a) ^ c) + (uint) K [1] + buff [1];
161 d = (d << 12) | (d >> 20);
162 d += a;
164 c += (((a ^ b) & d) ^ b) + (uint) K [2] + buff [2];
165 c = (c << 17) | (c >> 15);
166 c += d;
168 b += (((d ^ a) & c) ^ a) + (uint) K [3] + buff [3];
169 b = (b << 22) | (b >> 10);
170 b += c;
172 a += (((c ^ d) & b) ^ d) + (uint) K [4] + buff [4];
173 a = (a << 7) | (a >> 25);
174 a += b;
176 d += (((b ^ c) & a) ^ c) + (uint) K [5] + buff [5];
177 d = (d << 12) | (d >> 20);
178 d += a;
180 c += (((a ^ b) & d) ^ b) + (uint) K [6] + buff [6];
181 c = (c << 17) | (c >> 15);
182 c += d;
184 b += (((d ^ a) & c) ^ a) + (uint) K [7] + buff [7];
185 b = (b << 22) | (b >> 10);
186 b += c;
188 a += (((c ^ d) & b) ^ d) + (uint) K [8] + buff [8];
189 a = (a << 7) | (a >> 25);
190 a += b;
192 d += (((b ^ c) & a) ^ c) + (uint) K [9] + buff [9];
193 d = (d << 12) | (d >> 20);
194 d += a;
196 c += (((a ^ b) & d) ^ b) + (uint) K [10] + buff [10];
197 c = (c << 17) | (c >> 15);
198 c += d;
200 b += (((d ^ a) & c) ^ a) + (uint) K [11] + buff [11];
201 b = (b << 22) | (b >> 10);
202 b += c;
204 a += (((c ^ d) & b) ^ d) + (uint) K [12] + buff [12];
205 a = (a << 7) | (a >> 25);
206 a += b;
208 d += (((b ^ c) & a) ^ c) + (uint) K [13] + buff [13];
209 d = (d << 12) | (d >> 20);
210 d += a;
212 c += (((a ^ b) & d) ^ b) + (uint) K [14] + buff [14];
213 c = (c << 17) | (c >> 15);
214 c += d;
216 b += (((d ^ a) & c) ^ a) + (uint) K [15] + buff [15];
217 b = (b << 22) | (b >> 10);
218 b += c;
221 // ---- Round 2 --------
223 a += (((b ^ c) & d) ^ c) + (uint) K [16] + buff [1];
224 a = (a << 5) | (a >> 27);
225 a += b;
227 d += (((a ^ b) & c) ^ b) + (uint) K [17] + buff [6];
228 d = (d << 9) | (d >> 23);
229 d += a;
231 c += (((d ^ a) & b) ^ a) + (uint) K [18] + buff [11];
232 c = (c << 14) | (c >> 18);
233 c += d;
235 b += (((c ^ d) & a) ^ d) + (uint) K [19] + buff [0];
236 b = (b << 20) | (b >> 12);
237 b += c;
239 a += (((b ^ c) & d) ^ c) + (uint) K [20] + buff [5];
240 a = (a << 5) | (a >> 27);
241 a += b;
243 d += (((a ^ b) & c) ^ b) + (uint) K [21] + buff [10];
244 d = (d << 9) | (d >> 23);
245 d += a;
247 c += (((d ^ a) & b) ^ a) + (uint) K [22] + buff [15];
248 c = (c << 14) | (c >> 18);
249 c += d;
251 b += (((c ^ d) & a) ^ d) + (uint) K [23] + buff [4];
252 b = (b << 20) | (b >> 12);
253 b += c;
255 a += (((b ^ c) & d) ^ c) + (uint) K [24] + buff [9];
256 a = (a << 5) | (a >> 27);
257 a += b;
259 d += (((a ^ b) & c) ^ b) + (uint) K [25] + buff [14];
260 d = (d << 9) | (d >> 23);
261 d += a;
263 c += (((d ^ a) & b) ^ a) + (uint) K [26] + buff [3];
264 c = (c << 14) | (c >> 18);
265 c += d;
267 b += (((c ^ d) & a) ^ d) + (uint) K [27] + buff [8];
268 b = (b << 20) | (b >> 12);
269 b += c;
271 a += (((b ^ c) & d) ^ c) + (uint) K [28] + buff [13];
272 a = (a << 5) | (a >> 27);
273 a += b;
275 d += (((a ^ b) & c) ^ b) + (uint) K [29] + buff [2];
276 d = (d << 9) | (d >> 23);
277 d += a;
279 c += (((d ^ a) & b) ^ a) + (uint) K [30] + buff [7];
280 c = (c << 14) | (c >> 18);
281 c += d;
283 b += (((c ^ d) & a) ^ d) + (uint) K [31] + buff [12];
284 b = (b << 20) | (b >> 12);
285 b += c;
288 // ---- Round 3 --------
290 a += (b ^ c ^ d) + (uint) K [32] + buff [5];
291 a = (a << 4) | (a >> 28);
292 a += b;
294 d += (a ^ b ^ c) + (uint) K [33] + buff [8];
295 d = (d << 11) | (d >> 21);
296 d += a;
298 c += (d ^ a ^ b) + (uint) K [34] + buff [11];
299 c = (c << 16) | (c >> 16);
300 c += d;
302 b += (c ^ d ^ a) + (uint) K [35] + buff [14];
303 b = (b << 23) | (b >> 9);
304 b += c;
306 a += (b ^ c ^ d) + (uint) K [36] + buff [1];
307 a = (a << 4) | (a >> 28);
308 a += b;
310 d += (a ^ b ^ c) + (uint) K [37] + buff [4];
311 d = (d << 11) | (d >> 21);
312 d += a;
314 c += (d ^ a ^ b) + (uint) K [38] + buff [7];
315 c = (c << 16) | (c >> 16);
316 c += d;
318 b += (c ^ d ^ a) + (uint) K [39] + buff [10];
319 b = (b << 23) | (b >> 9);
320 b += c;
322 a += (b ^ c ^ d) + (uint) K [40] + buff [13];
323 a = (a << 4) | (a >> 28);
324 a += b;
326 d += (a ^ b ^ c) + (uint) K [41] + buff [0];
327 d = (d << 11) | (d >> 21);
328 d += a;
330 c += (d ^ a ^ b) + (uint) K [42] + buff [3];
331 c = (c << 16) | (c >> 16);
332 c += d;
334 b += (c ^ d ^ a) + (uint) K [43] + buff [6];
335 b = (b << 23) | (b >> 9);
336 b += c;
338 a += (b ^ c ^ d) + (uint) K [44] + buff [9];
339 a = (a << 4) | (a >> 28);
340 a += b;
342 d += (a ^ b ^ c) + (uint) K [45] + buff [12];
343 d = (d << 11) | (d >> 21);
344 d += a;
346 c += (d ^ a ^ b) + (uint) K [46] + buff [15];
347 c = (c << 16) | (c >> 16);
348 c += d;
350 b += (c ^ d ^ a) + (uint) K [47] + buff [2];
351 b = (b << 23) | (b >> 9);
352 b += c;
355 // ---- Round 4 --------
357 a += (((~d) | b) ^ c) + (uint) K [48] + buff [0];
358 a = (a << 6) | (a >> 26);
359 a += b;
361 d += (((~c) | a) ^ b) + (uint) K [49] + buff [7];
362 d = (d << 10) | (d >> 22);
363 d += a;
365 c += (((~b) | d) ^ a) + (uint) K [50] + buff [14];
366 c = (c << 15) | (c >> 17);
367 c += d;
369 b += (((~a) | c) ^ d) + (uint) K [51] + buff [5];
370 b = (b << 21) | (b >> 11);
371 b += c;
373 a += (((~d) | b) ^ c) + (uint) K [52] + buff [12];
374 a = (a << 6) | (a >> 26);
375 a += b;
377 d += (((~c) | a) ^ b) + (uint) K [53] + buff [3];
378 d = (d << 10) | (d >> 22);
379 d += a;
381 c += (((~b) | d) ^ a) + (uint) K [54] + buff [10];
382 c = (c << 15) | (c >> 17);
383 c += d;
385 b += (((~a) | c) ^ d) + (uint) K [55] + buff [1];
386 b = (b << 21) | (b >> 11);
387 b += c;
389 a += (((~d) | b) ^ c) + (uint) K [56] + buff [8];
390 a = (a << 6) | (a >> 26);
391 a += b;
393 d += (((~c) | a) ^ b) + (uint) K [57] + buff [15];
394 d = (d << 10) | (d >> 22);
395 d += a;
397 c += (((~b) | d) ^ a) + (uint) K [58] + buff [6];
398 c = (c << 15) | (c >> 17);
399 c += d;
401 b += (((~a) | c) ^ d) + (uint) K [59] + buff [13];
402 b = (b << 21) | (b >> 11);
403 b += c;
405 a += (((~d) | b) ^ c) + (uint) K [60] + buff [4];
406 a = (a << 6) | (a >> 26);
407 a += b;
409 d += (((~c) | a) ^ b) + (uint) K [61] + buff [11];
410 d = (d << 10) | (d >> 22);
411 d += a;
413 c += (((~b) | d) ^ a) + (uint) K [62] + buff [2];
414 c = (c << 15) | (c >> 17);
415 c += d;
417 b += (((~a) | c) ^ d) + (uint) K [63] + buff [9];
418 b = (b << 21) | (b >> 11);
419 b += c;
421 _H [0] += a;
422 _H [1] += b;
423 _H [2] += c;
424 _H [3] += d;
427 private void ProcessFinalBlock (byte[] inputBuffer, int inputOffset, int inputCount)
429 ulong total = count + (ulong)inputCount;
430 int paddingSize = (int)(56 - total % BLOCK_SIZE_BYTES);
432 if (paddingSize < 1)
433 paddingSize += BLOCK_SIZE_BYTES;
435 byte[] fooBuffer = new byte [inputCount+paddingSize+8];
437 for (int i=0; i<inputCount; i++) {
438 fooBuffer[i] = inputBuffer[i+inputOffset];
441 fooBuffer[inputCount] = 0x80;
442 for (int i=inputCount+1; i<inputCount+paddingSize; i++) {
443 fooBuffer[i] = 0x00;
446 // I deal in bytes. The algorithm deals in bits.
447 ulong size = total << 3;
448 AddLength (size, fooBuffer, inputCount+paddingSize);
449 ProcessBlock (fooBuffer, 0);
451 if (inputCount+paddingSize+8 == 128) {
452 ProcessBlock(fooBuffer, 64);
456 internal void AddLength (ulong length, byte[] buffer, int position)
458 buffer [position++] = (byte)(length);
459 buffer [position++] = (byte)(length >> 8);
460 buffer [position++] = (byte)(length >> 16);
461 buffer [position++] = (byte)(length >> 24);
462 buffer [position++] = (byte)(length >> 32);
463 buffer [position++] = (byte)(length >> 40);
464 buffer [position++] = (byte)(length >> 48);
465 buffer [position] = (byte)(length >> 56);
468 private readonly static uint[] K = {
469 0xd76aa478, 0xe8c7b756, 0x242070db, 0xc1bdceee,
470 0xf57c0faf, 0x4787c62a, 0xa8304613, 0xfd469501,
471 0x698098d8, 0x8b44f7af, 0xffff5bb1, 0x895cd7be,
472 0x6b901122, 0xfd987193, 0xa679438e, 0x49b40821,
473 0xf61e2562, 0xc040b340, 0x265e5a51, 0xe9b6c7aa,
474 0xd62f105d, 0x02441453, 0xd8a1e681, 0xe7d3fbc8,
475 0x21e1cde6, 0xc33707d6, 0xf4d50d87, 0x455a14ed,
476 0xa9e3e905, 0xfcefa3f8, 0x676f02d9, 0x8d2a4c8a,
477 0xfffa3942, 0x8771f681, 0x6d9d6122, 0xfde5380c,
478 0xa4beea44, 0x4bdecfa9, 0xf6bb4b60, 0xbebfbc70,
479 0x289b7ec6, 0xeaa127fa, 0xd4ef3085, 0x04881d05,
480 0xd9d4d039, 0xe6db99e5, 0x1fa27cf8, 0xc4ac5665,
481 0xf4292244, 0x432aff97, 0xab9423a7, 0xfc93a039,
482 0x655b59c3, 0x8f0ccc92, 0xffeff47d, 0x85845dd1,
483 0x6fa87e4f, 0xfe2ce6e0, 0xa3014314, 0x4e0811a1,
484 0xf7537e82, 0xbd3af235, 0x2ad7d2bb, 0xeb86d391