#if !BESTHTTP_DISABLE_ALTERNATE_SSL && (!UNITY_WEBGL || UNITY_EDITOR) #pragma warning disable using System; using System.IO; namespace Best.HTTP.SecureProtocol.Org.BouncyCastle.Tls.Crypto.Impl { /// The NULL cipher. public class TlsNullCipher : TlsCipher { protected readonly TlsCryptoParameters m_cryptoParams; protected readonly TlsSuiteHmac m_readMac, m_writeMac; /// public TlsNullCipher(TlsCryptoParameters cryptoParams, TlsHmac clientMac, TlsHmac serverMac) { if (TlsImplUtilities.IsTlsV13(cryptoParams)) throw new TlsFatalAlert(AlertDescription.internal_error); m_cryptoParams = cryptoParams; int keyBlockSize = clientMac.MacLength + serverMac.MacLength; #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER || UNITY_2021_2_OR_NEWER Span keyBlock = keyBlockSize <= 512 ? stackalloc byte[keyBlockSize] : new byte[keyBlockSize]; TlsImplUtilities.CalculateKeyBlock(cryptoParams, keyBlock); clientMac.SetKey(keyBlock[..clientMac.MacLength]); keyBlock = keyBlock[clientMac.MacLength..]; serverMac.SetKey(keyBlock[..serverMac.MacLength]); keyBlock = keyBlock[serverMac.MacLength..]; if (!keyBlock.IsEmpty) throw new TlsFatalAlert(AlertDescription.internal_error); #else byte[] keyBlock = TlsImplUtilities.CalculateKeyBlock(cryptoParams, keyBlockSize); int pos = 0; clientMac.SetKey(keyBlock, pos, clientMac.MacLength); pos += clientMac.MacLength; serverMac.SetKey(keyBlock, pos, serverMac.MacLength); pos += serverMac.MacLength; if (pos != keyBlockSize) throw new TlsFatalAlert(AlertDescription.internal_error); #endif if (cryptoParams.IsServer) { this.m_writeMac = new TlsSuiteHmac(cryptoParams, serverMac); this.m_readMac = new TlsSuiteHmac(cryptoParams, clientMac); } else { this.m_writeMac = new TlsSuiteHmac(cryptoParams, clientMac); this.m_readMac = new TlsSuiteHmac(cryptoParams, serverMac); } } public virtual int GetCiphertextDecodeLimit(int plaintextLimit) { return plaintextLimit + m_writeMac.Size; } public virtual int GetCiphertextEncodeLimit(int plaintextLength, int plaintextLimit) { return plaintextLength + m_writeMac.Size; } public virtual int GetPlaintextLimit(int ciphertextLimit) { return ciphertextLimit - m_writeMac.Size; } public virtual TlsEncodeResult EncodePlaintext(long seqNo, short contentType, ProtocolVersion recordVersion, int headerAllocation, byte[] plaintext, int offset, int len) { byte[] mac = m_writeMac.CalculateMac(seqNo, contentType, plaintext, offset, len); byte[] ciphertext = new byte[headerAllocation + len + mac.Length]; Array.Copy(plaintext, offset, ciphertext, headerAllocation, len); Array.Copy(mac, 0, ciphertext, headerAllocation + len, mac.Length); return new TlsEncodeResult(ciphertext, 0, ciphertext.Length, contentType); } #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER || UNITY_2021_2_OR_NEWER public virtual TlsEncodeResult EncodePlaintext(long seqNo, short contentType, ProtocolVersion recordVersion, int headerAllocation, ReadOnlySpan plaintext) { byte[] mac = m_writeMac.CalculateMac(seqNo, contentType, plaintext); byte[] ciphertext = new byte[headerAllocation + plaintext.Length + mac.Length]; plaintext.CopyTo(ciphertext.AsSpan(headerAllocation)); mac.CopyTo(ciphertext.AsSpan(headerAllocation + plaintext.Length)); return new TlsEncodeResult(ciphertext, 0, ciphertext.Length, contentType); } #endif public virtual TlsDecodeResult DecodeCiphertext(long seqNo, short recordType, ProtocolVersion recordVersion, byte[] ciphertext, int offset, int len) { int macSize = m_readMac.Size; if (len < macSize) throw new TlsFatalAlert(AlertDescription.decode_error); int macInputLen = len - macSize; byte[] expectedMac = m_readMac.CalculateMac(seqNo, recordType, ciphertext, offset, macInputLen); bool badMac = !TlsUtilities.ConstantTimeAreEqual(macSize, expectedMac, 0, ciphertext, offset + macInputLen); if (badMac) throw new TlsFatalAlert(AlertDescription.bad_record_mac); return new TlsDecodeResult(ciphertext, offset, macInputLen, recordType); } public virtual void RekeyDecoder() { throw new TlsFatalAlert(AlertDescription.internal_error); } public virtual void RekeyEncoder() { throw new TlsFatalAlert(AlertDescription.internal_error); } public virtual bool UsesOpaqueRecordType { get { return false; } } } } #pragma warning restore #endif