Our buildsystem is based on cmake
.
We use a bunch of custom cmake modules and python scripts
to detect libraries and generate code.
The entry point of the build system is /CMakeLists.txt
.
From there, all buildsystem code of buildsystem/
is then used.
There is a /configure
script that wraps the cmake invocation for convenience.
The /Makefile
is another wrapper to invoke the build and tests
compiler-independently.
The buildsystem is pretty sophisticated because openage consists C++
code, generated C++ code, generated Cython code and Python code. All C++
parts are packed in libopenage
, the Python stuff is in the openage
python package. We generate code with the openage.codegen
Python
package.
Steps in building openage:
- generate
openage/config.py
andlibopenage/config.{h, cpp}
from their*.in
template files, which contain install prefix and version info - run
openage.codegen
(Python module) to generate C++ source files (recipe:codegen
) - generate
.pxd
Cython extension declaration files from annotated.h
files (recipe:pxdgen
) - build and link
libopenage.so
(recipe:openage
) - build Cython extension modules (generate cpp files and compile them, via
buildsystem.cythonize
) (recipe:cython
); those link against libopenage.
Additional recipes:
- see
make help
install
doc
(generate docs via Doxygen)test
(runs the various tests)- various cleaning recipes:
cleanelf
,cleancodegen
,cleancython
,cleaninsourcebuild
,cleanpxdgen
,cleanbuilddirs
,mrproper
,mrproperer
- various compliance checkers:
checkfast
,checkall
, ...
cmake reads and interprets all cmake modules and CMakeLists.txt
files, populating the build directory with Makefiles and generating config.py
, config.h
and config.cpp
. In addition, the codegen script is invoked to determine the list of files it will generate, and the list of (python) files it depends on.
The ./configure
cmake-wrapper script may be used to achieve a pleasant build experience. ./configure
automatically creates the build directory and symlinks it to bin
, which makes the root Makefile work.
For each compiler invocation (gcc -Og, clang -O3, etc...), ./configure
creates its own build directory. This allows you to quickly switch compilers and flags (e.g. via ./configure -c clang -O2
) without having to re-build all object files when switching back.
cmake
supports many backends: Project files for various IDEs and more.
By default, GNU make
is used to interpret the cmake-generated Makefiles at build time. It will detect changes in the relevant cmake
source files, and re-trigger a cmake execution if necessary (just your standard CMake features; pretty cool stuff).
The recipes codegen
, libopenage
, pxdgen
, cythonize
, compilepy
and inplacemodules
are then run as needed.
The Makefile
in the project root directory may be used to invoke GNU make
in the build directory (make -C bin/
would be the manual way of doing this).
At install time, the openage library, python modules, Cython extension
modules and game assets are installed to the prefix that was set at cmake
time (default: /usr/local
), or, if the DESTDIR
variable is present,
to $DESTDIR/$INSTALLPREFIX
. Note: The Python/Cython modules have their
separate installation prefix, dictated by the Python environment, and
manually settable via the --py-prefix
option to ./configure
.
Some more CMake
magic: the -rpath
is automagically removed from the
installed binaries (if you don't know what that is: Lucky you. CMake
makes sure you don't need to).
The cpp module, apart from verifying the compiler version and setting flags, provides functions for assembling binaries step-by-step.
declare_binary
begins the assembly of a new binary.add_sources
adds translation units (cpp files) to the binary). TheGENERATED
keyword is required for addingcodegen
-generated translation units.finalize_binary
prints information about the binary, and calls theadd_executable
oradd_library
cmake function, depending on its second argument. No more sources may be added after this point. If any generated sources were added to the executable, thecodegen
recipe is added as amake
dependency.
At build time, this will cause all relevant object files and binaries to be built/linked. At install time, it will install the libs and binaries to /bin and /lib, respectively.
Plain invocations of add_executable
require all translation units to be specified at once; this system is designed to allow each subfolder to have its own CMakeLists.txt
file, adding all sources specified in that folder to the main binary.
The python module is similar to the cpp module. It checks the Python interpreter and provides functions to declare Cython and Python modules.
add_cython_modules
adds Cython modules (i.e..pyx
or.py
files) that will be compiled. See the code for option docs.pxdgen
adds C++ headers for.pxd
file generation (see pyinterface).add_pxd
adds any additional.pxd
or.pxi
files.add_py_modules
adds pure-Python modules; note that you also must declare.py
files that were declared inadd_cython_modules
.
Provides the function codegen_run
, which
- at cmake-time, runs the openage.codegen python module to obtain a list of sources it will generate
- at build time, actually generates the sources and writes them to the source directory where necessary
It sets the variable CODEGEN_TARGET_TUS
, containing a list of created cpp
files. That variable is designed to be used in a call to add_sources(openage GENERATED ${CODEGEN_TARGET_TUS})
.
The python script automatically determines the python dependencies of the codegen script (via sys.modules
) and adds them as dependencies of the codegen
recipe. When the list of python dependencies, or output TUS, have changed, it automatically triggers a new cmake
run.
The generated .cpp
files are placed in the libopenage
folder, and all end in .gen.cpp
(or .gen.h
).
read the source, or nag somebody to write more docs on them! wheeee!