|
|
@@ -10,15 +10,21 @@ namespace Shadowsocks.Encrypt |
|
|
|
const int CIPHER_SALSA20 = 1;
|
|
|
|
const int CIPHER_CHACHA20 = 2;
|
|
|
|
|
|
|
|
protected uint _encryptBytesRemaining;
|
|
|
|
protected uint _decryptBytesRemaining;
|
|
|
|
const int SODIUM_BLOCK_SIZE = 64;
|
|
|
|
|
|
|
|
protected int _encryptBytesRemaining;
|
|
|
|
protected int _decryptBytesRemaining;
|
|
|
|
protected ulong _encryptIC;
|
|
|
|
protected ulong _decryptIC;
|
|
|
|
protected byte[] _encryptBuf;
|
|
|
|
protected byte[] _decryptBuf;
|
|
|
|
|
|
|
|
public SodiumEncryptor(string method, string password)
|
|
|
|
: base(method, password)
|
|
|
|
{
|
|
|
|
InitKey(method, password);
|
|
|
|
_encryptBuf = new byte[MAX_INPUT_SIZE + SODIUM_BLOCK_SIZE];
|
|
|
|
_decryptBuf = new byte[MAX_INPUT_SIZE + SODIUM_BLOCK_SIZE];
|
|
|
|
}
|
|
|
|
|
|
|
|
protected override Dictionary<string, int[]> getCiphers()
|
|
|
@@ -31,18 +37,41 @@ namespace Shadowsocks.Encrypt |
|
|
|
|
|
|
|
protected override void cipherUpdate(bool isCipher, int length, byte[] buf, byte[] outbuf)
|
|
|
|
{
|
|
|
|
uint bytesRemaining;
|
|
|
|
// TODO write a unidirection cipher so we don't have to if if if
|
|
|
|
int bytesRemaining;
|
|
|
|
ulong ic;
|
|
|
|
byte[] sodiumBuf;
|
|
|
|
byte[] iv;
|
|
|
|
if (isCipher)
|
|
|
|
{
|
|
|
|
bytesRemaining = _encryptBytesRemaining;
|
|
|
|
ic = _encryptIC;
|
|
|
|
sodiumBuf = _encryptBuf;
|
|
|
|
iv = _encryptIV;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
bytesRemaining = _decryptBytesRemaining;
|
|
|
|
ic = _decryptIC;
|
|
|
|
sodiumBuf = _decryptBuf;
|
|
|
|
iv = _decryptIV;
|
|
|
|
}
|
|
|
|
int padding = bytesRemaining;
|
|
|
|
Buffer.BlockCopy(buf, 0, sodiumBuf, padding, length);
|
|
|
|
|
|
|
|
switch (_cipher)
|
|
|
|
{
|
|
|
|
case CIPHER_SALSA20:
|
|
|
|
Sodium.crypto_stream_salsa20_xor_ic(sodiumBuf, sodiumBuf, (ulong)(padding + length), iv, ic, _key);
|
|
|
|
break;
|
|
|
|
case CIPHER_CHACHA20:
|
|
|
|
Sodium.crypto_stream_chacha20_xor_ic(sodiumBuf, sodiumBuf, (ulong)(padding + length), iv, ic, _key);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
Buffer.BlockCopy(sodiumBuf, padding, outbuf, 0, length);
|
|
|
|
padding += length;
|
|
|
|
ic += (ulong)padding / SODIUM_BLOCK_SIZE;
|
|
|
|
bytesRemaining = padding % SODIUM_BLOCK_SIZE;
|
|
|
|
|
|
|
|
if (isCipher)
|
|
|
|
{
|
|
|
|