Xamarin Public Jenkins (auto-signing) 9050bbfc26 Imported Upstream version 6.6.0.108
Former-commit-id: 00c6fff7917ab7dddc0c67044b046850c2283096
2019-10-04 09:01:11 +00:00

542 lines
13 KiB
C#

/*
Copyright (C) 2009-2012 Jeroen Frijters
This software is provided 'as-is', without any express or implied
warranty. In no event will the authors be held liable for any damages
arising from the use of this software.
Permission is granted to anyone to use this software for any purpose,
including commercial applications, and to alter it and redistribute it
freely, subject to the following restrictions:
1. The origin of this software must not be misrepresented; you must not
claim that you wrote the original software. If you use this software
in a product, an acknowledgment in the product documentation would be
appreciated but is not required.
2. Altered source versions must be plainly marked as such, and must not be
misrepresented as being the original software.
3. This notice may not be removed or altered from any source distribution.
Jeroen Frijters
jeroen@frijters.net
*/
using System;
using System.Globalization;
using System.Configuration.Assemblies;
using System.IO;
using System.Security.Cryptography;
using System.Text;
using IKVM.Reflection.Reader;
namespace IKVM.Reflection
{
public sealed class AssemblyName
#if !CORECLR
: ICloneable
#endif
{
private string name;
private string culture;
private Version version;
private byte[] publicKeyToken;
private byte[] publicKey;
private StrongNameKeyPair keyPair;
private AssemblyNameFlags flags;
private AssemblyHashAlgorithm hashAlgorithm;
private AssemblyVersionCompatibility versionCompatibility = AssemblyVersionCompatibility.SameMachine;
private string codeBase;
internal byte[] hash;
public AssemblyName()
{
}
public AssemblyName(string assemblyName)
{
if (assemblyName == null)
{
throw new ArgumentNullException("assemblyName");
}
if (assemblyName == "")
{
throw new ArgumentException();
}
ParsedAssemblyName parsed;
switch (Fusion.ParseAssemblyName(assemblyName, out parsed))
{
case ParseAssemblyResult.GenericError:
case ParseAssemblyResult.DuplicateKey:
throw new FileLoadException();
}
if (!ParseVersion(parsed.Version, parsed.Retargetable.HasValue, out version))
{
throw new FileLoadException();
}
name = parsed.Name;
if (parsed.Culture != null)
{
if (parsed.Culture.Equals("neutral", StringComparison.OrdinalIgnoreCase))
{
culture = "";
}
else if (parsed.Culture == "")
{
throw new FileLoadException();
}
else
{
culture = new CultureInfo(parsed.Culture).Name;
}
}
if (parsed.PublicKeyToken != null)
{
if (parsed.PublicKeyToken.Equals("null", StringComparison.OrdinalIgnoreCase))
{
publicKeyToken = Empty<byte>.Array;
}
else if (parsed.PublicKeyToken.Length != 16)
{
throw new FileLoadException();
}
else
{
publicKeyToken = ParseKey(parsed.PublicKeyToken);
}
}
if (parsed.Retargetable.HasValue)
{
if (parsed.Culture == null || parsed.PublicKeyToken == null || version == null)
{
throw new FileLoadException();
}
if (parsed.Retargetable.Value)
{
flags |= AssemblyNameFlags.Retargetable;
}
}
ProcessorArchitecture = parsed.ProcessorArchitecture;
if (parsed.WindowsRuntime)
{
ContentType = AssemblyContentType.WindowsRuntime;
}
}
private static byte[] ParseKey(string key)
{
if ((key.Length & 1) != 0)
{
throw new FileLoadException();
}
byte[] buf = new byte[key.Length / 2];
for (int i = 0; i < buf.Length; i++)
{
buf[i] = (byte)(ParseHexDigit(key[i * 2]) * 16 + ParseHexDigit(key[i * 2 + 1]));
}
return buf;
}
private static int ParseHexDigit(char digit)
{
if (digit >= '0' && digit <= '9')
{
return digit - '0';
}
else
{
digit |= (char)0x20;
if (digit >= 'a' && digit <= 'f')
{
return 10 + digit - 'a';
}
else
{
throw new FileLoadException();
}
}
}
public override string ToString()
{
return FullName;
}
public string Name
{
get { return name; }
set { name = value; }
}
public CultureInfo CultureInfo
{
get { return culture == null ? null : new CultureInfo(culture); }
set { culture = value == null ? null : value.Name; }
}
public string CultureName
{
get { return culture; }
}
internal string Culture
{
get { return culture; }
set { culture = value; }
}
public Version Version
{
get { return version; }
set { version = value; }
}
public StrongNameKeyPair KeyPair
{
get { return keyPair; }
set { keyPair = value; }
}
public string CodeBase
{
get { return codeBase; }
set { codeBase = value; }
}
#if !CORECLR
public string EscapedCodeBase
{
get
{
// HACK use the real AssemblyName to escape the codebase
System.Reflection.AssemblyName tmp = new System.Reflection.AssemblyName();
tmp.CodeBase = codeBase;
return tmp.EscapedCodeBase;
}
}
#endif
public ProcessorArchitecture ProcessorArchitecture
{
get { return (ProcessorArchitecture)(((int)flags & 0x70) >> 4); }
set
{
if (value >= ProcessorArchitecture.None && value <= ProcessorArchitecture.Arm)
{
flags = (flags & ~(AssemblyNameFlags)0x70) | (AssemblyNameFlags)((int)value << 4);
}
}
}
public AssemblyNameFlags Flags
{
get { return flags & (AssemblyNameFlags)~0xEF0; }
set { flags = (flags & (AssemblyNameFlags)0xEF0) | (value & (AssemblyNameFlags)~0xEF0); }
}
public AssemblyVersionCompatibility VersionCompatibility
{
get { return versionCompatibility; }
set { versionCompatibility = value; }
}
public AssemblyContentType ContentType
{
get { return (AssemblyContentType)(((int)flags & 0xE00) >> 9); }
set
{
if (value >= AssemblyContentType.Default && value <= AssemblyContentType.WindowsRuntime)
{
flags = (flags & ~(AssemblyNameFlags)0xE00) | (AssemblyNameFlags)((int)value << 9);
}
}
}
public byte[] GetPublicKey()
{
return publicKey;
}
public void SetPublicKey(byte[] publicKey)
{
this.publicKey = publicKey;
flags = (flags & ~AssemblyNameFlags.PublicKey) | (publicKey == null ? 0 : AssemblyNameFlags.PublicKey);
}
public byte[] GetPublicKeyToken()
{
if (publicKeyToken == null && publicKey != null)
{
// note that GetPublicKeyToken() has a side effect in this case, because we retain this token even after the public key subsequently gets changed
publicKeyToken = ComputePublicKeyToken(publicKey);
}
return publicKeyToken;
}
public void SetPublicKeyToken(byte[] publicKeyToken)
{
this.publicKeyToken = publicKeyToken;
}
public AssemblyHashAlgorithm HashAlgorithm
{
get { return hashAlgorithm; }
set { hashAlgorithm = value; }
}
public byte[] __Hash
{
get { return hash; }
}
public string FullName
{
get
{
if (name == null)
{
return "";
}
ushort versionMajor = 0xFFFF;
ushort versionMinor = 0xFFFF;
ushort versionBuild = 0xFFFF;
ushort versionRevision = 0xFFFF;
if (version != null)
{
versionMajor = (ushort)version.Major;
versionMinor = (ushort)version.Minor;
versionBuild = (ushort)version.Build;
versionRevision = (ushort)version.Revision;
}
byte[] publicKeyToken = this.publicKeyToken;
if ((publicKeyToken == null || publicKeyToken.Length == 0) && publicKey != null)
{
publicKeyToken = ComputePublicKeyToken(publicKey);
}
return GetFullName(name, versionMajor, versionMinor, versionBuild, versionRevision, culture, publicKeyToken, (int)flags);
}
}
internal static string GetFullName(string name, ushort versionMajor, ushort versionMinor, ushort versionBuild, ushort versionRevision, string culture, byte[] publicKeyToken, int flags)
{
StringBuilder sb = new StringBuilder();
bool doubleQuotes = name.StartsWith(" ") || name.EndsWith(" ") || name.IndexOf('\'') != -1;
bool singleQuotes = name.IndexOf('"') != -1;
if (singleQuotes)
{
sb.Append('\'');
}
else if (doubleQuotes)
{
sb.Append('"');
}
if (name.IndexOf(',') != -1 || name.IndexOf('\\') != -1 || name.IndexOf('=') != -1 || (singleQuotes && name.IndexOf('\'') != -1))
{
for (int i = 0; i < name.Length; i++)
{
char c = name[i];
if (c == ',' || c == '\\' || c == '=' || (singleQuotes && c == '\''))
{
sb.Append('\\');
}
sb.Append(c);
}
}
else
{
sb.Append(name);
}
if (singleQuotes)
{
sb.Append('\'');
}
else if (doubleQuotes)
{
sb.Append('"');
}
if (versionMajor != 0xFFFF)
{
sb.Append(", Version=").Append(versionMajor);
if (versionMinor != 0xFFFF)
{
sb.Append('.').Append(versionMinor);
if (versionBuild != 0xFFFF)
{
sb.Append('.').Append(versionBuild);
if (versionRevision != 0xFFFF)
{
sb.Append('.').Append(versionRevision);
}
}
}
}
if (culture != null)
{
sb.Append(", Culture=").Append(culture == "" ? "neutral" : culture);
}
if (publicKeyToken != null)
{
sb.Append(", PublicKeyToken=");
if (publicKeyToken.Length == 0)
{
sb.Append("null");
}
else
{
AppendPublicKey(sb, publicKeyToken);
}
}
if ((flags & (int)AssemblyNameFlags.Retargetable) != 0)
{
sb.Append(", Retargetable=Yes");
}
if ((AssemblyContentType)((flags & 0xE00) >> 9) == AssemblyContentType.WindowsRuntime)
{
sb.Append(", ContentType=WindowsRuntime");
}
return sb.ToString();
}
internal static byte[] ComputePublicKeyToken(byte[] publicKey)
{
if (publicKey.Length == 0)
{
return publicKey;
}
using (var sha1 = SHA1.Create())
{
byte[] hash = sha1.ComputeHash(publicKey);
byte[] token = new byte[8];
for (int i = 0; i < token.Length; i++)
{
token[i] = hash[hash.Length - 1 - i];
}
return token;
}
}
internal static string ComputePublicKeyToken(string publicKey)
{
StringBuilder sb = new StringBuilder(16);
AppendPublicKey(sb, ComputePublicKeyToken(ParseKey(publicKey)));
return sb.ToString();
}
private static void AppendPublicKey(StringBuilder sb, byte[] publicKey)
{
for (int i = 0; i < publicKey.Length; i++)
{
sb.Append("0123456789abcdef"[publicKey[i] >> 4]);
sb.Append("0123456789abcdef"[publicKey[i] & 0x0F]);
}
}
public override bool Equals(object obj)
{
AssemblyName other = obj as AssemblyName;
return other != null && other.FullName == this.FullName;
}
public override int GetHashCode()
{
return FullName.GetHashCode();
}
public object Clone()
{
AssemblyName copy = (AssemblyName)MemberwiseClone();
copy.publicKey = Copy(publicKey);
copy.publicKeyToken = Copy(publicKeyToken);
return copy;
}
private static byte[] Copy(byte[] b)
{
return b == null || b.Length == 0 ? b : (byte[])b.Clone();
}
#if !CORECLR
public static bool ReferenceMatchesDefinition(AssemblyName reference, AssemblyName definition)
{
// HACK use the real AssemblyName to implement the (broken) ReferenceMatchesDefinition method
return System.Reflection.AssemblyName.ReferenceMatchesDefinition(new System.Reflection.AssemblyName(reference.FullName), new System.Reflection.AssemblyName(definition.FullName));
}
#endif
public static AssemblyName GetAssemblyName(string path)
{
try
{
path = Path.GetFullPath(path);
using (FileStream fs = new FileStream(path, FileMode.Open, FileAccess.Read, FileShare.Read))
{
ModuleReader module = new ModuleReader(null, null, fs, path, false);
if (module.Assembly == null)
{
throw new BadImageFormatException("Module does not contain a manifest");
}
return module.Assembly.GetName();
}
}
catch (IOException x)
{
throw new FileNotFoundException(x.Message, x);
}
catch (UnauthorizedAccessException x)
{
throw new FileNotFoundException(x.Message, x);
}
}
internal AssemblyNameFlags RawFlags
{
get { return flags; }
set { flags = value; }
}
private static bool ParseVersion(string str, bool mustBeComplete, out Version version)
{
if (str == null)
{
version = null;
return true;
}
string[] parts = str.Split('.');
if (parts.Length < 2 || parts.Length > 4)
{
version = null;
ushort dummy;
// if the version consists of a single integer, it is invalid, but not invalid enough to fail the parse of the whole assembly name
return parts.Length == 1 && ushort.TryParse(parts[0], NumberStyles.Integer, null, out dummy);
}
if (parts[0] == "" || parts[1] == "")
{
// this is a strange scenario, the version is invalid, but not invalid enough to fail the parse of the whole assembly name
version = null;
return true;
}
ushort major, minor, build = 65535, revision = 65535;
if (ushort.TryParse(parts[0], NumberStyles.Integer, null, out major)
&& ushort.TryParse(parts[1], NumberStyles.Integer, null, out minor)
&& (parts.Length <= 2 || parts[2] == "" || ushort.TryParse(parts[2], NumberStyles.Integer, null, out build))
&& (parts.Length <= 3 || parts[3] == "" || (parts[2] != "" && ushort.TryParse(parts[3], NumberStyles.Integer, null, out revision))))
{
if (mustBeComplete && (parts.Length < 4 || parts[2] == "" || parts[3] == ""))
{
version = null;
}
else if (major == 65535 || minor == 65535)
{
version = null;
}
else
{
version = new Version(major, minor, build, revision);
}
return true;
}
version = null;
return false;
}
}
}