276 lines
6.8 KiB
C#
Raw Normal View History

//
// AssemblyNameReference.cs
//
// Author:
// Jb Evain (jbevain@gmail.com)
//
// (C) 2005 Jb Evain
//
// 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.
//
namespace Mono.Cecil {
using System;
using System.Collections;
using System.Globalization;
using System.Security.Cryptography;
using System.Text;
using Mono.Cecil.Metadata;
internal class AssemblyNameReference : IMetadataScope, IAnnotationProvider, IReflectionStructureVisitable {
string m_name;
string m_culture;
Version m_version;
AssemblyFlags m_flags;
byte [] m_publicKey;
byte [] m_publicKeyToken;
AssemblyHashAlgorithm m_hashAlgo;
byte [] m_hash;
MetadataToken m_token;
IDictionary m_annotations;
bool m_fullNameDiscarded = true;
string m_fullName;
public string Name {
get { return m_name; }
set {
m_name = value;
m_fullNameDiscarded = true;
}
}
public string Culture {
get { return m_culture; }
set {
m_culture = value;
m_fullNameDiscarded = true;
}
}
public Version Version {
get { return m_version; }
set {
m_version = value;
m_fullNameDiscarded = true;
}
}
public AssemblyFlags Flags {
get { return m_flags; }
set { m_flags = value; }
}
public bool HasPublicKey {
get { return (m_flags & AssemblyFlags.PublicKey) != 0; }
set {
if (value)
m_flags |= AssemblyFlags.PublicKey;
else
m_flags &= ~AssemblyFlags.PublicKey;
}
}
public bool IsSideBySideCompatible {
get { return (m_flags & AssemblyFlags.SideBySideCompatible) != 0; }
set {
if (value)
m_flags |= AssemblyFlags.SideBySideCompatible;
else
m_flags &= ~AssemblyFlags.SideBySideCompatible;
}
}
public bool IsRetargetable {
get { return (m_flags & AssemblyFlags.Retargetable) != 0; }
set {
if (value)
m_flags |= AssemblyFlags.Retargetable;
else
m_flags &= ~AssemblyFlags.Retargetable;
}
}
public byte [] PublicKey {
get { return m_publicKey; }
set {
m_publicKey = value;
m_publicKeyToken = null;
m_fullNameDiscarded = true;
}
}
public byte [] PublicKeyToken {
get {
#if !CF_1_0
if ((m_publicKeyToken == null || m_publicKeyToken.Length == 0) && (m_publicKey != null && m_publicKey.Length > 0)) {
HashAlgorithm ha;
switch (m_hashAlgo) {
case AssemblyHashAlgorithm.Reserved:
ha = MD5.Create (); break;
default:
// None default to SHA1
ha = SHA1.Create (); break;
}
byte [] hash = ha.ComputeHash (m_publicKey);
// we need the last 8 bytes in reverse order
m_publicKeyToken = new byte [8];
Array.Copy (hash, (hash.Length - 8), m_publicKeyToken, 0, 8);
Array.Reverse (m_publicKeyToken, 0, 8);
}
#endif
return m_publicKeyToken;
}
set {
m_publicKeyToken = value;
m_fullNameDiscarded = true;
}
}
public string FullName {
get {
if (m_fullName != null && !m_fullNameDiscarded)
return m_fullName;
StringBuilder sb = new StringBuilder ();
string sep = ", ";
sb.Append (m_name);
if (m_version != null) {
sb.Append (sep);
sb.Append ("Version=");
sb.Append (m_version.ToString ());
}
sb.Append (sep);
sb.Append ("Culture=");
sb.Append (m_culture == null || m_culture.Length == 0 ? "neutral" : m_culture);
sb.Append (sep);
sb.Append ("PublicKeyToken=");
if (this.PublicKeyToken != null && m_publicKeyToken.Length > 0) {
for (int i = 0 ; i < m_publicKeyToken.Length ; i++) {
sb.Append (m_publicKeyToken [i].ToString ("x2"));
}
} else {
sb.Append ("null");
}
m_fullName = sb.ToString ();
m_fullNameDiscarded = false;
return m_fullName;
}
}
public static AssemblyNameReference Parse (string fullName)
{
if (fullName == null)
throw new ArgumentNullException ("fullName");
if (fullName.Length == 0)
throw new ArgumentException ("Name can not be empty");
AssemblyNameReference name = new AssemblyNameReference ();
string [] tokens = fullName.Split (',');
for (int i = 0; i < tokens.Length; i++) {
string token = tokens [i].Trim ();
if (i == 0) {
name.Name = token;
continue;
}
string [] parts = token.Split ('=');
if (parts.Length != 2)
throw new ArgumentException ("Malformed name");
switch (parts [0]) {
case "Version":
name.Version = new Version (parts [1]);
break;
case "Culture":
name.Culture = parts [1];
break;
case "PublicKeyToken":
string pkToken = parts [1];
if (pkToken == "null")
break;
name.PublicKeyToken = new byte [pkToken.Length / 2];
for (int j = 0; j < name.PublicKeyToken.Length; j++) {
name.PublicKeyToken [j] = Byte.Parse (pkToken.Substring (j * 2, 2), NumberStyles.HexNumber);
}
break;
}
}
return name;
}
public AssemblyHashAlgorithm HashAlgorithm
{
get { return m_hashAlgo; }
set { m_hashAlgo = value; }
}
public virtual byte [] Hash {
get { return m_hash; }
set { m_hash = value; }
}
public MetadataToken MetadataToken {
get { return m_token; }
set { m_token = value; }
}
IDictionary IAnnotationProvider.Annotations {
get {
if (m_annotations == null)
m_annotations = new Hashtable ();
return m_annotations;
}
}
public AssemblyNameReference () : this (string.Empty, string.Empty, new Version (0, 0, 0, 0))
{
}
public AssemblyNameReference (string name, string culture, Version version)
{
if (name == null)
throw new ArgumentNullException ("name");
if (culture == null)
throw new ArgumentNullException ("culture");
m_name = name;
m_culture = culture;
m_version = version;
m_hashAlgo = AssemblyHashAlgorithm.None;
}
public override string ToString ()
{
return this.FullName;
}
public virtual void Accept (IReflectionStructureVisitor visitor)
{
visitor.VisitAssemblyNameReference (this);
}
}
}