| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953 | CMake Tutorial**************.. only:: html   .. contents::The CMake tutorial provides a step-by-step guide that covers common buildsystem issues that CMake helps address. Seeing how various topics allwork together in an example project can be very helpful. The tutorialdocumentation and source code for examples can be found in the``Help/guide/tutorial`` directory of the CMake source code tree. Each step hasits own subdirectory containing code that may be used as a starting point. Thetutorial examples are progressive so that each step provides the completesolution for the previous step.A Basic Starting Point (Step 1)===============================The most basic project is an executable built from source code files.For simple projects, a three line ``CMakeLists.txt`` file is all that isrequired. This will be the starting point for our tutorial. Create a``CMakeLists.txt`` file in the ``Step1`` directory that looks like:.. code-block:: cmake  cmake_minimum_required(VERSION 3.10)  # set the project name  project(Tutorial)  # add the executable  add_executable(Tutorial tutorial.cxx)Note that this example uses lower case commands in the ``CMakeLists.txt`` file.Upper, lower, and mixed case commands are supported by CMake. The sourcecode for ``tutorial.cxx`` is provided in the ``Step1`` directory and can beused to compute the square root of a number.Adding a Version Number and Configured Header File--------------------------------------------------The first feature we will add is to provide our executable and project with aversion number. While we could do this exclusively in the source code, using``CMakeLists.txt`` provides more flexibility.First, modify the ``CMakeLists.txt`` file to use the :command:`project` commandto set the project name and version number... literalinclude:: Step2/CMakeLists.txt  :language: cmake  :end-before: # specify the C++ standardThen, configure a header file to pass the version number to the sourcecode:.. literalinclude:: Step2/CMakeLists.txt  :language: cmake  :start-after: # to the source code  :end-before: # add the executableSince the configured file will be written into the binary tree, wemust add that directory to the list of paths to search for includefiles. Add the following lines to the end of the ``CMakeLists.txt`` file:.. literalinclude:: Step2/CMakeLists.txt  :language: cmake  :start-after: # so that we will find TutorialConfig.hUsing your favorite editor, create ``TutorialConfig.h.in`` in the sourcedirectory with the following contents:.. literalinclude:: Step2/TutorialConfig.h.in  :language: cmakeWhen CMake configures this header file the values for``@Tutorial_VERSION_MAJOR@`` and ``@Tutorial_VERSION_MINOR@`` will bereplaced.Next modify ``tutorial.cxx`` to include the configured header file,``TutorialConfig.h``.Finally, let's print out the version number by updating ``tutorial.cxx`` asfollows:.. literalinclude:: Step2/tutorial.cxx  :language: c++  :start-after: {  :end-before: // convert input to doubleSpecify the C++ Standard-------------------------Next let's add some C++11 features to our project by replacing ``atof`` with``std::stod`` in ``tutorial.cxx``.  At the same time, remove``#include <cstdlib>``... literalinclude:: Step2/tutorial.cxx  :language: c++  :start-after: // convert input to double  :end-before: // calculate square rootWe will need to explicitly state in the CMake code that it should use thecorrect flags. The easiest way to enable support for a specific C++ standardin CMake is by using the :variable:`CMAKE_CXX_STANDARD` variable. For thistutorial, set the :variable:`CMAKE_CXX_STANDARD` variable in the``CMakeLists.txt`` file to 11 and :variable:`CMAKE_CXX_STANDARD_REQUIRED` toTrue:.. literalinclude:: Step2/CMakeLists.txt  :language: cmake  :end-before: # configure a header file to pass some of the CMake settingsBuild and Test--------------Run the :manual:`cmake <cmake(1)>` executable or the:manual:`cmake-gui <cmake-gui(1)>` to configure the project and then build itwith your chosen build tool.For example, from the command line we could navigate to the``Help/guide/tutorial`` directory of the CMake source code tree and run thefollowing commands:.. code-block:: console  mkdir Step1_build  cd Step1_build  cmake ../Step1  cmake --build .Navigate to the directory where Tutorial was built (likely the make directoryor a Debug or Release build configuration subdirectory) and run these commands:.. code-block:: console  Tutorial 4294967296  Tutorial 10  TutorialAdding a Library (Step 2)=========================Now we will add a library to our project. This library will contain our ownimplementation for computing the square root of a number. The executable canthen use this library instead of the standard square root function provided bythe compiler.For this tutorial we will put the library into a subdirectorycalled ``MathFunctions``. This directory already contains a header file,``MathFunctions.h``, and a source file ``mysqrt.cxx``. The source file has onefunction called ``mysqrt`` that provides similar functionality to thecompiler's ``sqrt`` function.Add the following one line ``CMakeLists.txt`` file to the ``MathFunctions``directory:.. literalinclude:: Step3/MathFunctions/CMakeLists.txt  :language: cmakeTo make use of the new library we will add an :command:`add_subdirectory`call in the top-level ``CMakeLists.txt`` file so that the library will getbuilt. We add the new library to the executable, and add ``MathFunctions`` asan include directory so that the ``mqsqrt.h`` header file can be found. Thelast few lines of the top-level ``CMakeLists.txt`` file should now look like:.. code-block:: cmake        # add the MathFunctions library        add_subdirectory(MathFunctions)        # add the executable        add_executable(Tutorial tutorial.cxx)        target_link_libraries(Tutorial PUBLIC MathFunctions)        # add the binary tree to the search path for include files        # so that we will find TutorialConfig.h        target_include_directories(Tutorial PUBLIC                                  "${PROJECT_BINARY_DIR}"                                  "${PROJECT_SOURCE_DIR}/MathFunctions"                                  )Now let us make the MathFunctions library optional. While for the tutorialthere really isn't any need to do so, for larger projects this is a commonoccurrence. The first step is to add an option to the top-level``CMakeLists.txt`` file... literalinclude:: Step3/CMakeLists.txt  :language: cmake  :start-after: # should we use our own math functions  :end-before: # add the MathFunctions libraryThis option will be displayed in the :manual:`cmake-gui <cmake-gui(1)>` and:manual:`ccmake <ccmake(1)>`with a default value of ON that can be changed by the user. This setting willbe stored in the cache so that the user does not need to set the value eachtime they run CMake on a build directory.The next change is to make building and linking the MathFunctions libraryconditional. To do this we change the end of the top-level ``CMakeLists.txt``file to look like the following:.. literalinclude:: Step3/CMakeLists.txt  :language: cmake  :start-after: # add the MathFunctions libraryNote the use of the variable ``EXTRA_LIBS`` to collect up any optionallibraries to later be linked into the executable. The variable``EXTRA_INCLUDES`` is used similarly for optional header files. This is aclassic approach when dealing with many optional components, we will coverthe modern approach in the next step.The corresponding changes to the source code are fairly straightforward. First,in ``tutorial.cxx``, include the ``MathFunctions.h`` header if we need it:.. literalinclude:: Step3/tutorial.cxx  :language: c++  :start-after: // should we include the MathFunctions header  :end-before: int mainThen, in the same file, make ``USE_MYMATH`` control which square rootfunction is used:.. literalinclude:: Step3/tutorial.cxx  :language: c++  :start-after: // which square root function should we use?  :end-before: std::cout << "The square root ofSince the source code now requires ``USE_MYMATH`` we can add it to``TutorialConfig.h.in`` with the following line:.. literalinclude:: Step3/TutorialConfig.h.in  :language: c  :lines: 4**Exercise**: Why is it important that we configure ``TutorialConfig.h.in``after the option for ``USE_MYMATH``? What would happen if we inverted the two?Run the :manual:`cmake  <cmake(1)>` executable or the:manual:`cmake-gui <cmake-gui(1)>` to configure the project and then build itwith your chosen build tool. Then run the built Tutorial executable.Use the :manual:`ccmake <ccmake(1)>` executable or the :manual:`cmake-gui <cmake-gui(1)>`to update the value of ``USE_MYMATH``. Rebuild and run the tutorial again.Which function gives better results, sqrt or mysqrt?Adding Usage Requirements for Library (Step 3)==============================================Usage requirements allow for far better control over a library or executable'slink and include line while also giving more control over the transitiveproperty of targets inside CMake. The primary commands that leverage usagerequirements are:  - :command:`target_compile_definitions`  - :command:`target_compile_options`  - :command:`target_include_directories`  - :command:`target_link_libraries`Let's refactor our code from `Adding a Library (Step 2)`_ to use the modernCMake approach of usage requirements. We first state that anybody linking toMathFunctions needs to include the current source directory, whileMathFunctions itself doesn't. So this can become an ``INTERFACE`` usagerequirement.Remember ``INTERFACE`` means things that consumers require but the producerdoesn't. Add the following lines to the end of``MathFunctions/CMakeLists.txt``:.. literalinclude:: Step4/MathFunctions/CMakeLists.txt  :language: cmake  :start-after: # to find MathFunctions.hNow that we've specified usage requirements for MathFunctions we can safelyremove our uses of the ``EXTRA_INCLUDES`` variable from the top-level``CMakeLists.txt``, here:.. literalinclude:: Step4/CMakeLists.txt  :language: cmake  :start-after: # add the MathFunctions library  :end-before: # add the executableAnd here:.. literalinclude:: Step4/CMakeLists.txt  :language: cmake  :start-after: # so that we will find TutorialConfig.hOnce this is done, run the :manual:`cmake  <cmake(1)>` executable or the:manual:`cmake-gui <cmake-gui(1)>` to configure the project and then build itwith your chosen build tool or by using ``cmake --build .`` from the builddirectory.Installing and Testing (Step 4)===============================Now we can start adding install rules and testing support to our project.Install Rules-------------The install rules are fairly simple: for MathFunctions we want to install thelibrary and header file and for the application we want to install theexecutable and configured header.So to the end of ``MathFunctions/CMakeLists.txt`` we add:.. literalinclude:: Step5/MathFunctions/CMakeLists.txt  :language: cmake  :start-after: # install rulesAnd to the end of the top-level ``CMakeLists.txt`` we add:.. literalinclude:: Step5/CMakeLists.txt  :language: cmake  :start-after: # add the install targets  :end-before: # enable testingThat is all that is needed to create a basic local install of the tutorial.Run the :manual:`cmake  <cmake(1)>` executable or the:manual:`cmake-gui <cmake-gui(1)>` to configure the project and then build itwith your chosen build tool. Run the install step by using the ``install``option of the :manual:`cmake  <cmake(1)>` command (introduced in 3.15, olderversions of CMake must use ``make install``) from the command line, or buildthe ``INSTALL`` target from an IDE. This will install the appropriate headerfiles, libraries, and executables.The CMake variable :variable:`CMAKE_INSTALL_PREFIX` is used to determine theroot of where the files will be installed. If using ``cmake --install`` acustom installation directory can be given via the ``--prefix`` argument. Formulti-configuration tools, use the ``--config`` argument to specify theconfiguration.Verify that the installed Tutorial runs.Testing Support---------------Next let's test our application. At the end of the top-level ``CMakeLists.txt``file we can enable testing and then add a number of basic tests to verify thatthe application is working correctly... literalinclude:: Step5/CMakeLists.txt  :language: cmake  :start-after: # enable testingThe first test simply verifies that the application runs, does not segfault orotherwise crash, and has a zero return value. This is the basic form of aCTest test.The next test makes use of the :prop_test:`PASS_REGULAR_EXPRESSION` testproperty to verify that the output of the test contains certain strings. Inthis case, verifying that the usage message is printed when an incorrect numberof arguments are provided.Lastly, we have a function called ``do_test`` that runs the application andverifies that the computed square root is correct for given input. For eachinvocation of ``do_test``, another test is added to the project with a name,input, and expected results based on the passed arguments.Rebuild the application and then cd to the binary directory and run the:manual:`ctest <ctest(1)>` executable: ``ctest -N`` and ``ctest -VV``. Formulti-config generators (e.g. Visual Studio), the configuration type must bespecified. To run tests in Debug mode, for example, use ``ctest -C Debug -VV``from the build directory (not the Debug subdirectory!). Alternatively, buildthe ``RUN_TESTS`` target from the IDE.Adding System Introspection (Step 5)====================================Let us consider adding some code to our project that depends on features thetarget platform may not have. For this example, we will add some code thatdepends on whether or not the target platform has the ``log`` and ``exp``functions. Of course almost every platform has these functions but for thistutorial assume that they are not common.If the platform has ``log`` and ``exp`` then we will use them to compute thesquare root in the ``mysqrt`` function. We first test for the availability ofthese functions using the :module:`CheckSymbolExists` module in the top-level``CMakeLists.txt``. We're going to use the new defines in``TutorialConfig.h.in``, so be sure to set them before that file is configured... literalinclude:: Step6/MathFunctions/CMakeLists.txt  :language: cmake  :start-after: # does this system provide the log and exp functions?  :end-before: if(HAVE_LOG AND HAVE_EXP)Now let's add these defines to ``TutorialConfig.h.in`` so that we can use themfrom ``mysqrt.cxx``:.. code-block:: console  // does the platform provide exp and log functions?  #cmakedefine HAVE_LOG  #cmakedefine HAVE_EXPIf ``log`` and ``exp`` are available on the system, then we will use them tocompute the square root in the ``mysqrt`` function. Add the following code tothe ``mysqrt`` function in ``MathFunctions/mysqrt.cxx`` (don't forget the``#endif`` before returning the result!):.. literalinclude:: Step6/MathFunctions/mysqrt.cxx  :language: c++  :start-after: // if we have both log and exp then use them  :end-before: // do ten iterationsWe will also need to modify ``mysqrt.cxx`` to include ``cmath``... literalinclude:: Step6/MathFunctions/mysqrt.cxx  :language: c++  :end-before: #include <iostream>Run the :manual:`cmake  <cmake(1)>` executable or the:manual:`cmake-gui <cmake-gui(1)>` to configure the project and then build itwith your chosen build tool and run the Tutorial executable.You will notice that we're not using ``log`` and ``exp``, even if we think theyshould be available. We should realize quickly that we have forgotten toinclude ``TutorialConfig.h`` in ``mysqrt.cxx``.We will also need to update ``MathFunctions/CMakeLists.txt`` so ``mysqrt.cxx``knows where this file is located:.. code-block:: cmake  target_include_directories(MathFunctions            INTERFACE ${CMAKE_CURRENT_SOURCE_DIR}            PRIVATE ${CMAKE_BINARY_DIR}            )After making this update, go ahead and build the project again and run thebuilt Tutorial executable. If ``log`` and ``exp`` are still not being used,open the generated ``TutorialConfig.h`` file from the build directory. Maybethey aren't available on the current system?Which function gives better results now, sqrt or mysqrt?Specify Compile Definition--------------------------Is there a better place for us to save the ``HAVE_LOG`` and ``HAVE_EXP`` valuesother than in ``TutorialConfig.h``? Let's try to use:command:`target_compile_definitions`.First, remove the defines from ``TutorialConfig.h.in``. We no longer need toinclude ``TutorialConfig.h`` from ``mysqrt.cxx`` or the extra include in``MathFunctions/CMakeLists.txt``.Next, we can move the check for ``HAVE_LOG`` and ``HAVE_EXP`` to``MathFunctions/CMakeLists.txt`` and then specify those values as ``PRIVATE``compile definitions... literalinclude:: Step6/MathFunctions/CMakeLists.txt  :language: cmake  :start-after: # does this system provide the log and exp functions?  :end-before: # install rulesAfter making these updates, go ahead and build the project again. Run thebuilt Tutorial executable and verify that the results are same as earlier inthis step.Adding a Custom Command and Generated File (Step 6)===================================================Suppose, for the purpose of this tutorial, we decide that we never want to usethe platform ``log`` and ``exp`` functions and instead would like togenerate a table of precomputed values to use in the ``mysqrt`` function.In this section, we will create the table as part of the build process,and then compile that table into our application.First, let's remove the check for the ``log`` and ``exp`` functions in``MathFunctions/CMakeLists.txt``. Then remove the check for ``HAVE_LOG`` and``HAVE_EXP`` from ``mysqrt.cxx``. At the same time, we can remove:code:`#include <cmath>`.In the ``MathFunctions`` subdirectory, a new source file named``MakeTable.cxx`` has been provided to generate the table.After reviewing the file, we can see that the table is produced as valid C++code and that the output filename is passed in as an argument.The next step is to add the appropriate commands to the``MathFunctions/CMakeLists.txt`` file to build the MakeTable executable andthen run it as part of the build process. A few commands are needed toaccomplish this.First, at the top of ``MathFunctions/CMakeLists.txt``, the executable for``MakeTable`` is added as any other executable would be added... literalinclude:: Step7/MathFunctions/CMakeLists.txt  :language: cmake  :start-after: # first we add the executable that generates the table  :end-before: # add the command to generate the source codeThen we add a custom command that specifies how to produce ``Table.h``by running MakeTable... literalinclude:: Step7/MathFunctions/CMakeLists.txt  :language: cmake  :start-after: # add the command to generate the source code  :end-before: # add the main libraryNext we have to let CMake know that ``mysqrt.cxx`` depends on the generatedfile ``Table.h``. This is done by adding the generated ``Table.h`` to the listof sources for the library MathFunctions... literalinclude:: Step7/MathFunctions/CMakeLists.txt  :language: cmake  :start-after: # add the main library  :end-before: # state that anybody linkingWe also have to add the current binary directory to the list of includedirectories so that ``Table.h`` can be found and included by ``mysqrt.cxx``... literalinclude:: Step7/MathFunctions/CMakeLists.txt  :start-after: # state that we depend on our bin  :end-before: # install rulesNow let's use the generated table. First, modify ``mysqrt.cxx`` to include``Table.h``. Next, we can rewrite the mysqrt function to use the table:.. literalinclude:: Step7/MathFunctions/mysqrt.cxx  :language: c++  :start-after: // a hack square root calculation using simple operationsRun the :manual:`cmake  <cmake(1)>` executable or the:manual:`cmake-gui <cmake-gui(1)>` to configure the project and then build itwith your chosen build tool.When this project is built it will first build the ``MakeTable`` executable.It will then run ``MakeTable`` to produce ``Table.h``. Finally, it willcompile ``mysqrt.cxx`` which includes ``Table.h`` to produce the MathFunctionslibrary.Run the Tutorial executable and verify that it is using the table.Building an Installer (Step 7)==============================Next suppose that we want to distribute our project to other people so thatthey can use it. We want to provide both binary and source distributions on avariety of platforms. This is a little different from the install we didpreviously in `Installing and Testing (Step 4)`_ , where we wereinstalling the binaries that we had built from the source code. In thisexample we will be building installation packages that support binaryinstallations and package management features. To accomplish this we will useCPack to create platform specific installers. Specifically we need to add afew lines to the bottom of our top-level ``CMakeLists.txt`` file... literalinclude:: Step8/CMakeLists.txt  :language: cmake  :start-after: # setup installerThat is all there is to it. We start by including:module:`InstallRequiredSystemLibraries`. This module will include any runtimelibraries that are needed by the project for the current platform. Next we setsome CPack variables to where we have stored the license and versioninformation for this project. The version information was set earlier in thistutorial and the ``license.txt`` has been included in the top-level sourcedirectory for this step.Finally we include the :module:`CPack module <CPack>` which will use thesevariables and some other properties of the current system to setup aninstaller.The next step is to build the project in the usual manner and then run the:manual:`cpack <cpack(1)>` executable. To build a binary distribution, from thebinary directory run:.. code-block:: console  cpackTo specify the generator, use the ``-G`` option. For multi-config builds, use``-C`` to specify the configuration. For example:.. code-block:: console  cpack -G ZIP -C DebugTo create a source distribution you would type:.. code-block:: console  cpack --config CPackSourceConfig.cmakeAlternatively, run ``make package`` or right click the ``Package`` target and``Build Project`` from an IDE.Run the installer found in the binary directory. Then run the installedexecutable and verify that it works.Adding Support for a Dashboard (Step 8)=======================================Adding support for submitting our test results to a dashboard is simple. Wealready defined a number of tests for our project in `Testing Support`_. Now wejust have to run those tests and submit them to a dashboard. To include supportfor dashboards we include the :module:`CTest` module in our top-level``CMakeLists.txt``.Replace:.. code-block:: cmake  # enable testing  enable_testing()With:.. code-block:: cmake  # enable dashboard scripting  include(CTest)The :module:`CTest` module will automatically call ``enable_testing()``, so wecan remove it from our CMake files.We will also need to create a ``CTestConfig.cmake`` file in the top-leveldirectory where we can specify the name of the project and where to submit thedashboard... literalinclude:: Step9/CTestConfig.cmake  :language: cmakeThe :manual:`ctest <ctest(1)>` executable will read in this file when it runs.To create a simple dashboard you can run the :manual:`cmake <cmake(1)>`executable or the :manual:`cmake-gui <cmake-gui(1)>` to configure the project,but do not build it yet. Instead, change directory to the binary tree, and thenrun:  ctest [-VV] -D ExperimentalRemember, for multi-config generators (e.g. Visual Studio), the configurationtype must be specified::  ctest [-VV] -C Debug -D ExperimentalOr, from an IDE, build the ``Experimental`` target.The :manual:`ctest <ctest(1)>` executable will build and test the project andsubmit the results to Kitware's public dashboard:https://my.cdash.org/index.php?project=CMakeTutorial.Mixing Static and Shared (Step 9)=================================In this section we will show how the :variable:`BUILD_SHARED_LIBS` variable canbe used to control the default behavior of :command:`add_library`,and allow control over how libraries without an explicit type (``STATIC``,``SHARED``, ``MODULE`` or ``OBJECT``) are built.To accomplish this we need to add :variable:`BUILD_SHARED_LIBS` to thetop-level ``CMakeLists.txt``. We use the :command:`option` command as it allowsusers to optionally select if the value should be ON or OFF.Next we are going to refactor MathFunctions to become a real library thatencapsulates using ``mysqrt`` or ``sqrt``, instead of requiring the callingcode to do this logic. This will also mean that ``USE_MYMATH`` will not controlbuilding MathFunctions, but instead will control the behavior of this library.The first step is to update the starting section of the top-level``CMakeLists.txt`` to look like:.. literalinclude:: Step10/CMakeLists.txt  :language: cmake  :end-before: # add the binary treeNow that we have made MathFunctions always be used, we will need to updatethe logic of that library. So, in ``MathFunctions/CMakeLists.txt`` we need tocreate a SqrtLibrary that will conditionally be built when ``USE_MYMATH`` isenabled. Now, since this is a tutorial, we are going to explicitly requirethat SqrtLibrary is built statically.The end result is that ``MathFunctions/CMakeLists.txt`` should look like:.. literalinclude:: Step10/MathFunctions/CMakeLists.txt  :language: cmake  :lines: 1-36,42-Next, update ``MathFunctions/mysqrt.cxx`` to use the ``mathfunctions`` and``detail`` namespaces:.. literalinclude:: Step10/MathFunctions/mysqrt.cxx  :language: c++We also need to make some changes in ``tutorial.cxx``, so that it no longeruses ``USE_MYMATH``:#. Always include ``MathFunctions.h``#. Always use ``mathfunctions::sqrt``#. Don't include cmathFinally, update ``MathFunctions/MathFunctions.h`` to use dll export defines:.. literalinclude:: Step10/MathFunctions/MathFunctions.h  :language: c++At this point, if you build everything, you will notice that linking failsas we are combining a static library without position independent code with alibrary that has position independent code. The solution to this is toexplicitly set the :prop_tgt:`POSITION_INDEPENDENT_CODE` target property ofSqrtLibrary to be True no matter the build type... literalinclude:: Step10/MathFunctions/CMakeLists.txt  :language: cmake  :lines: 37-42**Exercise**: We modified ``MathFunctions.h`` to use dll export defines.Using CMake documentation can you find a helper module to simplify this?Adding Generator Expressions (Step 10)======================================:manual:`Generator expressions <cmake-generator-expressions(7)>` are evaluatedduring build system generation to produce information specific to each buildconfiguration.:manual:`Generator expressions <cmake-generator-expressions(7)>` are allowed inthe context of many target properties, such as :prop_tgt:`LINK_LIBRARIES`,:prop_tgt:`INCLUDE_DIRECTORIES`, :prop_tgt:`COMPILE_DEFINITIONS` and others.They may also be used when using commands to populate those properties, such as:command:`target_link_libraries`, :command:`target_include_directories`,:command:`target_compile_definitions` and others.:manual:`Generator expressions <cmake-generator-expressions(7)>`  may be usedto enable conditional linking, conditional definitions used when compiling,conditional include directories and more. The conditions may be based on thebuild configuration, target properties, platform information or any otherqueryable information.There are different types of:manual:`generator expressions <cmake-generator-expressions(7)>` includingLogical, Informational, and Output expressions.Logical expressions are used to create conditional output. The basicexpressions are the 0 and 1 expressions. A ``$<0:...>`` results in the emptystring, and ``<1:...>`` results in the content of "...".  They can also benested.A common usage of:manual:`generator expressions <cmake-generator-expressions(7)>` is toconditionally add compiler flags, such as those for language levels orwarnings. A nice pattern is to associate this information to an ``INTERFACE``target allowing this information to propagate. Lets start by constructing an``INTERFACE`` target and specifying the required C++ standard level of ``11``instead of using :variable:`CMAKE_CXX_STANDARD`.So the following code:.. literalinclude:: Step10/CMakeLists.txt  :language: cmake  :start-after: project(Tutorial VERSION 1.0)  :end-before: # control where the static and shared libraries are built so that on windowsWould be replaced with:.. literalinclude:: Step11/CMakeLists.txt  :language: cmake  :start-after: project(Tutorial VERSION 1.0)  :end-before: # add compiler warning flags just when building this project viaNext we add the desired compiler warning flags that we want for our project. Aswarning flags vary based on the compiler we use the ``COMPILE_LANG_AND_ID``generator expression to control which flags to apply given a language and a setof compiler ids as seen below:.. literalinclude:: Step11/CMakeLists.txt  :language: cmake  :start-after: # the BUILD_INTERFACE genex  :end-before: # control where the static and shared libraries are built so that on windowsLooking at this we see that the warning flags are encapsulated inside a``BUILD_INTERFACE`` condition. This is done so that consumers of our installedproject will not inherit our warning flags.**Exercise**: Modify ``MathFunctions/CMakeLists.txt`` so that all targets havea :command:`target_link_libraries` call to ``tutorial_compiler_flags``.Adding Export Configuration (Step 11)=====================================During `Installing and Testing (Step 4)`_ of the tutorial we added the abilityfor CMake to install the library and headers of the project. During`Building an Installer (Step 7)`_ we added the ability to package up thisinformation so it could be distributed to other people.The next step is to add the necessary information so that other CMake projectscan use our project, be it from a build directory, a local install or whenpackaged.The first step is to update our :command:`install(TARGETS)` commands to notonly specify a ``DESTINATION`` but also an ``EXPORT``. The ``EXPORT`` keywordgenerates and installs a CMake file containing code to import all targetslisted in the install command from the installation tree. So let's go ahead andexplicitly ``EXPORT`` the MathFunctions library by updating the ``install``command in ``MathFunctions/CMakeLists.txt`` to look like:.. literalinclude:: Complete/MathFunctions/CMakeLists.txt  :language: cmake  :start-after: # install rulesNow that we have MathFunctions being exported, we also need to explicitlyinstall the generated ``MathFunctionsTargets.cmake`` file. This is done byadding the following to the bottom of the top-level ``CMakeLists.txt``:.. literalinclude:: Complete/CMakeLists.txt  :language: cmake  :start-after: # install the configuration targets  :end-before: include(CMakePackageConfigHelpers)At this point you should try and run CMake. If everything is setup properlyyou will see that CMake will generate an error that looks like:.. code-block:: console  Target "MathFunctions" INTERFACE_INCLUDE_DIRECTORIES property contains  path:    "/Users/robert/Documents/CMakeClass/Tutorial/Step11/MathFunctions"  which is prefixed in the source directory.What CMake is trying to say is that during generating the export informationit will export a path that is intrinsically tied to the current machine andwill not be valid on other machines. The solution to this is to update theMathFunctions :command:`target_include_directories` to understand that it needsdifferent ``INTERFACE`` locations when being used from within the builddirectory and from an install / package. This means converting the:command:`target_include_directories` call for MathFunctions to look like:.. literalinclude:: Step12/MathFunctions/CMakeLists.txt  :language: cmake  :start-after: # to find MathFunctions.h, while we don't.  :end-before: # should we use our own math functionsOnce this has been updated, we can re-run CMake and verify that it doesn'twarn anymore.At this point, we have CMake properly packaging the target information that isrequired but we will still need to generate a ``MathFunctionsConfig.cmake`` sothat the CMake :command:`find_package` command can find our project. So let's goahead and add a new file to the top-level of the project called``Config.cmake.in`` with the following contents:.. literalinclude:: Step12/Config.cmake.inThen, to properly configure and install that file, add the following to thebottom of the top-level ``CMakeLists.txt``:.. literalinclude:: Step12/CMakeLists.txt  :language: cmake  :start-after: # install the configuration targets  :end-before: # generate the exportAt this point, we have generated a relocatable CMake Configuration for ourproject that can be used after the project has been installed or packaged. Ifwe want our project to also be used from a build directory we only have to addthe following to the bottom of the top level ``CMakeLists.txt``:.. literalinclude:: Step12/CMakeLists.txt  :language: cmake  :start-after: # needs to be after the install(TARGETS ) commandWith this export call we now generate a ``Targets.cmake``, allowing theconfigured ``MathFunctionsConfig.cmake`` in the build directory to be used byother projects, without needing it to be installed.Packaging Debug and Release (Step 12)=====================================**Note:** This example is valid for single-configuration generators and willnot work for multi-configuration generators (e.g. Visual Studio).By default, CMake's model is that a build directory only contains a singleconfiguration, be it Debug, Release, MinSizeRel, or RelWithDebInfo. It ispossible, however, to setup CPack to bundle multiple build directories andconstruct a package that contains multiple configurations of the same project.First, we want to ensure that the debug and release builds use different namesfor the executables and libraries that will be installed. Let's use `d` as thepostfix for the debug executable and libraries.Set :variable:`CMAKE_DEBUG_POSTFIX` near the beginning of the top-level``CMakeLists.txt`` file:.. literalinclude:: Complete/CMakeLists.txt  :language: cmake  :start-after: project(Tutorial VERSION 1.0)  :end-before: target_compile_features(tutorial_compiler_flagsAnd the :prop_tgt:`DEBUG_POSTFIX` property on the tutorial executable:.. literalinclude:: Complete/CMakeLists.txt  :language: cmake  :start-after: # add the executable  :end-before: # add the binary tree to the search path for include filesLet's also add version numbering to the MathFunctions library. In``MathFunctions/CMakeLists.txt``, set the :prop_tgt:`VERSION` and:prop_tgt:`SOVERSION` properties:.. literalinclude:: Complete/MathFunctions/CMakeLists.txt  :language: cmake  :start-after: # setup the version numbering  :end-before: # install rulesFrom the ``Step12`` directory, create ``debug`` and ``release``subbdirectories. The layout will look like:.. code-block:: none  - Step12     - debug     - releaseNow we need to setup debug and release builds. We can use:variable:`CMAKE_BUILD_TYPE` to set the configuration type:.. code-block:: console  cd debug  cmake -DCMAKE_BUILD_TYPE=Debug ..  cmake --build .  cd ../release  cmake -DCMAKE_BUILD_TYPE=Release ..  cmake --build .Now that both the debug and release builds are complete, we can use a customconfiguration file to package both builds into a single release. In the``Step12`` directory, create a file called ``MultiCPackConfig.cmake``. In thisfile, first include the default configuration file that was created by the:manual:`cmake  <cmake(1)>` executable.Next, use the ``CPACK_INSTALL_CMAKE_PROJECTS`` variable to specify whichprojects to install. In this case, we want to install both debug and release... literalinclude:: Complete/MultiCPackConfig.cmake  :language: cmakeFrom the ``Step12`` directory, run :manual:`cpack <cpack(1)>` specifying ourcustom configuration file with the ``config`` option:.. code-block:: console  cpack --config MultiCPackConfig.cmake
 |