using System; using Org.BouncyCastle.Math; using Org.BouncyCastle.Utilities; namespace Org.BouncyCastle.Asn1 { public class DerInteger : Asn1Object { public const string AllowUnsafeProperty = "Org.BouncyCastle.Asn1.AllowUnsafeInteger"; internal static bool AllowUnsafe() { string allowUnsafeValue = Platform.GetEnvironmentVariable(AllowUnsafeProperty); return allowUnsafeValue != null && Platform.EqualsIgnoreCase("true", allowUnsafeValue); } private readonly byte[] bytes; /** * return an integer from the passed in object * * @exception ArgumentException if the object cannot be converted. */ public static DerInteger GetInstance( object obj) { if (obj == null || obj is DerInteger) { return (DerInteger)obj; } throw new ArgumentException("illegal object in GetInstance: " + Platform.GetTypeName(obj)); } /** * return an Integer from a tagged object. * * @param obj the tagged object holding the object we want * @param isExplicit true if the object is meant to be explicitly * tagged false otherwise. * @exception ArgumentException if the tagged object cannot * be converted. */ public static DerInteger GetInstance( Asn1TaggedObject obj, bool isExplicit) { if (obj == null) throw new ArgumentNullException("obj"); Asn1Object o = obj.GetObject(); if (isExplicit || o is DerInteger) { return GetInstance(o); } return new DerInteger(Asn1OctetString.GetInstance(o).GetOctets()); } public DerInteger(int value) { bytes = BigInteger.ValueOf(value).ToByteArray(); } public DerInteger(BigInteger value) { if (value == null) throw new ArgumentNullException("value"); bytes = value.ToByteArray(); } public DerInteger(byte[] bytes) : this(bytes, true) { } internal DerInteger(byte[] bytes, bool clone) { if (IsMalformed(bytes)) throw new ArgumentException("malformed integer", "bytes"); this.bytes = clone ? Arrays.Clone(bytes) : bytes; } /** * in some cases positive values Get crammed into a space, * that's not quite big enough... */ public BigInteger PositiveValue { get { return new BigInteger(1, bytes); } } public BigInteger Value { get { return new BigInteger(bytes); } } internal override void Encode( DerOutputStream derOut) { derOut.WriteEncoded(Asn1Tags.Integer, bytes); } protected override int Asn1GetHashCode() { return Arrays.GetHashCode(bytes); } protected override bool Asn1Equals( Asn1Object asn1Object) { DerInteger other = asn1Object as DerInteger; if (other == null) return false; return Arrays.AreEqual(this.bytes, other.bytes); } public override string ToString() { return Value.ToString(); } /** * Apply the correct validation for an INTEGER primitive following the BER rules. * * @param bytes The raw encoding of the integer. * @return true if the (in)put fails this validation. */ internal static bool IsMalformed(byte[] bytes) { switch (bytes.Length) { case 0: return true; case 1: return false; default: return (sbyte)bytes[0] == ((sbyte)bytes[1] >> 7) && !AllowUnsafe(); } } } }