2 // System.Security.Cryptography.ToBase64Transform
5 // Sergey Chaban (serge@wildwestsoftware.com)
7 // (C) 2004 Novell (http://www.novell.com)
8 // Copyright (C) 2004-2005 Novell, Inc (http://www.novell.com)
10 // Permission is hereby granted, free of charge, to any person obtaining
11 // a copy of this software and associated documentation files (the
12 // "Software"), to deal in the Software without restriction, including
13 // without limitation the rights to use, copy, modify, merge, publish,
14 // distribute, sublicense, and/or sell copies of the Software, and to
15 // permit persons to whom the Software is furnished to do so, subject to
16 // the following conditions:
18 // The above copyright notice and this permission notice shall be
19 // included in all copies or substantial portions of the Software.
21 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
22 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
23 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
24 // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
25 // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
26 // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
27 // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
30 using System.Globalization;
31 using System.Runtime.InteropServices;
33 namespace System.Security.Cryptography {
36 public class ToBase64Transform : ICryptoTransform {
38 private const int inputBlockSize = 3;
39 private const int outputBlockSize = 4;
40 private bool m_disposed;
42 public ToBase64Transform ()
51 public bool CanTransformMultipleBlocks {
55 public virtual bool CanReuseTransform {
59 public int InputBlockSize {
60 get { return inputBlockSize; }
63 public int OutputBlockSize {
64 get { return outputBlockSize; }
73 public void Dispose ()
75 void IDisposable.Dispose ()
79 GC.SuppressFinalize (this); // Finalization is now unnecessary
82 protected virtual void Dispose (bool disposing)
85 // dispose unmanaged objects
87 // dispose managed objects
93 // LAMESPEC: It's not clear from docs what should be happening
94 // here if inputCount > InputBlockSize. It just "Converts the
95 // specified region of the specified byte array" and that's all.
96 public int TransformBlock (byte[] inputBuffer, int inputOffset, int inputCount, byte[] outputBuffer, int outputOffset)
99 throw new ObjectDisposedException ("TransformBlock");
100 if (inputBuffer == null)
101 throw new ArgumentNullException ("inputBuffer");
102 if (outputBuffer == null)
103 throw new ArgumentNullException ("outputBuffer");
105 throw new ArgumentException ("inputCount", "< 0");
106 if (inputCount > inputBuffer.Length)
107 throw new ArgumentException ("inputCount", Locale.GetText ("Overflow"));
109 throw new ArgumentOutOfRangeException ("inputOffset", "< 0");
110 // ordered to avoid possible integer overflow
111 if (inputOffset > inputBuffer.Length - inputCount)
112 throw new ArgumentException ("inputOffset", Locale.GetText ("Overflow"));
113 // ordered to avoid possible integer overflow
114 if (outputOffset < 0)
115 throw new ArgumentOutOfRangeException ("outputOffset", "< 0");
116 if (outputOffset > outputBuffer.Length - inputCount)
117 throw new ArgumentException ("outputOffset", Locale.GetText ("Overflow"));
118 /// To match MS implementation
119 // if (inputCount != this.InputBlockSize)
120 // throw new CryptographicException (Locale.GetText ("Invalid input length"));
122 InternalTransformBlock (inputBuffer, inputOffset, inputCount, outputBuffer, outputOffset);
123 return this.OutputBlockSize;
126 internal static void InternalTransformBlock (byte[] inputBuffer, int inputOffset, int inputCount, byte[] outputBuffer, int outputOffset)
128 byte[] lookup = Base64Constants.EncodeTable;
130 int b1 = inputBuffer [inputOffset];
131 int b2 = inputBuffer [inputOffset + 1];
132 int b3 = inputBuffer [inputOffset + 2];
134 outputBuffer [outputOffset] = lookup [b1 >> 2];
135 outputBuffer [outputOffset+1] = lookup [((b1 << 4) & 0x30) | (b2 >> 4)];
136 outputBuffer [outputOffset+2] = lookup [((b2 << 2) & 0x3c) | (b3 >> 6)];
137 outputBuffer [outputOffset+3] = lookup [b3 & 0x3f];
140 public byte[] TransformFinalBlock (byte[] inputBuffer, int inputOffset, int inputCount)
143 throw new ObjectDisposedException ("TransformFinalBlock");
144 if (inputBuffer == null)
145 throw new ArgumentNullException ("inputBuffer");
147 throw new ArgumentException ("inputCount", "< 0");
148 if (inputOffset > inputBuffer.Length - inputCount)
149 throw new ArgumentException ("inputCount", Locale.GetText ("Overflow"));
150 if (inputCount > this.InputBlockSize)
151 throw new ArgumentOutOfRangeException (Locale.GetText ("Invalid input length"));
153 return InternalTransformFinalBlock (inputBuffer, inputOffset, inputCount);
156 // Mono System.Convert depends on the ability to process multiple blocks
157 internal static byte[] InternalTransformFinalBlock (byte[] inputBuffer, int inputOffset, int inputCount)
159 int blockLen = inputBlockSize;
160 int outLen = outputBlockSize;
161 int fullBlocks = inputCount / blockLen;
162 int tail = inputCount % blockLen;
164 byte[] res = new byte [(inputCount != 0)
165 ? ((inputCount + 2) / blockLen) * outLen
168 int outputOffset = 0;
170 for (int i = 0; i < fullBlocks; i++) {
171 InternalTransformBlock (inputBuffer, inputOffset, blockLen, res, outputOffset);
172 inputOffset += blockLen;
173 outputOffset += outLen;
176 byte[] lookup = Base64Constants.EncodeTable;
179 // When fewer than 24 input bits are available
180 // in an input group, zero bits are added
181 // (on the right) to form an integral number of
187 b1 = inputBuffer [inputOffset];
188 res [outputOffset] = lookup [b1 >> 2];
189 res [outputOffset+1] = lookup [(b1 << 4) & 0x30];
192 res [outputOffset+2] = (byte)'=';
193 res [outputOffset+3] = (byte)'=';
197 b1 = inputBuffer [inputOffset];
198 b2 = inputBuffer [inputOffset + 1];
199 res [outputOffset] = lookup [b1 >> 2];
200 res [outputOffset+1] = lookup [((b1 << 4) & 0x30) | (b2 >> 4)];
201 res [outputOffset+2] = lookup [(b2 << 2) & 0x3c];
204 res [outputOffset+3] = (byte)'=';