diff --git a/README-CMake.md b/README-CMake.md new file mode 100644 index 000000000..2fa99ef5f --- /dev/null +++ b/README-CMake.md @@ -0,0 +1,279 @@ +# Z3's CMake build system + +[CMake](https://cmake.org/) is a "meta build system" that reads a description +of the project written in the ``CMakeLists.txt`` files and emits a build +system for that project of your choice using one of CMake's "generators". +This allows CMake to support many different platforms and build tools. +You can run ``cmake --help`` to see the list of supported "generators" +on your platform. Example generators include "UNIX Makfiles" and "Visual Studio +12 2013". + +## Getting started + +The general workflow when using CMake is the following + +1. Create a new build directory +2. Configure the project +3. Generate the build system +4. Invoke the build system to build the project + +To perform steps 2 and 3 you can choose from three different tools + +* cmake +* ccmake +* cmake-gui + +``cmake`` is a command line tool and is what you should use if you are +writing a script to build Z3. This tool performs steps 1 and 2 in one +go without user interaction. The ``ccmake`` and ``cmake-gui`` tools are +more interactive and allow you to change various options. In both these +tools the basic steps to follow are: + +1. Configure. +2. Change any options you wish. Everytime you change a set of options + You should configure again. This may cause new options to appear +3. Generate. + +For information see https://cmake.org/runningcmake/ + +Note when invoking CMake you give it the path to the source directory. +This is the top-level directory in the Z3 repository containing a +``CMakeLists.txt``. That file should contain the line ``project(Z3 C CXX)``. +If you give it a path deeper into the Z3 repository (e.g. the ``src`` directory) +the configure step will fail. + +What follows is a brief walk through of how to build Z3 using various generators. + +### Unix Makefiles + +Run the following in the top level directory of the Z3 repository. + +``` +mkdir build +cd build +cmake -G "Unix Makefiles" ../ +make -j4 +``` + +Note that on some platforms "Unix Makesfiles" is the default generator so on those +platforms you don't need to pass ``-G "Unix Makefiles"`` command line option to +``cmake``. + +Note there is nothing special about the ``build`` directory name here. You can call +it whatever you like. + +Note the "Unix Makefile" generator is a "single" configuration generator which +means you pick the build type (e.g. ``Debug``, ``Release``) when you invoke CMake. +You can set the build type by passing it to the ``cmake`` invocation like so: + +``` +cmake -G "Unix Makefiles" -DCMAKE_BUILD_TYPE=Release ../ +``` + +See the section on "Build Types" for the different CMake build types. + +If you wish to use a different compiler set the CXX and CC environment variables +passed to cmake. This must be done at the very first invocation to ``cmake`` +in the build directory because once configuration has happened the compiler +is fixed. If you want to use a different compiler to the one you have already +configured you either need to make a new build directory or delete the contents +of the current build directory and start again. + +For example to use clang the ``cmake `` line would be + +``` +CC=clang CXX=clang++ cmake ../ +``` + +Note that CMake build will detect the target architecture that compiler is set up +to build for and the generated build system will build for that architecture. +If there is a way to tell your compiler to build for a different architecture via +compiler flags then you can set the ``CFLAGS`` and ``CXXFLAGS`` environment variables +to have the build target that architecture. + +For example if you are on a x86_64 machine and you want to do a 32-bit build and have +a multilib version of GCC you can run ``cmake`` like this + +``` +CFLAGS="-m32" CXXFLAGS="-m32" CC=gcc CXX=g++ cmake ../ +``` + +Note like with the ``CC`` and ``CXX`` flags this must be done on the very first invocation +to CMake in the build directory. + +### Ninja + +[Ninja](https://ninja-build.org/) is a simple build system that is built for speed. +It can be significantly faster than "UNIX Makefile"s because it is not a recursive +build system and thus doesn't create a new process everytime it traverses into a directory. +Ninja is particularly appropriate if you want fast incremental building. + +Basic usage is as follows: + +``` +mkdir build +cd build +cmake -G "Ninja" ../ +ninja +``` + +Note the discussion of the ``CC``, ``CXX``, ``CFLAGS`` and ``CXXFLAGS`` for "Unix Makefiles" +also applies here. + +Note also that like the "Unix Makefiles" generator, the "Ninja" generator is a single configuration +generator so you pick the build type when you invoke ``cmake`` by passing ``CMAKE_BUILD_TYPE=`` +to ``cmake``. See the section on "Build Types". + +Note that Ninja runs in parallel by default. Use the ``-j`` flag to change this. + +### Visual Studio + +For the Visual Studio generators you need to know which version of Visual Studio you wish +to use and also what architecture you want to build for. + +We'll use the ``cmake-gui`` here as it is easier to pick the right generator but this can +be scripted if need be. + +Here are the basic steps: + +1. Create an empty build directory +2. Start the cmake-gui program +3. Set "where is the source code" to the root of the Z3 project repository. You can do this by pressing + the "Browse Source..." button and picking the directory. +4. Set "where to build the binaries" to the empty build directory you just created. You can do this + by pressing the "Browse build..." button and picking the directory. +5. Press the "Configure" button +6. A window will appear asking you to pick the generator to use. Pick the + generator that matches the version of Visual Studio you are using. Note also + that some of the generator names contain ``Win64`` (e.g. ``Visual Studio 12 + 2013 Win64``) this indicates a x86 64-bit build. Generator names without + this (e.g. ``Visual Studio 12 2013``) are x86 32-bit build. +7. Press the "Finish" button and wait for CMake to finish it's first configure. +8. A set of configuration options will appear which will affect various aspects of the build. + Change them as you desire. If you change a set of options press the "Configure" + again. Additional options may appear when you do this. +9. When you have finished changing configuration options press the "Generate" button. +10. When generation is done close cmake-gui. +11. In the build directory open the generated ``Z3.sln`` solution file created by CMake with + Visual Studio. +12. In Visual Studio pick the build type (e.g. ``Debug``, ``Release``) you want. +13. Click "BUILD > Build Solution". + +Note that unlike the "Unix Makefile" and "Ninja" generators the Visual Studio generators +are multi-configuration generators which means you don't set the build type when invoking +CMake. Instead you set the build type inside Visual Studio. See the "Build Type" section +for more information. + +## Build Types + +Several build types are supported. + +* Release +* Debug +* RelWithDebInfo +* MinSizeRel + +For the single configuration generators (e.g. "Unix Makefile" and "Ninja") you set the +build type when invoking ``cmake`` by passing ``-DCMAKE_BUILD_TYPE=`` where +```` is one of the build types specified above. + +For multi-configuration generators (e.g. Visual Studio) you don't set the build type +when invoking CMake and instead set the build type within Visual Studio itself. + +## Useful options + +The following useful options can be passed to CMake whilst configuring. + +* ``CMAKE_BUILD_TYPE`` - STRING. The build type to use. Only relevant for single configuration generators (e.g. "Unix Makefile" and "Ninja"). +* ``CMAKE_INSTALL_PREFIX`` - STRING. The install prefix to use (e.g. ``/usr/local/``) +* ``ENABLE_TRACING`` - BOOL. If set to ``TRUE`` enable tracing, if set to ``FALSE`` disable tracing. +* ``BUILD_LIBZ3_SHARED`` - BOOL. If set to ``TRUE`` build libz3 as a shared library otherwise build as a static library. +* ``ENABLE_EXAMPLE_TARGETS`` - BOOL. If set to ``TRUE`` add the build targets for building the API examples. +* ``USE_OPENMP`` - BOOL. If set to ``TRUE`` and OpenMP support is detected build with OpenMP support. +* ``USE_LIB_GMP`` - BOOL. If set to ``TRUE`` use the GNU multiple precision library. If set to ``FALSE`` use an internal implementation. + +On the command line these can be passed to ``cmake`` using the ``-D`` option. In ``ccmake`` and ``cmake-gui`` these can be set in the user interface. + +## Developer/packager notes + +These notes are help developers and packagers of Z3. + +### Install/Uninstall + +Install and uninstall targets are supported. Use ``CMAKE_INSTALL_PREFIX`` to set the install +prefix. + +To install + +``` +make install +``` + +To uninstall + +``` +make uninstall +``` + +Note that ``DESTDIR`` is supported for [staged installs](https://www.gnu.org/prep/standards/html_node/DESTDIR.html). + +To install + +``` +mkdir staged +make install DESTDIR=/full/path/to/staged/ +``` + +to uninstall + +``` +make uninstall DESTDIR=/full/path/to/staged +``` + +The above also works for Ninja but ``DESTDIR`` must be an environment variable instead. + +### Examining invoked commands + +If you are using the "UNIX Makefiles" generator and want to see exactly the commands that are +being run you can pass ``VERBOSE=1`` to make. + +``` +make VERBOSE=1 +``` + +If you are using Ninja you can use the ``-v`` flag. + +### Additional targets + +To see the list of targets run + +``` +make help +``` + +There are a few special targets: + +* ``clean`` all the built targets in the current directory and below +* ``edit_cache`` will invoke one of the CMake tools (depending on which is available) to let you change configuration options. +* ``rebuild_cache`` will reinvoke ``cmake`` for the project. + +### Setting build type specific flags + +The build system supports single configuration and multi-configuration generators. This means +it is not possible to know the build type at configure time and therefore ``${CMAKE_BUILD_TYPE}`` +should not be conditionally used to set compiler flags or definitions. Instead you should use Generator expressions which are evaluated by the generator. + +For example + +``` +$<$:Z3DEBUG> +``` + +If the build type at build time is ``Debug`` this evaluates to ``Z3DEBUG`` but evaluates to nothing for all other configurations. You can see examples of this in the ``CMakeLists.txt`` files. + +### File-globbing + +It is tempting use file-globbing in ``CMakeLists.txt`` to find a set for files matching a pattern and +use them as the sources to build a target. This however is a bad idea because it prevents CMake from knowing when it needs to rerun itself. This is why source file names are explicitly listed in the ``CMakeLists.txt`` so that when changes are made the source files used to build a target automatically triggers a rerun of CMake. + +Long story short. Don't use file globbing.