You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
333 lines
9.9 KiB
333 lines
9.9 KiB
// Copyright (c) 2024 Siegfried Pammer
|
|
//
|
|
// 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.
|
|
|
|
#nullable enable
|
|
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Collections.Immutable;
|
|
using System.Diagnostics;
|
|
using System.Linq;
|
|
using System.Reflection.Metadata;
|
|
using System.Reflection.PortableExecutable;
|
|
|
|
using ICSharpCode.Decompiler.TypeSystem;
|
|
using ICSharpCode.Decompiler.Util;
|
|
|
|
namespace ICSharpCode.Decompiler.Metadata
|
|
{
|
|
/// <summary>
|
|
/// MetadataFile is the main class the decompiler uses to represent a metadata assembly/module.
|
|
/// Every file on disk can be loaded into a standalone MetadataFile instance.
|
|
///
|
|
/// A MetadataFile can be combined with its referenced assemblies/modules to form a type system,
|
|
/// in that case the <see cref="MetadataModule"/> class is used instead.
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// In addition to wrapping a <c>System.Reflection.Metadata.MetadataReader</c>, this class
|
|
/// contains a few decompiler-specific caches to allow efficiently constructing a type
|
|
/// system from multiple MetadataFiles. This allows the caches to be shared across multiple
|
|
/// decompiled type systems.
|
|
/// </remarks>
|
|
[DebuggerDisplay("{Kind}: {FileName}")]
|
|
public class MetadataFile
|
|
{
|
|
public enum MetadataFileKind
|
|
{
|
|
PortableExecutable,
|
|
ProgramDebugDatabase,
|
|
WebCIL,
|
|
Metadata
|
|
}
|
|
|
|
public string FileName { get; }
|
|
public MetadataFileKind Kind { get; }
|
|
public MetadataReader Metadata { get; }
|
|
|
|
public virtual int MetadataOffset { get; }
|
|
public virtual bool IsEmbedded { get; }
|
|
public virtual bool IsMetadataOnly { get; } = true;
|
|
|
|
public bool IsAssembly => Metadata.IsAssembly;
|
|
|
|
string? name;
|
|
|
|
public string Name {
|
|
get {
|
|
var value = LazyInit.VolatileRead(ref name);
|
|
if (value == null)
|
|
{
|
|
var metadata = Metadata;
|
|
value = metadata.IsAssembly
|
|
? metadata.GetString(metadata.GetAssemblyDefinition().Name)
|
|
: metadata.GetString(metadata.GetModuleDefinition().Name);
|
|
value = LazyInit.GetOrSet(ref name, value);
|
|
}
|
|
return value;
|
|
}
|
|
}
|
|
|
|
string? fullName;
|
|
|
|
public string FullName {
|
|
get {
|
|
var value = LazyInit.VolatileRead(ref fullName);
|
|
if (value == null)
|
|
{
|
|
var metadata = Metadata;
|
|
value = metadata.IsAssembly ? metadata.GetFullAssemblyName() : Name;
|
|
value = LazyInit.GetOrSet(ref fullName, value);
|
|
}
|
|
return value;
|
|
}
|
|
}
|
|
|
|
public TargetRuntime GetRuntime()
|
|
{
|
|
string version = Metadata.MetadataVersion;
|
|
if (version == null || version.Length <= 1)
|
|
return TargetRuntime.Unknown;
|
|
switch (version[1])
|
|
{
|
|
case '1':
|
|
if (version.Length <= 3)
|
|
return TargetRuntime.Unknown;
|
|
if (version[3] == 1)
|
|
return TargetRuntime.Net_1_0;
|
|
else
|
|
return TargetRuntime.Net_1_1;
|
|
case '2':
|
|
return TargetRuntime.Net_2_0;
|
|
case '4':
|
|
return TargetRuntime.Net_4_0;
|
|
default:
|
|
return TargetRuntime.Unknown;
|
|
}
|
|
}
|
|
|
|
ImmutableArray<AssemblyReference> assemblyReferences;
|
|
public ImmutableArray<AssemblyReference> AssemblyReferences {
|
|
get {
|
|
var value = assemblyReferences;
|
|
if (value.IsDefault)
|
|
{
|
|
value = Metadata.AssemblyReferences.Select(r => new AssemblyReference(this.Metadata, r)).ToImmutableArray();
|
|
assemblyReferences = value;
|
|
}
|
|
return value;
|
|
}
|
|
}
|
|
|
|
ImmutableArray<ModuleReferenceMetadata> moduleReferences;
|
|
public ImmutableArray<ModuleReferenceMetadata> ModuleReferences {
|
|
get {
|
|
var value = moduleReferences;
|
|
if (value.IsDefault)
|
|
{
|
|
value = Metadata.GetModuleReferences()
|
|
.Select(m => new ModuleReferenceMetadata(this.Metadata, m))
|
|
.ToImmutableArray();
|
|
|
|
moduleReferences = value;
|
|
}
|
|
return value;
|
|
}
|
|
}
|
|
|
|
public ImmutableArray<Resource> Resources => GetResources().ToImmutableArray();
|
|
|
|
IEnumerable<Resource> GetResources()
|
|
{
|
|
var metadata = Metadata;
|
|
foreach (var h in metadata.ManifestResources)
|
|
{
|
|
yield return new MetadataResource(this, h);
|
|
}
|
|
}
|
|
|
|
Dictionary<TopLevelTypeName, TypeDefinitionHandle>? typeLookup;
|
|
|
|
/// <summary>
|
|
/// Finds the top-level-type with the specified name.
|
|
/// </summary>
|
|
public TypeDefinitionHandle GetTypeDefinition(TopLevelTypeName typeName)
|
|
{
|
|
var lookup = LazyInit.VolatileRead(ref typeLookup);
|
|
if (lookup == null)
|
|
{
|
|
lookup = new Dictionary<TopLevelTypeName, TypeDefinitionHandle>();
|
|
foreach (var handle in Metadata.TypeDefinitions)
|
|
{
|
|
var td = Metadata.GetTypeDefinition(handle);
|
|
if (!td.GetDeclaringType().IsNil)
|
|
{
|
|
continue; // nested type
|
|
}
|
|
var nsHandle = td.Namespace;
|
|
string ns = nsHandle.IsNil ? string.Empty : Metadata.GetString(nsHandle);
|
|
string name = ReflectionHelper.SplitTypeParameterCountFromReflectionName(Metadata.GetString(td.Name), out int typeParameterCount);
|
|
lookup[new TopLevelTypeName(ns, name, typeParameterCount)] = handle;
|
|
}
|
|
lookup = LazyInit.GetOrSet(ref typeLookup, lookup);
|
|
}
|
|
if (lookup.TryGetValue(typeName, out var resultHandle))
|
|
return resultHandle;
|
|
else
|
|
return default;
|
|
}
|
|
|
|
Dictionary<FullTypeName, ExportedTypeHandle>? typeForwarderLookup;
|
|
|
|
/// <summary>
|
|
/// Finds the type forwarder with the specified name.
|
|
/// </summary>
|
|
public ExportedTypeHandle GetTypeForwarder(FullTypeName typeName)
|
|
{
|
|
var lookup = LazyInit.VolatileRead(ref typeForwarderLookup);
|
|
if (lookup == null)
|
|
{
|
|
lookup = new Dictionary<FullTypeName, ExportedTypeHandle>();
|
|
foreach (var handle in Metadata.ExportedTypes)
|
|
{
|
|
var td = Metadata.GetExportedType(handle);
|
|
lookup[td.GetFullTypeName(Metadata)] = handle;
|
|
}
|
|
lookup = LazyInit.GetOrSet(ref typeForwarderLookup, lookup);
|
|
}
|
|
if (lookup.TryGetValue(typeName, out var resultHandle))
|
|
return resultHandle;
|
|
else
|
|
return default;
|
|
}
|
|
|
|
MethodSemanticsLookup? methodSemanticsLookup;
|
|
|
|
internal MethodSemanticsLookup MethodSemanticsLookup {
|
|
get {
|
|
var r = LazyInit.VolatileRead(ref methodSemanticsLookup);
|
|
if (r != null)
|
|
return r;
|
|
else
|
|
return LazyInit.GetOrSet(ref methodSemanticsLookup, new MethodSemanticsLookup(Metadata));
|
|
}
|
|
}
|
|
|
|
public MetadataFile(MetadataFileKind kind, string fileName, MetadataReaderProvider metadata, MetadataReaderOptions metadataOptions = MetadataReaderOptions.Default, int metadataOffset = 0, bool isEmbedded = false)
|
|
{
|
|
this.Kind = kind;
|
|
this.FileName = fileName;
|
|
this.Metadata = metadata.GetMetadataReader(metadataOptions);
|
|
this.MetadataOffset = metadataOffset;
|
|
this.IsEmbedded = isEmbedded;
|
|
}
|
|
|
|
private protected MetadataFile(MetadataFileKind kind, string fileName, PEReader reader, MetadataReaderOptions metadataOptions = MetadataReaderOptions.Default)
|
|
{
|
|
this.Kind = kind;
|
|
this.FileName = fileName ?? throw new ArgumentNullException(nameof(fileName));
|
|
_ = reader ?? throw new ArgumentNullException(nameof(reader));
|
|
if (!reader.HasMetadata)
|
|
throw new MetadataFileNotSupportedException("PE file does not contain any managed metadata.");
|
|
this.Metadata = reader.GetMetadataReader(metadataOptions);
|
|
}
|
|
|
|
public virtual MethodBodyBlock GetMethodBody(int rva)
|
|
{
|
|
throw new BadImageFormatException("This metadata file does not contain method bodies.");
|
|
}
|
|
|
|
public virtual SectionData GetSectionData(int rva)
|
|
{
|
|
throw new BadImageFormatException("This metadata file does not support sections.");
|
|
}
|
|
|
|
public virtual int GetContainingSectionIndex(int rva)
|
|
{
|
|
throw new BadImageFormatException("This metadata file does not support sections.");
|
|
}
|
|
|
|
public virtual ImmutableArray<SectionHeader> SectionHeaders => throw new BadImageFormatException("This metadata file does not support sections.");
|
|
|
|
/// <summary>
|
|
/// Gets the CLI header or null if the image does not have one.
|
|
/// </summary>
|
|
public virtual CorHeader? CorHeader => null;
|
|
|
|
public IModuleReference WithOptions(TypeSystemOptions options)
|
|
{
|
|
return new MetadataFileWithOptions(this, options);
|
|
}
|
|
|
|
private class MetadataFileWithOptions : IModuleReference
|
|
{
|
|
readonly MetadataFile peFile;
|
|
readonly TypeSystemOptions options;
|
|
|
|
public MetadataFileWithOptions(MetadataFile peFile, TypeSystemOptions options)
|
|
{
|
|
this.peFile = peFile;
|
|
this.options = options;
|
|
}
|
|
|
|
IModule IModuleReference.Resolve(ITypeResolveContext context)
|
|
{
|
|
return new MetadataModule(context.Compilation, peFile, options);
|
|
}
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Abstraction over PEMemoryBlock
|
|
/// </summary>
|
|
public readonly unsafe struct SectionData
|
|
{
|
|
public byte* Pointer { get; }
|
|
public int Length { get; }
|
|
|
|
public SectionData(PEMemoryBlock block)
|
|
{
|
|
Pointer = block.Pointer;
|
|
Length = block.Length;
|
|
}
|
|
|
|
public SectionData(byte* startPointer, int length)
|
|
{
|
|
Pointer = startPointer;
|
|
Length = length;
|
|
}
|
|
|
|
public BlobReader GetReader()
|
|
{
|
|
return new BlobReader(Pointer, Length);
|
|
}
|
|
|
|
internal BlobReader GetReader(int offset, int size)
|
|
{
|
|
return new BlobReader(Pointer + offset, size);
|
|
}
|
|
}
|
|
|
|
public struct SectionHeader
|
|
{
|
|
public string Name;
|
|
public uint VirtualSize;
|
|
public uint VirtualAddress;
|
|
public uint RawDataSize;
|
|
public uint RawDataPtr;
|
|
}
|
|
}
|