193 lines
5.3 KiB
C#
193 lines
5.3 KiB
C#
#if !BESTHTTP_DISABLE_ALTERNATE_SSL && (!UNITY_WEBGL || UNITY_EDITOR)
|
|
#pragma warning disable
|
|
using System;
|
|
using System.IO;
|
|
using System.Diagnostics;
|
|
#if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER || _UNITY_2021_2_OR_NEWER_
|
|
using System.Buffers.Binary;
|
|
using System.Numerics;
|
|
#endif
|
|
|
|
using BestHTTP.SecureProtocol.Org.BouncyCastle.Utilities.IO;
|
|
|
|
namespace BestHTTP.SecureProtocol.Org.BouncyCastle.Asn1
|
|
{
|
|
public class Asn1OutputStream
|
|
: FilterStream
|
|
{
|
|
internal const int EncodingBer = 1;
|
|
internal const int EncodingDer = 2;
|
|
|
|
public static Asn1OutputStream Create(Stream output)
|
|
{
|
|
return new Asn1OutputStream(output);
|
|
}
|
|
|
|
public static Asn1OutputStream Create(Stream output, string encoding)
|
|
{
|
|
if (Asn1Encodable.Der.Equals(encoding))
|
|
{
|
|
return new DerOutputStream(output);
|
|
}
|
|
else
|
|
{
|
|
return new Asn1OutputStream(output);
|
|
}
|
|
}
|
|
|
|
internal Asn1OutputStream(Stream os)
|
|
: base(os)
|
|
{
|
|
}
|
|
|
|
public virtual void WriteObject(Asn1Encodable asn1Encodable)
|
|
{
|
|
if (null == asn1Encodable)
|
|
throw new ArgumentNullException("asn1Encodable");
|
|
|
|
asn1Encodable.ToAsn1Object().GetEncoding(this.Encoding).Encode(this);
|
|
FlushInternal();
|
|
}
|
|
|
|
public virtual void WriteObject(Asn1Object asn1Object)
|
|
{
|
|
if (null == asn1Object)
|
|
throw new ArgumentNullException("asn1Object");
|
|
|
|
asn1Object.GetEncoding(this.Encoding).Encode(this);
|
|
FlushInternal();
|
|
}
|
|
|
|
internal void EncodeContents(IAsn1Encoding[] contentsEncodings)
|
|
{
|
|
for (int i = 0, count = contentsEncodings.Length; i < count; ++i)
|
|
{
|
|
contentsEncodings[i].Encode(this);
|
|
}
|
|
}
|
|
|
|
internal virtual int Encoding
|
|
{
|
|
get { return EncodingBer; }
|
|
}
|
|
|
|
internal void FlushInternal()
|
|
{
|
|
// Placeholder to support future internal buffering
|
|
}
|
|
|
|
internal void WriteDL(int dl)
|
|
{
|
|
if (dl < 128)
|
|
{
|
|
Debug.Assert(dl >= 0);
|
|
WriteByte((byte)dl);
|
|
return;
|
|
}
|
|
|
|
#if false && (NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER || _UNITY_2021_2_OR_NEWER_)
|
|
Span<byte> encoding = stackalloc byte[5];
|
|
BinaryPrimitives.WriteUInt32BigEndian(encoding[1..], (uint)dl);
|
|
int leadingZeroBytes = BitOperations.LeadingZeroCount((uint)dl) / 8;
|
|
encoding[leadingZeroBytes] = (byte)(0x84 - leadingZeroBytes);
|
|
Write(encoding[leadingZeroBytes..]);
|
|
#else
|
|
byte[] stack = new byte[5];
|
|
int pos = stack.Length;
|
|
|
|
do
|
|
{
|
|
stack[--pos] = (byte)dl;
|
|
dl >>= 8;
|
|
}
|
|
while (dl > 0);
|
|
|
|
int count = stack.Length - pos;
|
|
stack[--pos] = (byte)(0x80 | count);
|
|
|
|
Write(stack, pos, count + 1);
|
|
#endif
|
|
}
|
|
|
|
internal void WriteIdentifier(int tagClass, int tagNo)
|
|
{
|
|
if (tagNo < 31)
|
|
{
|
|
WriteByte((byte)(tagClass | tagNo));
|
|
return;
|
|
}
|
|
|
|
#if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER || _UNITY_2021_2_OR_NEWER_
|
|
Span<byte> stack = stackalloc byte[6];
|
|
#else
|
|
byte[] stack = new byte[6];
|
|
#endif
|
|
int pos = stack.Length;
|
|
|
|
stack[--pos] = (byte)(tagNo & 0x7F);
|
|
while (tagNo > 127)
|
|
{
|
|
tagNo >>= 7;
|
|
stack[--pos] = (byte)(tagNo & 0x7F | 0x80);
|
|
}
|
|
|
|
stack[--pos] = (byte)(tagClass | 0x1F);
|
|
|
|
#if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER || _UNITY_2021_2_OR_NEWER_
|
|
Write(stack[pos..]);
|
|
#else
|
|
Write(stack, pos, stack.Length - pos);
|
|
#endif
|
|
}
|
|
|
|
internal static IAsn1Encoding[] GetContentsEncodings(int encoding, Asn1Encodable[] elements)
|
|
{
|
|
int count = elements.Length;
|
|
IAsn1Encoding[] contentsEncodings = new IAsn1Encoding[count];
|
|
for (int i = 0; i < count; ++i)
|
|
{
|
|
contentsEncodings[i] = elements[i].ToAsn1Object().GetEncoding(encoding);
|
|
}
|
|
return contentsEncodings;
|
|
}
|
|
|
|
internal static int GetLengthOfContents(IAsn1Encoding[] contentsEncodings)
|
|
{
|
|
int contentsLength = 0;
|
|
for (int i = 0, count = contentsEncodings.Length; i < count; ++i)
|
|
{
|
|
contentsLength += contentsEncodings[i].GetLength();
|
|
}
|
|
return contentsLength;
|
|
}
|
|
|
|
internal static int GetLengthOfDL(int dl)
|
|
{
|
|
if (dl < 128)
|
|
return 1;
|
|
|
|
int length = 2;
|
|
while ((dl >>= 8) > 0)
|
|
{
|
|
++length;
|
|
}
|
|
return length;
|
|
}
|
|
|
|
internal static int GetLengthOfIdentifier(int tagNo)
|
|
{
|
|
if (tagNo < 31)
|
|
return 1;
|
|
|
|
int length = 2;
|
|
while ((tagNo >>= 7) > 0)
|
|
{
|
|
++length;
|
|
}
|
|
return length;
|
|
}
|
|
}
|
|
}
|
|
#pragma warning restore
|
|
#endif
|