2

我最近在处理来自二级库/二进制模块的资源,遇到了一个奇怪的错误。

我有两个本机 WinAPI 参考:

[DllImport("kernel32.dll", SetLastError = true)]
public extern static bool EnumResourceNames(IntPtr hModule, int lpszType, EnumResNameProc lpEnumFunc, IntPtr lParam);

[DllImport("kernel32.dll", SetLastError=true)]
public extern static IntPtr LoadLibraryEx(string lpFileName, IntPtr hFile, int dwFlags);

当我调用 LoadLibraryEx 时,我得到了 IntPtr 实例——这正是我所需要的:

IntPtr x = WinApi.LoadLibraryEx(@"D:\Software\Reflector\Reflector.exe",IntPtr.Zero,2);
Debug.WriteLine(x.ToInt32());

但是,当我尝试枚举图标资源(由 ID = 3 定义)时:

Debug.WriteLine(WinApi.EnumResourceNames(x, 3, new EnumResNameProc(ListCallback), IntPtr.Zero));
Debug.WriteLine(Marshal.GetLastWin32Error());

我收到此错误代码(由 GetLastError 返回):

-532462766

据我所知,这通常意味着存在未知错误,但我只是好奇 - 从可执行文件中列出资源可能有什么问题?

4

2 回答 2

4

-532462766 == 0xe0434352。最后三个十六进制对拼写为“CCR”,这是 Microsoft 程序员用来尝试提出易于识别的异常代码的常用技巧。确切的含义非常神秘,除了它通常与托管代码相关联并且在通常不会产生有意义的托管异常的子系统中看似非常低级之外。

这个神秘异常有一个很好的候选原因,您的 EnumResources pinvoke 声明是错误的。第二个参数是 IntPtr,而不是 int。这有可能在 64 位操作系统上运行 kaboom。

如果您弄清楚 CCR 的含义,请回帖。


using System;
using System.ComponentModel;
using System.Runtime.InteropServices;
using System.Diagnostics;

class Program {
    static void Main(string[] args) {
        try {
            IntPtr module = LoadLibraryEx(@"C:\windows\system32\user32.dll", IntPtr.Zero, 2);
            if (module == IntPtr.Zero) throw new Win32Exception();
            if (!EnumResourceNames(module, (IntPtr)3, new EnumResNameProc(ListCallback), IntPtr.Zero))
                throw new Win32Exception();
        }
        catch (Win32Exception ex) {
            Console.WriteLine(ex.Message);
        }
        Console.ReadLine();
    }

    static bool ListCallback(IntPtr hModule, IntPtr type, IntPtr name, IntPtr lp) {
        long idorname = (long)name;
        if (idorname >> 16 == 0) Console.WriteLine("#{0}", idorname);
        else Console.WriteLine(Marshal.PtrToStringAnsi(name));
        return true;
    }

    public delegate bool EnumResNameProc(IntPtr hModule, IntPtr type, IntPtr name, IntPtr lp);
    [DllImport("kernel32.dll", SetLastError = true)]
    public extern static bool EnumResourceNames(IntPtr hModule, IntPtr type, EnumResNameProc lpEnumFunc, IntPtr lParam);
    [DllImport("kernel32.dll", SetLastError = true)]
    public extern static IntPtr LoadLibraryEx(string lpFileName, IntPtr hFile, int dwFlags);
}
于 2011-01-22T01:32:35.983 回答
0

Hans Passant 是正确的,但要详细说明错误消息,0xe0434352 是 .NET 异常的通用错误代码。如果你从 Visual Studio 调试器运行它,你会看到System.ArgumentException在 EnumResourceNames 尝试调用回调时引发了 a。错误信息是:

作为字符串传入的指针不得位于进程地址空间的底部 64K 中。

此异常被 EnumResourceNames 捕获并变成失败。正如 Hans 所展示的,解决方案是回调函数必须将第二个和第三个参数作为 IntPtr 而不是字符串。

于 2013-11-12T00:58:37.013 回答