.. _target CMake Build System Overview: Overview and Architecture ========================= CMake system --------------------------- All of the shared CMake logic lives in ``Scripts/CMake``: .. list-table:: :header-rows: 1 :widths: 25 75 * - File - Purpose * - ``OTProject.cmake`` - The public API. It provides every ``ot_*`` function a project calls, and including it also applies the global MSVC compile flags, the common definitions and the default runtime library. * - ``OTEnvironment.cmake`` - Reads the OpenTwin environment variables that ``SetupEnvironment.bat`` sets and exposes them as CMake variables and path helpers. * - ``OTQt.cmake`` - Finds Qt 6 and defines the imported ``Qt6::*`` targets used by the ``Qt*`` dependency tokens. - Instead of using ``find_package()`` we directly read set environment for our ``Qt`` packges. * - ``OTPresets.json`` - The shared ``CMakePresets`` definition that every project includes, so the configurations (such as ``windows-debug`` and ``windows-release``) are defined in one place. A project finds the meta system through the ``OT_CMAKE_DIR`` environment variable, which points at ``Scripts/CMake``. ``SetupEnvironment.bat`` exports it together with the per project root variables (``OT__ROOT``) and the third party paths. .. note:: The System depends on the set environment variable ``OT_CMAKE_DIR``, which is defined in the ``SetupEnvironment.bat`` of OpenTwin. If unset, CMake wont be able to find all the CMake scripts defined in ``/Scripts/CMake`` as well as the presets. The two target model: ``_core`` plus the final target ----------------------------------------------------- Every project builds as two CMake targets. One is the core object library ``_core``. It compiles everything under ``src/`` and ``include/`` and carries the compile flags, preprocessor definitions, include directories, the runtime library and the export macro. The other is the final target. ``ot_finalize_lib`` builds the DLL from those core objects, while ``ot_finalize_bin`` builds an executable. Either one picks up the link directories and link libraries. .. code-block:: text ot_initialize_lib(Foo ...) -> creates OBJECT lib Foo_core ot_add_dependency(Foo ...) -> records tokens on Foo_core ot_finalize_lib(Foo) -> creates SHARED lib Foo (from Foo_core objects) ot_add_test(Foo) -> adds tests/ -> Foo_tests exe (reuses Foo_core objects) Expected project layout ------------------------ The initialize functions discover sources by convention, so a project needs this layout: .. code-block:: text / CMakeLists.txt # the project definition (see next page) include/ # *.h / *.hpp src/ # *.cpp / *.c tests/ # optional unit tests CMakeLists.txt src/ # *.cpp for the test executable Sources are found with ``file(GLOB_RECURSE ... CONFIGURE_DEPENDS)``, so adding or removing a file under ``src/`` or ``include/`` is picked up on the next build. You do not list files by hand. OTProject.cmake --------------------- Including ``OTProject.cmake`` applies the OpenTwin standard MSVC setup to every target. The main pieces are: * C++20 with compiler extensions disabled. * Flags such as ``/permissive-``, ``/Zc:__cplusplus``, ``/Zc:preprocessor``, ``/EHsc``, ``/MP`` for parallel compilation, and external header warning suppression. * The dynamic CRT by default: ``/MD`` in Release, ``/MDd`` in Debug. A target overrides this only when it has to. * Common definitions everywhere (``WIN32``, ``_WIN32``, ``_WINDOWS``, ``UNICODE``, ``_UNICODE``), plus a few Debug only ones. * ``/OPT:REF`` and ``/OPT:ICF`` for Release linking. * ``_DEBUG`` in Debug and ``NDEBUG`` in Release on the core (Python targets differ, see below). .. note:: The overal compiler settings will be adjusted in the future. E.g. ``/W3`` (Warning level 3) will be added globally. .. _target CMake Runtime Library: Runtime library and the Debug/Release mapping --------------------------------------------- There are three runtime profiles. You pick one by choosing the initialize function, plus an optional helper: .. list-table:: :header-rows: 1 :widths: 44 28 28 * - Profile - Debug - Release * - Default (libraries and most binaries) - ``/MDd`` + ``_DEBUG`` - ``/MD`` + ``NDEBUG`` * - ``ot_set_runtime_static_release`` (tools) - ``/MDd`` - ``/MT`` (static) * - ``ot_initialize_bin_python`` (subprocess) - ``/MD`` + ``_RELEASEDEBUG`` - ``/MD`` + ``NDEBUG`` The default profile uses the dynamic CRT in both configurations. All libraries and services use it, and so do binaries that run next to the OpenTwin DLLs. Command line tools that ship as a single executable use ``ot_set_runtime_static_release``, so Release links the static CRT while Debug stays on ``/MDd`` for normal debugging. Python only ships a release build with debug information, so a binary that embeds Python always uses the release runtime, even in Debug. ``ot_initialize_bin_python`` handles that distinction: it keeps ``/MD`` in every configuration and, in Debug, defines ``_RELEASEDEBUG`` instead of ``_DEBUG`` so the service's debug code paths are still active. Configurations and presets -------------------------- Configurations come from ``OTPresets.json`` through each project's ``CMakePresets.json``. The Visual Studio generator is multi config, so one generated ``.vcxproj`` contains ``Debug``, ``Release``, ``RelWithDebInfo`` and ``MinSizeRel``. Day to day you mostly use ``windows-debug`` and ``windows-release``.