a575963da9
Former-commit-id: da6be194a6b1221998fc28233f2503bd61dd9d14
276 lines
6.8 KiB
C#
276 lines
6.8 KiB
C#
//
|
|
// 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);
|
|
}
|
|
}
|
|
}
|