mirror of https://github.com/icsharpcode/ILSpy.git
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; |
|
} |
|
}
|
|
|