using System;
using System.Collections;
using System.Diagnostics;
using System.IO;
#if PORTABLE
using System.Collections.Generic;
using System.Linq;
#endif
using Org.BouncyCastle.Utilities;
using Org.BouncyCastle.Utilities.Collections;
namespace Org.BouncyCastle.Asn1
{
public abstract class Asn1Set
: Asn1Object, IEnumerable
{
/**
* return an ASN1Set from the given object.
*
* @param obj the object we want converted.
* @exception ArgumentException if the object cannot be converted.
*/
public static Asn1Set GetInstance(object obj)
{
if (obj == null || obj is Asn1Set)
{
return (Asn1Set)obj;
}
//else if (obj is Asn1SetParser)
else if (obj is IAsn1Convertible)
{
Asn1Object asn1Object = ((IAsn1Convertible)obj).ToAsn1Object();
if (asn1Object is Asn1Set)
return (Asn1Set)asn1Object;
}
else if (obj is byte[])
{
try
{
return GetInstance(FromByteArray((byte[])obj));
}
catch (IOException e)
{
throw new ArgumentException("failed to construct set from byte[]: " + e.Message);
}
}
throw new ArgumentException("illegal object in GetInstance: " + Platform.GetTypeName(obj), "obj");
}
/**
* Return an ASN1 set from a tagged object. There is a special
* case here, if an object appears to have been explicitly tagged on
* reading but we were expecting it to be implicitly tagged in the
* normal course of events it indicates that we lost the surrounding
* set - so we need to add it back (this will happen if the tagged
* object is a sequence that contains other sequences). If you are
* dealing with implicitly tagged sets you really should
* be using this method.
*
* @param taggedObject the tagged object.
* @param declaredExplicit true if the object is meant to be explicitly tagged
* false otherwise.
* @exception ArgumentException if the tagged object cannot
* be converted.
*/
public static Asn1Set GetInstance(Asn1TaggedObject taggedObject, bool declaredExplicit)
{
Asn1Object baseObject = taggedObject.GetObject();
if (declaredExplicit)
{
if (!taggedObject.IsExplicit())
throw new ArgumentException("object implicit - explicit expected.");
return (Asn1Set)baseObject;
}
// If parsed as explicit though declared implicit, it should have been a set of one
if (taggedObject.IsExplicit())
{
if (taggedObject is BerTaggedObject)
return new BerSet(baseObject);
return new DLSet(baseObject);
}
if (baseObject is Asn1Set)
return (Asn1Set)baseObject;
// Parser assumes implicit constructed encodings are sequences
if (baseObject is Asn1Sequence)
return ((Asn1Sequence)baseObject).ToAsn1Set();
throw new ArgumentException("illegal object in GetInstance: " + Platform.GetTypeName(taggedObject),
"taggedObject");
}
// NOTE: Only non-readonly to support LazyDLSet
internal Asn1Encodable[] elements;
internal bool isSorted;
protected internal Asn1Set()
{
this.elements = Asn1EncodableVector.EmptyElements;
this.isSorted = true;
}
protected internal Asn1Set(Asn1Encodable element)
{
if (null == element)
throw new ArgumentNullException("element");
this.elements = new Asn1Encodable[]{ element };
this.isSorted = true;
}
protected internal Asn1Set(Asn1Encodable[] elements, bool doSort)
{
if (Arrays.IsNullOrContainsNull(elements))
throw new NullReferenceException("'elements' cannot be null, or contain null");
Asn1Encodable[] tmp = Asn1EncodableVector.CloneElements(elements);
if (doSort && tmp.Length >= 2)
{
tmp = Sort(tmp);
}
this.elements = tmp;
this.isSorted = doSort || tmp.Length < 2;
}
protected internal Asn1Set(Asn1EncodableVector elementVector, bool doSort)
{
if (null == elementVector)
throw new ArgumentNullException("elementVector");
Asn1Encodable[] tmp;
if (doSort && elementVector.Count >= 2)
{
tmp = Sort(elementVector.CopyElements());
}
else
{
tmp = elementVector.TakeElements();
}
this.elements = tmp;
this.isSorted = doSort || tmp.Length < 2;
}
protected internal Asn1Set(bool isSorted, Asn1Encodable[] elements)
{
this.elements = elements;
this.isSorted = isSorted || elements.Length < 2;
}
public virtual IEnumerator GetEnumerator()
{
return elements.GetEnumerator();
}
/**
* return the object at the set position indicated by index.
*
* @param index the set number (starting at zero) of the object
* @return the object at the set position indicated by index.
*/
public virtual Asn1Encodable this[int index]
{
get { return elements[index]; }
}
public virtual int Count
{
get { return elements.Length; }
}
public virtual Asn1Encodable[] ToArray()
{
return Asn1EncodableVector.CloneElements(elements);
}
private class Asn1SetParserImpl
: Asn1SetParser
{
private readonly Asn1Set outer;
private readonly int max;
private int index;
public Asn1SetParserImpl(
Asn1Set outer)
{
this.outer = outer;
// NOTE: Call Count here to 'force' a LazyDerSet
this.max = outer.Count;
}
public IAsn1Convertible ReadObject()
{
if (index == max)
return null;
Asn1Encodable obj = outer[index++];
if (obj is Asn1Sequence)
return ((Asn1Sequence)obj).Parser;
if (obj is Asn1Set)
return ((Asn1Set)obj).Parser;
// NB: Asn1OctetString implements Asn1OctetStringParser directly
// if (obj is Asn1OctetString)
// return ((Asn1OctetString)obj).Parser;
return obj;
}
public virtual Asn1Object ToAsn1Object()
{
return outer;
}
}
public Asn1SetParser Parser
{
get { return new Asn1SetParserImpl(this); }
}
protected override int Asn1GetHashCode()
{
// NOTE: Call Count here to 'force' a LazyDerSet
int i = Count;
int hc = i + 1;
while (--i >= 0)
{
hc *= 257;
hc ^= elements[i].ToAsn1Object().CallAsn1GetHashCode();
}
return hc;
}
protected override bool Asn1Equals(Asn1Object asn1Object)
{
Asn1Set that = asn1Object as Asn1Set;
if (null == that)
return false;
// NOTE: Call Count here (on both) to 'force' a LazyDerSet
int count = this.Count;
if (that.Count != count)
return false;
for (int i = 0; i < count; ++i)
{
Asn1Object o1 = this.elements[i].ToAsn1Object();
Asn1Object o2 = that.elements[i].ToAsn1Object();
if (o1 != o2 && !o1.CallAsn1Equals(o2))
return false;
}
return true;
}
internal override bool EncodeConstructed()
{
return true;
}
public override string ToString()
{
return CollectionUtilities.ToString(elements);
}
private static Asn1Encodable[] Sort(Asn1Encodable[] elements)
{
int count = elements.Length;
if (count < 2)
return elements;
#if PORTABLE
return elements
.Cast()
.Select(a => new { Item = a, Key = a.GetEncoded(Asn1Encodable.Der) })
.OrderBy(t => t.Key, new DerComparer())
.Select(t => t.Item)
.ToArray();
#else
byte[][] keys = new byte[count][];
for (int i = 0; i < count; ++i)
{
keys[i] = elements[i].GetEncoded(Der);
}
Array.Sort(keys, elements, new DerComparer());
return elements;
#endif
}
#if PORTABLE
private class DerComparer
: IComparer
{
public int Compare(byte[] x, byte[] y)
{
byte[] a = x, b = y;
#else
private class DerComparer
: IComparer
{
public int Compare(object x, object y)
{
byte[] a = (byte[])x, b = (byte[])y;
#endif
Debug.Assert(a.Length >= 2 && b.Length >= 2);
/*
* NOTE: Set elements in DER encodings are ordered first according to their tags (class and
* number); the CONSTRUCTED bit is not part of the tag.
*
* For SET-OF, this is unimportant. All elements have the same tag and DER requires them to
* either all be in constructed form or all in primitive form, according to that tag. The
* elements are effectively ordered according to their content octets.
*
* For SET, the elements will have distinct tags, and each will be in constructed or
* primitive form accordingly. Failing to ignore the CONSTRUCTED bit could therefore lead to
* ordering inversions.
*/
int a0 = a[0] & ~Asn1Tags.Constructed;
int b0 = b[0] & ~Asn1Tags.Constructed;
if (a0 != b0)
return a0 < b0 ? -1 : 1;
int len = System.Math.Min(a.Length, b.Length);
for (int i = 1; i < len; ++i)
{
byte ai = a[i], bi = b[i];
if (ai != bi)
return ai < bi ? -1 : 1;
}
Debug.Assert(a.Length == b.Length);
return 0;
}
}
}
}