-
Notifications
You must be signed in to change notification settings - Fork 112
Building and developing engine without docker
https://github.com/beyond-all-reason/spring/tree/BAR105/docker-build/scripts is the source of truth and best reference to figure out how to invoke and configure things.
It's arguable that compilation of spring for both Linux and Windows is just easier from Linux. So, we will only describe compilation on Linux.
- Windows: There is WSL and it works great.
- Linux: To not have to install all the dev dependencies, compilers etc in the base system, and for compatibility, you can use distrobox and develop there. It's fine to do it without that, but it's just very convenient.
You should use Debian based system to compile spring. For Linux stable builds, Debian Stable is fine, for anything else, Debian Unstable can be also fine, you might also just try some flavor of Ubunut, e.g 20.04, 22.04 etc.
Compilers, basic utilities, and helpful for developing:
sudo apt-get install -y cmake g++ ccache ninja-build mingw-w64 clang lld git clangd socat python3-pip
sudo pip install compdb
Spring engine dependencies
sudo apt-get install -y doxygen libsdl2-dev default-jdk libdevil-dev valgrind libcurl4-openssl-dev \
p7zip-full libopenal-dev libogg-dev libvorbis-dev libunwind-dev libfreetype-dev libglew-dev \
libminizip-dev libfontconfig-dev libjsoncpp-dev liblua5.1-0-dev libzip-dev libboost-test-dev
git clone https://github.com/beyond-all-reason/spring.git
cd spring
git submodule update --init --recursive
# for windows compilation
git clone https://github.com/beyond-all-reason/mingwlibs64.git mingwlibs64
# for linux static build
git clone https://github.com/beyond-all-reason/spring-static-libs.git -b 18.04 spring-static-libs
This part is the most annoying: configuring build is done using cmake, and the command lines are quite large.
First, you should have a few toolchains configured. Toolchains select the compiler and target operating system. You can store them in the toolchain
directory in the spring repo. Linux toolchains use lld
linker as it's much faster.
toolchain/clang_x86_64-pc-linux-gnu.cmake
:
SET(CMAKE_SYSTEM_NAME Linux)
SET(CMAKE_C_COMPILER "clang")
SET(CMAKE_CXX_COMPILER "clang++")
SET(CMAKE_EXE_LINKER_FLAGS_INIT "-fuse-ld=lld")
SET(CMAKE_MODULE_LINKER_FLAGS_INIT "-fuse-ld=lld")
SET(CMAKE_SHARED_LINKER_FLAGS_INIT "-fuse-ld=lld")
toolchain/gcc_x86_64-pc-linux-gnu.cmake
:
SET(CMAKE_SYSTEM_NAME Linux)
SET(CMAKE_C_COMPILER "gcc")
SET(CMAKE_CXX_COMPILER "g++")
SET(CMAKE_EXE_LINKER_FLAGS_INIT "-fuse-ld=lld")
SET(CMAKE_MODULE_LINKER_FLAGS_INIT "-fuse-ld=lld")
SET(CMAKE_SHARED_LINKER_FLAGS_INIT "-fuse-ld=lld")
toolchain/gcc_x86_64-pc-windows-gnu.cmake
:
SET(CMAKE_SYSTEM_NAME Windows)
SET(CMAKE_C_COMPILER "x86_64-w64-mingw32-gcc-posix")
SET(CMAKE_CXX_COMPILER "x86_64-w64-mingw32-g++-posix")
SET(CMAKE_RC_COMPILER "x86_64-w64-mingw32-windres")
SET(WINDRES_BIN "x86_64-w64-mingw32-windres")
SET(CMAKE_DLLTOOL "x86_64-w64-mingw32-dlltool")
SET(DLLTOOL "x86_64-w64-mingw32-dlltool")
With cmake we are building outside of the source, so create directory like builddir-win
, or builddir-dbg
and inside them we can run cmake invocations. There are plenty of possible configuration so we will just list a bunch that can be used as a starting point.
In all of them:
- We use Ninja generator, as Ninja is the quickest to actually execute the build process, scan for changes etc.
- Using ccache to make next compilation quicker
- Install dir is simply
install
, so that after configuing build with cmake, you can just runninja && ninja install
and get all the files ready for usage in theinstall
directory in builddir.
Basic release with debug info, shared libraries Linux build with GCC:
cmake \
-DCMAKE_TOOLCHAIN_FILE="../toolchain/gcc_x86_64-pc-linux-gnu.cmake" \
-DCMAKE_CXX_COMPILER_LAUNCHER=ccache \
-DCMAKE_CXX_FLAGS_RELWITHDEBINFO="-O3 -g -DNDEBUG -fdiagnostics-color=always" \
-DCMAKE_C_FLAGS_RELWITHDEBINFO="-O3 -g -DNDEBUG -fdiagnostics-color=always" \
-DCMAKE_BUILD_TYPE=RELWITHDEBINFO \
-DAI_TYPES=NATIVE \
-DINSTALL_PORTABLE=ON \
-DCMAKE_USE_RELATIVE_PATHS:BOOL=1
-DBINDIR:PATH=./ \
-DLIBDIR:PATH=./ \
-DDATADIR:PATH=./ \
-DCMAKE_INSTALL_PREFIX=install \
-G Ninja \
..
Fast unoptimized debug Linux shared libraries build with Clang and generation of compile_commands.json
file.
cmake \
-DCMAKE_TOOLCHAIN_FILE="../toolchain/clang_x86_64-pc-linux-gnu.cmake" \
-DCMAKE_CXX_COMPILER_LAUNCHER=ccache \
-DCMAKE_CXX_FLAGS_DEBUG="-O1 -g -fdiagnostics-color=always" \
-DCMAKE_C_FLAGS_DEBUG="-O1 -g -fdiagnostics-color=always" \
-DCMAKE_BUILD_TYPE=DEBUG \
-DDEBUG_MAX_WARNINGS=OFF \
-DAI_TYPES=NATIVE \
-DINSTALL_PORTABLE=ON \
-DCMAKE_USE_RELATIVE_PATHS:BOOL=1 \
-DBINDIR:PATH=./ \
-DLIBDIR:PATH=./ \
-DDATADIR:PATH=./ \
-DCMAKE_INSTALL_PREFIX=install \
-DCMAKE_EXPORT_COMPILE_COMMANDS=ON \
-G Ninja \
..
Windows release with minimal line debug info static cross compilation with mingw64:
cmake \
-DCMAKE_TOOLCHAIN_FILE="../toolchain/gcc_x86_64-pc-windows-gnu.cmake" \
-DCMAKE_CXX_COMPILER_LAUNCHER=ccache \
-DCMAKE_BUILD_TYPE=RELWITHDEBINFO \
-DCMAKE_CXX_FLAGS_RELWITHDEBINFO="-O3 -g1 -DNDEBUG -fdiagnostics-color=always" \
-DCMAKE_C_FLAGS_RELWITHDEBINFO="-O3 -g1 -DNDEBUG -fdiagnostics-color=always" \
-DAI_TYPES=NATIVE \
-DINSTALL_PORTABLE=ON \
-DCMAKE_INSTALL_PREFIX:PATH=install \
-GNinja ..
The longest, most awful one, and the most brittle one: release with debug info static libraries Linux build with GCC:
cmake \
-DCMAKE_TOOLCHAIN_FILE="../toolchain/gcc_x86_64-pc-linux-gnu.cmake" \
-DCMAKE_CXX_COMPILER_LAUNCHER=ccache \
-DCMAKE_CXX_FLAGS_RELWITHDEBINFO="-O3 -g -DNDEBUG -fdiagnostics-color=always" \
-DCMAKE_C_FLAGS_RELWITHDEBINFO="-O3 -g -DNDEBUG -fdiagnostics-color=always" \
-DCMAKE_BUILD_TYPE=RELWITHDEBINFO \
-DAI_TYPES=NATIVE \
-DINSTALL_PORTABLE=ON \
-DBINDIR:PATH=./ \
-DLIBDIR:PATH=./ \
-DDATADIR:PATH=./ \
-DCMAKE_INSTALL_PREFIX=install \
-DPREFER_STATIC_LIBS:BOOL=1 \
-DCMAKE_USE_RELATIVE_PATHS:BOOL=1 \
-DOGG_INCLUDE_DIR:PATH=$(pwd)/../spring-static-libs/include \
-DOGG_LIBRARY=$(pwd)/../spring-static-libs/lib/libogg.a \
-DMINIZIP_INCLUDE_DIR:PATH=$(pwd)/../spring-static-libs/include \
-DMINIZIP_LIBRARY=$(pwd)/../spring-static-libs/lib/libminizip.a \
-DLZMA_LIBRARY=$(pwd)/../spring-static-libs/lib/liblzma.a \
-DLIBUUID_LIBRARY=$(pwd)/../spring-static-libs/lib/libuuid.a \
-DVORBIS_INCLUDE_DIR:PATH=$(pwd)/../spring-static-libs/include \
-DVORBISENC_LIBRARY=$(pwd)/../spring-static-libs/lib/libvorbisenc.a \
-DVORBISFILE_LIBRARY=$(pwd)/../spring-static-libs/lib/libvorbisfile.a \
-DVORBIS_LIBRARY=$(pwd)/../spring-static-libs/lib/libvorbis.a \
-DIL_IL_HEADER:PATH=$(pwd)/../spring-static-libs/include/IL/il.h \
-DIL_INCLUDE_DIR:PATH=$(pwd)/../spring-static-libs/include \
-DIL_IL_LIBRARY:PATH=$(pwd)/../spring-static-libs/lib/libIL.a \
-DIL_LIBRARIES:PATH=$(pwd)/../spring-static-libs/lib/libIL.a \
-DILU_LIBRARIES:PATH=$(pwd)/../spring-static-libs/lib/libILU.a \
-DGIF_LIBRARY:PATH=$(pwd)/../spring-static-libs/lib/libgif.a \
-DGIF_INCLUDE_DIR:PATH=$(pwd)/../spring-static-libs/include \
-DPNG_PNG_INCLUDE_DIR:PATH=$(pwd)/../spring-static-libs/include \
-DPNG_LIBRARY_RELEASE:PATH=$(pwd)/../spring-static-libs/lib/libpng.a \
-DJPEG_INCLUDE_DIR:PATH=$(pwd)/../spring-static-libs/include \
-DJPEG_LIBRARY:PATH=$(pwd)/../spring-static-libs/lib/libjpeg.a \
-DTIFF_INCLUDE_DIR:PATH=$(pwd)/../spring-static-libs/include \
-DTIFF_LIBRARY_RELEASE:PATH=$(pwd)/../spring-static-libs/lib/libtiff.a \
-DZLIB_INCLUDE_DIR:PATH=$(pwd)/../spring-static-libs/include \
-DZLIB_LIBRARY_RELEASE:PATH=$(pwd)/../spring-static-libs/lib/libz.a \
-DGLEW_INCLUDE_DIR:PATH=$(pwd)/../spring-static-libs/include \
-DGLEW_LIBRARIES:PATH=$(pwd)/../spring-static-libs/lib/libGLEW.a \
-DLIBUNWIND_INCLUDE_DIRS:PATH=$(pwd)/../spring-static-libs/include \
-DLIBUNWIND_LIBRARY:PATH=$(pwd)/../spring-static-libs/lib/libunwind.a \
-DCURL_INCLUDE_DIR:PATH=$(pwd)/../spring-static-libs/include \
-DCURL_LIBRARY:PATH="$(pwd)/../spring-static-libs/lib/libcurl.a;$(pwd)/../spring-static-libs/lib/libnghttp2.a" \
-DOPENSSL_INCLUDE_DIR:PATH=$(pwd)/../spring-static-libs/include \
-DOPENSSL_SSL_LIBRARY:PATH=$(pwd)/../spring-static-libs/lib/libssl.a \
-DOPENSSL_CRYPTO_LIBRARY:PATH=$(pwd)/../spring-static-libs/lib/libcrypto.a \
-DVORBIS_INCLUDE_DIR:PATH=$(pwd)/../spring-static-libs/include \
-DVORBISENC_LIBRARY:PATH=$(pwd)/../spring-static-libs/lib/libvorbisenc.a \
-DVORBISFILE_LIBRARY:PATH=$(pwd)/../spring-static-libs/lib/libvorbisfile.a \
-DVORBIS_LIBRARY:PATH=$(pwd)/../spring-static-libs/lib/libvorbis.a \
-DFREETYPE_LIBRARY:PATH="/usr/lib/x86_64-linux-gnu/libfreetype.a;/usr/lib/x86_64-linux-gnu/libbrotlidec.a;/usr/lib/x86_64-linux-gnu/libbrotlicommon.a" \
-G Ninja \
..
In the clang debug above, we also enabled creation of compile_commands.json
file that can be then used by IMHO the best C++ language server clangd. The main problem with cmake generated compilation databases is that it doesn't contain entries for header files. That can be fixed with compdb utility installed in the beggining. Running from top repo directory:
compdb -p builddir-clang/ list > compile_commands.json
Clangd will then just pick it up.
If running in distrobox, and with clangd in distrobox, you might need to override clangd invocation in the LSP supporting editor to something like: distrobox enter --no-tty {container_name} spring -- socat tcp-listen:{port},reuseaddr exec:clangd
When developing on Windows, with clangd on Linux, and WSL container mapped to L:
drive, the invocation might looks like this: wsl.exe socat tcp-listen:${port},reuseaddr exec:'clangd --path-mappings=L:/home=/home,L:/usr=/usr'
. Only drawback is that ofc #ifdef _WIN32
blocks won't have completion in such setup.
Example spring.sublime-project
for Sublime Text on Linux with LSP server pckages:
{
"folders":
[
{
"path": "."
}
],
"settings": {
"tab_size": 4,
"LSP": {
"clangd": {
"enabled": true,
"command": [
"distrobox",
"enter",
"--no-tty",
"spring",
"--",
"socat",
"tcp-listen:{port},reuseaddr",
"exec:clangd"
],
"tcp_port": 0,
"scopes": ["source.c", "source.c++"],
"syntaxes": [
"Packages/C++/C.sublime-syntax",
"Packages/C++/C++.sublime-syntax",
],
"languageId": "cpp",
}
}
}
}