CtsBlockCipher
using System;
namespace Org.BouncyCastle.Crypto.Modes
{
    public class CtsBlockCipher : BufferedBlockCipher
    {
        private readonly int blockSize;
        public CtsBlockCipher(IBlockCipher cipher)
            : this(EcbBlockCipher.GetBlockCipherMode(cipher))
        {
        }
        public CtsBlockCipher(IBlockCipherMode cipherMode)
        {
            if (!(cipherMode is CbcBlockCipher) && !(cipherMode is EcbBlockCipher))
                throw new ArgumentException("CtsBlockCipher can only accept ECB, or CBC ciphers");
            m_cipherMode = cipherMode;
            blockSize = cipherMode.GetBlockSize();
            buf = new byte[blockSize * 2];
            bufOff = 0;
        }
        public override int GetUpdateOutputSize(int length)
        {
            int num = length + bufOff;
            int num2 = num % buf.Length;
            if (num2 == 0)
                return num - buf.Length;
            return num - num2;
        }
        public override int GetOutputSize(int length)
        {
            return length + bufOff;
        }
        public override int ProcessByte(byte input, byte[] output, int outOff)
        {
            int result = 0;
            if (bufOff == buf.Length) {
                result = m_cipherMode.ProcessBlock(buf, 0, output, outOff);
                Array.Copy(buf, blockSize, buf, 0, blockSize);
                bufOff = blockSize;
            }
            buf[bufOff++] = input;
            return result;
        }
        public override int ProcessBytes(byte[] input, int inOff, int length, byte[] output, int outOff)
        {
            if (length < 0)
                throw new ArgumentException("Can't have a negative input length!");
            int num = GetBlockSize();
            int updateOutputSize = GetUpdateOutputSize(length);
            if (updateOutputSize > 0)
                Check.OutputLength(output, outOff, updateOutputSize, "output buffer too short");
            int num2 = 0;
            int num3 = buf.Length - bufOff;
            if (length > num3) {
                Array.Copy(input, inOff, buf, bufOff, num3);
                num2 = m_cipherMode.ProcessBlock(buf, 0, output, outOff);
                Array.Copy(buf, num, buf, 0, num);
                bufOff = num;
                length -= num3;
                inOff += num3;
                while (length > num) {
                    Array.Copy(input, inOff, buf, bufOff, num);
                    num2 += m_cipherMode.ProcessBlock(buf, 0, output, outOff + num2);
                    Array.Copy(buf, num, buf, 0, num);
                    length -= num;
                    inOff += num;
                }
            }
            Array.Copy(input, inOff, buf, bufOff, length);
            bufOff += length;
            return num2;
        }
        public override int DoFinal(byte[] output, int outOff)
        {
            if (base.bufOff + outOff > output.Length)
                throw new DataLengthException("output buffer too small in DoFinal");
            int num = m_cipherMode.GetBlockSize();
            int length = base.bufOff - num;
            byte[] array = new byte[num];
            if (forEncryption) {
                m_cipherMode.ProcessBlock(buf, 0, array, 0);
                if (base.bufOff < num)
                    throw new DataLengthException("need at least one block of input for CTS");
                for (int i = base.bufOff; i != buf.Length; i++) {
                    buf[i] = array[i - num];
                }
                for (int j = num; j != base.bufOff; j++) {
                    buf[j] ^= array[j - num];
                }
                m_cipherMode.UnderlyingCipher.ProcessBlock(buf, num, output, outOff);
                Array.Copy(array, 0, output, outOff + num, length);
            } else {
                byte[] array2 = new byte[num];
                m_cipherMode.UnderlyingCipher.ProcessBlock(buf, 0, array, 0);
                for (int k = num; k != base.bufOff; k++) {
                    array2[k - num] = (byte)(array[k - num] ^ buf[k]);
                }
                Array.Copy(buf, num, array, 0, length);
                m_cipherMode.ProcessBlock(array, 0, output, outOff);
                Array.Copy(array2, 0, output, outOff + num, length);
            }
            int bufOff = base.bufOff;
            Reset();
            return bufOff;
        }
    }
}