从C#使用时如何调试LoadLibraryEx

时间:2015-08-06 05:47:07

标签: c# pinvoke loadlibrary

我正在尝试使用LoadLibraryEx在C#中加载win32 dll。

它无法正常工作 - 我在Visual Studio中收到一条消息" vshost32.exe已停止工作"。没有例外或任何关于它为什么不起作用的线索。

我不相信这是一个依赖性问题,因为如果我改变了依赖项的搜索路径,我会收到一个消息框,说明"找不到xyz.dll"。

我想知道是否有办法可以找出它没有正确加载的原因。该程序停止工作:

IntPtr pDll = LoadLibraryEx(@"C:\Program Files\XXX\XXX.dll", IntPtr.Zero, flags);
[UnmanagedFunctionPointer(CallingConvention.StdCall)]
    private delegate void ImportResults();

    [DllImport("kernel32.dll", SetLastError = true)]
    public static extern IntPtr LoadLibraryEx(string dllToLoad, IntPtr hFile, LoadLibraryFlags flags);

    [DllImport("kernel32.dll", SetLastError = true)]
    public static extern IntPtr GetProcAddress(IntPtr hModule, string procedureName);

    [DllImport("kernel32.dll", SetLastError = true)]
    public static extern bool FreeLibrary(IntPtr hModule);

    [System.Flags]
    public enum LoadLibraryFlags : uint
    {
        DONT_RESOLVE_DLL_REFERENCES = 0x00000001,
        LOAD_IGNORE_CODE_AUTHZ_LEVEL = 0x00000010,
        LOAD_LIBRARY_AS_DATAFILE = 0x00000002,
        LOAD_LIBRARY_AS_DATAFILE_EXCLUSIVE = 0x00000040,
        LOAD_LIBRARY_AS_IMAGE_RESOURCE = 0x00000020,
        LOAD_WITH_ALTERED_SEARCH_PATH = 0x00000008,
        LOAD_LIBRARY_SEARCH_DLL_LOAD_DIR = 0x00000100,
        LOAD_LIBRARY_SEARCH_SYSTEM32 = 0x00000800,
        LOAD_LIBRARY_SEARCH_DEFAULT_DIRS = 0x00001000
    }

    public void Import()
    {
        LoadLibraryFlags flags = LoadLibraryFlags.LOAD_LIBRARY_SEARCH_DEFAULT_DIRS |
                                 LoadLibraryFlags.LOAD_LIBRARY_SEARCH_DLL_LOAD_DIR;

        IntPtr pDll = LoadLibraryEx(@"C:\Program Files\XXX\XXX.dll", IntPtr.Zero, flags);

        IntPtr pAddressOfFunctionToCall = GetProcAddress(pDll, "ImportResults");


        ImportResults import = (ImportResults)Marshal.GetDelegateForFunctionPointer(pAddressOfFunctionToCall,
                                                                               typeof(ImportResults));
        import();

        bool result = FreeLibrary(pDll);

    }

更新

我已经下载了windows sdk,并尝试使用cdb.exe来调试问题,使用此处描述的过程:http://blogs.msdn.com/b/junfeng/archive/2006/11/20/debugging-loadlibrary-failures.aspx

运行该实用程序时,我正在使用此命令行:

"C:\Program Files (x86)\Windows Kits\8.1\Debuggers\x86\cdb.exe" loadlib "C:\Program Files\XXX\XXX\XXX.DLL"

但是我收到了这个错误: Error

该文件肯定存在,所以我不确定我在这里做错了什么。

1 个答案:

答案 0 :(得分:1)

我安装了Windows调试工具:https://msdn.microsoft.com/en-US/windows/desktop/bg162891

然后我下载了Windows Symbol包:https://msdn.microsoft.com/en-us/windows/hardware/gg463028.aspx

然后设置一个环境变量来告诉调试器在哪里寻找符号:

_NT_SYMBOL_PATH = SRV*C:\dev\symbols*http://msdl.microsoft.com/download/symbols;C:\Symbols

(我将符号包安装到C:\ Symbols)

然后我启动了我的应用程序并使用以下命令附加调试器:

"C:\Program Files (x86)\Windows Kits\8.1\Debuggers\x86\cdb.exe" -pb -p <pid>

-pb意味着它不会在例外情况下中断 -p是要调试的线程的pid

由此我能够确定错误发生的位置。尽管如此,仍然无法理解为什么会发生这种情况。