debugging 如何使用IMetaDataDispenser.OpenScope访问嵌入程序集的元数据?

tsm1rwdh  于 2023-03-23  发布在  其他
关注(0)|答案(1)|浏览(73)

我有一个.NET解决方案,它由几个项目组成。可以说其中一个项目在逻辑上是主项目,而所有其他项目都是辅助项目。我们的团队决定用另一种方式构建该项目。主项目将生成一个程序集(我将其称为主项目)。所有其他项目的程序集都是辅助项目,它们将作为资源嵌入到主项目中。
Primary项目中的SourceCodeForExceptionHelper类负责使用PDB文件获取每个遇到的异常的原始源代码。为此,我使用了here描述的方法。它在我单独的概念验证项目中工作正常。但是当我试图将该类移动到真实的解决方案中时,我遇到了一个问题:IMetaDataDispenser.OpenScope方法需要对程序集文件路径的非空引用。当然,我没有对任何二级程序集的这样的引用(因为它们的文件被嵌入在主目录中)。由于这个原因,我不能创建一个类型为ISymbolReader的对象并读取源代码。我如何解决这个问题?顺便说一下,这个问题甚至更糟,因为我们只嵌入了辅助程序集,而没有它们的PDB文件(尽管我们会在必要时这样做)。
提前感谢您的任何帮助和建议!

6ju8rftf

6ju8rftf1#

我不认为你可以使用.NET Framework内置函数来实现这一点,因为它们依赖于物理文件。然而,有一个使用Mono Cecil库的解决方案,因为它有一个重载,它将Stream作为输入,而不是其符号读取器的文件路径。
下面是一个名为“Testpdb”的控制台应用程序的示例,它将其IL代码转储到控制台,包括PDB信息:

using System;
using System.IO;
using Mono.Cecil;
using Mono.Cecil.Cil;
using Mono.Cecil.Pdb;

namespace TestPdb
{
    class Program
    {
        static void Main(string[] args)
        {
            // we use a Stream for the assembly
            AssemblyDefinition asm;
            using (FileStream asmStream = new FileStream("testpdb.exe", FileMode.Open, FileAccess.Read, FileShare.Read))
            {
                asm = AssemblyDefinition.ReadAssembly(asmStream);
            }

            // we use a Stream for the PDB
            using (FileStream symbolStream = new FileStream("testpdb.pdb", FileMode.Open, FileAccess.Read, FileShare.Read))
            {
                asm.MainModule.ReadSymbols(new PdbReaderProvider().GetSymbolReader(asm.MainModule, symbolStream));
            }

            TypeDefinition type = asm.MainModule.GetType("TestPdb.Program");

            foreach (MethodDefinition method in type.Methods)
            {
                Console.WriteLine("Method:" + method.Name);
                foreach (Instruction ins in method.Body.Instructions)
                {
                    Console.WriteLine(" " + ins);
                    if (ins.SequencePoint != null)
                    {
                        Console.WriteLine("  Url:" + ins.SequencePoint.Document.Url);
                        // see http://blogs.msdn.com/b/jmstall/archive/2005/06/19/feefee-sequencepoints.aspx
                        if (ins.SequencePoint.StartLine != 0xFEEFEE)
                        {
                            Console.WriteLine("  StartLine:" + ins.SequencePoint.StartLine + " StartColumn:" + ins.SequencePoint.StartColumn);
                            Console.WriteLine("  EndLine:" + ins.SequencePoint.EndLine + " EndColumn:" + ins.SequencePoint.EndColumn);
                        }
                        // etc...
                    }
                }
            }   
        }
    }
}

注意:由于您只需要从PDB读取,因此可以重新编译定义READ_ONLY条件符号的塞西尔库以保存一些字节。您还可以将Cecil源代码直接嵌入到程序集中,而无需使用.DLL版本。

相关问题