using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Runtime.CompilerServices; using System.Text; using System.Threading.Tasks; using ICSharpCode.Decompiler.DebugInfo; using ICSharpCode.Decompiler.Metadata; using ICSharpCode.Decompiler.TypeSystem; namespace ICSharpCode.ILSpy { public static class LoadedAssemblyExtensions { /// /// This method creates a Cecil object model from a PEFile. It is intended as helper method for plugins. /// Note that this method is expensive and creates high memory pressure! /// Note that accessing the Cecil objects created by this method after an assembly has been unloaded by ILSpy /// might lead to or similar. /// /// Use only as last resort if there is something missing in the official ILSpy API. /// Consider creating an issue at https://github.com/icsharpcode/ILSpy/issues/new /// and discussing the problem with us. public unsafe static Mono.Cecil.ModuleDefinition CreateCecilObjectModel(this PEFile file) { if (!file.Reader.IsEntireImageAvailable) throw new InvalidOperationException("Need full image to create Cecil object model!"); var image = file.Reader.GetEntireImage(); return Mono.Cecil.ModuleDefinition.ReadModule(new UnmanagedMemoryStream(image.Pointer, image.Length)); } public static IAssemblyResolver GetAssemblyResolver(this PEFile file) { return GetLoadedAssembly(file).GetAssemblyResolver(); } public static IDebugInfoProvider GetDebugInfoOrNull(this PEFile file) { return GetLoadedAssembly(file).GetDebugInfoOrNull(); } public static ICompilation GetTypeSystemOrNull(this PEFile file) { return GetLoadedAssembly(file).GetTypeSystemOrNull(); } static LoadedAssembly GetLoadedAssembly(PEFile file) { if (file == null) throw new ArgumentNullException(nameof(file)); LoadedAssembly loadedAssembly; lock (LoadedAssembly.loadedAssemblies) { if (!LoadedAssembly.loadedAssemblies.TryGetValue(file, out loadedAssembly)) throw new ArgumentException("The specified file is not associated with a LoadedAssembly!"); } return loadedAssembly; } } }