如何从资源中获取版本信息?

时间:2012-12-18 21:32:49

标签: winapi visual-c++ resources

我在声明的资源中有版本信息:

100 VERSIONINFO
FILEVERSION 1,0,0,2
PRODUCTVERSION 1,0,0,2
FILEOS VOS_NT
FILETYPE VFT_APP
{
    BLOCK "StringFileInfo"
    {
        BLOCK "000004b0"
        {
            VALUE "FileDescription", "My application"
            VALUE "FileVersion", "1.0.0.2"
            VALUE "InternalName", "app.exe"
            VALUE "LegalCopyright", "Copyright ©  2012 by David."
            VALUE "OriginalFilename", "app.exe"
            VALUE "ProductName", "app"
            VALUE "ProductVersion", "1.0.0.2"
            VALUE "Assembly Version", "1.0.0.2"
        }
    }

    BLOCK "VarFileInfo"
    {
        VALUE "Translation", 0x0000 0x04B0
    }
}

我以这种方式获取版本信息:

HRSRC hResInfo;
HGLOBAL hResData;
LPCVOID pRes;
UINT uLen;
VS_FIXEDFILEINFO *lpFfi;

hResInfo = FindResource(hInst, MAKEINTRESOURCE(100), RT_VERSION);
MessageBox(0, "FindResource", 0,0);

hResData = LoadResource(hInst, hResInfo);
MessageBox(0, "LoadResource", 0,0);

pRes = LockResource(hResData);
MessageBox(0, "LockResource", 0,0);

VerQueryValue(pRes, "\\" ,(LPVOID*)&lpFfi, &uLen);
MessageBox(0, "VerQueryValue", 0,0);

FreeResource(hResData);

DWORD dwFileVersionMS = lpFfi->dwFileVersionMS;
DWORD dwFileVersionLS = lpFfi->dwFileVersionLS;

DWORD dwLeftMost     = HIWORD(dwFileVersionMS);
DWORD dwSecondLeft   = LOWORD(dwFileVersionMS);
DWORD dwSecondRight  = HIWORD(dwFileVersionLS);
DWORD dwRightMost    = LOWORD(dwFileVersionLS);

在函数VerQueryValue是一个错误,因为程序中断(带有文本“VerQueryValue”的MessageBox没有显示),Visual C ++向我显示以下消息:

  

ProxyCU.exe中0x77bf15a5的第一次机会异常:0xC0000005:   访问冲突写入位置0x00483192。

如何修复此代码?

问候,大卫

2 个答案:

答案 0 :(得分:12)

VerQueryValue()无法直接从原始资源访问版本信息。您必须在内存中复制资源,然后将该内存传递给VerQueryValue()。原因是因为VerQueryValue()旨在与GetFileVersionInfo()一起使用,这需要用户分配的可写内存块并在该内存中执行某些修正。访问VS_FIXEDFILEINFO结构不需要修复,但内存块必须仍然可写。您无法将原始资源直接传递给VerQueryValue(),因为它是只读内存。

请改为尝试:

HRSRC hResInfo;
DWORD dwSize;
HGLOBAL hResData;
LPVOID pRes, pResCopy;
UINT uLen;
VS_FIXEDFILEINFO *lpFfi;

hResInfo = FindResource(hInst, MAKEINTRESOURCE(100), RT_VERSION);
dwSize = SizeofResource(hInst, hResInfo);
hResData = LoadResource(hInst, hResInfo);
pRes = LockResource(hResData);
pResCopy = LocalAlloc(LMEM_FIXED, dwSize);
CopyMemory(pResCopy, pRes, dwSize);
FreeResource(hResData);

VerQueryValue(pResCopy, TEXT("\\"), (LPVOID*)&lpFfi, &uLen);

DWORD dwFileVersionMS = lpFfi->dwFileVersionMS;
DWORD dwFileVersionLS = lpFfi->dwFileVersionLS;

DWORD dwLeftMost     = HIWORD(dwFileVersionMS);
DWORD dwSecondLeft   = LOWORD(dwFileVersionMS);
DWORD dwSecondRight  = HIWORD(dwFileVersionLS);
DWORD dwRightMost    = LOWORD(dwFileVersionLS);

LocalFree(pResCopy);

更新:仅在您仅访问VS_FIXEDFILEINFO结构时才有效。如果您需要访问任何其他值,则必须使用GetFileVersionInfo()。 Per Raymond Chen的博客:

The first parameter to VerQueryValue really must be a buffer you obtained from GetFileVersionInfo

  

文档说明,VerQueryValue的第一个参数必须是GetFileVersionInfo函数返回的缓冲区。 GetFileVersionInfo返回的缓冲区是一个专门格式化的不透明数据块,以便VerQueryValue可以工作。你不应该查看那个缓冲区,你当然不能尝试“以其他方式获取数据”。因为如果这样做,VerQueryValue将在缓冲区中查找未按函数期望的方式格式化的内容。

答案 1 :(得分:0)

以上将在调试模式下创建堆损坏错误消息,例如"在释放"后在b753ed4处修改的免费堆块b753e70。很多年前有人在http://microsoft.public.win32.programmer.kernel.narkive.com/mqoHgVwM/verqueryvalue-bug发布了这个问题。它今天仍在变化。通过使dwSize足够大,例如将其乘以4,可以使消息消失。