CPP deployment using AppVeyor - MIPT-ILab/mipt-mips GitHub Wiki
Introduction
AppVeyor is a continuous intergration and deployment service for Windows developers. However, there is no good manual how to deploy a build of C++ project like MIPT-MIPS.
There are three major gamers in C++ development on Windows:
- Microsoft Visual Studio
- Minimalistic GNU on Windows (MinGW)
- Clang
Our appveyor.yml
supports build by Visual Studio and MinGW. In this manual I'll explain how it works, line-by-line.
Build with Visual Studio
Visual Studio support is tricky for our non-commercial project, as usually there are no licenses for the latest Visual Studio version. Thus, sometimes you have to work in a 'blind' mode. Below I'll try to cover the most tricky corner cases I faced.
Set up
First of all, you have to select MSVC image. We are using the latest one:
version: "{branch}-ci-{build}"
image: Visual Studio 2017
There are some differences between versions of Visual Studio, I'll explain below.
Then, you add Visual Studio environment to the matrix. Please note the path to Boost linkable libraries: these paths are not mentioned on AppVeyor manuals!
environment:
matrix:
- build: msvc
platform: x86
BOOST_ROOT: C:\Libraries\boost_1_64_0
BOOST_LIBRARYDIR: C:\Libraries\boost_1_64_0\lib32-msvc-14.1
LIBELF_INCLUDE: C:\Projects\mipt-mips\libelf\lib
- build: msvc
platform: x64
BOOST_ROOT: C:\Libraries\boost_1_64_0
BOOST_LIBRARYDIR: C:\Libraries\boost_1_64_0\lib64-msvc-14.1
LIBELF_INCLUDE: C:\Projects\mipt-mips\libelf\lib
Before build
After matrices are set, we may starting environment preparation:
# set variables for MSVC
- if "%build%"=="msvc" call "C:\Program Files (x86)\Microsoft Visual tudio\2017\Community\Common7\Tools\VsDevCmd.bat" -arch=%PLATFORM%
- set INCLUDE=%LIBELF_INCLUDE%;%BOOST_ROOT%;%INCLUDE%
- set PATH=%LIBELF_INCLUDE%;%PATH%
- set LIB=%BOOST_LIBRARYDIR%;%LIB%
INCLUDE
, PATH
, and LIB
are MSVC environment variables which usually do not interfere with MinGW, so we may set them unconditionally. As one may guess, they have following purposes:
INCLUDE
is a set of paths where MSVC will search for headersLIB
is a set of paths where MSVC will search for static libraries (e.g. Boost)PATH
should contain path to all*.dll
s we may use in runtime (for MIPT-MIPS it islibelf.dll
)
Please note that MSVC 2015 uses different command to set up instead of VsDevCmd.bat:
# set variables for MSVC
- if "%build%"=="msvc" call "%VS140COMNTOOLS%\..\..\VC\vcvarsall.bat" %PLATFORM%
Build
For MIPT-MIPS we use [CMake] to generate Visual Studio solution file:
build_script:
- cd C:\projects\MIPT-MIPS\simulator\build
- if "%build%"=="msvc" cmake .. -G "Visual Studio 15 Win64"
AppVeyor provides test_script
scenario as well, and we do testing here:
test_script:
- ctest -C Release --verbose
Build with MinGW
In short, MinGW is a GCC that uses Windows API. Because of that, there may be some differences between system calls. The most widespread error occurs because Windows distinguish 'binary' and 'non-binary' file opens. In 'non-binary' file mode, Windows will automatically translate \n
symbol to \r\n
, which is acceptable for text files but not for binaries. To prevent doing that, do the similar change in your C/C++ code:
- FILE* file_descr = fopen( elf_file_name.c_str(), "r");
+ FILE* file_descr = fopen( elf_file_name.c_str(), "rb");
You don't have to check any macroes, as b
character will be ignored by Linux.
Set up
MinGW is installed to the all of AppVeyor images, so we can stick with 2017:
version: "{branch}-ci-{build}"
image: Visual Studio 2017
Then, set up the matrix. We are going to use Msys2 environment provided by AppVeyor.
environment:
matrix:
- build: g++
platform: x86
MINGW_ROOT: C:\msys64\mingw32
BOOST_ROOT: C:\msys64\mingw32\include\boost
LIBELF_INCLUDE: C:\msys64\mingw32\include\libelf
MINGW_ARCH: i686
MSYSTEM: MINGW32
- build: g++
platform: x64
MINGW_ROOT: C:\msys64\mingw64
BOOST_ROOT: C:\msys64\mingw64\include\boost
LIBELF_INCLUDE: C:\msys64\mingw64\include\libelf
MINGW_ARCH: x86_64
MSYSTEM: MINGW64
Before build
By default, MinGW is not included to Windows' PATH
variable. Ok, it is not a problem for us:
- set PATH=%MINGW_ROOT%;%MINGW_ROOT%\bin;C:\msys64\usr\bin\;"C:\Program Files\LLVM\bin";%PATH%
Also, we need to download libraries. They are fetched by Msys2's package manager. Don't forget to use right target architecture name for downloading.
- bash -lc "pacman --needed --noconfirm -S mingw-w64-%MINGW_ARCH%-libelf"
- bash -lc "pacman --needed --noconfirm -S mingw-w64-%MINGW_ARCH%-boost"
As in Linux, MinGW uses CPATH
, C_INCLUDE_PATH
, and CPLUS_INCLUDE_PATH
variables to search for includes and libraries. It is worth to mention that environment variables are shared by Msys2 and Windows CMD, so you can set them in CMD and use in bash later.
- set CPATH=%LIBELF_INCLUDE%;%CPATH%
- set CPLUS_INCLUDE_PATH=%LIBELF_INCLUDE%;%BOOST_ROOT%;%CPLUS_INCLUDE_PATH%
Build
Build is run under the Msys2 bash, so we generate MSYS Makefiles with CMake:
build_script:
- cd C:\projects\MIPT-MIPS\simulator\build
- if not "%build%"=="msvc" cmake .. -G "MSYS Makefiles"
- if not "%build%"=="msvc" bash -lc "cd /c/projects/mipt-mips/simulator/build; make"
We use same scenario for testing:
test_script:
- ctest -C Release --verbose
Clang
WIP on Clang support in low priority as have clang builds in Linux and OSX on Travis. Additionally, supporting Clang for Windows is tricky as it may behave either MSVC-like compiler or GCC-like compiler