linux下使用mingw编译NSIS-3.03

简述

最近在研究使用NSIS做安装包,语法不算复杂,插件也很多,中文资料也不少,还挺好用的。先后用NSIS做出了安装和卸载需要输入密码,通过自定义页面实现安装时候选择多个目录、安装的时候输入配置文件信息,禁止在某些平台或环境下安装,检测是否已经安装或正在运行等,稍后将把这些都放出来,做个记录。

有一个问题就是NSIS打包的文件可以直接使用7zip解压,安装过程做的事情就被跳过了。为了解决这个问题,我想修改一下NSIS的源码,来使得打包的程序无法使用7zip等软件解压。这里记录一下编译过程。

修改过的文件及编译好的文件下载nsis-3.03-src_修改.7z

准备工作

我是在linux下使用mingw32进行编译的,所以先要安装mingw32

然后是下载zlib库,我是在这里下载的https://jaist.dl.sourceforge.net/project/mingw/MinGW/Extension/zlib/zlib-1.2.3-1-mingw32/libz-1.2.3-1-mingw32-dev.tar.gz,这个说不定哪天就过期了,这是项目的页面https://sourceforge.net/projects/mingw/files/MinGW/Extension/zlib/

下载之后解压,将其中lib目录下的文件拷贝到/usr/lib/gcc/i686-w64-mingw32/6.3-win32/lib目录下,将include目录下的文件拷贝到/usr/lib/gcc/i686-w64-mingw32/6.3-win32/include目录下。

