| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339 | 
							- //
 
- // ASN1.cs: Abstract Syntax Notation 1 - micro-parser and generator
 
- //
 
- // Authors:
 
- //	Sebastien Pouliot  <sebastien@ximian.com>
 
- //	Jesper Pedersen  <jep@itplus.dk>
 
- //
 
- // (C) 2002, 2003 Motus Technologies Inc. (http://www.motus.com)
 
- // Copyright (C) 2004 Novell, Inc (http://www.novell.com)
 
- // (C) 2004 IT+ A/S (http://www.itplus.dk)
 
- //
 
- // Permission is hereby granted, free of charge, to any person obtaining
 
- // a copy of this software and associated documentation files (the
 
- // "Software"), to deal in the Software without restriction, including
 
- // without limitation the rights to use, copy, modify, merge, publish,
 
- // distribute, sublicense, and/or sell copies of the Software, and to
 
- // permit persons to whom the Software is furnished to do so, subject to
 
- // the following conditions:
 
- // 
 
- // The above copyright notice and this permission notice shall be
 
- // included in all copies or substantial portions of the Software.
 
- // 
 
- // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
 
- // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
 
- // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
 
- // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
 
- // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
 
- // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
 
- // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
 
- //
 
- using System;
 
- using System.Collections;
 
- using System.IO;
 
- using System.Text;
 
