123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279 |
- #if !BESTHTTP_DISABLE_ALTERNATE_SSL && (!UNITY_WEBGL || UNITY_EDITOR)
- using System;
- using System.Diagnostics;
- using System.Text;
- using Org.BouncyCastle.Math;
- using Org.BouncyCastle.Utilities;
- namespace Org.BouncyCastle.Asn1
- {
- public class DerBitString
- : DerStringBase
- {
- private static readonly char[] table
- = { '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'A', 'B', 'C', 'D', 'E', 'F' };
- protected readonly byte[] mData;
- protected readonly int mPadBits;
- /**
- * return a Bit string from the passed in object
- *
- * @exception ArgumentException if the object cannot be converted.
- */
- public static DerBitString GetInstance(
- object obj)
- {
- if (obj == null || obj is DerBitString)
- {
- return (DerBitString) obj;
- }
- if (obj is byte[])
- {
- try
- {
- return (DerBitString)FromByteArray((byte[])obj);
- }
- catch (Exception e)
- {
- throw new ArgumentException("encoding error in GetInstance: " + e.ToString());
- }
- }
- throw new ArgumentException("illegal object in GetInstance: " + Org.BouncyCastle.Utilities.Platform.GetTypeName(obj));
- }
- /**
- * return a Bit string from a tagged object.
- *
- * @param obj the tagged object holding the object we want
- * @param explicitly true if the object is meant to be explicitly
- * tagged false otherwise.
- * @exception ArgumentException if the tagged object cannot
- * be converted.
- */
- public static DerBitString GetInstance(
- Asn1TaggedObject obj,
- bool isExplicit)
- {
- Asn1Object o = obj.GetObject();
- if (isExplicit || o is DerBitString)
- {
- return GetInstance(o);
- }
- return FromAsn1Octets(((Asn1OctetString)o).GetOctets());
- }
- /**
- * @param data the octets making up the bit string.
- * @param padBits the number of extra bits at the end of the string.
- */
- public DerBitString(
- byte[] data,
- int padBits)
- {
- if (data == null)
- throw new ArgumentNullException("data");
- if (padBits < 0 || padBits > 7)
- throw new ArgumentException("must be in the range 0 to 7", "padBits");
- if (data.Length == 0 && padBits != 0)
- throw new ArgumentException("if 'data' is empty, 'padBits' must be 0");
- this.mData = Arrays.Clone(data);
- this.mPadBits = padBits;
- }
- public DerBitString(
- byte[] data)
- : this(data, 0)
- {
- }
- public DerBitString(
- int namedBits)
- {
- if (namedBits == 0)
- {
- this.mData = new byte[0];
- this.mPadBits = 0;
- return;
- }
- int bits = BigInteger.BitLen(namedBits);
- int bytes = (bits + 7) / 8;
- Debug.Assert(0 < bytes && bytes <= 4);
- byte[] data = new byte[bytes];
- --bytes;
- for (int i = 0; i < bytes; i++)
- {
- data[i] = (byte)namedBits;
- namedBits >>= 8;
- }
- Debug.Assert((namedBits & 0xFF) != 0);
- data[bytes] = (byte)namedBits;
- int padBits = 0;
- while ((namedBits & (1 << padBits)) == 0)
- {
- ++padBits;
- }
- Debug.Assert(padBits < 8);
- this.mData = data;
- this.mPadBits = padBits;
- }
- public DerBitString(
- Asn1Encodable obj)
- : this(obj.GetDerEncoded())
- {
- }
- /**
- * Return the octets contained in this BIT STRING, checking that this BIT STRING really
- * does represent an octet aligned string. Only use this method when the standard you are
- * following dictates that the BIT STRING will be octet aligned.
- *
- * @return a copy of the octet aligned data.
- */
- public virtual byte[] GetOctets()
- {
- if (mPadBits != 0)
- throw new InvalidOperationException("attempt to get non-octet aligned data from BIT STRING");
- return Arrays.Clone(mData);
- }
- public virtual byte[] GetBytes()
- {
- byte[] data = Arrays.Clone(mData);
- // DER requires pad bits be zero
- if (mPadBits > 0)
- {
- data[data.Length - 1] &= (byte)(0xFF << mPadBits);
- }
- return data;
- }
- public virtual int PadBits
- {
- get { return mPadBits; }
- }
- /**
- * @return the value of the bit string as an int (truncating if necessary)
- */
- public virtual int IntValue
- {
- get
- {
- int value = 0, length = System.Math.Min(4, mData.Length);
- for (int i = 0; i < length; ++i)
- {
- value |= (int)mData[i] << (8 * i);
- }
- if (mPadBits > 0 && length == mData.Length)
- {
- int mask = (1 << mPadBits) - 1;
- value &= ~(mask << (8 * (length - 1)));
- }
- return value;
- }
- }
- internal override void Encode(
- DerOutputStream derOut)
- {
- if (mPadBits > 0)
- {
- int last = mData[mData.Length - 1];
- int mask = (1 << mPadBits) - 1;
- int unusedBits = last & mask;
- if (unusedBits != 0)
- {
- byte[] contents = Arrays.Prepend(mData, (byte)mPadBits);
- /*
- * X.690-0207 11.2.1: Each unused bit in the final octet of the encoding of a bit string value shall be set to zero.
- */
- contents[contents.Length - 1] = (byte)(last ^ unusedBits);
- derOut.WriteEncoded(Asn1Tags.BitString, contents);
- return;
- }
- }
- derOut.WriteEncoded(Asn1Tags.BitString, (byte)mPadBits, mData);
- }
- protected override int Asn1GetHashCode()
- {
- return mPadBits.GetHashCode() ^ Arrays.GetHashCode(mData);
- }
- protected override bool Asn1Equals(
- Asn1Object asn1Object)
- {
- DerBitString other = asn1Object as DerBitString;
- if (other == null)
- return false;
- return this.mPadBits == other.mPadBits
- && Arrays.AreEqual(this.mData, other.mData);
- }
- public override string GetString()
- {
- StringBuilder buffer = new StringBuilder("#");
- byte[] str = GetDerEncoded();
- for (int i = 0; i != str.Length; i++)
- {
- uint ubyte = str[i];
- buffer.Append(table[(ubyte >> 4) & 0xf]);
- buffer.Append(table[str[i] & 0xf]);
- }
- return buffer.ToString();
- }
- internal static DerBitString FromAsn1Octets(byte[] octets)
- {
- if (octets.Length < 1)
- throw new ArgumentException("truncated BIT STRING detected", "octets");
- int padBits = octets[0];
- byte[] data = Arrays.CopyOfRange(octets, 1, octets.Length);
- if (padBits > 0 && padBits < 8 && data.Length > 0)
- {
- int last = data[data.Length - 1];
- int mask = (1 << padBits) - 1;
- if ((last & mask) != 0)
- {
- return new BerBitString(data, padBits);
- }
- }
- return new DerBitString(data, padBits);
- }
- }
- }
- #endif
|