然后是下载NSIS-3.03的源码,地址在这里[https://jaist.dl.sourceforge.net/project/nsis/NSIS 3/3.03/nsis-3.03-src.tar.bz2](https://jaist.dl.sourceforge.net/project/nsis/NSIS 3/3.03/nsis-3.03-src.tar.bz2),下载之后解压即可。

因为NSIS使用scons进行构建,所以还需要安装python2.7scons工具。

编译过程

关于NSIS的编译,在这里Appendix G: Building NSIS有部分介绍。

准备工作做好后,使用下面命令进行构建。

scons SKIPSTUBS=all XGCC_W32_PREFIX=i686-w64-mingw32-    SKIPPLUGINS=all SKIPUTILS=all SKIPMISC=all NSIS_CONFIG_CONST_DATA_PATH=no PREFIX=./build

如果没有问题的话,正常会构建成功。但是会发现一个问题,build/urelease/makensis下面没有生成makensis.exe文件,只有一个makensis文件,而且使用readelf查看,这是一个ELF文件,而不是PE格式文件。
使用PETool查看结果如下:

这个错误的原因在这里:

根据这个报错,查看了Sconstruct文件后,找到错误的原因,是因为其中一个环境变量没有设置对。
打开nsis-3.03-src/SCons/Config/gnu文件,找到364行,在下面添加一行内容makensis_env.Replace(CXX = stub_env['CXX'])

然后重新执行构建命令。

解决完g++的问题后,继续编译遇到下面的问题。

下面有些修改其实是因为PLATFROM没有识别或者设置为win32的原因。

错误 Source/scriptpp.cpp:1054:93: error: call to non-constexpr function 'size_t wcslen(const wchar_t*)'

i686-w64-mingw32-g++ -o build/urelease/makensis/scriptpp.o -c -Wno-non-virtual-dtor -Wall -O2 -DNSISCALL= -D_UNICODE -DUNICODE -DMAKENSIS -D_WIN32_IE=0x0500 -Ibuild/urelease/config Source/scriptpp.cpp
In file included from /usr/share/mingw-w64/include/minwindef.h:163:0,
                 from /usr/share/mingw-w64/include/windef.h:8,
                 from /usr/share/mingw-w64/include/windows.h:69,
                 from Source/Platform.h:38,
                 from Source/scriptpp.cpp:17:
Source/scriptpp.cpp: In member function 'int CEXEBuild::pp_finalize(LineParser&)':
Source/scriptpp.cpp:1054:93: error: call to non-constexpr function 'size_t wcslen(const wchar_t*)'
   newcmd = (struct postbuild_cmd*) (new BYTE[FIELD_OFFSET(struct postbuild_cmd, cmd[_tcsclen(cmdstr)+1])]);

这里FIELD_OFFSET宏的参数应该是结构体的名称成员的名称,但这里明显不是,结合代码上下文来看,我把这里给修改为了

 newcmd = (struct postbuild_cmd*) (new BYTE[FIELD_OFFSET(struct postbuild_cmd, cmd)* (_tcsclen(cmdstr)+1)]);

没有仔细研究它的代码,所以这里就多分配一点了。

继续编译。

错误 Source/util.cpp:1072:11: error: jump to label 'finalwrite' [-fpermissive]

i686-w64-mingw32-g++ -o build/urelease/makensis/util.o -c -Wno-non-virtual-dtor -Wall -O2 -DNSISCALL= -D_UNICODE -DUNICODE -DMAKENSIS -D_WIN32_IE=0x0500 -Ibuild/urelease/config Source/util.cpp
Source/util.cpp: In function 'int RunChildProcessRedirected(LPCWSTR, LPCWSTR, bool)':
Source/util.cpp:1072:11: error: jump to label 'finalwrite' [-fpermissive]
         { finalwrite:
           ^~~~~~~~~~
Source/util.cpp:1085:25: note:   from here
         if (cchwb) goto finalwrite; // End of stream without a ending newline, write out the remaining data.
                         ^~~~~~~~~~
Source/util.cpp:1033:17: note:   skips initialization of 'DWORD i'
       for(DWORD i = 0; i < cbRead;)
                 ^
Source/util.cpp: At global scope:
Source/util.cpp:1159:15: warning: 'void* NSISRT_GetConsoleScreenHandle()' defined but not used [-Wunused-function]
 static HANDLE NSISRT_GetConsoleScreenHandle()
               ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~
scons: *** [build/urelease/makensis/util.o] Error 1
scons: building terminated because of errors.

看来一下这里的代码,比较繁杂,就不改了,直接使用下面的命令先编译通过

i686-w64-mingw32-g++ -o build/urelease/makensis/util.o -c -fpermissive  -Wno-non-virtual-dtor -Wall -O2 -DNSISCALL= -D_UNICODE -DUNICODE -DMAKENSIS -D_WIN32_IE=0x0500 -Ibuild/urelease/config Source/util.cpp

这里需要再次修改一下nsis-3.03-src/SCons/Config/gnu文件,在119行下添加一行makensis_env.Append(CXXFLAGS = ['-fpermissive'])

修改直接继续编译。

错误 build/urelease/makensis/crc32.o: file not recognized: File format not recognized

解决上面的错误之后,编译就没有问题了,一直到最后的链接这一步,又有点小问题了。

gcc -o build/urelease/makensis/crc32.o -c -Wall -O2 -DNSISCALL= -D_UNICODE -DUNICODE -DMAKENSIS -D_WIN32_IE=0x0500 -Ibuild/urelease/config Source/crc32.c
i686-w64-mingw32-g++ -o build/urelease/makensis/makensis -Wl,-Map,build/urelease/makensis/makensis.map -s -pthread build/urelease/makensis/build.o build/urelease/makensis/clzma.o build/urelease/makensis/crc32.o build/urelease/makensis/DialogTemplate.o build/urelease/makensis/dirreader.o build/urelease/makensis/fileform.o build/urelease/makensis/growbuf.o build/urelease/makensis/icon.o build/urelease/makensis/lang.o build/urelease/makensis/lineparse.o build/urelease/makensis/makenssi.o build/urelease/makensis/manifest.o build/urelease/makensis/mmap.o build/urelease/makensis/Plugins.o build/urelease/makensis/ResourceEditor.o build/urelease/makensis/ResourceVersionInfo.o build/urelease/makensis/BinInterop.o build/urelease/makensis/script.o build/urelease/makensis/scriptpp.o build/urelease/makensis/ShConstants.o build/urelease/makensis/strlist.o build/urelease/makensis/tokens.o build/urelease/makensis/tstring.o build/urelease/makensis/utf.o build/urelease/makensis/util.o build/urelease/makensis/winchar.o build/urelease/makensis/writer.o build/urelease/makensis/bzip2/blocksort.o build/urelease/makensis/bzip2/bzlib.o build/urelease/makensis/bzip2/compress.o build/urelease/makensis/bzip2/huffman.o build/urelease/makensis/7zip/7zGuids.o build/urelease/makensis/7zip/7zip/Common/OutBuffer.o build/urelease/makensis/7zip/7zip/Common/StreamUtils.o build/urelease/makensis/7zip/7zip/Compress/LZ/LZInWindow.o build/urelease/makensis/7zip/7zip/Compress/LZMA/LZMAEncoder.o build/urelease/makensis/7zip/7zip/Compress/RangeCoder/RangeCoderBit.o build/urelease/makensis/7zip/Common/Alloc.o build/urelease/makensis/7zip/Common/CRC.o -lpthread -lz
build/urelease/makensis/crc32.o: file not recognized: File format not recognized
collect2: error: ld returned 1 exit status
scons: *** [build/urelease/makensis/makensis] Error 1
scons: building terminated because of errors.

看我选中的部分,这里编译crc32.c的时候使用的是gcc而不是mingw32-gcc,所以这里的crc32.o文件是不能被正常识别的。
因为这是编译makensis部分出现的问题,所以这里还是环境没有设置对。
这里再次修改nsis-3.03-src/SCons/Config/gnu文件,在之前改CXX变量的前面吧CC也改了。这里在一开始就应该想到两个都应该重新设置的。

修改之后继续编译。

错误 build/urelease/makensis/BinInterop.o:BinInterop.cpp:(.text+0xf5): undefined reference to 'GetFileVersionInfoSizeW@8' 等

这是最后链接的一点小问题

build/urelease/makensis/BinInterop.o:BinInterop.cpp:(.text+0xf5): undefined reference to `GetFileVersionInfoSizeW@8'
build/urelease/makensis/BinInterop.o:BinInterop.cpp:(.text+0x136): undefined reference to `GetFileVersionInfoW@16'
build/urelease/makensis/BinInterop.o:BinInterop.cpp:(.text+0x183): undefined reference to `VerQueryValueW@16'

这三个函数找不到,其实是没有链接versions.lib的原因,这里手动添加一下,执行链接即可。

这里也可以修改nsis-3.03-src/Source/SConscript文件,在libs中添加version即可。

这时候编译出来的makensis就是一个有效的PE文件了。

posted @ 2018-08-23 13:59  乌合之众  阅读(821)  评论(0编辑  收藏  举报
clear