5

我正在寻找在可执行文件的版本信息中更改内部版本号(VersionLS 的低阶)。因此,我应该阅读VS_VERSIONINFO结构,更改内部版本号,然后将其更新回 PE。

我正在使用此代码作为基础:https ://stackoverflow.com/a/7999813/1970843 。此代码非常适合更改VS_FIXEDFILEINFO数据,但它不会更改(也不会访问)StringFileInfo信息。

我很确定我应该在 VERSIONHEADER 打包记录中包含一些内容以添加 VS_VERSIONINFO 的 Children 条目,但我不知道具体该怎么做。这是我到目前为止所拥有的:

type
    StringStruc = Packed Record
        wLength: Word;
        wValueLength: Word;
        wType: Word;
        //szKey: ?;
        //Value: ?;
    End;

    StringTable = Packed Record
        wLength: Word;
        wValueLength: Word;
        wType: Word;
        szKey: Array[0..8] Of WideChar;
        Children: StringStruc;
    End;

    StringFileInfo = Packed Record
        wLength: Word;
        wValueLength: Word;
        wType: Word;
        szKey: Array[0..14] Of WideChar;   // 'STRINGFILEINFO'
        Children: StringTable;
    End;

    VERSIONHEADER = Packed Record
        wLength: Word;
        wValueLength: Word;
        wType: Word;
        szKey: Array[0..16] Of WideChar;   // 'VS_VERSION_INFO'
        Version: VS_FIXEDFILEINFO;
        Children: StringFileInfo;
    End;

...

var VersionHandle, VersionRes: THandle;
    VersionSize: Cardinal;
    Version: Array Of AnsiChar;
    Ver: ^VERSIONHEADER;
Begin
    VersionSize := GetFileVersionInfoSize(PChar(sExe), VersionHandle);

    SetLength(Version, VersionSize);
    Ver := Pointer(Version);
    GetFileVersionInfo(PChar(sExe), 0, VersionSize, Ver);

因此,信息似乎正确地出现在第一个 StringStruc 上。但是由于 szKey 和 Value 都不是固定大小,我不知道如何正确定义我的打包记录(甚至可能吗?)来获取这些值。我也遇到了数组问题......我该如何定义它们?我正在做的方式是,我只是在每个 Struc 上获得第一个孩子。请注意,我忽略了填充...可以吗?

任何帮助表示赞赏。我在这里所做的大部分工作都是通过反复试验,所以我真的不明白发生了什么。

PS:我还在做这个,所以我可能会经常更新这个帖子。

4

1 回答 1

1

感谢您的关注和帮助。我在这里找到了一个现成的解决方案。事实上,它就在我所链接的问题的评论中(我感到羞耻!)。

它基于Colin Wilson的库。它使用指针算法来提取和写入信息,因此是硬(也许是唯一的)方式。Jason Penny 还提供了一个关于如何使用该库的好例子:SetVersion。因为我使用的是 D7,所以我从这里(在 Resource Utilities 下)下载了 Colin Wilson 的库,但是这里提供了一个使用 UnicodeString 和更好的指针算法的更新版本。

这是我现在的实际实现:

uses ..., unitResourceVersionInfo, unitPEFile;

...

var VersionInfo: TVersionInfoResourceDetails;
    PEResModule: TPEResourceModule;
    VersionNumber: ULARGE_INTEGER;
    sVersion: String;
    I: Integer;
Begin
    PEResModule := TPEResourceModule.Create;
    Try
        PEResModule.LoadFromFile(sExe);

        For I := 0 To PEResModule.ResourceCount - 1 Do Begin
            If PEResModule.ResourceDetails[I] Is TVersionInfoResourceDetails Then Begin
                VersionInfo := (PEResModule.ResourceDetails[I] As TVersionInfoResourceDetails);
                Break;
            End;
        End;

        VersionNumber.LowPart := MakeLong(NewBuildNumber, HiWord(VersionInfo.FileVersion.LowPart));
        VersionNumber.HighPart := VersionInfo.FileVersion.HighPart;
        VersionInfo.FileVersion := VersionNumber;
        VersionInfo.ProductVersion := VersionNumber;

        VersionInfo.CodePage := $04e4;

        sVersion := Format('%d.%d.%d.%d', [HiWord(VersionInfo.FileVersion.HighPart), LoWord(VersionInfo.FileVersion.HighPart), HiWord(VersionInfo.FileVersion.LowPart), LoWord(VersionInfo.FileVersion.LowPart)]);
        VersionInfo.SetKeyValue('FileVersion', sVersion);

        PEResModule.SaveToFile(ChangeFileExt(sExe, '.exe2'));

    Finally
        FreeAndNil(PEResModule);
    End;
End;

代码页行是由于库中的一个错误(我相信是)。它不读取代码页(为 0),因此,当您保存回来时,它显示为 0。

于 2014-02-14T14:32:33.597 回答