- namespace Emby.Common.Implementations.Security {
 
- 	// References:
 
- 	// a.	ITU ASN.1 standards (free download)
 
- 	//	http://www.itu.int/ITU-T/studygroups/com17/languages/
 
-     public class ASN1 {
 
- 		private byte m_nTag;
 
- 		private byte[] m_aValue;
 
- 		private ArrayList elist;
 
- 		public ASN1 () : this (0x00, null) {}
 
- 		public ASN1 (byte tag) : this (tag, null) {}
 
- 		public ASN1 (byte tag, byte[] data) 
 
- 		{
 
- 			m_nTag = tag;
 
- 			m_aValue = data;
 
- 		}
 
- 		public ASN1 (byte[] data) 
 
- 		{
 
- 			m_nTag = data [0];
 
- 			int nLenLength = 0;
 
- 			int nLength = data [1];
 
- 			if (nLength > 0x80) {
 
- 				// composed length
 
- 				nLenLength = nLength - 0x80;
 
- 				nLength = 0;
 
- 				for (int i = 0; i < nLenLength; i++) {
 
- 					nLength *= 256;
 
- 					nLength += data [i + 2];
 
- 				}
 
- 			}
 
- 			else if (nLength == 0x80) {
 
- 				// undefined length encoding
 
- 				throw new NotSupportedException ("Undefined length encoding.");
 
- 			}
 
- 			m_aValue = new byte [nLength];
 
- 			Buffer.BlockCopy (data, (2 + nLenLength), m_aValue, 0, nLength);
 
- 			if ((m_nTag & 0x20) == 0x20) {
 
- 				int nStart = (2 + nLenLength);
 
- 				Decode (data, ref nStart, data.Length);
 
- 			}
 
- 		}
 
- 		public int Count {
 
- 			get { 
 
- 				if (elist == null)
 
- 					return 0;
 
- 				return elist.Count; 
 
- 			}
 
- 		}
 
- 		public byte Tag {
 
- 			get { return m_nTag; }
 
- 		}
 
- 		public int Length {
 
- 			get { 
 
- 				if (m_aValue != null)
 
- 					return m_aValue.Length; 
 
- 				else
 
- 					return 0;
 
- 			}
 
- 		}
 
- 		public byte[] Value {
 
- 			get { 
 
- 				if (m_aValue == null)
 
- 					GetBytes ();
 
- 				return (byte[]) m_aValue.Clone (); 
 
- 			}
 
- 			set { 
 
- 				if (value != null)
 
- 					m_aValue = (byte[]) value.Clone (); 
 
- 			}
 
- 		}
 
- 		private bool CompareArray (byte[] array1, byte[] array2)
 
- 		{
 
- 			bool bResult = (array1.Length == array2.Length);
 
- 			if (bResult) {
 
- 				for (int i = 0; i < array1.Length; i++) {
 
- 					if (array1[i] != array2[i])
 
- 						return false;
 
- 				}
 
- 			}
 
- 			return bResult;
 
- 		}
 
- 		public bool Equals (byte[] asn1) 
 
- 		{
 
- 			return CompareArray (this.GetBytes (), asn1);
 
- 		}
 
- 		public bool CompareValue (byte[] value) 
 
- 		{
 
- 			return CompareArray (m_aValue, value);
 
- 		}
 
- 		public ASN1 Add (ASN1 asn1) 
 
- 		{
 
- 			if (asn1 != null) {
 
- 				if (elist == null)
 
- 					elist = new ArrayList ();
 
- 				elist.Add (asn1);
 
- 			}
 
- 			return asn1;
 
- 		}
 
- 		public virtual byte[] GetBytes () 
 
- 		{
 
- 			byte[] val = null;
 
- 			
 
- 			if (Count > 0) {
 
- 				int esize = 0;
 
- 				ArrayList al = new ArrayList ();
 
- 				foreach (ASN1 a in elist) {
 
- 					byte[] item = a.GetBytes ();
 
- 					al.Add (item);
 
- 					esize += item.Length;
 
- 				}
 
- 				val = new byte [esize];
 
- 				int pos = 0;
 
- 				for (int i=0; i < elist.Count; i++) {
 
- 					byte[] item = (byte[]) al[i];
 
- 					Buffer.BlockCopy (item, 0, val, pos, item.Length);
 
- 					pos += item.Length;
 
- 				}
 
- 			} else if (m_aValue != null) {
 
- 				val = m_aValue;
 
- 			}
 
- 			byte[] der;
 
- 			int nLengthLen = 0;
 
- 			if (val != null) {
 
- 				int nLength = val.Length;
 
- 				// special for length > 127
 
- 				if (nLength > 127) {
 
- 					if (nLength <= Byte.MaxValue) {
 
- 						der = new byte [3 + nLength];
 
- 						Buffer.BlockCopy (val, 0, der, 3, nLength);
 
- 						nLengthLen = 0x81;
 
- 						der[2] = (byte)(nLength);
 
- 					}
 
- 					else if (nLength <= UInt16.MaxValue) {
 
- 						der = new byte [4 + nLength];
 
- 						Buffer.BlockCopy (val, 0, der, 4, nLength);
 
- 						nLengthLen = 0x82;
 
- 						der[2] = (byte)(nLength >> 8);
 
- 						der[3] = (byte)(nLength);
 
- 					}
 
- 					else if (nLength <= 0xFFFFFF) {
 
- 						// 24 bits
 
- 						der = new byte [5 + nLength];
 
- 						Buffer.BlockCopy (val, 0, der, 5, nLength);
 
- 						nLengthLen = 0x83;
 
- 						der [2] = (byte)(nLength >> 16);
 
- 						der [3] = (byte)(nLength >> 8);
 
- 						der [4] = (byte)(nLength);
 
- 					}
 
- 					else {
 
- 						// max (Length is an integer) 32 bits
 
- 						der = new byte [6 + nLength];
 
- 						Buffer.BlockCopy (val, 0, der, 6, nLength);
 
- 						nLengthLen = 0x84;
 
- 						der [2] = (byte)(nLength >> 24);
 
- 						der [3] = (byte)(nLength >> 16);
 
- 						der [4] = (byte)(nLength >> 8);
 
- 						der [5] = (byte)(nLength);
 
- 					}
 
- 				}
 
- 				else {
 
- 					// basic case (no encoding)
 
- 					der = new byte [2 + nLength];
 
- 					Buffer.BlockCopy (val, 0, der, 2, nLength);
 
- 					nLengthLen = nLength;
 
- 				}
 
- 				if (m_aValue == null)
 
- 					m_aValue = val;
 
- 			}
 
- 			else
 
- 				der = new byte[2];
 
- 			der[0] = m_nTag;
 
- 			der[1] = (byte)nLengthLen;
 
- 			return der;
 
- 		}
 
- 		// Note: Recursive
 
- 		protected void Decode (byte[] asn1, ref int anPos, int anLength) 
 
- 		{
 
- 			byte nTag;
 
- 			int nLength;
 
- 			byte[] aValue;
 
- 			// minimum is 2 bytes (tag + length of 0)
 
- 			while (anPos < anLength - 1) {
 
- 				DecodeTLV (asn1, ref anPos, out nTag, out nLength, out aValue);
 
- 				// sometimes we get trailing 0
 
- 				if (nTag == 0)
 
- 					continue;
 
- 				ASN1 elm = Add (new ASN1 (nTag, aValue));
 
- 				if ((nTag & 0x20) == 0x20) {
 
- 					int nConstructedPos = anPos;
 
- 					elm.Decode (asn1, ref nConstructedPos, nConstructedPos + nLength);
 
- 				}
 
- 				anPos += nLength; // value length
 
- 			}
 
- 		}
 
- 		// TLV : Tag - Length - Value
 
- 		protected void DecodeTLV (byte[] asn1, ref int pos, out byte tag, out int length, out byte[] content) 
 
- 		{
 
- 			tag = asn1 [pos++];
 
- 			length = asn1 [pos++];
 
- 			// special case where L contains the Length of the Length + 0x80
 
- 			if ((length & 0x80) == 0x80) {
 
- 				int nLengthLen = length & 0x7F;
 
- 				length = 0;
 
- 				for (int i = 0; i < nLengthLen; i++)
 
- 					length = length * 256 + asn1 [pos++];
 
- 			}
 
- 			content = new byte [length];
 
- 			Buffer.BlockCopy (asn1, pos, content, 0, length);
 
- 		}
 
- 		public ASN1 this [int index] {
 
- 			get { 		
 
- 				try {
 
- 					if ((elist == null) || (index >= elist.Count))
 
- 						return null;
 
- 					return (ASN1) elist [index];
 
- 				}
 
- 				catch (ArgumentOutOfRangeException) {
 
- 					return null;
 
- 				}
 
- 			}
 
- 		}
 
- 		public ASN1 Element (int index, byte anTag) 
 
- 		{
 
- 			try {
 
- 				if ((elist == null) || (index >= elist.Count))
 
- 					return null;
 
- 				ASN1 elm = (ASN1) elist [index];
 
- 				if (elm.Tag == anTag)
 
- 					return elm;
 
- 				else
 
- 					return null;
 
- 			}
 
- 			catch (ArgumentOutOfRangeException) {
 
- 				return null;
 
- 			}
 
- 		}
 
- 		public override string ToString()
 
- 		{
 
- 			StringBuilder hexLine = new StringBuilder ();
 
-             
 
- 			// Add tag
 
- 			hexLine.AppendFormat ("Tag: {0} {1}", m_nTag.ToString ("X2"), Environment.NewLine);
 
- 			// Add length
 
- 			hexLine.AppendFormat ("Length: {0} {1}", Value.Length, Environment.NewLine);
 
- 			// Add value
 
- 			hexLine.Append ("Value: ");
 
- 			hexLine.Append (Environment.NewLine);
 
- 			for (int i = 0; i < Value.Length; i++) {
 
- 				hexLine.AppendFormat ("{0} ", Value [i].ToString ("X2"));
 
- 				if ((i+1) % 16 == 0)
 
- 					hexLine.AppendFormat (Environment.NewLine);
 
- 			}
 
- 			return hexLine.ToString ();
 
- 		}
 
- 		public void SaveToFile (string filename)
 
- 		{
 
- 			if (filename == null)
 
- 				throw new ArgumentNullException ("filename");
 
- 			using (FileStream fs = File.Create (filename)) {
 
- 				byte[] data = GetBytes ();
 
- 				fs.Write (data, 0, data.Length);
 
- 			}
 
- 		}
 
- 	}
 
- }
 
 
  |