PSO2SERVER/Server/Crypto/ARC4Managed.cs
2024-09-11 17:13:00 +08:00

213 lines
6.7 KiB
C#

//
// ARC4Managed.cs: Alleged RC4(tm) compatible symmetric stream cipher
// RC4 is a trademark of RSA Security
//
//
// 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;
using System.Security.Cryptography;
namespace PSO2SERVER.Crypto
{
// References:
// a. Usenet 1994 - RC4 Algorithm revealed
// http://www.qrst.de/html/dsds/rc4.htm
#if !INSIDE_CORLIB
public
#endif
class Arc4Managed : Rc4, ICryptoTransform
{
private byte[] _key;
private bool _mDisposed;
private byte[] _state;
private byte _x;
private byte _y;
public Arc4Managed()
{
_state = new byte[256];
_mDisposed = false;
}
public override byte[] Key
{
get
{
if (KeyValue == null)
GenerateKey();
return (byte[])KeyValue.Clone();
}
set
{
if (value == null)
throw new ArgumentNullException("Key");
KeyValue = _key = (byte[]) value.Clone();
KeySetup(_key);
}
}
public bool CanReuseTransform
{
get { return false; }
}
public bool CanTransformMultipleBlocks
{
get { return true; }
}
public int InputBlockSize
{
get { return 1; }
}
public int OutputBlockSize
{
get { return 1; }
}
public int TransformBlock(byte[] inputBuffer, int inputOffset, int inputCount, byte[] outputBuffer,
int outputOffset)
{
CheckInput(inputBuffer, inputOffset, inputCount);
// check output parameters
if (outputBuffer == null)
throw new ArgumentNullException("outputBuffer");
if (outputOffset < 0)
throw new ArgumentOutOfRangeException("outputOffset", "< 0");
// ordered to avoid possible integer overflow
if (outputOffset > outputBuffer.Length - inputCount)
throw new ArgumentException("outputBuffer overflow");
return InternalTransformBlock(inputBuffer, inputOffset, inputCount, outputBuffer, outputOffset);
}
public byte[] TransformFinalBlock(byte[] inputBuffer, int inputOffset, int inputCount)
{
CheckInput(inputBuffer, inputOffset, inputCount);
var output = new byte[inputCount];
InternalTransformBlock(inputBuffer, inputOffset, inputCount, output, 0);
return output;
}
~Arc4Managed()
{
Dispose(true);
}
protected override void Dispose(bool disposing)
{
if (!_mDisposed)
{
_x = 0;
_y = 0;
if (_key != null)
{
Array.Clear(_key, 0, _key.Length);
_key = null;
}
Array.Clear(_state, 0, _state.Length);
_state = null;
GC.SuppressFinalize(this);
_mDisposed = true;
}
}
public override ICryptoTransform CreateEncryptor(byte[] rgbKey, byte[] rgvIv)
{
Key = rgbKey;
return this;
}
public override ICryptoTransform CreateDecryptor(byte[] rgbKey, byte[] rgvIv)
{
Key = rgbKey;
return CreateEncryptor();
}
public override void GenerateIV()
{
// not used for a stream cipher
IV = new byte[0];
}
public override void GenerateKey()
{
throw new NotImplementedException();
}
private void KeySetup(byte[] key)
{
byte index1 = 0;
byte index2 = 0;
for (var counter = 0; counter < 256; counter++)
_state[counter] = (byte) counter;
_x = 0;
_y = 0;
for (var counter = 0; counter < 256; counter++)
{
index2 = (byte) (key[index1] + _state[counter] + index2);
// swap byte
var tmp = _state[counter];
_state[counter] = _state[index2];
_state[index2] = tmp;
index1 = (byte) ((index1 + 1)%key.Length);
}
}
private void CheckInput(byte[] inputBuffer, int inputOffset, int inputCount)
{
if (inputBuffer == null)
throw new ArgumentNullException("inputBuffer");
if (inputOffset < 0)
throw new ArgumentOutOfRangeException("inputOffset", "< 0");
if (inputCount < 0)
throw new ArgumentOutOfRangeException("inputCount", "< 0");
// ordered to avoid possible integer overflow
if (inputOffset > inputBuffer.Length - inputCount)
throw new ArgumentException("inputBuffer overflow");
}
private int InternalTransformBlock(byte[] inputBuffer, int inputOffset, int inputCount, byte[] outputBuffer,
int outputOffset)
{
for (var counter = 0; counter < inputCount; counter++)
{
_x = (byte) (_x + 1);
_y = (byte) (_state[_x] + _y);
// swap byte
var tmp = _state[_x];
_state[_x] = _state[_y];
_state[_y] = tmp;
var xorIndex = (byte) (_state[_x] + _state[_y]);
outputBuffer[outputOffset + counter] = (byte) (inputBuffer[inputOffset + counter] ^ _state[xorIndex]);
}
return inputCount;
}
}
}