2005-06-10 Sebastien Pouliot <sebastien@ximian.com>
[mono.git] / mcs / class / corlib / System.Security.Cryptography / SHA384Managed.cs
index d5679038c80969029cc0d9a7f79b0c9926217a22..45eb1551e1dca3beee99d1af2da9eec016194fc9 100644 (file)
-//\r
-// System.Security.Cryptography.SHA384Managed.cs\r
-//\r
-// Authors:\r
-//   Dan Lewis (dihlewis@yahoo.co.uk)\r
-//\r
-// (C) 2002\r
-//\r
-// Stubbed.\r
-//\r
-\r
-using System;\r
-\r
-namespace System.Security.Cryptography {\r
-       \r
-       [MonoTODO]\r
-       public class SHA384Managed : SHA384 {\r
-               public SHA384Managed () {\r
-               }\r
-\r
-               public override void Initialize () {\r
-               }\r
-\r
-               // protected\r
-\r
-               protected override void HashCore (byte[] rgb, int start, int count) {\r
-               }\r
-\r
-               protected override byte[] HashFinal () {\r
-                       return null;\r
-               }\r
-       }\r
-}\r
+//
+// System.Security.Cryptography.SHA384Managed.cs
+//
+// Authors:
+//     Dan Lewis (dihlewis@yahoo.co.uk)
+//     Sebastien Pouliot (sebastien@ximian.com)
+//
+// (C) 2002
+// Implementation translated from Bouncy Castle JCE (http://www.bouncycastle.org/)
+// See bouncycastle.txt for license.
+// Copyright (C) 2004-2005 Novell, Inc (http://www.novell.com)
+//
+// Permission is hereby granted, free of charge, to any person obtaining
+// a copy of this software and associated documentation files (the
+// "Software"), to deal in the Software without restriction, including
+// without limitation the rights to use, copy, modify, merge, publish,
+// distribute, sublicense, and/or sell copies of the Software, and to
+// permit persons to whom the Software is furnished to do so, subject to
+// the following conditions:
+// 
+// The above copyright notice and this permission notice shall be
+// included in all copies or substantial portions of the Software.
+// 
+// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
+// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
+// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
+// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
+// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
+// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
+// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
+//
+
+using System.Runtime.InteropServices;
+
+namespace System.Security.Cryptography {
+       
+#if NET_2_0
+[ComVisible (true)]
+#endif
+public class SHA384Managed : SHA384 {
+
+       private byte[] xBuf;
+       private int xBufOff;
+
+       private ulong byteCount1;
+       private ulong byteCount2;
+
+       private ulong H1, H2, H3, H4, H5, H6, H7, H8;
+       private ulong[] W;
+       private int wOff;
+
+       public SHA384Managed () 
+       {
+               xBuf = new byte [8];
+               W = new ulong [80];
+               Initialize (false); // limited initialization
+       }
+
+       private void Initialize (bool reuse) 
+       {
+               // SHA-384 initial hash value
+               // The first 64 bits of the fractional parts of the square roots
+               // of the 9th through 16th prime numbers
+               H1 = 0xcbbb9d5dc1059ed8L;
+               H2 = 0x629a292a367cd507L;
+               H3 = 0x9159015a3070dd17L;
+               H4 = 0x152fecd8f70e5939L;
+               H5 = 0x67332667ffc00b31L;
+               H6 = 0x8eb44a8768581511L;
+               H7 = 0xdb0c2e0d64f98fa7L;
+               H8 = 0x47b5481dbefa4fa4L;
+
+               if (reuse) {
+                       byteCount1 = 0;
+                       byteCount2 = 0;
+
+                       xBufOff = 0;
+                       for (int i = 0; i < xBuf.Length; i++) 
+                               xBuf [i] = 0;
+
+                       wOff = 0;
+                       for (int i = 0; i != W.Length; i++)
+                               W [i] = 0;
+               }
+       }
+
+       public override void Initialize () 
+       {
+               Initialize (true); // reuse instance
+       }
+
+       // protected
+
+       protected override void HashCore (byte[] rgb, int start, int count) 
+       {
+               // fill the current word
+               while ((xBufOff != 0) && (count > 0)) {
+                       update (rgb [start]);
+                       start++;
+                       count--;
+               }
+
+               // process whole words.
+               while (count > xBuf.Length) {
+                       processWord (rgb, start);
+                       start += xBuf.Length;
+                       count -= xBuf.Length;
+                       byteCount1 += (ulong) xBuf.Length;
+               }
+
+               // load in the remainder.
+               while (count > 0) {
+                       update (rgb [start]);
+                       start++;
+                       count--;
+               }
+       }
+
+       protected override byte[] HashFinal () 
+       {
+               adjustByteCounts ();
+
+               ulong lowBitLength = byteCount1 << 3;
+               ulong hiBitLength = byteCount2;
+
+               // add the pad bytes.
+               update ( (byte) 128);
+               while (xBufOff != 0)
+                       update ( (byte)0);
+
+               processLength (lowBitLength, hiBitLength);
+               processBlock ();
+               
+               byte[] output = new byte [48];
+               unpackWord(H1, output, 0);
+               unpackWord(H2, output, 8);
+               unpackWord(H3, output, 16);
+               unpackWord(H4, output, 24);
+               unpackWord(H5, output, 32);
+               unpackWord(H6, output, 40);
+
+               Initialize ();
+               return output;
+       }
+
+       private void update (byte input)
+       {
+               xBuf [xBufOff++] = input;
+               if (xBufOff == xBuf.Length) {
+                       processWord(xBuf, 0);
+                       xBufOff = 0;
+               }
+               byteCount1++;
+       }
+
+       private void processWord (byte[] input, int inOff)
+       {
+               W [wOff++] = ( (ulong) input [inOff] << 56)
+                       | ( (ulong) input [inOff + 1] << 48)
+                       | ( (ulong) input [inOff + 2] << 40)
+                       | ( (ulong) input [inOff + 3] << 32)
+                       | ( (ulong) input [inOff + 4] << 24)
+                       | ( (ulong) input [inOff + 5] << 16)
+                       | ( (ulong) input [inOff + 6] << 8)
+                       | ( (ulong) input [inOff + 7]); 
+               if (wOff == 16)
+                       processBlock ();
+       }
+
+       private void unpackWord (ulong word, byte[] output, int outOff)
+       {
+               output[outOff]     = (byte) (word >> 56);
+               output[outOff + 1] = (byte) (word >> 48);
+               output[outOff + 2] = (byte) (word >> 40);
+               output[outOff + 3] = (byte) (word >> 32);
+               output[outOff + 4] = (byte) (word >> 24);
+               output[outOff + 5] = (byte) (word >> 16);
+               output[outOff + 6] = (byte) (word >> 8);
+               output[outOff + 7] = (byte) word;
+       }
+
+       // adjust the byte counts so that byteCount2 represents the
+       // upper long (less 3 bits) word of the byte count.
+       private void adjustByteCounts()
+       {
+               if (byteCount1 > 0x1fffffffffffffffL) {
+                       byteCount2 += (byteCount1 >> 61);
+                       byteCount1 &= 0x1fffffffffffffffL;
+               }
+       }
+
+       private void processLength (ulong lowW, ulong hiW)
+       {
+               if (wOff > 14)
+                       processBlock ();
+               W[14] = hiW;
+               W[15] = lowW;
+       }
+
+       private void processBlock ()
+       {
+               adjustByteCounts ();
+               // expand 16 word block into 80 word blocks.
+               for (int t = 16; t <= 79; t++)
+                       W[t] = Sigma1 (W [t - 2]) + W [t - 7] + Sigma0 (W [t - 15]) + W [t - 16];
+
+               // set up working variables.
+               ulong a = H1;
+               ulong b = H2;
+               ulong c = H3;
+               ulong d = H4;
+               ulong e = H5;
+               ulong f = H6;
+               ulong g = H7;
+               ulong h = H8;
+
+               for (int t = 0; t <= 79; t++) {
+                       ulong T1 = h + Sum1 (e) + Ch (e, f, g) + SHAConstants.K2 [t] + W [t];
+                       ulong T2 = Sum0 (a) + Maj (a, b, c);
+                       h = g;
+                       g = f;
+                       f = e;
+                       e = d + T1;
+                       d = c;
+                       c = b;
+                       b = a;
+                       a = T1 + T2;
+               }
+
+               H1 += a;
+               H2 += b;
+               H3 += c;
+               H4 += d;
+               H5 += e;
+               H6 += f;
+               H7 += g;
+               H8 += h;
+               // reset the offset and clean out the word buffer.
+               wOff = 0;
+               for (int i = 0; i != W.Length; i++)
+                       W[i] = 0;
+       }
+
+       private ulong rotateRight (ulong x, int n)
+       {
+               return (x >> n) | (x << (64 - n));
+       }
+
+       /* SHA-384 and SHA-512 functions (as for SHA-256 but for longs) */
+       private ulong Ch (ulong x, ulong y, ulong z)
+       {
+               return ((x & y) ^ ((~x) & z));
+       }
+
+       private ulong Maj (ulong x, ulong y, ulong z)
+       {
+               return ((x & y) ^ (x & z) ^ (y & z));
+       }
+
+       private ulong Sum0 (ulong x)
+       {
+               return rotateRight (x, 28) ^ rotateRight (x, 34) ^ rotateRight (x, 39);
+       }
+
+       private ulong Sum1 (ulong x)
+       {
+               return rotateRight (x, 14) ^ rotateRight (x, 18) ^ rotateRight (x, 41);
+       }
+
+       private ulong Sigma0 (ulong x)
+       {
+               return rotateRight (x, 1) ^ rotateRight(x, 8) ^ (x >> 7);
+       }
+
+       private ulong Sigma1 (ulong x)
+       {
+               return rotateRight (x, 19) ^ rotateRight (x, 61) ^ (x >> 6);
+       }
+}
+
+}