Cloned SEACAS for EXODUS library with extra build files for internal package management.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

4440 lines
180 KiB

2 years ago
.. Common references to other documents
.. _TriBITS Users Guide and Reference: TribitsUsersGuide.html
.. _Package Dependencies and Enable/Disable Logic: TribitsUsersGuide.html#package-dependencies-and-enable-disable-logic
.. _TriBITS Dependency Handling Behaviors: TribitsUsersGuide.html#tribits-dependency-handling-behaviors
.. _tribits_tpl_find_include_dirs_and_libraries(): TribitsUsersGuide.html#tribits-tpl-find-include-dirs-and-libraries
.. _tribits_ctest_driver(): TribitsUsersGuide.html#tribits-ctest-driver
.. _Ninja: https://ninja-build.org
.. _CMake Ninja Fortran Support: https://cmake.org/cmake/help/latest/generator/Ninja.html
.. _CTest Resource Allocation System: https://cmake.org/cmake/help/latest/manual/ctest.1.html#resource-allocation
.. _CTest Resource Specification File: https://cmake.org/cmake/help/latest/manual/ctest.1.html#ctest-resource-specification-file
.. _CTest Resource Allocation Environment Variables: https://cmake.org/cmake/help/latest/manual/ctest.1.html#environment-variables
.. _RESOURCE_GROUPS: https://cmake.org/cmake/help/latest/prop_test/RESOURCE_GROUPS.html#prop_test:RESOURCE_GROUPS
Getting set up to use CMake
===========================
Before one can configure <Project> to be built, one must first obtain a
version of CMake on the system newer than <MinCMakeVer> This guide assumes
that once CMake is installed that it will be in the default path with the name
``cmake``.
Installing a binary release of CMake [casual users]
---------------------------------------------------
Download and install the binary (version <MinCMakeVer> or greater is
recommended) from:
http://www.cmake.org/cmake/resources/software.html
Installing CMake from source [developers and experienced users]
---------------------------------------------------------------
If you have access to the <Project> git repositories (which which includes a
snapshot of TriBITS), then install CMake with::
$ cd <some-scratch-space>/
$ TRIBITS_BASE_DIR=<project-base-dir>/cmake/tribits
$ $TRIBITS_BASE_DIR/devtools_install/install-cmake.py \
--install-dir-base=<INSTALL_BASE_DIR> --cmake-version=X.Y.Z \
--do-all
This will result in cmake and related CMake tools being installed in
``<INSTALL_BASE_DIR>/cmake-X.Y.Z/bin/`` (see the instructions printed at the
end on how to update your ``PATH`` env var).
To get help for installing CMake with this script use::
$ $TRIBITS_BASE_DIR/devtools_install/install-cmake.py --help
NOTE: You will want to read the help message about how to install CMake to
share with other users and maintainers and how to install with sudo if needed.
Installing Ninja from Source
----------------------------
The `Ninja`_ tool allows for much faster parallel builds for some large CMake
projects and performs much faster dependency analysis than the Makefiles
back-end build system. It also provides some other nice features like ``ninja
-n -d explain`` to show why the build system decides to (re)build the targets
that it decides to build.
As of Ninja 1.10+, Fortran support is part of the official GitHub version of
Ninja as can be obtained from:
https://github.com/ninja-build/ninja/releases
(see `CMake Ninja Fortran Support`_).
Ninja is easy to install from source on almost any machine. On Unix/Linux
systems it is as simple as ``configure --prefix=<dir>``, ``make`` and ``make
install``.
Getting CMake Help
==================
Finding CMake help at the website
---------------------------------
http://www.cmake.org
Building CMake help locally
---------------------------
To get help on CMake input options, run::
$ cmake --help
To get help on a single CMake function, run::
$ cmake --help-command <command>
To generate the entire documentation at once, run::
$ cmake --help-full cmake.help.html
(Open your web browser to the file cmake.help.html)
Configuring (Makefile, Ninja and other Generators)
===================================================
CMake supports a number of different build generators (e.g. Ninja, Eclipse,
XCode, MS Visual Studio, etc.) but the primary generator most people use on
Unix/Linux system is ``make`` (using the default cmake option ``-G"Unix
Makefiles"``) and CMake generated Makefiles. Another (increasingly) popular
generator is Ninja (using cmake option ``-GNinja``). Most of the material in
this section applies to all generators but most experience is for the
Makefiles and Ninja generators.
Setting up a build directory
----------------------------
In order to configure, one must set up a build directory. <Project> does
**not** support in-source builds so the build tree must be separate from the
source tree. The build tree can be created under the source tree such as
with::
$ cd <src-dir>/
$ mkdir <build-dir>
$ cd <build-dir>/
but it is generally recommended to create a build directory parallel from the
source tree such as with::
<some-base-dir>/
<src-dir>/
<build-dir>/
NOTE: If you mistakenly try to configure for an in-source build (e.g. with
'cmake .') you will get an error message and instructions on how to resolve
the problem by deleting the generated CMakeCache.txt file (and other generated
files) and then follow directions on how to create a different build directory
as shown above.
Basic configuration
-------------------
A few different approaches for configuring are given below.
* `Create a do-configure script [Recommended]`_
* `Create a *.cmake file and point to it [Most Recommended]`_
* `Using the QT CMake configuration GUI`_
.. _Create a do-configure script [Recommended]:
a) Create a 'do-configure' script such as [Recommended]::
#!/bin/bash
cmake \
-D CMAKE_BUILD_TYPE=DEBUG \
-D <Project>_ENABLE_TESTS=ON \
"$@" \
${SOURCE_BASE}
and then run it with::
./do-configure [OTHER OPTIONS] -D<Project>_ENABLE_<TRIBITS_PACKAGE>=ON
where ``<TRIBITS_PACKAGE>`` is a valid Package name (see above), etc. and
``SOURCE_BASE`` is set to the <Project> source base directory (or your can
just give it explicitly in the script).
See ``<Project>/sampleScripts/*`` for examples of real ``do-configure``
scripts for different platforms.
NOTE: If one has already configured once and one needs to configure from
scratch (needs to wipe clean defaults for cache variables, updates
compilers, other types of changes) then one will want to delete the local
CMakeCache.txt and other CMake-generated files before configuring again (see
`Reconfiguring completely from scratch`_).
.. _<Project>_CONFIGURE_OPTIONS_FILE:
.. _Create a *.cmake file and point to it [Most Recommended]:
b) Create a ``*.cmake`` file and point to it [Most Recommended].
Create a do-configure script like::
#!/bin/bash
cmake \
-D <Project>_CONFIGURE_OPTIONS_FILE=MyConfigureOptions.cmake \
-D <Project>_ENABLE_TESTS=ON \
"$@" \
${SOURCE_BASE}
where MyConfigureOptions.cmake (in the current working directory) might look
like::
set(CMAKE_BUILD_TYPE DEBUG CACHE STRING "Set in MyConfigureOptions.cmake")
set(<Project>_ENABLE_CHECKED_STL ON CACHE BOOL "Set in MyConfigureOptions.cmake")
set(BUILD_SHARED_LIBS ON CACHE BOOL "Set in MyConfigureOptions.cmake")
...
Using a configuration fragment ``*.cmake`` file allows for better reuse of
configure options across different configure scripts and better version
control of configure options. Using the comment ``"Set in
MyConfigureOptions.cmake"`` makes it easy see where that variable got set
when looking an the generated ``CMakeCache.txt`` file. Also, when this
``*.cmake`` fragment file changes, CMake will automatically trigger a
reconfigure during a make (because it knows about the file and will check its
time stamp, unlike when using ``-C <file-name>.cmake``, see below).
One can use the ``FORCE`` option in the ``set()`` commands shown above and
that will override any value of the options that might already be set (but
when using ``-C`` to include this forced ``set(<var> ... FORCE)`` will only
override the value if the file with the ``set()`` is listed after the
``-D<var>=<val>`` command-line option). However, that will not allow the
user to override the options on the CMake command-line using
``-D<VAR>=<value>`` so it is generally **not** desired to use ``FORCE``.
One can also pass in a list of configuration fragment files separated by
commas ``','`` which will be read in the order they are given as::
-D <Project>_CONFIGURE_OPTIONS_FILE=<file0>.cmake,<file1>.cmake,...
One can read in configure option files under the project source directory by
using the type ``STRING`` such as with::
-D <Project>_CONFIGURE_OPTIONS_FILE:STRING=cmake/MpiConfig1.cmake
In this case, the relative paths will be with respect to the project base
source directory, not the current working directory (unlike when using ``-C
<file-name>.cmake``, see below). (By specifying the type ``STRING``, one
turns off CMake interpretation as a ``FILEPATH``. Otherwise, the type
``FILEPATH`` causes CMake to always interpret relative paths with respect to
the current working directory and set the absolute path).
Note that CMake options files can also be read in using the built-in CMake
argument ``-C <file>.cmake`` as::
cmake -C <file0>.cmake -C <file1>.cmake ... [other options] \
${SOURCE_BASE}
However, there are some differences to using
``<Project>_CONFIGURE_OPTIONS_FILE`` vs. ``-C`` to read in ``*.cmake`` files
to be aware of as described below:
1) One can use
``-D<Project>_CONFIGURE_OPTIONS_FILE:STRING=<rel-path>/<file-name>.cmake``
with a relative path w.r.t. to the source tree to make it easier to point to
options files in the project source. Using ``cmake -C
<abs-path>/<file-name>.cmake`` would require having to give the absolute
path ``<abs-path>`` or a longer relative path from the build directory back
to the source directory. Having to give the absolute path to files in the
source tree complicates configure scripts in some cases (i.e. where the
project source directory location may not be known or easy to get).
2) When configuration files are read in using
``<Project>_CONFIGURE_OPTIONS_FILE``, they will get reprocessed on every
reconfigure (such as when reconfigure happens automatically when running
``make``). That means that if options change in those included ``*.cmake``
files from the initial configure, then those updated options will get
automatically picked up in a reconfigure. But when processing ``*.cmake``
files using the built-in ``-C <frag>.cmake`` argument, updated options will
not get set. Therefore, if one wants to have the ``*.cmake`` files
automatically be reprocessed, then one should use
``<Project>_CONFIGURE_OPTIONS_FILE``. But if one does not want to have the
contents of the ``<frag>.cmake`` file reread on reconfigures, then one would
want to use ``-C <frag>.cmake``.
3) When using ``<Project>_CONFIGURE_OPTIONS_FILE``, one can create and use
parameterized ``*.cmake`` files that can be used with multiple TriBITS
projects. For example, one can have set statements like
``set(${PROJECT_NAME}_ENABLE_Fortran OFF ...)`` since ``PROJECT_NAME`` is
known before the file is included. One cannot do that with ``-C`` and
instead would have to provide the full variables names specific for a given
TriBITS project.
4) When using ``<Project>_CONFIGURE_OPTIONS_FILE``, non-cache project-level
variables can be set in a ``*.cmake`` file that will impact the
configuration. When using the ``-C`` option, only variables set with
``set(<varName> <val> CACHE <TYPE> ...)`` will impact the configuration.
5) Cache variables forced set with ``set(<varName> <val> CACHE <TYPE>
"<doc>" FORCE)`` in a ``<frag>.cmake`` file pulled in with ``-C
<frag>.cmake`` will only override a cache variable ``-D<varName>=<val2>``
passed on the command-line if the ``-C <frag>.cmake`` argument comes
**after** the ``-D<varName>=<val2>`` argument (i.e. ``cmake
-D<varName>=<val2> -C <frag>.cmake``). Otherwise, if the order of the
``-D`` and ``-C`` arguments is reversed (i.e. ``cmake -C <frag>.cmake
-D<varName>=<val2>``) then the forced ``set()`` statement **WILL NOT**
override the cache var set on the command-line with ``-D<varName>=<val2>``.
However, note that a forced ``set()`` statement **WILL** override other
cache vars set with non-forced ``set()`` statements ``set(<varName> <val1>
CACHE <TYPE> "<doc>")`` in the same ``*.cmake`` file or in previously read
``-C <frag2>.cmake`` files included on the command-line before the file ``-C
<frag>.cmake``. Alternatively, if the file is pulled in with
``-D<Project>_CONFIGURE_OPTIONS_FILE=<frag>.cmake``, then a ``set(<varName>
<val> CACHE <TYPE> "<doc>" FORCE)`` statement in a ``<frag>.cmake`` **WILL**
override a cache variable passed in on the command-line
``-D<varName>=<val2>`` no matter the order of the arguments
``-D<Project>_CONFIGURE_OPTIONS_FILE=<frag>.cmake`` and
``-D<varName>=<val2>``. (This is because the file ``<frag>.cmake`` is
included as part of the processing of the project's top-level
``CMakeLists.txt`` file.)
6) However, the ``*.cmake`` files specified by
``<Project>_CONFIGURE_OPTIONS_FILE`` will only get read in **after** the
project's ``ProjectName.cmake`` and other ``set()`` statements are called at
the top of the project's top-level ``CMakeLists.txt`` file. So any CMake
cache variables that are set in this early CMake code will override cache
defaults set in the included ``*.cmake`` file. (This is why TriBITS
projects must be careful **not** to set default values for cache variables
directly like this but instead should set indirect
``<Project>_<VarName>_DEFAULT`` non-cache variables.) But when a
``*.cmake`` file is read in using ``-C``, then the ``set()`` statements in
those files will get processed before any in the project's
``CMakeLists.txt`` file. So be careful about this difference in behavior
and carefully watch cache variable values actually set in the generated
``CMakeCache.txt`` file.
In other words, the context and impact of what get be set from a ``*.cmake``
file read in through the built-in CMake ``-C`` argument is more limited
while the code listed in the ``*.cmake`` file pulled in with
``-D<Project>_CONFIGURE_OPTIONS_FILE=<frag>.cmake`` behaves just like
regular CMake statements executed in the project's top-level
``CMakeLists.txt`` file. In addition, any forced set statements in a
``*.cmake`` file pulled in with ``-C`` **may or may not** override cache
vars sets on the command-line with ``-D<varName>=<val>`` depending on the
order of the ``-C`` and ``-D`` options. (There is no order dependency for
``*.cmake`` files passed in through
``-D<Project>_CONFIGURE_OPTIONS_FILE=<frag>.cmake``.)
.. _Using the QT CMake configuration GUI:
c) Using the QT CMake configuration GUI:
On systems where the QT CMake GUI is installed (e.g. Windows) the CMake GUI
can be a nice way to configure <Project> (or just explore options) if you
are a user. To make your configuration easily repeatable, you might want to
create a fragment file and just load it by setting
`<Project>_CONFIGURE_OPTIONS_FILE`_ in the GUI.
Likely the most recommended approach to manage complex configurations is to
use ``*.cmake`` fragment files passed in through the
`<Project>_CONFIGURE_OPTIONS_FILE`_ option. This offers the greatest
flexibility and the ability to version-control the configuration settings.
Selecting the list of packages to enable
----------------------------------------
The <Project> project is broken up into a set of packages that can be enabled
(or disabled). For details and generic examples, see `Package Dependencies and
Enable/Disable Logic`_ and `TriBITS Dependency Handling Behaviors`_.
See the following use cases:
* `Determine the list of packages that can be enabled`_
* `Print package dependencies`_
* `Enable a set of packages`_
* `Enable or disable tests for specific packages`_
* `Enable to test all effects of changing a given package(s)`_
* `Enable all packages (and optionally all tests)`_
* `Disable a package and all its dependencies`_
* `Remove all package enables in the cache`_
Determine the list of packages that can be enabled
++++++++++++++++++++++++++++++++++++++++++++++++++
In order to see the list of available <Project> Packages to enable, just
run a basic CMake configure, enabling nothing, and then grep the output to see
what packages are available to enable. The full set of defined packages is
contained the lines starting with ``'Final set of enabled packages'`` and
``'Final set of non-enabled packages'``. If no packages are enabled by
default (which is base behavior), the full list of packages will be listed on
the line ``'Final set of non-enabled packages'``. Therefore, to see the
full list of defined packages, run::
./do-configure 2>&1 | grep "Final set of .*enabled packages"
Any of the packages shown on those lines can potentially be enabled using ``-D
<Project>_ENABLE_<TRIBITS_PACKAGE>=ON`` (unless they are set to disabled
for some reason, see the CMake output for package disable warnings).
Another way to see the full list of packages that can be enabled is to
configure with `<Project>_DUMP_PACKAGE_DEPENDENCIES`_ = ``ON`` and then grep
for ``<Project>_INTERNAL_PACKAGES`` using, for example::
./do-configure 2>&1 | grep "<Project>_INTERNAL_PACKAGES: "
Print package dependencies
++++++++++++++++++++++++++
.. _<Project>_DUMP_PACKAGE_DEPENDENCIES:
The set of package dependencies can be printed in the ``cmake`` STDOUT by
setting the configure option::
-D <Project>_DUMP_PACKAGE_DEPENDENCIES=ON
This will print the basic forward/upstream dependencies for each package.
To find this output, look for the line::
Printing package dependencies ...
and the dependencies are listed below this for each package in the form::
-- <PKG>_LIB_DEFINED_DEPENDENCIES: <PKG0>[O] <[PKG1>[R] ...
-- <PKG>_TEST_DEFINED_DEPENDENCIES: <PKG6>[R] <[PKG8>[R] ...
(Dependencies that don't exist are left out of the output. For example, if
there are no extra test dependencies, then ``<PKG>_TEST_DEFINED_DEPENDENCIES``
will not be printed.)
To also see the direct forward/downstream dependencies for each package,
also include::
-D <Project>_DUMP_FORWARD_PACKAGE_DEPENDENCIES=ON
These dependencies are printed along with the backward/upstsream dependencies
as described above.
Both of these variables are automatically enabled when
`<Project>_VERBOSE_CONFIGURE`_ = ``ON``.
Enable a set of packages
++++++++++++++++++++++++
.. _<Project>_ENABLE_ALL_OPTIONAL_PACKAGES:
.. _<Project>_ENABLE_TESTS:
To enable a package ``<TRIBITS_PACKAGE>`` (and optionally also its tests and
examples), configure with::
-D <Project>_ENABLE_<TRIBITS_PACKAGE>=ON \
-D <Project>_ENABLE_ALL_OPTIONAL_PACKAGES=ON \
-D <Project>_ENABLE_TESTS=ON \
This set of arguments allows a user to turn on ``<TRIBITS_PACKAGE>`` as well
as all packages that ``<TRIBITS_PACKAGE>`` can use. All of the package's
optional "can use" upstream dependent packages are enabled with
``-D<Project>_ENABLE_ALL_OPTIONAL_PACKAGES=ON``. However,
``-D<Project>_ENABLE_TESTS=ON`` will only enable tests and examples for
``<TRIBITS_PACKAGE>`` (and any other packages explicitly enabled).
If a TriBITS package ``<TRIBITS_PACKAGE>`` has subpackages (e.g. subpackages
``<A>``, ``<B>``, ...), then enabling the package is equivalent to enabling
all of the required **and optional** subpackagses::
-D <Project>_ENABLE_<TRIBITS_PACKAGE><A>=ON \
-D <Project>_ENABLE_<TRIBITS_PACKAGE><B>=ON \
...
(In this case, the parent package's optional subpackages are enabled
regardless the value of ``<Project>_ENABLE_ALL_OPTIONAL_PACKAGES``.)
However, a TriBITS subpackage will only be enabled if it is not already
disabled either explicitly or implicitly.
NOTE: The CMake cache variable type for all ``XXX_ENABLE_YYY`` variables is
actually ``STRING`` and not ``BOOL``. That is because these enable variables
take on the string enum values of ``"ON"``, ``"OFF"``, end empty ``""``. An
empty enable means that the TriBITS dependency system is allowed to decide if
an enable should be turned on or off based on various logic. The CMake GUI
will enforce the values of ``"ON"``, ``"OFF"``, and empty ``""`` but it will
not enforce this if you set the value on the command line or in a ``set()``
statement in an input ```*.cmake`` options files. However, setting
``-DXXX_ENABLE_YYY=TRUE`` and ``-DXXX_ENABLE_YYY=FALSE`` is allowed and will
be interpreted correctly..
Enable or disable tests for specific packages
+++++++++++++++++++++++++++++++++++++++++++++
The enable tests for explicitly enabled packages, configure with::
-D <Project>_ENABLE_<TRIBITS_PACKAGE_1>=ON \
-D <Project>_ENABLE_<TRIBITS_PACKAGE_2>=ON \
-D <Project>_ENABLE_TESTS=ON \
This will result in the enable of the test suites for any package that
explicitly enabled with ``-D <Project>_ENABLE_<TRIBITS_PACKAGE>=ON``. Note
that his will **not** result in the enable of the test suites for any packages
that may only be implicitly enabled in order to build the explicitly enabled
packages.
.. _<TRIBITS_PACKAGE>_ENABLE_TESTS:
If one wants to enable a package along with the enable of other packages, but
not the test suite for that package, then one can use a "exclude-list"
appraoch to disable the tests for that package by configuring with, for
example::
-D <Project>_ENABLE_<TRIBITS_PACKAGE_1>=ON \
-D <Project>_ENABLE_<TRIBITS_PACKAGE_2>=ON \
-D <Project>_ENABLE_<TRIBITS_PACKAGE_3>=ON \
-D <Project>_ENABLE_TESTS=ON \
-D <TRIBITS_PACKAGE_2>_ENABLE_TESTS=OFF \
The above will enable the package test suites for ``<TRIBITS_PACKGE_1>`` and
``<TRIBITS_PACKGE_3>`` but **not** for ``<TRIBITS_PACKAGE_2>`` (or any other
packages that might get implicitly enabled). One might use this approach if
one wants to build and install package ``<TRIBITS_PACKAGE_2>`` but does not
want to build and run the test suite for that package.
Alternatively, one can use an "include-list" appraoch to enable packages and
only enable tests for specific packages, for example, configuring with::
-D <Project>_ENABLE_<TRIBITS_PACKAGE_1>=ON \
-D <TRIBITS_PACKAGE_1>_ENABLE_TESTS=ON \
-D <Project>_ENABLE_<TRIBITS_PACKAGE_2>=ON \
-D <Project>_ENABLE_<TRIBITS_PACKAGE_3>=ON \
-D <TRIBITS_PACKAGE_3>_ENABLE_TESTS=ON \
That will have the same result as using the "exclude-list" approach above.
**NOTE:** Setting ``<TRIBITS_PACKAGE>_ENABLE_TESTS=ON`` will set
``<TRIBITS_PACKAGE>_ENABLE_EXAMPLES=ON`` by default. Also, setting
``<TRIBITS_PACKAGE>_ENABLE_TESTS=ON`` will result in setting
``<TRIBITS_PACKAGE><SP>_ENABLE_TESTS=ON`` for all subpackages in a parent
package that are explicitly enabled or are enabled in the forward sweep as a
result of `<Project>_ENABLE_ALL_FORWARD_DEP_PACKAGES`_ being set to ``ON``.
These and other options give the user complete control of what packages get
enabled or disabled and what package test suites are enabled or disabled.
Enable to test all effects of changing a given package(s)
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++
.. _<Project>_ENABLE_ALL_FORWARD_DEP_PACKAGES:
To enable a package ``<TRIBITS_PACKAGE>`` to test it and all of its
down-stream packages, configure with::
-D <Project>_ENABLE_<TRIBITS_PACKAGE>=ON \
-D <Project>_ENABLE_ALL_FORWARD_DEP_PACKAGES=ON \
-D <Project>_ENABLE_TESTS=ON \
The above set of arguments will result in package ``<TRIBITS_PACKAGE>`` and
all packages that depend on ``<TRIBITS_PACKAGE>`` to be enabled and have all
of their tests turned on. Tests will not be enabled in packages that do not
depend (at least implicitly) on ``<TRIBITS_PACKAGE>`` in this case. This
speeds up and robustifies testing for changes in specific packages (like in
per-merge testing in a continuous integration process).
NOTE: setting ``<Project>_ENABLE_ALL_FORWARD_DEP_PACKAGES=ON`` also
automatically sets and overrides `<Project>_ENABLE_ALL_OPTIONAL_PACKAGES`_ to
be ``ON`` as well. (It makes no sense to want to enable forward dependent
packages for testing purposes unless you are enabling all optional packages.)
Enable all packages (and optionally all tests)
++++++++++++++++++++++++++++++++++++++++++++++
To enable all defined packages, add the configure option::
-D <Project>_ENABLE_ALL_PACKAGES=ON \
To also optionally enable the tests and examples in all of those enabled
packages, add the configure option::
-D <Project>_ENABLE_TESTS=ON \
Specific packages can be disabled (i.e. "exclude-listed") by adding
``<Project>_ENABLE_<TRIBITS_PACKAGE>=OFF``. This will also disable all
packages that depend on ``<TRIBITS_PACKAGE>``.
Note, all examples are also enabled by default when setting
``<Project>_ENABLE_TESTS=ON``.
By default, setting ``<Project>_ENABLE_ALL_PACKAGES=ON`` only enables primary
tested (PT) packages and code. To have this also enable all secondary tested
(ST) packages and ST code in PT packages code, one must also set::
-D <Project>_ENABLE_SECONDARY_TESTED_CODE=ON \
NOTE: If this project is a "meta-project", then
``<Project>_ENABLE_ALL_PACKAGES=ON`` may not enable *all* the packages but
only the project's primary meta-project packages. See `Package Dependencies
and Enable/Disable Logic`_ and `TriBITS Dependency Handling Behaviors`_ for
details.
Disable a package and all its dependencies
++++++++++++++++++++++++++++++++++++++++++
To disable a package and all of the packages that depend on it, add the
configure option::
-D <Project>_ENABLE_<TRIBITS_PACKAGE>=OFF
For example::
-D <Project>_ENABLE_<TRIBITS_PACKAGE_A>=ON \
-D <Project>_ENABLE_ALL_OPTIONAL_PACKAGES=ON \
-D <Project>_ENABLE_<TRIBITS_PACKAGE_B>=OFF \
will enable ``<TRIBITS_PACKAGE_A>`` and all of the packages that it depends on
except for ``<TRIBITS_PACKAGE_B>`` and all of its forward dependencies.
If a TriBITS package ``<TRIBITS_PACKAGE>`` has subpackages (e.g. a parent
package with subpackages ``<A>``, ``<B>``, ...), then disabling the parent
package is equivalent to disabling all of the required and optional
subpackages::
-D <Project>_ENABLE_<TRIBITS_PACKAGE><A>=OFF \
-D <Project>_ENABLE_<TRIBITS_PACKAGE><B>=OFF \
...
The disable of the subpackages in this case will override any enables.
.. _<Project>_DISABLE_ENABLED_FORWARD_DEP_PACKAGES:
If a disabled package is a required dependency of some explicitly enabled
downstream package, then the configure will error out if::
-D <Project>_DISABLE_ENABLED_FORWARD_DEP_PACKAGES=OFF \
is set. Otherwise, if ``<Project>_DISABLE_ENABLED_FORWARD_DEP_PACKAGES=ON``,
a ``NOTE`` will be printed and the downstream package will be disabled and
configuration will continue.
Remove all package enables in the cache
+++++++++++++++++++++++++++++++++++++++
To wipe the set of package enables in the ``CMakeCache.txt`` file so they can
be reset again from scratch, re-configure with::
$ cmake -D <Project>_UNENABLE_ENABLED_PACKAGES=TRUE .
This option will set to empty '' all package enables, leaving all other cache
variables as they are. You can then reconfigure with a new set of package
enables for a different set of packages. This allows you to avoid more
expensive configure time checks (like the standard CMake compiler checks) and
to preserve other cache variables that you have set and don't want to loose.
For example, one would want to do this to avoid more expensive compiler and
TPL checks.
Selecting compiler and linker options
-------------------------------------
The compilers for C, C++, and Fortran will be found by default by CMake if
they are not otherwise specified as described below (see standard CMake
documentation for how default compilers are found). The most direct way to
set the compilers are to set the CMake cache variables::
-D CMAKE_<LANG>_COMPILER=<path-to-compiler>
The path to the compiler can be just a name of the compiler
(e.g. ``-DCMAKE_C_COMPILER=gcc``) or can be an absolute path
(e.g. ``-DCMAKE_C_COMPILER=/usr/local/bin/cc``). The safest and more direct
approach to determine the compilers is to set the absolute paths using, for
example, the cache variables::
-D CMAKE_C_COMPILER=/opt/my_install/bin/gcc \
-D CMAKE_CXX_COMPILER=/opt/my_install/bin/g++ \
-D CMAKE_Fortran_COMPILER=/opt/my_install/bin/gfortran
or if ``TPL_ENABLE_MPI=ON`` (see `Configuring with MPI support`_) something
like::
-D CMAKE_C_COMPILER=/opt/my_install/bin/mpicc \
-D CMAKE_CXX_COMPILER=/opt/my_install/bin/mpicxx \
-D CMAKE_Fortran_COMPILER=/opt/my_install/bin/mpif90
If these the CMake cache variables are not set, then CMake will use the
compilers specified in the environment variables ``CC``, ``CXX``, and ``FC``
for C, C++ and Fortran, respectively. If one needs to drill down through
different layers of scripts, then it can be useful to set the compilers using
these environment variables. But in general is it recommended to be explicit
and use the above CMake cache variables to set the absolute path to the
compilers to remove all ambiguity.
If absolute paths to the compilers are not specified using the CMake cache
variables or the environment variables as described above, then in MPI mode
(i.e. ``TPL_ENABLE_MPI=ON``) TriBITS performs its own search for the MPI
compiler wrappers that will find the correct compilers for most MPI
distributions (see `Configuring with MPI support`_). However, if in serial
mode (i.e. ``TPL_ENABLE_MPI=OFF``), then CMake will do its own default
compiler search. The algorithm by which raw CMake finds these compilers is
not precisely documented (and seems to change based on the platform).
However, on Linux systems, the observed algorithm appears to be:
1. Search for the C compiler first by looking in ``PATH`` (or the equivalent
on Windows), starting with a compiler with the name ``cc`` and then moving
on to other names like ``gcc``, etc. This first compiler found is set to
``CMAKE_C_COMPILER``.
2. Search for the C++ compiler with names like ``c++``, ``g++``, etc., but
restrict the search to the same directory specified by base path to the C
compiler given in the variable ``CMAKE_C_COMPILER``. The first compiler
that is found is set to ``CMAKE_CXX_COMPILER``.
3. Search for the Fortran compiler with names like ``f90``, ``gfortran``,
etc., but restrict the search to the same directory specified by base path
to the C compiler given in the variable ``CMAKE_C_COMPILER``. The first
compiler that is found is set to ``CMAKE_Fortran_COMPILER``.
**WARNING:** While this built-in CMake compiler search algorithm may seems
reasonable, it fails to find the correct compilers in many cases for a non-MPI
serial build. For example, if a newer version of GCC is installed and is put
first in ``PATH``, then CMake will fail to find the updated ``gcc`` compiler
and will instead find the default system ``cc`` compiler (usually under
``/usr/bin/cc`` on Linux may systems) and will then only look for the C++ and
Fortran compilers under that directory. This will fail to find the correct
updated compilers because GCC does not install a C compiler named ``cc``!
Therefore, if you want to use the default CMake compiler search to find the
updated GCC compilers, you can set the CMake cache variable::
-D CMAKE_C_COMPILER=gcc
or can set the environment variable ``CC=gcc``. Either one of these will
result in CMake finding the updated GCC compilers found first in ``PATH``.
Once one has specified the compilers, one can also set the compiler flags, but
the way that CMake does this is a little surprising to many people. But the
<Project> TriBITS CMake build system offers the ability to tweak the built-in
CMake approach for setting compiler flags. First some background is in order.
When CMake creates the object file build command for a given source file, it
passes in flags to the compiler in the order::
${CMAKE_<LANG>_FLAGS} ${CMAKE_<LANG>_FLAGS_<CMAKE_BUILD_TYPE>}
where ``<LANG>`` = ``C``, ``CXX``, or ``Fortran`` and ``<CMAKE_BUILD_TYPE>`` =
``DEBUG`` or ``RELEASE``. Note that the options in
``CMAKE_<LANG>_FLAGS_<CMAKE_BUILD_TYPE>`` come after and override those in
``CMAKE_<LANG>_FLAGS``! The flags in ``CMAKE_<LANG>_FLAGS`` apply to all
build types. Optimization, debug, and other build-type-specific flags are set
in ``CMAKE_<LANG>_FLAGS_<CMAKE_BUILD_TYPE>``. CMake automatically provides a
default set of debug and release optimization flags for
``CMAKE_<LANG>_FLAGS_<CMAKE_BUILD_TYPE>`` (e.g. ``CMAKE_CXX_FLAGS_DEBUG`` is
typically ``"-g -O0"`` while ``CMAKE_CXX_FLAGS_RELEASE`` is typically
``"-O3"``). This means that if you try to set the optimization level with
``-DCMAKE_CXX_FLAGS="-04"``, then this level gets overridden by the flags
specified in ``CMAKE_<LANG>_FLAGS_BUILD`` or ``CMAKE_<LANG>_FLAGS_RELEASE``.
TriBITS will set defaults for ``CMAKE_<LANG>_FLAGS`` and
``CMAKE_<LANG>_FLAGS_<CMAKE_BUILD_TYPE>``, which may be different that what
raw CMake would set. TriBITS provides a means for project and package
developers and users to set and override these compiler flag variables
globally and on a package-by-package basis. Below, the facilities for
manipulating compiler flags is described.
To see that the full set of compiler flags one has to actually build a target
by running, for example, ``make VERBOSE=1 <target_name>`` (see `Building with
verbose output without reconfiguring`_). (NOTE: One can also see the exact
set of flags used for each target in the generated ``build.ninja`` file when
using the Ninja generator.) One cannot just look at the cache variables for
``CMAKE_<LANG>_FLAGS`` and ``CMAKE_<LANG>_FLAGS_<CMAKE_BUILD_TYPE>`` in the
file ``CMakeCache.txt`` and see the full set of flags are actually being used.
These variables can override the cache variables by TriBITS as project-level
local non-cache variables as described below (see `Overriding CMAKE_BUILD_TYPE
debug/release compiler options`_).
The <Project> TriBITS CMake build system will set up default compile flags for
GCC ('GNU') in development mode
(i.e. ``<Project>_ENABLE_DEVELOPMENT_MODE=ON``) on order to help produce
portable code. These flags set up strong warning options and enforce language
standards. In release mode (i.e. ``<Project>_ENABLE_DEVELOPMENT_MODE=OFF``),
these flags are not set. These flags get set internally into the variables
``CMAKE_<LANG>_FLAGS`` (when processing packages, not at the global cache
variable level) but the user can append flags that override these as described
below.
Configuring to build with default debug or release compiler flags
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
.. _CMAKE_BUILD_TYPE:
To build a debug version, pass into 'cmake'::
-D CMAKE_BUILD_TYPE=DEBUG
This will result in debug flags getting passed to the compiler according to
what is set in ``CMAKE_<LANG>_FLAGS_DEBUG``.
To build a release (optimized) version, pass into 'cmake'::
-D CMAKE_BUILD_TYPE=RELEASE
This will result in optimized flags getting passed to the compiler according
to what is in ``CMAKE_<LANG>_FLAGS_RELEASE``.
The default build type is typically ``CMAKE_BUILD_TYPE=RELEASE`` unless ``-D
USE_XSDK_DEFAULTS=TRUE`` is set in which case the default build type is
``CMAKE_BUILD_TYPE=DEBUG`` as per the xSDK configure standard.
Adding arbitrary compiler flags but keeping default build-type flags
++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
To append arbitrary compiler flags to ``CMAKE_<LANG>_FLAGS`` (which may be
set internally by TriBITS) that apply to all build types, configure with::
-D CMAKE_<LANG>_FLAGS="<EXTRA_COMPILER_OPTIONS>"
where ``<EXTRA_COMPILER_OPTIONS>`` are your extra compiler options like
``"-DSOME_MACRO_TO_DEFINE -funroll-loops"``. These options will get
appended to (i.e. come after) other internally defined compiler option and
therefore override them. The options are then pass to the compiler in the
order::
<DEFAULT_TRIBITS_LANG_FLAGS> <EXTRA_COMPILER_OPTIONS> \
${CMAKE_<LANG>_FLAGS_<CMAKE_BUILD_TYPE>}
This that setting ``CMAKE_<LANG>_FLAGS`` can override the default flags that
TriBITS will set for ``CMAKE_<LANG>_FLAGS`` but will **not** override flags
specified in ``CMAKE_<LANG>_FLAGS_<CMAKE_BUILD_TYPE>``.
Instead of directly setting the CMake cache variables ``CMAKE_<LANG>_FLAGS``
one can instead set environment variables ``CFLAGS``, ``CXXFLAGS`` and
``FFLAGS`` for ``CMAKE_C_FLAGS``, ``CMAKE_CXX_FLAGS`` and
``CMAKE_Fortran_FLAGS``, respectively.
In addition, if ``-DUSE_XSDK_DEFAULTS=TRUE`` is set, then one can also pass
in Fortran flags using the environment variable ``FCFLAGS`` (raw CMake does
not recognize ``FCFLAGS``). But if ``FFLAGS`` and ``FCFLAGS`` are both set,
then they must be the same or a configure error will occur.
Options can also be targeted to a specific TriBITS package using::
-D <TRIBITS_PACKAGE>_<LANG>_FLAGS="<PACKAGE_EXTRA_COMPILER_OPTIONS>"
The package-specific options get appended **after** those already in
``CMAKE_<LANG>_FLAGS`` and therefore override (but not replace) those set
globally in ``CMAKE_<LANG>_FLAGS`` (either internally in the CMakeLists.txt
files or by the user in the cache).
In addition, flags can be targeted to a specific TriBITS subpackage using the
same syntax::
-D <TRIBITS_SUBPACKAGE>_<LANG>_FLAGS="<SUBPACKAGE_EXTRA_COMPILER_OPTIONS>"
If top-level package-specific flags and subpackage-specific flags are both set
for the same parent package such as with::
-D SomePackage_<LANG>_FLAGS="<Package-flags>" \
-D SomePackageSpkgA_<LANG>_FLAGS="<Subpackage-flags>" \
then the flags for the subpackage ``SomePackageSpkgA`` will be listed after
those for its parent package ``SomePackage`` on the compiler command-line as::
<Package-flags> <SubPackage-flags>
That way, compiler options for a subpackage override flags set for the parent
package.
NOTES:
1) Setting ``CMAKE_<LANG>_FLAGS`` as a cache variable by the user on input be
listed after and therefore override, but will not replace, any internally set
flags in ``CMAKE_<LANG>_FLAGS`` defined by the <Project> CMake system. To get
rid of these project/TriBITS set compiler flags/options, see the below items.
2) Given that CMake passes in flags in
``CMAKE_<LANG>_FLAGS_<CMAKE_BUILD_TYPE>`` after those in
``CMAKE_<LANG>_FLAGS`` means that users setting the ``CMAKE_<LANG>_FLAGS``
and ``<TRIBITS_PACKAGE>_<LANG>_FLAGS`` will **not** override the flags in
``CMAKE_<LANG>_FLAGS_<CMAKE_BUILD_TYPE>`` which come after on the compile
line. Therefore, setting ``CMAKE_<LANG>_FLAGS`` and
``<TRIBITS_PACKAGE>_<LANG>_FLAGS`` should only be used for options that will
not get overridden by the debug or release compiler flags in
``CMAKE_<LANG>_FLAGS_<CMAKE_BUILD_TYPE>``. However, setting
``CMAKE_<LANG>_FLAGS`` will work well for adding extra compiler defines
(e.g. -DSOMETHING) for example.
WARNING: Any options that you set through the cache variable
``CMAKE_<LANG>_FLAGS_<CMAKE_BUILD_TYPE>`` will get overridden in the
<Project> CMake system for GNU compilers in development mode so don't try to
manually set ``CMAKE_<LANG>_FLAGS_<CMAKE_BUILD_TYPE>`` directly! To
override those options, see
``CMAKE_<LANG>_FLAGS_<CMAKE_BUILD_TYPE>_OVERRIDE`` below.
Overriding CMAKE_BUILD_TYPE debug/release compiler options
++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
To override the default CMake-set options in
``CMAKE_<LANG>_FLAGS_<CMAKE_BUILD_TYPE>``, use::
-D CMAKE_<LANG>_FLAGS_<CMAKE_BUILD_TYPE>_OVERRIDE="<OPTIONS_TO_OVERRIDE>"
For example, to default debug options use::
-D CMAKE_C_FLAGS_DEBUG_OVERRIDE="-g -O1" \
-D CMAKE_CXX_FLAGS_DEBUG_OVERRIDE="-g -O1"
-D CMAKE_Fortran_FLAGS_DEBUG_OVERRIDE="-g -O1"
and to override default release options use::
-D CMAKE_C_FLAGS_RELEASE_OVERRIDE="-O3 -funroll-loops" \
-D CMAKE_CXX_FLAGS_RELEASE_OVERRIDE="-03 -funroll-loops"
-D CMAKE_Fortran_FLAGS_RELEASE_OVERRIDE="-03 -funroll-loops"
NOTES: The TriBITS CMake cache variable
``CMAKE_<LANG>_FLAGS_<CMAKE_BUILD_TYPE>_OVERRIDE`` is used and not
``CMAKE_<LANG>_FLAGS_<CMAKE_BUILD_TYPE>`` because is given a default
internally by CMake and the new variable is needed to make the override
explicit.
Turning off strong warnings for individual packages
+++++++++++++++++++++++++++++++++++++++++++++++++++
.. _<TRIBITS_PACKAGE>_DISABLE_STRONG_WARNINGS:
To turn off strong warnings (for all languages) for a given TriBITS package,
set::
-D <TRIBITS_PACKAGE>_DISABLE_STRONG_WARNINGS=ON
This will only affect the compilation of the sources for
``<TRIBITS_PACKAGES>``, not warnings generated from the header files in
downstream packages or client code.
Note that strong warnings are only enabled by default in development mode
(``<Project>_ENABLE_DEVELOPMENT_MODE==ON``) but not release mode
(``<Project>_ENABLE_DEVELOPMENT_MODE==ON``). A release of <Project> should
therefore not have strong warning options enabled.
Overriding all (strong warnings and debug/release) compiler options
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
To override all compiler options, including both strong warning options
and debug/release options, configure with::
-D CMAKE_C_FLAGS="-O3 -funroll-loops" \
-D CMAKE_CXX_FLAGS="-03 -fexceptions" \
-D CMAKE_BUILD_TYPE=NONE \
-D <Project>_ENABLE_STRONG_C_COMPILE_WARNINGS=OFF \
-D <Project>_ENABLE_STRONG_CXX_COMPILE_WARNINGS=OFF \
-D <Project>_ENABLE_SHADOW_WARNINGS=OFF \
-D <Project>_ENABLE_COVERAGE_TESTING=OFF \
-D <Project>_ENABLE_CHECKED_STL=OFF \
NOTE: Options like ``<Project>_ENABLE_SHADOW_WARNINGS``,
``<Project>_ENABLE_COVERAGE_TESTING``, and ``<Project>_ENABLE_CHECKED_STL``
do not need to be turned off by default but they are shown above to make it
clear what other CMake cache variables can add compiler and link arguments.
NOTE: By setting ``CMAKE_BUILD_TYPE=NONE``, then ``CMAKE_<LANG>_FLAGS_NONE``
will be empty and therefore the options set in ``CMAKE_<LANG>_FLAGS`` will
be all that is passed in.
Enable and disable shadowing warnings for all <Project> packages
++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
To enable shadowing warnings for all <Project> packages (that don't already
have them turned on) then use::
-D <Project>_ENABLE_SHADOW_WARNINGS=ON
To disable shadowing warnings for all <Project> packages (even those that
have them turned on by default) then use::
-D <Project>_ENABLE_SHADOW_WARNINGS=OFF
NOTE: The default value is empty '' which lets each <Project> package
decide for itself if shadowing warnings will be turned on or off for that
package.
Removing warnings as errors for CLEANED packages
++++++++++++++++++++++++++++++++++++++++++++++++
To remove the ``-Werror`` flag (or some other flag that is set) from being
applied to compile CLEANED packages (like the Trilinos package Teuchos), set
the following when configuring::
-D <Project>_WARNINGS_AS_ERRORS_FLAGS=""
Adding debug symbols to the build
+++++++++++++++++++++++++++++++++
To get the compiler to add debug symbols to the build, configure with::
-D <Project>_ENABLE_DEBUG_SYMBOLS=ON
This will add ``-g`` on most compilers. NOTE: One does **not** generally need
to create a full debug build to get debug symbols on most compilers.
Printing out compiler flags for each package
++++++++++++++++++++++++++++++++++++++++++++
To print out the exact ``CMAKE_<LANG>_FLAGS`` that will be used for each
package, set::
-D <Project>_PRINT_PACKAGE_COMPILER_FLAGS=ON
That will print lines in STDOUT that are formatted as::
<TRIBITS_SUBPACKAGE>: CMAKE_<LANG>_FLAGS="<exact-flags-usedy-by-package>"
<TRIBITS_SUBPACKAGE>: CMAKE_<LANG>_FLAGS_<BUILD_TYPE>="<build-type-flags>"
This will print the value of the ``CMAKE_<LANG>_FLAGS`` and
``CMAKE_<LANG>_FLAGS_<BUILD_TYPE>`` variables that are used as each package is
being processed and will contain the flags in the exact order they are applied
by CMake
Appending arbitrary libraries and link flags every executable
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
In order to append any set of arbitrary libraries and link flags to your
executables use::
-D<Project>_EXTRA_LINK_FLAGS="<EXTRA_LINK_LIBRARIES>" \
-DCMAKE_EXE_LINKER_FLAGS="<EXTRA_LINK_FLAGG>"
Above, you can pass any type of library and they will always be the last
libraries listed, even after all of the TPLs.
NOTE: This is how you must set extra libraries like Fortran libraries and
MPI libraries (when using raw compilers). Please only use this variable
as a last resort.
NOTE: You must only pass in libraries in ``<Project>_EXTRA_LINK_FLAGS`` and
*not* arbitrary linker flags. To pass in extra linker flags that are not
libraries, use the built-in CMake variable ``CMAKE_EXE_LINKER_FLAGS``
instead. The TriBITS variable ``<Project>_EXTRA_LINK_FLAGS`` is badly named
in this respect but the name remains due to backward compatibility
requirements.
Enabling support for Ninja
--------------------------
The `Ninja`_ build tool can be used as the back-end build tool instead of
Makefiles by adding::
-GNinja
to the CMake configure line (the default on most Linux and OSX platforms is
``-G"Unix Makefiles"``). This instructs CMake to create the back-end
``ninja`` build files instead of back-end Makefiles (see `Building (Ninja
generator)`_).
.. _<Project>_WRITE_NINJA_MAKEFILES:
In addition, the TriBITS build system will, by default, generate Makefiles in
every binary directory where there is a CMakeLists.txt file in the source
tree. These Makefiles have targets scoped to that subdirectory that use
``ninja`` to build targets in that subdirectory just like with the native
CMake recursive ``-G "Unix Makefiles"`` generator. This allows one to ``cd``
into any binary directory and type ``make`` to build just the targets in that
directory. These TriBITS-generated Ninja makefiles also support ``help`` and
``help-objects`` targets making it easy to build individual executables,
libraries and object files in any binary subdirectory.
**WARNING:** Using ``make -j<N>`` with these TriBITS-generated Ninja Makefiles
will **not** result in using ``<N>`` processes to build in parallel and will
instead use **all** of the free cores to build on the machine! To control the
number of processes used, run ``make NP=<N>`` instead! See `Building in
parallel with Ninja`_.
The generation of these Ninja makefiles can be disabled by setting::
-D<Project>_WRITE_NINJA_MAKEFILES=OFF
(But these Ninja Makefiles get created very quickly even for a very large
CMake project so there is usually little reason to not generate them.)
Limiting parallel compile and link jobs for Ninja builds
--------------------------------------------------------
When the CMake generator Ninja is used (i.e. ``-GNinja``), one can limit the
number of parallel jobs that are used for compiling object files by setting::
-D <Project>_PARALLEL_COMPILE_JOBS_LIMIT=<N>
and/or limit the number of parallel jobs that are used for linking libraries
and executables by setting::
-D <Project>_PARALLEL_LINK_JOBS_LIMIT=<M>
where ``<N>`` and ``<M>`` are integers like ``20`` and ``4``. If these are
not set, then the number of parallel jobs will be determined by the ``-j<P>``
argument passed to ``ninja -j<P>`` or by ninja automatically according to
machine load when running ``ninja``.
Limiting the number of link jobs can be useful, for example, for certain
builds of large projects where linking many jobs in parallel can consume all
of the RAM on a given system and crash the build.
NOTE: These options are ignored when using Makefiles or other CMake
generators. They only work for the Ninja generator.
Disabling explicit template instantiation for C++
-------------------------------------------------
By default, support for optional explicit template instantiation (ETI) for C++
code is enabled. To disable support for optional ETI, configure with::
-D <Project>_ENABLE_EXPLICIT_INSTANTIATION=OFF
When ``OFF``, all packages that have templated C++ code will use implicit
template instantiation (unless they have hard-coded usage of ETI).
ETI can be enabled (``ON``) or disabled (``OFF``) for individual packages
with::
-D <TRIBITS_PACKAGE>_ENABLE_EXPLICIT_INSTANTIATION=[ON|OFF]
The default value for ``<TRIBITS_PACKAGE>_ENABLE_EXPLICIT_INSTANTIATION`` is
set by ``<Project>_ENABLE_EXPLICIT_INSTANTIATION``.
For packages that support it, explicit template instantiation can massively
reduce the compile times for the C++ code involved and can even avoid compiler
crashes in some cases. To see what packages support explicit template
instantiation, just search the CMakeCache.txt file for variables with
``ENABLE_EXPLICIT_INSTANTIATION`` in the name.
Disabling the Fortran compiler and all Fortran code
---------------------------------------------------
To disable the Fortran compiler and all <Project> code that depends on Fortran
set::
-D <Project>_ENABLE_Fortran=OFF
NOTE: The Fortran compiler may be disabled automatically by default on systems
like MS Windows.
NOTE: Most Apple Macs do not come with a compatible Fortran compiler by
default so you must turn off Fortran if you don't have a compatible Fortran
compiler.
Enabling runtime debug checking
-------------------------------
a) Enabling <Project> ifdefed runtime debug checking:
To turn on optional ifdefed runtime debug checking, configure with::
-D <Project>_ENABLE_DEBUG=ON
This will result in a number of ifdefs to be enabled that will perform a
number of runtime checks. Nearly all of the debug checks in <Project> will
get turned on by default by setting this option. This option can be set
independent of ``CMAKE_BUILD_TYPE`` (which sets the compiler debug/release
options).
NOTES:
* The variable ``CMAKE_BUILD_TYPE`` controls what compiler options are
passed to the compiler by default while ``<Project>_ENABLE_DEBUG``
controls what defines are set in config.h files that control ifdefed debug
checks.
* Setting ``-DCMAKE_BUILD_TYPE=DEBUG`` will automatically set the
default ``<Project>_ENABLE_DEBUG=ON``.
b) Enabling checked STL implementation:
To turn on the checked STL implementation set::
-D <Project>_ENABLE_CHECKED_STL=ON
NOTES:
* By default, this will set -D_GLIBCXX_DEBUG as a compile option for all C++
code. This only works with GCC currently.
* This option is disabled by default because to enable it by default can
cause runtime segfaults when linked against C++ code that was compiled
without -D_GLIBCXX_DEBUG.
Configuring with MPI support
----------------------------
To enable MPI support you must minimally set::
-D TPL_ENABLE_MPI=ON
There is built-in logic to try to find the various MPI components on your
system but you can override (or make suggestions) with::
-D MPI_BASE_DIR="path"
(Base path of a standard MPI installation which has the subdirs 'bin', 'libs',
'include' etc.)
or::
-D MPI_BIN_DIR="path1;path2;...;pathn"
which sets the paths where the MPI executables (e.g. mpiCC, mpicc, mpirun,
mpiexec) can be found. By default this is set to ``${MPI_BASE_DIR}/bin`` if
``MPI_BASE_DIR`` is set.
**NOTE:** TriBITS uses the MPI compiler wrappers (e.g. mpiCC, mpicc, mpic++,
mpif90, etc.) which is more standard with other builds systems for HPC
computing using MPI (and the way that MPI implementations were meant to be
used). But directly using the MPI compiler wrappers as the direct compilers
is inconsistent with the way that the standard CMake module ``FindMPI.cmake``
which tries to "unwrap" the compiler wrappers and grab out the raw underlying
compilers and the raw compiler and linker command-line arguments. In this
way, TriBITS is more consistent with standard usage in the HPC community but
is less consistent with CMake (see "HISTORICAL NOTE" below).
There are several different different variations for configuring with MPI
support:
a) **Configuring build using MPI compiler wrappers:**
The MPI compiler wrappers are turned on by default. There is built-in logic
in TriBITS that will try to find the right MPI compiler wrappers. However,
you can specifically select them by setting, for example::
-D MPI_C_COMPILER:FILEPATH=mpicc \
-D MPI_CXX_COMPILER:FILEPATH=mpic++ \
-D MPI_Fortan_COMPILER:FILEPATH=mpif77
which gives the name of the MPI C/C++/Fortran compiler wrapper executable.
In this case, just the names of the programs are given and absolute path of
the executables will be searched for under ``${MPI_BIN_DIR}/`` if the cache
variable ``MPI_BIN_DIR`` is set, or in the default path otherwise. The
found programs will then be used to set the cache variables
``CMAKE_[C,CXX,Fortran]_COMPILER``.
One can avoid the search and just use the absolute paths with, for example::
-D MPI_C_COMPILER:FILEPATH=/opt/mpich/bin/mpicc \
-D MPI_CXX_COMPILER:FILEPATH=/opt/mpich/bin/mpic++ \
-D MPI_Fortan_COMPILER:FILEPATH=/opt/mpich/bin/mpif77
However, you can also directly set the variables
``CMAKE_[C,CXX,Fortran]_COMPILER`` with, for example::
-D CMAKE_C_COMPILER:FILEPATH=/opt/mpich/bin/mpicc \
-D CMAKE_CXX_COMPILER:FILEPATH=/opt/mpich/bin/mpic++ \
-D CMAKE_Fortan_COMPILER:FILEPATH=/opt/mpich/bin/mpif77
**WARNING:** If you set just the compiler names and not the absolute paths
with ``CMAKE_<LANG>_COMPILER`` in MPI mode, then a search will not be done
and these will be expected to be in the path at build time. (Note that his
is inconsistent the behavior of raw CMake in non-MPI mode described in
`Selecting compiler and linker options`_). If both
``CMAKE_<LANG>_COMPILER`` and ``MPI_<LANG>_COMPILER`` are set, however, then
``CMAKE_<LANG>_COMPILER`` will be used and ``MPI_<LANG>_COMPILER`` will be
ignored.
Note that when ``USE_XSDK_DEFAULTS=FALSE`` (see `xSDK Configuration
Options`_), then the environment variables ``CC``, ``CXX`` and ``FC`` are
ignored. But when ``USE_XSDK_DEFAULTS=TRUE`` and the CMake cache variables
``CMAKE_[C,CXX,Fortran]_COMPILER`` are not set, then the environment
variables ``CC``, ``CXX`` and ``FC`` will be used for
``CMAKE_[C,CXX,Fortran]_COMPILER``, even if the CMake cache variables
``MPI_[C,CXX,Fortran]_COMPILER`` are set! So if one wants to make sure and
set the MPI compilers irrespective of the xSDK mode, then one should set
cmake cache variables ``CMAKE_[C,CXX,Fortran]_COMPILER`` to the absolute
path of the MPI compiler wrappers.
**HISTORICAL NOTE:** The TriBITS system has its own custom MPI integration
support and does not (currently) use the standard CMake module
``FindMPI.cmake``. This custom support for MPI was added to TriBITS in 2008
when it was found the built-in ``FindMPI.cmake`` module was not sufficient
for the needs of Trilinos and the approach taken by the module (still in use
as of CMake 3.4.x) which tries to unwrap the raw compilers and grab the list
of include directories, link libraries, etc, was not sufficiently portable
for the systems where Trilinos needed to be used. But earlier versions of
TriBITS used the ``FindMPI.cmake`` module and that is why the CMake cache
variables ``MPI_[C,CXX,Fortran]_COMPILER`` are defined and still supported.
b) **Configuring to build using raw compilers and flags/libraries:**
While using the MPI compiler wrappers as described above is the preferred
way to enable support for MPI, you can also just use the raw compilers and
then pass in all of the other information that will be used to compile and
link your code.
To turn off the MPI compiler wrappers, set::
-D MPI_USE_COMPILER_WRAPPERS=OFF
You will then need to manually pass in the compile and link lines needed to
compile and link MPI programs. The compile flags can be set through::
-D CMAKE_[C,CXX,Fortran]_FLAGS="$EXTRA_COMPILE_FLAGS"
The link and library flags must be set through::
-D <Project>_EXTRA_LINK_FLAGS="$EXTRA_LINK_FLAGS"
Above, you can pass any type of library or other linker flags in and they
will always be the last libraries listed, even after all of the TPLs.
NOTE: A good way to determine the extra compile and link flags for MPI is to
use::
export EXTRA_COMPILE_FLAGS="`$MPI_BIN_DIR/mpiCC --showme:compile`"
export EXTRA_LINK_FLAGS="`$MPI_BIN_DIR/mpiCC --showme:link`"
where ``MPI_BIN_DIR`` is set to your MPI installations binary directory.
c) **Setting up to run MPI programs:**
In order to use the ctest program to run MPI tests, you must set the mpi
run command and the options it takes. The built-in logic will try to find
the right program and options but you will have to override them in many
cases.
MPI test and example executables are passed to CTest ``add_test()`` as::
add_test(
${MPI_EXEC} ${MPI_EXEC_PRE_NUMPROCS_FLAGS}
${MPI_EXEC_NUMPROCS_FLAG} <NP>
${MPI_EXEC_POST_NUMPROCS_FLAGS}
<TEST_EXECUTABLE_PATH> <TEST_ARGS> )
where ``<TEST_EXECUTABLE_PATH>``, ``<TEST_ARGS>``, and ``<NP>`` are specific
to the test being run.
The test-independent MPI arguments are::
-D MPI_EXEC:FILEPATH="exec_name"
(The name of the MPI run command (e.g. mpirun, mpiexec) that is used to run
the MPI program. This can be just the name of the program in which case
the full path will be looked for in ``${MPI_BIN_DIR}`` as described above.
If it is an absolute path, it will be used without modification.)
::
-D MPI_EXEC_DEFAULT_NUMPROCS=4
(The default number of processes to use when setting up and running
MPI test and example executables. The default is set to '4' and only
needs to be changed when needed or desired.)
::
-D MPI_EXEC_MAX_NUMPROCS=4
(The maximum number of processes to allow when setting up and running MPI
tests and examples that use MPI. The default is set to '4' but should be
set to the largest number that can be tolerated for the given machine or the
most cores on the machine that you want the test suite to be able to use.
Tests and examples that require more processes than this are excluded from
the CTest test suite at configure time. ``MPI_EXEC_MAX_NUMPROCS`` is also
used to exclude tests in a non-MPI build (i.e. ``TPL_ENABLE_MPI=OFF``) if
the number of required cores for a given test is greater than this value.)
::
-D MPI_EXEC_NUMPROCS_FLAG=-np
(The command-line option just before the number of processes to use
``<NP>``. The default value is based on the name of ``${MPI_EXEC}``, for
example, which is ``-np`` for OpenMPI.)
::
-D MPI_EXEC_PRE_NUMPROCS_FLAGS="arg1;arg2;...;argn"
(Other command-line arguments that must come *before* the numprocs
argument. The default is empty "".)
::
-D MPI_EXEC_POST_NUMPROCS_FLAGS="arg1;arg2;...;argn"
(Other command-line arguments that must come *after* the numprocs
argument. The default is empty "".)
NOTE: Multiple arguments listed in ``MPI_EXEC_PRE_NUMPROCS_FLAGS`` and
``MPI_EXEC_POST_NUMPROCS_FLAGS`` must be quoted and separated by ``';'`` as
these variables are interpreted as CMake arrays.
Configuring for OpenMP support
------------------------------
To enable OpenMP support, one must set::
-D <Project>_ENABLE_OpenMP=ON
Note that if you enable OpenMP directly through a compiler option (e.g.,
``-fopenmp``), you will NOT enable OpenMP inside <Project> source code.
To skip adding flags for OpenMP for ``<LANG>`` = ``C``, ``CXX``, or
``Fortran``, use::
-D OpenMP_<LANG>_FLAGS_OVERRIDE=" "
The single space " " will result in no flags getting added. This is needed
since one can't set the flags ``OpenMP_<LANG>_FLAGS`` to an empty string or
the ``find_package(OpenMP)`` command will fail. Setting the variable
``-DOpenMP_<LANG>_FLAGS_OVERRIDE= " "`` is the only way to enable OpenMP but
skip adding the OpenMP flags provided by ``find_package(OpenMP)``.
Building shared libraries
-------------------------
.. _BUILD_SHARED_LIBS:
To configure to build shared libraries, set::
-D BUILD_SHARED_LIBS=ON
The above option will result in all shared libraries to be build on all
systems (i.e., ``.so`` on Unix/Linux systems, ``.dylib`` on Mac OS X, and
``.dll`` on Windows systems).
NOTE: If the project has ``USE_XSDK_DEFAULTS=ON`` set, then this will set
``BUILD_SHARED_LIBS=TRUE`` by default. Otherwise, the default is
``BUILD_SHARED_LIBS=FALSE``
Many systems support a feature called ``RPATH`` when shared libraries are used
that embeds the default locations to look for shared libraries when an
executable is run. By default on most systems, CMake will automatically add
RPATH directories to shared libraries and executables inside of the build
directories. This allows running CMake-built executables from inside the
build directory without needing to set ``LD_LIBRARY_PATH`` on any other
environment variables. However, this can be disabled by setting::
-D CMAKE_SKIP_BUILD_RPATH=TRUE
but it is hard to find a use case where that would be useful.
Building static libraries and executables
-----------------------------------------
To build static libraries, turn off the shared library support::
-D BUILD_SHARED_LIBS=OFF
Some machines, such as the Cray XT5, require static executables. To build
<Project> executables as static objects, a number of flags must be set::
-D BUILD_SHARED_LIBS=OFF \
-D TPL_FIND_SHARED_LIBS=OFF \
-D <Project>_LINK_SEARCH_START_STATIC=ON
The first flag tells cmake to build static versions of the <Project>
libraries. The second flag tells cmake to locate static library versions of
any required TPLs. The third flag tells the auto-detection routines that
search for extra required libraries (such as the mpi library and the gfortran
library for gnu compilers) to locate static versions.
Changing include directories in downstream CMake projects to non-system
-----------------------------------------------------------------------
By default, include directories from IMPORTED library targets from the
<Project> project's installed ``<Package>Config.cmake`` files will be
considered ``SYSTEM`` headers and therefore will be included on the compile
lines of downstream CMake projects with ``-isystem`` with most compilers.
However, when using CMake 3.23+, by configuring with::
-D <Project>_IMPORTED_NO_SYSTEM=ON
then all of the IMPORTED library targets in the set of installed
``<Package>Config.cmake`` files will have the ``IMPORTED_NO_SYSTEM`` target
property set. This will cause downstream customer CMake projects to apply the
include directories from these IMPORTED library targets as non-SYSTEM include
directories. On most compilers, that means that the include directories will
be listed on the compile lines with ``-I`` instead of with ``-isystem`` (for
compilers that support the ``-isystem`` option). (Changing from ``-isystem
<incl-dir>`` to ``-I <incl-dir>`` moves ``<incl-dir>`` forward in the
compiler's include directory search order and could also result in the found
header files emitting compiler warnings that would other otherwise be silenced
when the headers were found in include directories pulled in with
``-isystem``.)
**NOTE:** Setting ``<Project>_IMPORTED_NO_SYSTEM=ON`` when using a CMake
version less than 3.23 will result in a fatal configure error (so don't do
that).
**A workaround for CMake versions less than 3.23** is for **downstream
customer CMake projects** to set the native CMake cache variable::
-D CMAKE_NO_SYSTEM_FROM_IMPORTED=TRUE
This will result in **all** include directories from **all** IMPORTED library
targets used in the downstream customer CMake project to be listed on the
compile lines using ``-I`` instead of ``-isystem``, and not just for the
IMPORTED library targets from this <Project> project's installed
``<Package>Config.cmake`` files!
**NOTE:** Setting ``CMAKE_NO_SYSTEM_FROM_IMPORTED=TRUE`` in the <Project>
CMake configure will **not** result in changing how include directories from
<Project>'s IMPORTED targets are handled in a downstream customer CMake
project! It will only change how include directories from upstream package's
IMPORTED targets are handled in the <Project> CMake project build itself.
Enabling the usage of resource files to reduce length of build lines
--------------------------------------------------------------------
When using the ``Unix Makefile`` generator and the ``Ninja`` generator, CMake
supports some very useful (undocumented) options for reducing the length of
the command-lines used to build object files, create libraries, and link
executables. Using these options can avoid troublesome "command-line too
long" errors, "Error 127" library creation errors, and other similar errors
related to excessively long command-lines to build various targets.
When using the ``Unix Makefile`` generator, CMake responds to the three cache
variables ``CMAKE_CXX_USE_RESPONSE_FILE_FOR_INCLUDES``,
``CMAKE_CXX_USE_RESPONSE_FILE_FOR_OBJECTS`` and
``CMAKE_CXX_USE_RESPONSE_FILE_FOR_LIBRARIES`` described below.
To aggregate the list of all of the include directories (e.g. ``'-I
<full_path>'``) into a single ``*.rsp`` file for compiling object files, set::
-D CMAKE_CXX_USE_RESPONSE_FILE_FOR_INCLUDES=ON
To aggregate the list of all of the object files (e.g. ``'<path>/<name>.o'``)
into a single ``*.rsp`` file for creating libraries or linking executables,
set::
-D CMAKE_CXX_USE_RESPONSE_FILE_FOR_OBJECTS=ON
To aggregate the list of all of the libraries (e.g. ``'<path>/<libname>.a'``)
into a single ``*.rsp`` file for creating shared libraries or linking
executables, set::
-D CMAKE_CXX_USE_RESPONSE_FILE_FOR_LIBRARIES=ON
When using the ``Ninja`` generator, CMake only responds to the single option::
-D CMAKE_NINJA_FORCE_RESPONSE_FILE=ON
which turns on the usage of ``*.rsp`` response files for include directories,
object files, and libraries (and therefore is equivalent to setting the above
three ``Unix Makefiles`` generator options to ``ON``).
This feature works well on most standard systems but there are problems in
some situations and therefore these options can only be safely enabled on
case-by-case basis -- experimenting to ensure they are working correctly.
Some examples of some known problematic cases (as of CMake 3.11.2) are:
* CMake will only use resource files with static libraries created with GNU
``ar`` (e.g. on Linux) but not BSD ``ar`` (e.g. on MacOS). With BSD ``ar``,
CMake may break up long command-lines (i.e. lots of object files) with
multiple calls to ``ar`` but that may only work with the ``Unix Makefiles``
generator, not the ``Ninja`` generator.
* Some versions of ``gfortran`` do not accept ``*.rsp`` files.
* Some versions of ``nvcc`` (e.g. with CUDA 8.044) do not accept ``*.rsp``
files for compilation or linking.
Because of problems like these, TriBITS cannot robustly automatically turn on
these options. Therefore, it is up to the user to try these options out to
see if they work with their specific version of CMake, compilers, and OS.
NOTE: When using the ``Unix Makefiles`` generator, one can decide to set any
combination of these three options based on need and preference and what
actually works with a given OS, version of CMake, and provided compilers. For
example, on one system ``CMAKE_CXX_USE_RESPONSE_FILE_FOR_OBJECTS=ON`` may work
but ``CMAKE_CXX_USE_RESPONSE_FILE_FOR_INCLUDES=ON`` may not (which is the case
for ``gfortran`` mentioned above). Therefore, one should experiment carefully
and inspect the build lines using ``make VERBOSE=1 <target>`` as described in
`Building with verbose output without reconfiguring`_ when deciding which of
these options to enable.
NOTE: Newer versions of CMake may automatically determine when these options
need to be turned on so watch for that in looking at the build lines.
External Packages/Third-Party Library (TPL) support
---------------------------------------------------
A set of external packages/third-party libraries (TPL) can be enabled and
disabled and the locations of those can be specified at configure time (if
they are not found in the default path).
Enabling support for an optional Third-Party Library (TPL)
++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
To enable a given external packages/TPL, set::
-D TPL_ENABLE_<TPLNAME>=ON
where ``<TPLNAME>`` = ``BLAS``, ``LAPACK`` ``Boost``, ``Netcdf``, etc.
(Requires TPLs for enabled package will automatically be enabled.)
The full list of TPLs that is defined and can be enabled is shown by doing a
configure with CMake and then grepping the configure output for ``Final set of
.* TPLs``. The set of TPL names listed in ``'Final set of enabled external
packages/TPLs'`` and ``'Final set of non-enabled external packages/TPLs'``
gives the full list of TPLs that can be enabled (or disabled).
Optional package-specific support for a TPL can be turned off by setting::
-D <TRIBITS_PACKAGE>_ENABLE_<TPLNAME>=OFF
This gives the user full control over what TPLs are supported by which package
independent of whether the TPL is enabled or not.
Support for an optional TPL can also be turned on implicitly by setting::
-D <TRIBITS_PACKAGE>_ENABLE_<TPLNAME>=ON
where ``<TRIBITS_PACKAGE>`` is a TriBITS package that has an optional
dependency on ``<TPLNAME>``. That will result in setting
``TPL_ENABLE_<TPLNAME>=ON`` internally (but not set in the cache) if
``TPL_ENABLE_<TPLNAME>=OFF`` is not already set.
Specifying the location of the parts of an enabled external package/TPL
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Once an external package/TPL is enabled, the parts of that TPL must be found.
For many external packages/TPLs, this will be done automatically by searching
the environment paths.
Some external packages/TPLs are specified with a call to
``find_package(<externalPkg>)`` (see CMake documentation for
``find_package()``). Many other external packages/TPLs use a legacy TriBITS
system that locates the parts using the CMake commands ``find_file()`` and
``find_library()`` as described below.
Every defined external package/TPL uses a specification provided in a
``FindTPL<TPLNAME>.cmake`` module file. This file describes how the package
is found in a way that provides modern CMake IMPORTED targets (including the
``<TPLNAME>::all_libs`` target) that is used by the downstream packages.
Some TPLs require only libraries (e.g. Fortran libraries like BLAS or LAPACK),
some TPL require only include directories, and some TPLs require both.
For ``FindTPL<TPLNAME>.cmake`` files using the legacy TriBITS TPL system, a
TPL is fully specified through the following cache variables:
* ``TPL_<TPLNAME>_INCLUDE_DIRS:PATH``: List of paths to header files for the
TPL (if the TPL supplies header files).
* ``TPL_<TPLNAME>_LIBRARIES:PATH``: List of (absolute) paths to libraries,
ordered as they will be on the link line (of the TPL supplies libraries).
These variables are the only variables are used to create IMPORTED CMake
targets for the TPL. One can set these two variables as CMake cache
variables, for ``SomeTPL`` for example, with::
-D TPL_SomeTPL_INCLUDE_DIRS="${LIB_BASE}/include/a;${LIB_BASE}/include/b" \
-D TPL_SomeTPL_LIBRARIES="${LIB_BASE}/lib/liblib1.so;${LIB_BASE}/lib/liblib2.so" \
Using this approach, one can be guaranteed that these libraries and these
include directories and will used in the compile and link lines for the
packages that depend on this TPL ``SomeTPL``.
**NOTE:** When specifying ``TPL_<TPLNAME>_INCLUDE_DIRS`` and/or
``TPL_<TPLNAME>_LIBRARIES``, the build system will use these without question.
It will **not** check for the existence of these directories or files so make
sure that these files and directories exist before these are used in the
compiles and links. (This can actually be a feature in rare cases the
libraries and header files don't actually get created until after the
configure step is complete but before the build step.)
**NOTE:** It is generally *not recommended* to specify the TPLs libraries as
just a set of link options as, for example::
TPL_SomeTPL_LIBRARIES="-L/some/dir;-llib1;-llib2;..."
But this is supported as long as this link line contains only link library
directories and library names. (Link options that are not order-sensitive are
also supported like ``-mkl``.)
When the variables ``TPL_<TPLNAME>_INCLUDE_DIRS`` and
``TPL_<TPLNAME>_LIBRARIES`` are not specified, then most
``FindTPL<TPLNAME>.cmake`` modules use a default find operation. Some will
call ``find_package(<externalPkg>)`` internally by default and some may
implement the default find in some other way. To know for sure, see the
documentation for the specific external package/TPL (e.g. looking in the
``FindTPL<TPLNAME>.cmake`` file to be sure). NOTE: if a given
``FindTPL<TPLNAME>.cmake`` will use ``find_package(<externalPkg>)`` by
default, this can be disabled by configuring with::
-D<TPLNAME>_ALLOW_PACKAGE_PREFIND=OFF
(Not all ``FindTPL<TPLNAME>.cmake`` files support this option.)
Many ``FindTPL<TPLNAME>.cmake`` files, use the legacy TriBITS TPL system for
finding include directories and/or libraries based on the function
`tribits_tpl_find_include_dirs_and_libraries()`_. These simple standard
``FindTPL<TPLNAME>.cmake`` modules specify a set of header files and/or
libraries that must be found. The directories where these header files and
library files are looked for are specified using the CMake cache variables:
* ``<TPLNAME>_INCLUDE_DIRS:PATH``: List of paths to search for header files
using ``find_file()`` for each header file, in order.
* ``<TPLNAME>_LIBRARY_NAMES:STRING``: List of unadorned library names, in the
order of the link line. The platform-specific prefixes (e.g.. 'lib') and
postfixes (e.g. '.a', '.lib', or '.dll') will be added automatically by
CMake. For example, the library ``libblas.so``, ``libblas.a``, ``blas.lib``
or ``blas.dll`` will all be found on the proper platform using the name
``blas``.
* ``<TPLNAME>_LIBRARY_DIRS:PATH``: The list of directories where the library
files will be searched for using ``find_library()``, for each library, in
order.
Most of these ``FindTPL<TPLNAME>.cmake`` modules will define a default set of
libraries to look for and therefore ``<TPLNAME>_LIBRARY_NAMES`` can typically
be left off.
Therefore, to find the same set of libraries for ``SimpleTPL`` shown
above, one would specify::
-D SomeTPL_LIBRARY_DIRS="${LIB_BASE}/lib"
and if the set of libraries to be found is different than the default, one can
override that using::
-D SomeTPL_LIBRARY_NAMES="lib1;lib2"
Therefore, this is in fact the preferred way to specify the libraries for
these legacy TriBITS TPLs.
In order to allow a TPL that normally requires one or more libraries to ignore
the libraries, one can set ``<TPLNAME>_LIBRARY_NAMES`` to empty, for example::
-D <TPLNAME>_LIBRARY_NAMES=""
If all the parts of a TPL are not found on an initial configure, the configure
will error out with a helpful error message. In that case, one can change the
variables ``<TPLNAME>_INCLUDE_DIRS``, ``<TPLNAME>_LIBRARY_NAMES``, and/or
``<TPLNAME>_LIBRARY_DIRS`` in order to help fund the parts of the TPL. One
can do this over and over until the TPL is found. By reconfiguring, one avoids
a complete configure from scratch which saves time. Or, one can avoid the
find operations by directly setting ``TPL_<TPLNAME>_INCLUDE_DIRS`` and
``TPL_<TPLNAME>_LIBRARIES`` as described above.
**TPL Example 1: Standard BLAS Library**
Suppose one wants to find the standard BLAS library ``blas`` in the
directory::
/usr/lib/
libblas.so
libblas.a
...
The ``FindTPLBLAS.cmake`` module should be set up to automatically find the
BLAS TPL by simply enabling BLAS with::
-D TPL_ENABLE_BLAS=ON
This will result in setting the CMake cache variable ``TPL_BLAS_LIBRARIES`` as
shown in the CMake output::
-- TPL_BLAS_LIBRARIES='/user/lib/libblas.so'
(NOTE: The CMake ``find_library()`` command that is used internally will
always select the shared library by default if both shared and static
libraries are specified, unless told otherwise. See `Building static
libraries and executables`_ for more details about the handling of shared and
static libraries.)
However, suppose one wants to find the ``blas`` library in a non-default
location, such as in::
/projects/something/tpls/lib/libblas.so
In this case, one could simply configure with::
-D TPL_ENABLE_BLAS=ON \
-D BLAS_LIBRARY_DIRS=/projects/something/tpls/lib \
That will result in finding the library shown in the CMake output::
-- TPL_BLAS_LIBRARIES='/projects/something/tpls/libblas.so'
And if one wants to make sure that this BLAS library is used, then one can
just directly set::
-D TPL_BLAS_LIBRARIES=/projects/something/tpls/libblas.so
**TPL Example 2: Intel Math Kernel Library (MKL) for BLAS**
There are many cases where the list of libraries specified in the
``FindTPL<TPLNAME>.cmake`` module is not correct for the TPL that one wants to
use or is present on the system. In this case, one will need to set the CMake
cache variable ``<TPLNAME>_LIBRARY_NAMES`` to tell the
`tribits_tpl_find_include_dirs_and_libraries()`_ function what libraries to
search for, and in what order.
For example, the Intel Math Kernel Library (MKL) implementation for the BLAS
is usually given in several libraries. The exact set of libraries needed
depends on the version of MKL, whether 32bit or 64bit libraries are needed,
etc. Figuring out the correct set and ordering of these libraries for a given
platform may be non-trivial. But once the set and the order of the libraries
is known, then one can provide the correct list at configure time.
For example, suppose one wants to use the threaded MKL libraries listed in the
directories::
/usr/local/intel/Compiler/11.1/064/mkl/lib/em64t/
/usr/local/intel/Compiler/11.1/064/lib/intel64/
and the list of libraries being searched for is ``mkl_intel_lp64``,
``mkl_intel_thread``, ``mkl_core`` and ``iomp5``.
In this case, one could specify this with the following do-configure script::
#!/bin/bash
INTEL_DIR=/usr/local/intel/Compiler/11.1/064
cmake \
-D TPL_ENABLE_BLAS=ON \
-D BLAS_LIBRARY_DIRS="${INTEL_DIR}/em64t;${INTEL_DIR}/intel64" \
-D BLAS_LIBRARY_NAMES="mkl_intel_lp64;mkl_intel_thread;mkl_core;iomp5" \
...
${PROJECT_SOURCE_DIR}
This would call ``find_library()`` on each of the listed library names in
these directories and would find them and list them in::
-- TPL_BLAS_LIBRARIES='/usr/local/intel/Compiler/11.1/064/em64t/libmkl_intel_lp64.so;...'
(where ``...`` are the rest of the found libraries.)
NOTE: When shared libraries are used, one typically only needs to list the
direct libraries, not the indirect libraries, as the shared libraries are
linked to each other.
In this example, one could also play it super safe and manually list out the
libraries in the right order by configuring with::
-D TPL_BLAS_LIBRARIES="${INTEL_DIR}/em64t/libmkl_intel_lp64.so;..."
(where ``...`` are the rest of the libraries found in order).
Adjusting upstream dependencies for a Third-Party Library (TPL)
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Some TPLs have dependencies on one or more upstream TPLs. These dependencies
must be specified correctly for the compile and links to work correctly. The
<Project> Project already defines these dependencies for the average situation
for all of these TPLs. However, there may be situations where the
dependencies may need to be tweaked to match how these TPLs were actually
installed on some systems. To redefine what dependencies a TPL can have (if
the upstream TPLs are enabled), set::
-D <TPLNAME>_LIB_DEFINED_DEPENDENCIES="<tpl_1>;<tpl_2>;..."
A dependency on an upstream TPL ``<tpl_i>`` will be set if the an upstream TPL
``<tpl_i>`` is actually enabled.
If any of the specified dependent TPLs ``<tpl_i>`` are listed after
``<TPLNAME>`` in the ``TPLsList.cmake`` file (or are not listed at all), then
a configure-time error will occur.
To take complete control over what dependencies an TPL has, set::
-D <TPLNAME>_LIB_ENABLED_DEPENDENCIES="<tpl_1>;<tpl_2>;..."
If the upstream TPLs listed here are not defined upstream and enabled TPLs,
then a configure-time error will occur.
Disabling support for a Third-Party Library (TPL)
+++++++++++++++++++++++++++++++++++++++++++++++++
Disabling a TPL explicitly can be done using::
-D TPL_ENABLE_<TPLNAME>=OFF
This will result in the disabling of any direct or indirect downstream
packages that have a required dependency on ``<TPLNAME>`` as described in
`Disable a package and all its dependencies`_.
NOTE: If a disabled TPL is a required dependency of some explicitly enabled
downstream package, then the configure will error out if
`<Project>_DISABLE_ENABLED_FORWARD_DEP_PACKAGES`_ ``= OFF``. Otherwise, a
NOTE will be printed and the downstream package will be disabled and
configuration will continue.
Disabling tentatively enabled TPLs
++++++++++++++++++++++++++++++++++
To disable a tentatively enabled TPL, set::
-D TPL_ENABLE_<TPLNAME>=OFF
where ``<TPLNAME>`` = ``BinUtils``, ``Boost``, etc.
NOTE: Some TPLs in <Project> are always tentatively enabled (e.g. BinUtils
for C++ stacktracing) and if all of the components for the TPL are found
(e.g. headers and libraries) then support for the TPL will be enabled,
otherwise it will be disabled. This is to allow as much functionality as
possible to get automatically enabled without the user having to learn about
the TPL, explicitly enable the TPL, and then see if it is supported or not
on the given system. However, if the TPL is not supported on a given
platform, then it may be better to explicitly disable the TPL (as shown
above) so as to avoid the output from the CMake configure process that shows
the tentatively enabled TPL being processes and then failing to be enabled.
Also, it is possible that the enable process for the TPL may pass, but the
TPL may not work correctly on the given platform. In this case, one would
also want to explicitly disable the TPL as shown above.
Require all TPL libraries be found
++++++++++++++++++++++++++++++++++
By default, some TPLs don't require that all of the libraries listed in
``<tplName>_LIBRARY_NAMES`` be found. To change this behavior so that all
libraries for all enabled TPLs be found, one can set::
-D <Project>_MUST_FIND_ALL_TPL_LIBS=TRUE
This makes the configure process catch more mistakes with the env.
Disable warnings from TPL header files
++++++++++++++++++++++++++++++++++++++
To disable warnings coming from included TPL header files for C and C++ code,
set::
-D<Project>_TPL_SYSTEM_INCLUDE_DIRS=TRUE
On some systems and compilers (e.g. GNU), that will result is include
directories for all TPLs to be passed in to the compiler using ``-isystem``
instead of ``-I``.
WARNING: On some systems this will result in build failures involving gfortran
and module files. Therefore, don't enable this if Fortran code in your
project is pulling in module files from TPLs.
Building against pre-installed packages
---------------------------------------
The <Project> project can build against any pre-installed packages defined in
the project and ignore the internally defined packages. To trigger the enable
of a pre-installed internal package treated as an external package, configure
with::
-D TPL_ENABLE_<TRIBITS_PACKAGE>=ON
That will cause the <Project> CMake project to pull in the pre-installed
package ``<TRIBITS_PACKAGE>`` as an external package using
``find_package(<TRIBITS_PACKAGE>)`` instead of configuring and building the
internally defined ``<TRIBITS_PACKAGE>`` package.
Configuring and building against a pre-installed package treated as an
external packages has several consequences:
* Any internal packages that are upstream from ``<TRIBITS_PACKAGE>`` from an
enabled set of dependencies will also be treated as external packages (and
therefore must be pre-installed as well).
* The TriBITS package ``Dependencies.cmake`` files for the
``<TRIBITS_PACKAGE>`` package and all of its upstream packages must still
exist and will still be read in by the <Project> CMake project and the same
enable/disable logic will be performed as if the packages were being treated
internal. (However, the base ``CMakeLists.txt`` and all of other files for
these internally defined packages being treated as external packages can be
missing and will be ignored.)
* The same set of enabled and disabled upstream dependencies must be specified
to the <Project> CMake project that was used to pre-build and pre-install
these internally defined packages being treated as external packages.
(Otherwise, a configure error will result from the mismatch.)
* The definition of any TriBITS external packages/TPLs that are enabled
upstream dependencies from any of these internally defined packages being
treated as external packages will be defined by the calls to
``find_package(<TRIBITS_PACKAGE>)`` and will **not** be found again.
The logic for treating internally defined packages as external packages will
be printed in the CMake configure output in the section ``Adjust the set of
internal and external packages`` with output like::
Adjust the set of internal and external packages ...
-- Treating internal package <PKG2> as EXTERNAL because TPL_ENABLE_<PKG2>=ON
-- Treating internal package <PKG1> as EXTERNAL because downstream package <PKG2> being treated as EXTERNAL
-- NOTE: <TPL2> is indirectly downstream from a TriBITS-compliant external package
-- NOTE: <TPL1> is indirectly downstream from a TriBITS-compliant external package
All of these internally defined being treated as external (and all of their
upstream dependencies) are processed in a loop over these just these
TriBITS-compliant external packages and ``find_package()`` is only called on
the terminal TriBITS-compliant external packages. This is shown in the CMake
output in the section ``Getting information for all enabled TriBITS-compliant
or upstream external packages/TPLs`` and looks like::
Getting information for all enabled TriBITS-compliant or upstream external packages/TPLs ...
Processing enabled external package/TPL: <TPL1> (...)
-- The external package/TPL <TPL1> will be read in by a downstream TriBITS-compliant external package
Processing enabled external package/TPL: <TPL2> (...)
-- The external package/TPL <TPL2> will be read in by a downstream TriBITS-compliant external package
Processing enabled external package/TPL: <PKG1> (...)
-- The external package/TPL <PKG1> will be read in by a downstream TriBITS-compliant external package
Processing enabled external package/TPL: <PKG2> (...)
-- Calling find_package(<PKG2> for TriBITS-compliant external package
In the above example ``<TPL1>``, ``<TPL2>`` and ``<PKG1>`` are all direct or
indirect dependencies of ``<PKG2>`` and therefore calling just
``find_package(<PKG2>)`` fully defines those TriBITS-compliant external
packages as well.
xSDK Configuration Options
--------------------------
The configure of <Project> will adhere to the `xSDK Community Package
Policies`_ simply by setting the CMake cache variable::
-D USE_XSDK_DEFAULTS=TRUE
Setting this will have the following impact:
* ``BUILD_SHARED_LIBS`` will be set to ``TRUE`` by default instead of
``FALSE``, which is the default for raw CMake projects (see `Building shared
libraries`_).
* ``CMAKE_BUILD_TYPE`` will be set to ``DEBUG`` by default instead of
``RELEASE`` which is the standard TriBITS default (see `CMAKE_BUILD_TYPE`_).
* The compilers in MPI mode ``TPL_ENABLE_MPI=ON`` or serial mode
``TPL_ENABLE_MPI=OFF`` will be read from the environment variables ``CC``,
``CXX`` and ``FC`` if they are set but the cmake cache variables
``CMAKE_C_COMPILER``, ``CMAKE_C_COMPILER`` and ``CMAKE_C_COMPILER`` are not
set. Otherwise, the TriBITS default behavior is to ignore these environment
variables in MPI mode.
* The Fortran flags will be read from environment variable ``FCFLAGS`` if the
environment variable ``FFLAGS`` and the CMake cache variable
``CMAKE_Fortran_FLAGS`` are empty. Otherwise, raw CMake ignores ``FCFLAGS``
(see `Adding arbitrary compiler flags but keeping default build-type
flags`_).
The rest of the required xSDK configure standard is automatically satisfied by
every TriBITS CMake project, including the <Project> project.
Generating verbose output
-------------------------
There are several different ways to generate verbose output to debug problems
when they occur:
.. _<Project>_TRACE_FILE_PROCESSING:
a) **Trace file processing during configure:**
::
-D <Project>_TRACE_FILE_PROCESSING=ON
This will cause TriBITS to print out a trace for all of the project's,
repository's, and package's files get processed on lines using the prefix
``File Trace:``. This shows what files get processed and in what order they
get processed. To get a clean listing of all the files processed by TriBITS
just grep out the lines starting with ``-- File Trace:``. This can be
helpful in debugging configure problems without generating too much extra
output.
Note that `<Project>_TRACE_FILE_PROCESSING`_ is set to ``ON`` automatically
when `<Project>_VERBOSE_CONFIGURE`_ = ``ON``.
.. _<Project>_VERBOSE_CONFIGURE:
b) **Getting verbose output from TriBITS configure:**
To do a complete debug dump for the TriBITS configure process, use::
-D <Project>_VERBOSE_CONFIGURE=ON
However, this produces a *lot* of output so don't enable this unless you are
very desperate. But this level of details can be very useful when debugging
configuration problems.
To just view the package and TPL dependencies, it is recommended to use
``-D`` `<Project>_DUMP_PACKAGE_DEPENDENCIES`_ ``= ON``.
To just print the link libraries for each library and executable created,
use::
-D <Project>_DUMP_LINK_LIBS=ON
Of course ``<Project>_DUMP_PACKAGE_DEPENDENCIES`` and
``<Project>_DUMP_LINK_LIBS`` can be used together. Also, note that
``<Project>_DUMP_PACKAGE_DEPENDENCIES`` and ``<Project>_DUMP_LINK_LIBS``
both default t ``ON`` when ``<Project>_VERBOSE_CONFIGURE=ON`` on the first
configure.
c) **Getting verbose output from the makefile:**
::
-D CMAKE_VERBOSE_MAKEFILE=TRUE
NOTE: It is generally better to just pass in ``VERBOSE=`` when directly
calling ``make`` after configuration is finished. See `Building with
verbose output without reconfiguring`_.
d) **Getting very verbose output from configure:**
::
-D <Project>_VERBOSE_CONFIGURE=ON --debug-output --trace
NOTE: This will print a complete stack trace to show exactly where you are.
Enabling/disabling deprecated warnings
--------------------------------------
To turn off all deprecated warnings, set::
-D <Project>_SHOW_DEPRECATED_WARNINGS=OFF
This will disable, by default, all deprecated warnings in packages in
<Project>. By default, deprecated warnings are enabled.
To enable/disable deprecated warnings for a single <Project> package, set::
-D <TRIBITS_PACKAGE>_SHOW_DEPRECATED_WARNINGS=OFF
This will override the global behavior set by
``<Project>_SHOW_DEPRECATED_WARNINGS`` for individual package
``<TRIBITS_PACKAGE>``.
Adjusting CMake DEPRECATION warnings
------------------------------------
By default, deprecated TriBITS features being used in the project's CMake
files will result in CMake deprecation warning messages (issued by calling
``message(DEPRECATION ...)`` internally). The handling of these deprecation
warnings can be changed by setting the CMake cache variable
``TRIBITS_HANDLE_TRIBITS_DEPRECATED_CODE``. For example, to remove all
deprecation warnings, set::
-D TRIBITS_HANDLE_TRIBITS_DEPRECATED_CODE=IGNORE
Other valid values include:
* ``DEPRECATION``: Issue a CMake ``DEPRECATION`` message and continue (default).
* ``AUTHOR_WARNING``: Issue a CMake ``AUTHOR_WARNING`` message and continue.
* ``SEND_ERROR``: Issue a CMake ``SEND_ERROR`` message and continue.
* ``FATAL_ERROR``: Issue a CMake ``FATAL_ERROR`` message and exit.
Disabling deprecated code
-------------------------
To actually disable and remove deprecated code from being included in
compilation, set::
-D <Project>_HIDE_DEPRECATED_CODE=ON
and a subset of deprecated code will actually be removed from the build. This
is to allow testing of downstream client code that might otherwise ignore
deprecated warnings. This allows one to certify that a downstream client code
is free of calling deprecated code.
To hide deprecated code for a single <Project> package set::
-D <TRIBITS_PACKAGE>_HIDE_DEPRECATED_CODE=ON
This will override the global behavior set by
``<Project>_HIDE_DEPRECATED_CODE`` for individual package
``<TRIBITS_PACKAGE>``.
Outputting package dependency information
-----------------------------------------
.. _<Project>_DEPS_DEFAULT_OUTPUT_DIR:
To generate the various XML and HTML package dependency files, one can set the
output directory when configuring using::
-D <Project>_DEPS_DEFAULT_OUTPUT_DIR:FILEPATH=<SOME_PATH>
This will generate, by default, the output files
``<Project>PackageDependencies.xml``,
``<Project>PackageDependenciesTable.html``, and
``CDashSubprojectDependencies.xml``. If ``<Project>_DEPS_DEFAULT_OUTPUT_DIR``
is not set, then the individual output files can be specified as described below.
.. _<Project>_DEPS_XML_OUTPUT_FILE:
The filepath for <Project>PackageDependencies.xml can be overridden (or set
independently) using::
-D <Project>_DEPS_XML_OUTPUT_FILE:FILEPATH=<SOME_FILE_PATH>
.. _<Project>_DEPS_HTML_OUTPUT_FILE:
The filepath for ``<Project>PackageDependenciesTable.html`` can be overridden
(or set independently) using::
-D <Project>_DEPS_HTML_OUTPUT_FILE:FILEPATH=<SOME_FILE_PATH>
.. _<Project>_CDASH_DEPS_XML_OUTPUT_FILE:
The filepath for CDashSubprojectDependencies.xml can be overridden (or set
independently) using::
-D <Project>_CDASH_DEPS_XML_OUTPUT_FILE:FILEPATH=<SOME_FILE_PATH>
NOTES:
* One must start with a clean CMake cache for all of these defaults to work.
* The files ``<Project>PackageDependenciesTable.html`` and
``CDashSubprojectDependencies.xml`` will only get generated if support for
Python is enabled.
Test-related configuration settings
-----------------------------------
Many options can be set at configure time to determine what tests are enabled
and how they are run. The following subsections described these various
settings.
Enabling different test categories
++++++++++++++++++++++++++++++++++
To turn on a set a given set of tests by test category, set::
-D <Project>_TEST_CATEGORIES="<CATEGORY0>;<CATEGORY1>;..."
Valid categories include ``BASIC``, ``CONTINUOUS``, ``NIGHTLY``, ``HEAVY`` and
``PERFORMANCE``. ``BASIC`` tests get built and run for pre-push testing, CI
testing, and nightly testing. ``CONTINUOUS`` tests are for post-push testing
and nightly testing. ``NIGHTLY`` tests are for nightly testing only.
``HEAVY`` tests are for more expensive tests that require larger number of MPI
processes and longer run times. These test categories are nested
(e.g. ``HEAVY`` contains all ``NIGHTLY``, ``NIGHTLY`` contains all
``CONTINUOUS`` and ``CONTINUOUS`` contains all ``BASIC`` tests). However,
``PERFORMANCE`` tests are special category used only for performance testing
and don't nest with the other categories.
Disabling specific tests
++++++++++++++++++++++++
Any TriBITS-added ctest test (i.e. listed in ``ctest -N``) can be disabled at
configure time by setting::
-D <fullTestName>_DISABLE=ON
where ``<fullTestName>`` must exactly match the test listed out by ``ctest
-N``. This will result in the printing of a line for the excluded test when
`Trace test addition or exclusion`_ is enabled and the test will not be added
with ``add_test()`` and therefore CTest (and CDash) will never see the
disabled test.
Another approach to disable a test is the set the ctest property ``DISABLED``
and print and a message at configure time by setting::
-D <fullTestName>_SET_DISABLED_AND_MSG="<messageWhyDisabled>"
In this case, the test will still be added with ``add_test()`` and seen by
CTest, but CTest will not run the test locally but will mark it as "Not Run"
(and post to CDash as "Not Run" tests with test details "Not Run (Disabled)"
in processes where tests get posted to CDash). Also, ``<messageWhyDisabled>``
will get printed to STDOUT when CMake is run to configure the project and
``-D<Project>_TRACE_ADD_TEST=ON`` is set.
Also, note that if a test is currently disabled using the ``DISABLED`` option
in the CMakeLists.txt file, then that ``DISABLE`` property can be removed by
configuring with::
-D <fullTestName>_SET_DISABLED_AND_MSG=FALSE
(or any value that CMake evaluates to FALSE like "FALSE", "false", "NO", "no",
"", etc.).
Also note that other specific defined tests can also be excluded using the
``ctest -E`` argument.
Disabling specific test executable builds
+++++++++++++++++++++++++++++++++++++++++
Any TriBITS-added executable (i.e. listed in ``make help``) can be disabled
from being built by setting::
-D <exeTargetName>_EXE_DISABLE=ON
where ``<exeTargetName>`` is the name of the target in the build system.
Note that one should also disable any ctest tests that might use this
executable as well with ``-D<fullTestName>_DISABLE=ON`` (see above). This
will result in the printing of a line for the executable target being disabled
at configure time to CMake STDOUT.
Disabling just the ctest tests but not the test executables
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
To allow the building of the tests and examples in a package (enabled either
through setting `<Project>_ENABLE_TESTS`_ ``= ON`` or
`<TRIBITS_PACKAGE>_ENABLE_TESTS`_ ``= ON``) but not actually define the ctest
tests that will get run, configure with::
-D <TRIBITS_PACKAGE>_SKIP_CTEST_ADD_TEST=TRUE \
(This has the effect of skipping calling the ``add_test()`` command in the
CMake code for the package ``<TRIBITS_PACKAGE>``.)
To avoid defining ctest tests for all of the enabled packages, configure
with::
-D <Project>_SKIP_CTEST_ADD_TEST=TRUE \
(The default for ``<TRIBITS_PACKAGE>_SKIP_CTEST_ADD_TEST`` for each TriBITS
package ``<TRIBITS_PACKAGE>`` is set to the project-wide option
``<Project>_SKIP_CTEST_ADD_TEST``.)
One can also use these options to "white-list" and "black-list" the set of
package tests that one will run. For example, to enable the building of all
test and example targets but only actually defining ctest tests for two
specific packages (i.e. "white-listing"), one would configure with::
-D <Project>_ENABLE_ALL_PACKAGES=ON \
-D <Project>_ENABLE_TESTS=ON \
-D <Project>_SKIP_CTEST_ADD_TEST=TRUE \
-D <TRIBITS_PACKAGE_1>_SKIP_CTEST_ADD_TEST=FALSE \
-D <TRIBITS_PACKAGE_2>_SKIP_CTEST_ADD_TEST=FALSE \
Alternatively, to enable the building of all test and example targets and
allowing the ctest tests to be defined for all packages except for a couple of
specific packages (i.e. "black-listing"), one would configure with::
-D <Project>_ENABLE_ALL_PACKAGES=ON \
-D <Project>_ENABLE_TESTS=ON \
-D <TRIBITS_PACKAGE_1>_SKIP_CTEST_ADD_TEST=TRUE \
-D <TRIBITS_PACKAGE_2>_SKIP_CTEST_ADD_TEST=TRUE \
Using different values for ``<Project>_SKIP_CTEST_ADD_TEST`` and
``<TRIBITS_PACKAGE>_SKIP_CTEST_ADD_TEST`` in this way allows for building all
of the test and example targets for the enabled packages but not defining
ctest tests for any set of packages desired. This allows setting up testing
scenarios where one wants to test the building of all test-related targets but
not actually run the tests with ctest for a subset of all of the enabled
packages. (This can be useful in cases where the tests are very expensive and
one can't afford to run all of them given the testing budget, or when running
tests on a given platform is very flaky, or when some packages have fragile or
poor quality tests that don't port to new platforms very well.)
NOTE: These options avoid having to pass specific sets of labels when running
``ctest`` itself (such as when defining ``ctest -S <script>.cmake`` scripts)
and instead the decisions as to the exact set of ctest tests to define is made
at configure time. Therefore, all of the decisions about what test targets
should be build and which tests should be run can be made at configure time.
Set specific tests to run in serial
+++++++++++++++++++++++++++++++++++
In order to cause a specific test to run by itself on the machine and not at
the same time as other tests (such as when running multiple tests at the same
time with something like ``ctest -j16``), set at configure time::
-D <fullTestName>_SET_RUN_SERIAL=ON
This will set the CTest test property ``RUN_SERIAL`` for the test
``<fullTestName>``.
This can help to avoid longer runtimes and timeouts when some individual tests
don't run as quickly when run beside other tests running at the same time on
the same machine. These longer runtimes can often occur when running tests
with CUDA code on GPUs and with OpenMP code on some platforms with some OpenMP
options.
Also, if individual tests have ``RUN_SERIAL`` set by default internally, they
can have the ``RUN_SERIAL`` property removed by setting::
-D <fullTestName>_SET_RUN_SERIAL=OFF
Trace test addition or exclusion
++++++++++++++++++++++++++++++++
To see what tests get added and see those that don't get added for various
reasons, configure with::
-D <Project>_TRACE_ADD_TEST=ON
That will print one line per test and will show if the test got added or not.
If the test is added, it shows some of the key test properties. If the test
did not get added, then this line will show why the test was not added
(i.e. due to criteria related to the test's ``COMM``, ``NUM_MPI_PROCS``,
``CATEGORIES``, ``HOST``, ``XHOST``, ``HOSTTYPE``, or ``XHOSTTYPE``
arguments).
Enable advanced test start and end times and timing blocks
++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
For tests added using ``tribits_add_advanced_test()``, one can see start and
end times for the tests and the timing for each ``TEST_<IDX>`` block in the
detailed test output by configuring with::
-D<Project>_SHOW_TEST_START_END_DATE_TIME=ON
The implementation of this feature currently uses ``execute_process(date)``
and therefore will only work on many (but perhaps not all) Linux/Unix/Mac
systems and not native Windows systems.
Setting test timeouts at configure time
+++++++++++++++++++++++++++++++++++++++
.. _DART_TESTING_TIMEOUT:
A maximum default time limit (timeout) for all the tests can be set at
configure time using the cache variable::
-D DART_TESTING_TIMEOUT=<maxSeconds>
where ``<maxSeconds>`` is the number of wall-clock seconds. The default for
most projects is 1500 seconds (see the default value set in the CMake cache).
This value gets scaled by `<Project>_SCALE_TEST_TIMEOUT`_ and then set as the
field ``TimeOut`` in the CMake-generated file ``DartConfiguration.tcl``. The
value ``TimeOut`` from this file is what is directly read by the ``ctest``
executable. Timeouts for tests are important. For example, when an MPI
program has a defect, it can easily hang (forever) until it is manually
killed. If killed by a timeout, CTest will kill the test process and all of
its child processes correctly.
NOTES:
* If ``DART_TESTING_TIMEOUT`` is not explicitly set by the user, then the
projects gives it a default value (typically 1500 seconds but see the value
in the CMakeCache.txt file).
* If ``DART_TESTING_TIMEOUT`` is explicitly set to empty
(i.e. ``-DDART_TESTING_TIMEOUT=``), then by default tests have no timeout
and can run forever until manually killed.
* Individual tests may have their timeout limit set on a test-by-test basis
internally in the project's ``CMakeLists.txt`` files (see the ``TIMEOUT``
argument for ``tribits_add_test()`` and ``tribits_add_advanced_test()``).
When this is the case, the global timeout set with ``DART_TESTING_TIMEOUT``
has no impact on these individually set test timeouts.
* Be careful not set the global test timeout too low since if a machine
becomes loaded tests can take longer to run and may result in timeouts that
would not otherwise occur.
* The value of ``DART_TESTING_TIMEOUT`` and the timeouts for individual tests
can be scaled up or down using the cache variable
`<Project>_SCALE_TEST_TIMEOUT`_.
* To set or override the default global test timeout limit at runtime, see
`Overriding test timeouts`_.
Scaling test timeouts at configure time
+++++++++++++++++++++++++++++++++++++++
.. _<Project>_SCALE_TEST_TIMEOUT:
The global default test timeout `DART_TESTING_TIMEOUT`_ as well as all of the
timeouts for the individual tests that have their own timeout set (through the
``TIMEOUT`` argument for each individual test) can be scaled by a constant
factor ``<testTimeoutScaleFactor>`` by configuring with::
-D <Project>_SCALE_TEST_TIMEOUT=<testTimeoutScaleFactor>
Here, ``<testTimeoutScaleFactor>`` can be an integral number like ``5`` or can
be fractional number like ``1.5``.
This feature is generally used to compensate for slower machines or overloaded
test machines and therefore only scaling factors greater than 1 are to be
used. The primary use case for this feature is to add large scale factors
(e.g. ``40`` to ``100``) to compensate for running tests using valgrind (see
`Running memory checking`_) but this can also be used for debug-mode builds
that create tests which run more slowly than for full release-mode optimized
builds.
NOTES:
* If ``<Project>_SCALE_TEST_TIMEOUT`` is not set, the the default value is set
to ``1.0`` (i.e. no scaling of test timeouts).
* When scaling the timeouts, the timeout is first truncated to integral
seconds so an original timeout like ``200.5`` will be truncated to ``200``
before it gets scaled.
* Only the first fractional digit of ``<Project>_SCALE_TEST_TIMEOUT`` is used
so ``1.57`` is truncated to ``1.5``, for example, before scaling the test
timeouts.
* The value of the variable `DART_TESTING_TIMEOUT`_ is not changed in the
``CMakeCache.txt`` file. Only the value of ``TimeOut`` written into the
``DartConfiguration.tcl`` file (which is directly read by ``ctest``) will be
scaled. (This ensures that running configure over and over again will not
increase ``DART_TESTING_TIMEOUT`` or ``TimeOut`` with each new configure.)
Spreading out and limiting tests running on GPUs
++++++++++++++++++++++++++++++++++++++++++++++++
For CUDA builds (i.e. ``TPL_ENABLE_CUDA=ON``) with tests that run on a single
node which has multiple GPUs, there are settings that can help ``ctest``
spread out the testing load over all of the GPUs and limit the number of
kernels that can run at the same time on a single GPU.
To instruct ``ctest`` to spread out the load on multiple GPUs, one can set the
following configure-time options::
-D TPL_ENABLE_CUDA=ON \
-D <Project>_AUTOGENERATE_TEST_RESOURCE_FILE=ON \
-D <Project>_CUDA_NUM_GPUS=<num-gpus> \
-D <Project>_CUDA_SLOTS_PER_GPU=<slots-per-gpu> \
This will cause a file ``ctest_resources.json`` to get generated in the base
build directory that CTest will use to spread out the work across the
``<num-gpus>`` GPUs with a maximum of ``<slots-per-gpu>`` processes running
kernels on any one GPU. (This uses the `CTest Resource Allocation System`_
first added in CMake 3.16 and made more usable in CMake 3.18.)
For example, when running on one node on a system with 4 GPUs per node
(allowing 5 kernels to run at a time on a single GPU) one would configure
with::
-D TPL_ENABLE_CUDA=ON \
-D <Project>_AUTOGENERATE_TEST_RESOURCE_FILE=ON \
-D <Project>_CUDA_NUM_GPUS=4 \
-D <Project>_CUDA_SLOTS_PER_GPU=5 \
This allows, for example, up to 5 tests using 4-rank MPI jobs, or 10 tests
using 2-rank MPI jobs, or 20 tests using 1-rank MPI jobs, to run at the same
time (or any combination of tests that add up to 20 or less total MPI
processes to run a the same time). But a single 21-rank or above MPI test job
would not be allowed to run and would be listed as "Not Run" because it would
have required more than ``<slots-per-gpu> = 5`` MPI processes running kernels
at one time on a single GPU. (Therefore, one must set ``<slots-per-gpu>``
large enough to allow all of the defined tests to run or one should avoid
defining tests that require too many slots for available GPUs.)
The CTest implementation uses a breath-first approach to spread out the work
across all the available GPUs before adding more work for each GPU. For
example, when running two 2-rank MPI tests at the same time (e.g. using
``ctest -j4``) in the above example, CTest will instruct these tests at
runtime to spread out across all 4 GPUs and therefore run the CUDA kernels for
just one MPI process on each GPU. But when running four 2-rank MPI tests at
the same time (e.g. using ``ctest -j8``), then each of the 4 GPUs would get
the work of two MPI processes (i.e. running two kernels at a time on each of
the 4 GPUs).
One can also manually create a `CTest Resource Specification File`_ and point
to it by setting::
-D TPL_ENABLE_CUDA=ON \
-D CTEST_RESOURCE_SPEC_FILE=<file-path> \
In all cases, ctest will not spread out and limit running on the GPUs unless
``TPL_ENABLE_CUDA=ON`` is set which causes TriBITS to add the
`RESOURCE_GROUPS`_ test property to each test.
NOTES:
* This setup assumes that a single MPI process will run just one kernel on its
assigned GPU and therefore take up one GPU "slot". So a 2-rank MPI test
will take up 2 total GPU "slots" (either on the same or two different GPUs,
as determined by CTest).
* The underlying test executables/scripts themselves must be set up to read in
the `CTest Resource Allocation Environment Variables`_ set specifically by
``ctest`` on the fly for each test and then must run on the specific GPUs
specified in those environment variables. (If the project is using a Kokkos
back-end implementation for running CUDA code on the GPU then this will work
automatically since Kokkos is set up to automatically look for these
CTest-set environment variables. Without this CTest and TriBITS
implementation, when running 2-rank MPI tests on a node with 4 GPUs, Kokkos
would just utilize the first two GPUs and leave the other two GPUs idle.
One when running 1-rank MPI tests, Kokkos would only utilize the first GPU
and leave the last three GPUs idle.)
* The option ``<Project>_AUTOGENERATE_TEST_RESOURCE_FILE=ON`` sets the
built-in CMake variable ``CTEST_RESOURCE_SPEC_FILE`` to point to the
generated file ``ctest_resources.json`` in the build directory.
* One can avoid setting the CMake cache variables
``<Project>_AUTOGENERATE_TEST_RESOURCE_FILE`` or
``CTEST_RESOURCE_SPEC_FILE`` at configure time and can instead directly pass
the path to the `CTest Resource Specification File`_ directly into ``ctest``
using the command-line option ``--resource-spec-file`` or the
``ctest_test()`` function argument ``RESOURCE_SPEC_FILE`` (when using a
``ctest -S`` script driver). (This allows using CMake 3.16+ since support
for the ``CTEST_RESOURCE_SPEC_FILE`` cache variable was not added until
CMake 3.18.)
* **WARNING:** This currently only works for a single node, not multiple
nodes. (CTest needs to be extended to work correctly for multiple nodes
where each node has multiple GPUs. Alternatively, TriBITS could be extended
to make this work for multiple nodes but will require considerable work and
will need to closely interact with the MPI launcher to control what nodes
are run on for each MPI job/test.)
* **WARNING:** This feature is still evolving in CMake/CTest and TriBITS and
therefore the input options and behavior of this may change in the future.
Enabling support for coverage testing
-------------------------------------
To turn on support for coverage testing set::
-D <Project>_ENABLE_COVERAGE_TESTING=ON
This will set compile and link options -fprofile-arcs -ftest-coverage for GCC.
Use 'make dashboard' (see below) to submit coverage results to CDash
Viewing configure options and documentation
-------------------------------------------
a) Viewing available configure-time options with documentation:
::
$ cd $BUILD_DIR
$ rm -rf CMakeCache.txt CMakeFiles/
$ cmake -LAH -D <Project>_ENABLE_ALL_PACKAGES=ON \
$SOURCE_BASE
You can also just look at the text file CMakeCache.txt after configure which
gets created in the build directory and has all of the cache variables and
documentation.
b) Viewing available configure-time options without documentation:
::
$ cd $BUILD_DIR
$ rm -rf CMakeCache.txt CMakeFiles/
$ cmake -LA <SAME_AS_ABOVE> $SOURCE_BASE
c) Viewing current values of cache variables:
::
$ cmake -LA $SOURCE_BASE
or just examine and grep the file CMakeCache.txt.
Enabling extra repositories with add-on packages:
-------------------------------------------------
.. _<Project>_EXTRA_REPOSITORIES:
To configure <Project> with an post extra set of packages in extra TriBITS
repositories, configure with::
-D<Project>_EXTRA_REPOSITORIES="<REPO0>,<REPO1>,..."
Here, ``<REPOi>`` is the name of an extra repository that typically has been
cloned under the main <Project> source directory as::
<Project>/<REPOi>/
For example, to add the packages from SomeExtraRepo one would configure as::
$ cd $SOURCE_BASE_DIR
$ git clone some_url.com/some/dir/SomeExtraRepo
$ cd $BUILD_DIR
$ ./do-configure -D<Project>_EXTRA_REPOSITORIES=SomeExtraRepo \
[Other Options]
After that, all of the extra packages defined in ``SomeExtraRepo`` will appear
in the list of official <Project> packages (after all of the native packages)
and one is free to enable any of the defined add-on packages just like any
other native <Project> package.
NOTE: If ``<Project>_EXTRAREPOS_FILE`` and
``<Project>_ENABLE_KNOWN_EXTERNAL_REPOS_TYPE`` are specified, then the list of
extra repositories in ``<Project>_EXTRA_REPOSITORIES`` must be a subset and in
the same order as the list extra repos read in from the file specified by
`<Project>_EXTRAREPOS_FILE`_. (Also see the variable
`<Project>_PRE_REPOSITORIES`_ as well.)
Enabling extra repositories through a file
------------------------------------------
.. _<Project>_EXTRAREPOS_FILE:
.. _<Project>_ENABLE_KNOWN_EXTERNAL_REPOS_TYPE:
In order to provide the list of extra TriBITS repositories containing add-on
packages from a file, configure with::
-D<Project>_EXTRAREPOS_FILE:FILEPATH=<EXTRAREPOSFILE> \
-D<Project>_ENABLE_KNOWN_EXTERNAL_REPOS_TYPE=Continuous
Specifying extra repositories through an extra repos file allows greater
flexibility in the specification of extra repos. This is not needed for a
basic configure of the project but is useful in generating version information
using `<Project>_GENERATE_VERSION_DATE_FILES`_ and
`<Project>_GENERATE_REPO_VERSION_FILE`_ as well as in automated testing using
the ctest -S scripts with the ``tribits_ctest_driver()`` function and the
``checkin-test.py`` tool.
The valid values of ``<Project>_ENABLE_KNOWN_EXTERNAL_REPOS_TYPE`` include
``Continuous``, ``Nightly``, and ``Experimental``. Only repositories listed
in the file ``<EXTRAREPOSFILE>`` that match this type will be included. Note
that ``Nightly`` matches ``Continuous`` and ``Experimental`` matches
``Nightly`` and ``Continuous`` and therefore includes all repos by default.
If ``<Project>_IGNORE_MISSING_EXTRA_REPOSITORIES`` is set to ``TRUE``, then
any extra repositories selected who's directory is missing will be ignored.
This is useful when the list of extra repos that a given developer develops or
tests is variable and one just wants TriBITS to pick up the list of existing
repos automatically.
If the file ``<projectDir>/cmake/ExtraRepositoriesList.cmake`` exists, then it
is used as the default value for ``<Project>_EXTRAREPOS_FILE``. However, the
default value for ``<Project>_ENABLE_KNOWN_EXTERNAL_REPOS_TYPE`` is empty so
no extra repositories are defined by default unless
``<Project>_ENABLE_KNOWN_EXTERNAL_REPOS_TYPE`` is specifically set to one of
the allowed values.
.. _<Project>_PRE_REPOSITORIES:
NOTE: The set of extra repositories listed in the file
``<Project>_EXTRAREPOS_FILE`` can be filtered down by setting the variables
``<Project>_PRE_REPOSITORIES`` if PRE extra repos are listed and/or
``<Project>_EXTRA_REPOSITORIES`` if POST extra repos are listed.
Selecting a different source location for a package
---------------------------------------------------
The source location for any package can be changed by configuring with::
-D<TRIBITS_PACKAGE>_SOURCE_DIR_OVERRIDE:STRING=<path>
Here, ``<path>`` can be a relative path or an absolute path, but in both cases
must be under the project source directory (otherwise, an error will occur).
The relative path will then become the relative path for the package under the
binary tree as well.
This can be used, for example, to use a different repository for the
implementation of a package that is otherwise snapshotted into the base
project source repository (e.g. Kokkos in Trilinos).
Reconfiguring completely from scratch
-------------------------------------
To reconfigure from scratch, one needs to delete the the ``CMakeCache.txt``
and base-level ``CMakeFiles/`` directory, for example, as::
$ rm -rf CMakeCache.txt CMakeFiles/
$ ./do-configure [options]
Removing the ``CMakeCache.txt`` file is often needed when removing variables
from the configure line since they are already in the cache. Removing the
``CMakeFiles/`` directories is needed if there are changes in some CMake
modules or the CMake version itself. However, usually removing just the
top-level ``CMakeCache.txt`` and ``CMakeFiles/`` directory is enough to
guarantee a clean reconfigure from a dirty build directory.
If one really wants a clean slate, then try::
$ rm -rf `ls | grep -v do-configure`
$ ./do-configure [options]
Viewing configure errors
-------------------------
To view various configure errors, read the file::
$BUILD_BASE_DIR/CMakeFiles/CMakeError.log
This file contains detailed output from try-compile commands, Fortran/C name
mangling determination, and other CMake-specific information.
Adding configure timers
-----------------------
To add timers to various configure steps, configure with::
-D <Project>_ENABLE_CONFIGURE_TIMING=ON
This will do bulk timing for the major configure steps which is independent
of the number of packages in the project.
To additionally add timing for the configure of individual packages, configure
with::
-D <Project>_ENABLE_CONFIGURE_TIMING=ON \
-D <Project>_ENABLE_PACKAGE_CONFIGURE_TIMING=ON
If you are configuring a large number of packages (perhaps by including a lot
of add-on packages in extra repos) then you might not want to enable
package-by-package timing since it can add some significant overhead to the
configure times.
If you just want to time individual packages instead, you can enable that
with::
-D <Project>_ENABLE_CONFIGURE_TIMING=ON \
-D <TRIBITS_PACKAGE_0>_PACKAGE_CONFIGURE_TIMING=ON \
-D <TRIBITS_PACKAGE_1>_PACKAGE_CONFIGURE_TIMING=ON \
...
NOTES:
* This requires that you are running on a Linux/Unix system that has the
standard shell command ``date``. CMake does not have built-in timing
functions so this system command needs to be used instead. This will report
timings to 0.001 seconds but note that the overall configure time will go up
due to the increased overhead of calling ``date`` as a process shell
command.
* '''WARNING:''' Because this feature has to call the ``data`` using CMake's
``execute_process()`` command, it can be expensive. Therefore, this should
really only be turned on for large projects (where the extra overhead is
small) or for smaller projects for extra informational purposes.
Generating export files
-----------------------
The project <Project> can generate export files for external CMake projects.
These export files provide the lists of libraries, include directories, compilers
and compiler options, etc.
To configure to generate CMake export files for the project, configure with::
-D <Project>_ENABLE_INSTALL_CMAKE_CONFIG_FILES=ON
This will generate the file ``<Project>Config.cmake`` for the project and the
files ``<Package>Config.cmake`` for each enabled package in the build tree.
In addition, this will install versions of these files into the install tree.
The list of export files generated can be reduced by specifying the exact list
of packages the files are requested for with::
-D <Project>_GENERATE_EXPORT_FILES_FOR_ONLY_LISTED_PACKAGES="<pkg0>;<pkg1>"
To only install the package ``<Package>Config.cmake`` files and **not** the
project-level ``<Project>Config.cmake`` file, configure with::
-D <Project>_ENABLE_INSTALL_CMAKE_CONFIG_FILES=ON \
-D <Project>_SKIP_INSTALL_PROJECT_CMAKE_CONFIG_FILES=ON \
NOTES:
* Only enabled packages will have their export files generated.
* One would only want to limit the export files generated for very large
projects where the cost my be high for doing so.
* One would want to skip the installation of the project-level
``<Project>Config.cmake`` file in cases where the TriBITS project's packages
may be built in smaller subsets of packages in different individual CMake
project builds where there is no clear completion to the installation of the
packages for a given TriBITS project containing a larger collection of
packages.
Generating a project repo version file
--------------------------------------
.. _<Project>_GENERATE_REPO_VERSION_FILE:
When working with local git repos for the project sources, one can generate a
``<Project>RepoVersion.txt`` file which lists all of the repos and their
current versions using::
-D <Project>_GENERATE_REPO_VERSION_FILE=ON
This will cause a ``<Project>RepoVersion.txt`` file to get created in the
binary directory, get installed in the install directory, and get included in
the source distribution tarball.
NOTE: If the base ``.git/`` directory is missing, then no
``<Project>RepoVersion.txt`` file will get generated and a ``NOTE`` message is
printed to cmake STDOUT.
Generating git version date files
---------------------------------
.. _<Project>_GENERATE_VERSION_DATE_FILES:
When working with local git repos for the project sources, one can generate
the files ``VersionDate.cmake`` and ``<Project>_version_date.h`` in the build
directory by setting::
-D <Project>_GENERATE_VERSION_DATE_FILES=ON
These files are generated in the build directory and the file
``<Project>_version_date.h`` is installed in the installation directory. (In
addition, these files are also generated for each extra repository that are
also version-controlled repos, see `<Project>_EXTRAREPOS_FILE`_.)
These files contain ``<PROJECT_NAME_UC>_VERSION_DATE`` which is a 10-digit
date-time version integer. This integer is created by first using git to
extract the commit date for ``HEAD`` using the command::
env TZ=GMT git log --format="%cd" --date=iso-local -1 HEAD
which returns the date and time for the commit date of ``HEAD`` in the form::
"YYYY-MM-DD hh:mm:ss +0000"
This git commit date is then is used to create a 10-digit date/time integer of
the form::
YYYYMMDDhh
This 10-digit integer is set to a CMake variable
``<PROJECT_NAME_UC>_VERSION_DATE`` in the generated ``VersionDate.cmake`` file
and a C/C++ preprocessor macro ``<PROJECT_NAME_UC>_VERSION_DATE`` in the
generated ``<Project>_version_date.h`` header file.
This 10-digit date/time integer ``YYYYMMDDhh`` will fit in a signed 32-bit
integer with a maximum value of ``2^32 / 2 - 1`` = ``2147483647``. Therefore,
the maximum date that can be handled is the year 2147 with the max date/time
of ``2147 12 31 23`` = ``2147123123``.
The file ``<Project>_version_date.h`` is meant to be included by downstream
codes to determine the version of ``<Project>`` being used and allows
``<PROJECT_NAME_UC>_VERSION_DATE`` to be used in C/C++ ``ifdefs`` like::
#if defined(<PROJECT_NAME_UC>_VERSION_DATE) && <PROJECT_NAME_UC>_VERSION_DATE >= 2019032704
/* The version is newer than 2019-03-27 04:00:00 UTC */
...
#else
/* The version is older than 2019-03-27 04:00:00 UTC */
...
#endif
This allows downstream codes to know the fine-grained version of <Project> at
configure and build time to adjust for the addition of new features,
deprecation of code, or breaks in backward compatibility (which occur in
specific commits with unique commit dates).
NOTE: If the branch is not hard-reset then the first-parent commits on that
branch will have monotonically increasing git commit dates (adjusted for UTC).
This assumption is required for the correct usage of the
``<PROJECT_NAME_UC>_VERSION_DATE`` macro as demonstrated above.
NOTE: If the base ``.git/`` directory is missing or the version of git is not
2.10.0 or greater (needed for the ``--date=iso-local`` argument), then the
``<Project>_version_date.h`` file will still get generated but will have an
undefined macro ``<PROJECT_NAME_UC>_VERSION_DATE`` and a ``NOTE`` message will
be printed to cmake STDOUT.
CMake configure-time development mode and debug checking
--------------------------------------------------------
To turn off CMake configure-time development-mode checking, set::
-D <Project>_ENABLE_DEVELOPMENT_MODE=OFF
This turns off a number of CMake configure-time checks for the <Project>
TriBITS/CMake files including checking the package dependencies and other
usage of TriBITS. These checks can be expensive and may also not be
appropriate for a tarball release of the software. However, this also turns
off strong compiler warnings so this is not recommended by default (see
`<TRIBITS_PACKAGE>_DISABLE_STRONG_WARNINGS`_). For a release of <Project>
this option is set OFF by default.
One of the CMake configure-time debug-mode checks performed as part of
``<Project>_ENABLE_DEVELOPMENT_MODE=ON`` is to assert the existence of TriBITS
package directories. In development mode, the failure to find a package
directory is usually a programming error (i.e. a miss-spelled package
directory name). But in a tarball release of the project, package directories
may be purposefully missing (see `Creating a tarball of the source tree`_) and
must be ignored.
When building from a reduced source tarball created from the
development sources, set::
-D <Project>_ASSERT_DEFINED_DEPENDENCIES=OFF
or to ``IGNORE``. (valid values include ``FATAL_ERROR``, ``SEND_ERROR``,
``WARNING``, ``NOTICE``, ``IGNORE`` and ``OFF``)
Setting this ``OFF`` will cause the TriBITS CMake configure to simply ignore
any undefined packages and turn off all dependencies on these missing
packages.
Another type of checking is for optional inserted/external packages
(e.g. packages who's source can optionally be included and is flagged with
``tribits_allow_missing_external_packages()``). Any of these package
directories that are missing result in the packages being silently ignored by
default. However, notes on what missing packages are being ignored can
printed by configuring with::
-D <Project>_WARN_ABOUT_MISSING_EXTERNAL_PACKAGES=TRUE
These warnings starting with 'NOTE' (not starting with 'WARNING' that would
otherwise trigger warnings in CDash) about missing inserted/external packages
will print regardless of the setting for
``<Project>_ASSERT_DEFINED_DEPENDENCIES``.
Finally, ``<Project>_ENABLE_DEVELOPMENT_MODE=ON`` results in a number of
checks for invalid usage of TriBITS in the project's ``CMakeLists.txt`` files
and will, by default, abort configure with a fatal error on the first failed
check. This is appropriate for development mode when a project is clean of all
such invalid usage patterns but there are times when it makes sense to report
these check failures in different ways (such as when upgrading TriBITS in a
project that has some invalid usage patterns that just happen work but may be
disallowed in future versions of TriBITS). To change how these invalid usage
checks are handled, set::
-D <Project>_ASSERT_CORRECT_TRIBITS_USAGE=<check-mode>
where ``<check-mode>`` can be ``FATAL_ERROR``, ``SEND_ERROR``, ``WARNING``,
``IGNORE`` or ``OFF`` (where ``IGNORE`` or ``OFF`` avoids any error reporting
or warnings).
For ``<Project>_ENABLE_DEVELOPMENT_MODE=OFF``, the default for
``<Project>_ASSERT_CORRECT_TRIBITS_USAGE`` is set to ``IGNORE``.
Building (Makefile generator)
=============================
This section described building using the default CMake Makefile generator.
Building with the Ninja is described in section `Building (Ninja generator)`_.
But every other CMake generator is also supported such as Visual Studio on
Windows, XCode on Macs, and Eclipse project files but using those build
systems are not documented here (consult standard CMake and concrete build
tool documentation).
Building all targets
--------------------
To build all targets use::
$ make [-jN]
where ``N`` is the number of processes to use (i.e. 2, 4, 16, etc.) .
Discovering what targets are available to build
-----------------------------------------------
CMake generates Makefiles with a 'help' target! To see the targets at the
current directory level type::
$ make help
NOTE: In general, the ``help`` target only prints targets in the current
directory, not targets in subdirectories. These targets can include object
files and all, anything that CMake defines a target for in the current
directory. However, running ``make help`` it from the base build directory
will print all major targets in the project (i.e. libraries, executables,
etc.) but not minor targets like object files. Any of the printed targets can
be used as a target for ``make <some-target>``. This is super useful for just
building a single object file, for example.
Building all of the targets for a package
-----------------------------------------
To build only the targets for a given TriBITS package, one can use::
$ make <TRIBITS_PACKAGE>_all
or::
$ cd packages/<TRIBITS_PACKAGE>
$ make
This will build only the targets for TriBITS package ``<TRIBITS_PACKAGE>`` and
its required upstream targets.
Building all of the libraries for a package
-------------------------------------------
To build only the libraries for given TriBITS package, use::
$ make <TRIBITS_PACKAGE>_libs
Building all of the libraries for all enabled packages
------------------------------------------------------
To build only the libraries for all enabled TriBITS packages, use::
$ make libs
NOTE: This target depends on the ``<PACKAGE>_libs`` targets for all of the
enabled ``<Project>`` packages. You can also use the target name
``'<Project>_libs``.
Building a single object file
-----------------------------
To build just a single object file (i.e. to debug a compile problem), first,
look for the target name for the object file build based on the source file,
for example for the source file ``SomeSourceFile.cpp``, use::
$ make help | grep SomeSourceFile
The above will return a target name like::
... SomeSourceFile.o
To find the name of the actual object file, do::
$ find . -name "*SomeSourceFile*.o"
that will return something like::
./CMakeFiles/<source-dir-path>.dir/SomeSourceFile.cpp.o
(but this file location and name depends on the source directory structure,
the version of CMake, and other factors). Use the returned name (exactly) for
the object file returned in the above find operation to remove the object file
first, for example, as::
$ rm ./CMakeFiles/<source-dir-path>.dir/SomeSourceFile.cpp.o
and then build it again, for example, with::
$ make SomeSourceFile.o
Again, the names of the target and the object file name an location depend on
the CMake version, the structure of your source directories and other factors
but the general process of using ``make help | grep <some-file-base-name>`` to
find the target name and then doing a find ``find . -name
"*<some-file-base-name>*"`` to find the actual object file path always works.
For this process to work correctly, you must be in the subdirectory where the
``tribits_add_library()`` or ``tribits_add_executable()`` command is called
from its ``CMakeLists.txt`` file, otherwise the object file targets will not be
listed by ``make help``.
NOTE: CMake does not seem to not check on dependencies when explicitly
building object files as shown above so you need to always delete the object
file first to make sure that it gets rebuilt correctly.
Building with verbose output without reconfiguring
--------------------------------------------------
One can get CMake to generate verbose make output at build time by just
setting the Makefile variable ``VERBOSE=1``, for example, as::
$ make VERBOSE=1 [<SOME_TARGET>]
Any number of compile or linking problem can be quickly debugged by seeing the
raw compile and link lines. See `Building a single object file`_ for more
details.
NOTE: The libraries listed on the link line are often in the form
``-L<lib-dir> -l<lib1> -l<lib2>`` even if one passed in full library paths for
TPLs through ``TPL_<TPLNAME>_LIBRARIES`` (see `Enabling support for an
optional Third-Party Library (TPL)`_). That is because CMake tries to keep
the link lines as short as possible and therefore it often does this
translation automatically (whether you want it to or not).
Relink a target without considering dependencies
------------------------------------------------
CMake provides a way to rebuild a target without considering its dependencies
using::
$ make <SOME_TARGET>/fast
Building (Ninja generator)
==========================
When using the Ninja back-end (see `Enabling support for Ninja`_), one can
build with simply::
ninja -j<N>
or use any options and workflows that the raw ``ninja`` executable supports
(see ``ninja --help``). In general, the ``ninja`` command can only be run
from the base project binary directory and running it from the subdirectory
will not work without having to use the ``-C <dir>`` option pointing to the
base dir and one will need to pass in specific target names or the entire
project targets will get built with the default ``all`` target.
But if the TriBITS-created Ninja makefiles are also generated (see
`<Project>_WRITE_NINJA_MAKEFILES`_), then ``make`` can be run from any
subdirectory to build targets in that subdirectory. Because of this and other
advantages of these makefiles, the majority of the instructions below will be
for running with these makefiles, not the raw ``ninja`` command. These
makefiles define many of the standard targets that are provided by the default
CMake-generated makefiles like ``all``, ``clean``, ``install``, and
``package_source`` (run ``make help`` to see all of the targets).
Building in parallel with Ninja
-------------------------------
By default, running the raw ``ninja`` command::
ninja
will use **all** of the free cores on the node to build targets in parallel on
the machine! This will not overload the machine but it will not leave any
unused cores either (see Ninja documentation).
To run the raw ``ninja`` command to build with a specific number of build
processes (regardless of machine load), e.g. ``16`` build processes, use::
ninja -j16
When using the TriBITS-generated Ninja makefiles, running with::
make
will also use all of the free cores, and **not** just one process like with
the default CMake-generated makefiles.
But with the TriBITS-generated Ninja makefiles, to build with a specific
number of build processes (regardless of machine load), e.g. ``16`` build
processes, one can **not** use ``-j<N>`` but instead must use the ``NP=<N>``
argument with::
make NP=16
which will call ``ninja -j16`` internally.
That reason that ``-j16`` cannot be used with these TriBITS-generated Ninja
Makefiles is that the ``make`` program does not inform the executed
``Makefile`` the value of this option and therefore this can't be passed on to
the underlying ``ninja`` command. Therefore the ``make`` option ``-j<N>`` is
essentially ignored. Therefore, running ``make -j16`` will result in calling
raw ``ninja`` which will use all of the free cores on the machine. Arguably
that is better than using only one core and will not overload the machine but
still this is behavior the user must be aware.
Building in a subdirectory with Ninja
-------------------------------------
To build from a binary subdirectory in the build tree with the
TriBITS-generated Ninja makefiles, just ``cd`` into that directory and build
with::
cd <some-subdir>/
make NP=16
and this will only build targets that are defined in that subdirectory. (See
the raw ``ninja`` command that gets called in this case which is echoed at the
top.)
Building verbose without reconfiguring with Ninja
-------------------------------------------------
To build targets and see the full build lines for each with the Ninja
makefiles, build with::
make NP=10 VERBOSE=1 <target_name>
But note that ``ninja`` will automatically provide the full build command for
a build target when that target fails so the ``VERBOSE=1`` option is not
needed in the case were a build target is failing but is useful in other cases
none the less.
Discovering what targets are available to build with Ninja
----------------------------------------------------------
To determine the target names for library, executable (or any other general
target except for object files) that can be built in any binary directory with
the TriBITS-generated Ninja Makefiles, use::
make help
which will return::
This Makefile supports the following standard targets:
all (default)
clean
help
install
test
package
package_source
edit_cache
rebuild_cache
and the following project targets:
<target0>
<target1>
...
Run 'make help-objects' to list object files.
To determine the target names for building any object files that can be run in
any directory with the TriBITS-generated Ninja Makefiles, use::
make help-objects
which will return::
This Makefile supports the following object files:
<object-target-0>
<object-target-1>
...
NOTE: The raw ``ninja`` command does not provide a compact way to list all of
the targets that can be built in any given directory.
Building specific targets with Ninja
------------------------------------
To build with any specific target, use::
make NP=16 <target>
See `Discovering what targets are available to build with Ninja`_ for how to get
a list of targets.
Building single object files with Ninja
---------------------------------------
To build any object file, use::
make NP=16 <object-target>
See `Discovering what targets are available to build with Ninja`_ for how to get
a list of the object file targets.
Note that unlike the native CMake-generated Makefiles, when an object target
like this gets built, Ninja will build all of the upstream targets as well.
For example, if you change an upstream header file and just want to see the
impact of building a single ``*.o`` file, this target will build **all** of
the targets for the library where the object fill will gets used. But this is
not generally what one wants to do to iteratively develop the compilation of a
single object file.
To avoid that behavior and instead just build a single ``*.o`` file, first one
must instead use::
make VERBOSE=1 <object-target>
to print the command-line for building the one object file, and then ``cd`` to
the base project binary directory and manually run that command to build only
that object file. (This can be considered a regression w.r.t. the native
CMake-generated Makefiles.)
NOTE: The raw ``ninja`` command does not provide a compact way to list all of
the object files that can be built and does not make it easy to build a single
object file.
Cleaning build targets with Ninja
---------------------------------
With the TriBITS-generated Ninja Makefiles, when one runs::
make clean
in a subdirectory to clean out the targets in that subdirectory, the
underlying ``ninja`` command will actually delete not only the targets in that
subdirectory but instead will clean **all** the targets upstream from the
targets in the current subdirectory as well! This is **not** the behavior of
the default CMake-generated Makefiles where only the generated files in that
subdirectory will be removed and files for upstream dependencies.
Therefore, if one then wants to clean only the object files, libraries, and
executables in a subdirectory, one should just manually delete them with::
cd <some-subdir>/
find . -name "*.o" -exec rm {} \;
find . -name "lib*.a" -exec rm {} \;
find . -name "lib*.so*" -exec rm {} \;
find . -name "*.exe" -exec rm {} \;
then one can rebuild just the targets in that subdirectory with::
make NP=10
Testing with CTest
==================
This section assumes one is using the CMake Makefile generator described
above. Also, the ``ctest`` does not consider make dependencies when running
so the software must be completely built before running ``ctest`` as described
here.
Running all tests
-----------------
To run all of the defined tests (i.e. created using ``tribits_add_test()`` or
``tribits_add_advanced_test()``) use::
$ ctest -j<N>
(where ``<N>`` is an integer for the number of processes to try to run tests
in parallel). A summary of what tests are run and their pass/fail status will
be printed to the screen. Detailed output about each of the tests is archived
in the generate file::
Testing/Temporary/LastTest.log
where CTest creates the ``Testing`` directory in the local directory where you
run it from.
NOTE: The ``-j<N>`` argument allows CTest to use more processes to run tests.
This will intelligently load balance the defined tests with multiple processes
(i.e. MPI tests) and will try not exceed the number of processes ``<N>``.
However, if tests are defined that use more that ``<N>`` processes, then CTest
will still run the test but will not run any other tests while the limit of
``<N>`` processes is exceeded. To exclude tests that require more than
``<N>`` processes, set the cache variable ``MPI_EXEC_MAX_NUMPROCS`` (see
`Configuring with MPI support`_).
Only running tests for a single package
---------------------------------------
Tests for just a single TriBITS package can be run with::
$ ctest -j4 -L <TRIBITS_PACKAGE>
or::
$ cd packages/<TRIBITS_PACKAGE>
$ ctest -j4
This will run tests for packages and subpackages inside of the parent package
``<TRIBITS_PACKAGE>``.
NOTE: CTest has a number of ways to filter what tests get run. You can use
the test name using ``-E``, you can exclude tests using ``-I``, and there are
other approaches as well. See ``ctest --help`` and on-line documentation, and
experiment for more details.
Running a single test with full output to the console
-----------------------------------------------------
To run just a single test and send detailed output directly to the console,
one can run::
$ ctest -R ^<FULL_TEST_NAME>$ -VV
However, when running just a single test, it is usually better to just run the
test command manually to allow passing in more options. To see what the
actual test command is, use::
$ ctest -R ^<FULL_TEST_NAME>$ -VV -N
This will only print out the test command that ``ctest`` runs and show the
working directory. To run the test exactly as ``ctest`` would, cd into the
shown working directory and run the shown command.
Overriding test timeouts
-------------------------
The configured global test timeout described in ``Setting test timeouts at
configure time`` can be overridden on the CTest command-line as::
$ ctest --timeout <maxSeconds>
This will override the configured cache variable `DART_TESTING_TIMEOUT`_
(actually, the scaled value set as ``TimeOut`` in the file
``DartConfiguration.tcl``). However, this will **not** override the test
time-outs set on individual tests on a test-by-test basis!
**WARNING:** Do not try to use ``--timeout=<maxSeconds>`` or CTest will just
ignore the argument!
Running memory checking
-----------------------
To configure for running memory testing with ``valgrind``, use::
-D MEMORYCHECK_COMMAND=<abs-path-to-valgrind>/valgrind \
-D MEMORYCHECK_SUPPRESSIONS_FILE=<abs-path-to-supp-file0> \
-D MEMORYCHECK_COMMAND_OPTIONS="-q --trace-children=yes --tool=memcheck \
--leak-check=yes --workaround-gcc296-bugs=yes \
--num-callers=50 --suppressions=<abs-path-to-supp-file1> \
... --suppressions=<abs-path-to-supp-fileN>"
Above, you have to set the absolute path to the valgrind executable to run
using ``MEMORYCHECK_COMMAND`` as CMake will not find this for you by default.
To use a single valgrind suppression file, just set
``MEMORYCHECK_SUPPRESSIONS_FILE`` to the path of that suppression file as
shown above. To add other suppression files, they have to be added as other
general valgrind arguments in ``MEMORYCHECK_COMMAND_OPTIONS`` as shown.
After configuring with the above options, to run the memory tests for all
enabled tests, from the **base** project build directory, do::
$ ctest -T memcheck
This will run valgrind on **every** test command that is run by ctest.
To run valgrind on the tests for a single package, from the **base** project
directory, do::
$ ctest -T memcheck -L <TRIBITS_PACKAGE>
To run valgrind on a specific test, from the **base** project directory, do::
$ ctest -T memcheck -R ^<FULL_TEST_NAME>$
Detailed output from valgrind is printed in the file::
Testing/Temporary/LastDynamicAnalysis_<DATE_TIME>.log
NOTE: If you try to run memory tests from any subdirectories, it will not
work. You have to run them from the ***base** project build directory as
shown above. A nice way to view valgrind results is to submit to CDash using
the ``dashboard`` target (see `Dashboard submissions`_).
NOTE: You have to use the valgrind option ``--trace-children=yes`` to trace
through child processes. This is needed if you have tests that are given as
CMake -P scripts (such as advanced tests) or tests driven in bash, Perl,
Python, or other languages.
Installing
==========
After a build and test of the software is complete, the software can be
installed. Actually, to get ready for the install, the install directory must
be specified at configure time by setting the variable
``CMAKE_INSTALL_PREFIX`` in addition to other variables that affect the
installation (see the following sections). The other commands described below
can all be run after the build and testing is complete.
For the most typical case where the software is build and installed on the
same machine in the same location where it will be used, one just needs to
configure with::
$ cmake -DCMAKE_INSTALL_PREFIX=<install-base-dir> [other options] \
${SOURCE_DIR}
$ make -j<N> install
For more details, see the following subsections:
* `Setting the install prefix`_
* `Setting install RPATH`_
* `Avoiding installing libraries and headers`_
* `Installing the software`_
* `Using the installed software in downstream CMake projects`_
* `Using packages from the build tree in downstream CMake projects`_
Setting the install prefix
--------------------------
In order to set up for the install, the install prefix should be set up at
configure time by setting, for example::
-D CMAKE_INSTALL_PREFIX=$HOME/install/<Project>/mpi/opt
The default location for the installation of libraries, headers, and
executables is given by the variables (with defaults)::
-D <Project>_INSTALL_INCLUDE_DIR="include" \
-D <Project>_INSTALL_LIB_DIR="lib" \
-D <Project>_INSTALL_RUNTIME_DIR="bin" \
-D <Project>_INSTALL_EXAMPLE_DIR="example"
If these paths are relative (i.e. don't start with "/" and use type
``STRING``) then they are relative to ``${CMAKE_INSTALL_PREFIX}``. Otherwise
the paths can be absolute (use type ``PATH``) and don't have to be under
``${CMAKE_INSTALL_PREFIX}``. For example, to install each part in any
arbitrary location use::
-D <Project>_INSTALL_INCLUDE_DIR="/usr/<Project>_include" \
-D <Project>_INSTALL_LIB_DIR="/usr/<Project>_lib" \
-D <Project>_INSTALL_RUNTIME_DIR="/usr/<Project>_bin" \
-D <Project>_INSTALL_EXAMPLE_DIR="/usr/share/<Project>/examples"
NOTE: The defaults for the above include paths will be set by the standard
CMake module ``GNUInstallDirs`` if ``<Project>_USE_GNUINSTALLDIRS=TRUE`` is
set. Some projects have this set by default (see the ``CMakeCache.txt`` after
configuring to see default being used by this project).
WARNING: To overwrite default relative paths, you must use the data type
``STRING`` for the cache variables. If you don't, then CMake will use the
current binary directory for the base path. Otherwise, if you want to specify
absolute paths, use the data type ``PATH`` as shown above.
Setting install ownership and permissions
-----------------------------------------
By default, when installing with the ``install`` (or
``install_package_by_package``) target, any files and directories created are
given the default permissions for the user that runs the install command (just
as if they typed ``mkdir <some-dir>`` or ``touch <some-file>``). On most
Unix/Linux systems, one can use ``umask`` to set default permissions and one
can set the default group and the group sticky bit to control what groups owns
the newly created files and directories. However, some computer systems do
not support the group sticky bit and there are cases where one wants or needs
to provide different group ownership and write permissions.
To control what group owns the install-created files and directories related
to ``CMAKE_INSTALL_PREFIX`` and define the permissions on those, one can set
one or more of the following options::
-D <Project>_SET_GROUP_AND_PERMISSIONS_ON_INSTALL_BASE_DIR=<install-base-dir> \
-D <Project>_MAKE_INSTALL_GROUP=[<owning-group>] \
-D <Project>_MAKE_INSTALL_GROUP_READABLE=[TRUE|FALSE] \
-D <Project>_MAKE_INSTALL_GROUP_WRITABLE=[TRUE|FALSE] \
-D <Project>_MAKE_INSTALL_WORLD_READABLE=[TRUE|FALSE] \
(where ``<Project>_SET_GROUP_AND_PERMISSIONS_ON_INSTALL_BASE_DIR`` must be a
base directory of ``CMAKE_INSTALL_PREFIX``). This has the impact of both
setting the built-in CMake variable
``CMAKE_INSTALL_DEFAULT_DIRECTORY_PERMISSIONS`` with the correct permissions
according to these and also triggers the automatic running of the recursive
``chgrp`` and ``chmod`` commands starting from the directory
``<install-base-dir>`` on down, after all of the other project files have been
installed. The directory set by
``<Project>_SET_GROUP_AND_PERMISSIONS_ON_INSTALL_BASE_DIR`` and those below it
may be created by the ``install`` command by CMake (as it may not exist before
the install). If ``<Project>_SET_GROUP_AND_PERMISSIONS_ON_INSTALL_BASE_DIR``
is not given, then it is set internally to the same directory as
``CMAKE_INSTALL_PREFIX``.
For an example, to configure for an install based on a dated base directory
where a non-default group should own the installation and have group
read/write permissions, and "others" only have read access, one would
configure with::
-D CMAKE_INSTALL_PREFIX=$HOME/2020-04-25/my-proj \
-D <Project>_SET_GROUP_AND_PERMISSIONS_ON_INSTALL_BASE_DIR=$HOME/2020-04-25 \
-D <Project>_MAKE_INSTALL_GROUP=some-other-group \
-D <Project>_MAKE_INSTALL_GROUP_WRITABLE=TRUE \
-D <Project>_MAKE_INSTALL_WORLD_READABLE=TRUE \
Using these settings, after all of the files and directories have been
installed using the ``install`` or ``install_package_by_package`` build
targets, the following commands are automatically run at the very end::
chgrp some-other-group $HOME/2020-04-25
chmod g+rwX,o+rX $HOME/2020-04-25
chgrp some-other-group -R $HOME/2020-04-25/my-proj
chmod g+rwX,o+rX -R $HOME/2020-04-25/my-proj
That allows the owning group ``some-other-group`` to later modify or delete
the installation and allows all users to use the installation.
NOTES:
* Setting ``<Project>_MAKE_INSTALL_GROUP_WRITABLE=TRUE`` implies
``<Project>_MAKE_INSTALL_GROUP_READABLE=TRUE``.
* Non-recursive ``chgrp`` and ``chmod`` commands are run on the directories
above ``CMAKE_INSTALL_PREFIX``. Recursive ``chgrp`` and ``chmod`` commands
are only run on the base ``CMAKE_INSTALL_PREFIX`` directory itself. (This
avoids touching any files or directories not directly involved in this
install.)
Setting install RPATH
---------------------
Setting RPATH for installed shared libraries and executables
(i.e. ``BUILD_SHARED_LIBS=ON``) can be a little tricky. Some discussion of
how raw CMake handles RPATH and installations can be found at:
.. _CMake RPATH handling reference:
https://cmake.org/Wiki/CMake_RPATH_handling
The TriBITS/CMake build system being used for this <Project> CMake project
defines the following default behavior for installed RPATH (which is not the
same as the raw CMake default behavior):
* ``CMAKE_INSTALL_RPATH`` for all libraries and executables built and installed
by this CMake project is set to ``${<Project>_INSTALL_LIB_DIR}``. (This
default is controlled by the variable `<Project>_SET_INSTALL_RPATH`_.)
* The path for all shared external libraries (i.e. TPLs) is set to the
location of the external libraries passed in (or automatically discovered)
at configure time. (This is controlled by the built-in CMake cache variable
``CMAKE_INSTALL_RPATH_USE_LINK_PATH`` which is set to ``TRUE`` by default
for most TriBITS projects but is empty "" for raw CMake.)
The above default behavior allows the installed executables and libraries to
be run without needing to set ``LD_LIBRARY_PATH`` or any other system
environment variables. However, this setting does not allow the installed
libraries and executables to be easily moved or relocated. There are several
built-in CMake variables that control how RPATH is handled related to
installations. The built-in CMake variables that control RPATH handling
include ``CMAKE_INSTALL_RPATH``, ``CMAKE_SKIP_BUILD_RPATH``,
``CMAKE_SKIP_INSTALL_RPATH``, ``CMAKE_SKIP_RPATH``,
``CMAKE_BUILD_WITH_INSTALL_RPATH``, ``CMAKE_INSTALL_RPATH_USE_LINK_PATH``.
The TriBITS/CMake build system for <Project> respects all of these raw CMake
variables and their documented effect on the build and install.
In addition, this TriBITS/CMake project defines the cache variable:
.. _<Project>_SET_INSTALL_RPATH:
**<Project>_SET_INSTALL_RPATH**: If ``TRUE``, then the global CMake variable
``CMAKE_INSTALL_RPATH`` is set to ``<Project>_INSTALL_LIB_DIR``. If
``CMAKE_INSTALL_RPATH`` is set by the user, then that is used instead. This
avoids having to manually set ``CMAKE_INSTALL_RPATH`` to the correct default
install directory.
Rather than re-documenting all of the native CMake RPATH variables mentioned
above, instead, we describe how these variables should be set for different
installation and distribution scenarios:
0. `Use default CMake behavior`_
1. `Libraries and executables are built, installed and used on same machine`_ (TriBITS default)
2. `Targets will move after installation`_
3. `Targets and TPLs will move after installation`_
4. `Explicitly set RPATH for the final target system`_
5. `Define all shared library paths at runtime using environment variables`_
These scenarios in detail are:
.. _Use default CMake behavior:
0. *Use default CMake behavior:* If one just wants the default raw CMake
behavior with respect to RPATH, then configure with::
-D<Project>_SET_INSTALL_RPATH=FALSE \
-DCMAKE_INSTALL_RPATH_USE_LINK_PATH=FALSE \
This will not put any directories into RPATH for the installed libraries or
executables. This is the same behavior as setting
``CMAKE_SKIP_INSTALL_RPATH=TRUE`` (see `Define all shared library paths at
runtime using environment variables`_).
.. _Libraries and executables are built, installed and used on same machine:
1. *Libraries and executables are built, installed and used on same machine
(TriBITS default):* One needs no options for this behavior but to make this
explicit then configure with::
-D<Project>_SET_INSTALL_RPATH=TRUE \
-DCMAKE_INSTALL_RPATH_USE_LINK_PATH=TRUE \
As described above, this allows libraries and executables to be used right
away once installed without needing to set any environment variables.
Note that this also allows the installed libraries and executables to be
moved to the same location on an different but identical machine as well.
.. _Targets will move after installation:
2. *Targets will move after installation:* In this scenario, the final
location of built libraries and executables will be different on the same
machine or an otherwise identical machine. In this case, we assume that
all of the external library references and directories would be the same.
In this case, one would generally configure with::
-D<Project>_SET_INSTALL_RPATH=FALSE \
-DCMAKE_INSTALL_RPATH_USE_LINK_PATH=TRUE \
Then, to run any executables using these shared libraries, one must update
LD_LIBRARY_PATH as::
$ export LD_LIBRARY_PATH=<final-install-dir>/lib:$LD_LIBRARY_PATH
Or, if the final directory location is known, then one can directly
``CMAKE_INSTALL_RPATH`` at configure time to match the final target system
and then one does not need to mess with ``LD_LIBRARY_PATH`` or any other
env variables (see `Explicitly set RPATH for the final target system`_).
.. _Targets and TPLs will move after installation:
3. *Targets and TPLs will move after installation*: In this scenario, the
final location of the installed libraries and executables will not be the
same as the initial install location and the external library (i.e. TPL)
locations may not be the same on the final target machine. This can be
handled in one of two ways. First, if one knows the final target machine
structure, then one can directly set ``CMAKE_INSTALL_RPATH`` to the
locations on the final target machine (see `Explicitly set RPATH for the
final target system`_). Second, if one does not know the final machine
directory structure (or the same distribution needs to support several
different systems with different directory structures), then one can set
``CMAKE_SKIP_INSTALL_RPATH=TRUE`` and then require setting the paths in the
env (see `Define all shared library paths at runtime using environment
variables`_).
.. _Explicitly set RPATH for the final target system:
4. *Explicitly set RPATH for the final target system:* If one knows the
directory structure of the final target machine where the installed
libraries and executables will be used, then one can set those paths at
configure time with::
-D<Project>_SET_INSTALL_RPATH=FALSE \
-DCMAKE_INSTALL_RPATH_USE_LINK_PATH=FALSE \
-DCMAKE_INSTALL_RPATH="<path0>;<path1>;..." \
In this case ``CMAKE_INSTALL_RPATH`` is explicitly set. (The value of
``<Project>_SET_INSTALL_RPATH`` has no effect but setting it to ``FALSE``
may help to avoid confusion.)
Once the install directories are moved to the final location, the
executables can be run without any need to set environment variables.
Note that TriBITS also accepts the directory separator "``:``" for::
-DCMAKE_INSTALL_RPATH="<path0>:<path1>:..." \
and replaces it internally with "``;``" which raw CMake requires. (This
makes it more robust to pass around inside of CMake code since "``;``"
means array boundary with CMake.). However, since ":" is not a valid
character for a path for any Unix system, this is a safe substitution (and
``CMAKE_INSTALL_RPATH`` is not used on Windows systems that allow "``:``"
in a directory path).
Also note that Linux supports RPATHs with the special value ``$ORIGIN`` to
allow for relative paths and for relocatable installations. (Mac OSX has
similar variables like ``@executable_path``.) With this, one can define
``CMAKE_INSTALL_RPATH`` using something like ``$ORIGIN/../lib``. See the
above `CMake RPATH handling reference`_ for more details.
.. _Define all shared library paths at runtime using environment variables:
5. *Define all shared library paths at runtime using environment variables:*
If one wants complete freedom to define the paths for the shared libraries
at runtime with ``LD_LIBRARY_PATH`` on Linux (and similar variables on
other platforms), then one can completely strip RPATH out of the installed
libraries and executables by configuring with::
-DCMAKE_SKIP_INSTALL_RPATH=TRUE \
This will result in all paths being stripped out of RPATH regardless of the
values of ``<Project>_SET_INSTALL_RPATH`` or
``CMAKE_INSTALL_RPATH_USE_LINK_PATH``. (This is the same default behavior
as raw CMake, see `Use default CMake behavior`_).
Then the runtime environment must be set up to find the correct shared
libraries in the correct order at runtime (e.g. by setting
``LD_LIBRARY_PATH``) . But this approach provides the most flexibility
about where executables and libraries are installed and run from.
Also note that, while not necessary, in order to avoid confusion, it is
likely desired to configure with::
-D<Project>_SET_INSTALL_RPATH=FALSE \
-DCMAKE_INSTALL_RPATH_USE_LINK_PATH=FALSE \
-DCMAKE_SKIP_INSTALL_RPATH=TRUE \
This will produce the least confusing CMake configure output.
One additional issue about RPATH handling on Mac OSX systems needs to be
mentioned. That is, in order for this default RPATH approach to work on OSX
systems, all of the upstream shared libraries must have
``@rpath/lib<libname>.dylib`` embedded into them (as shown by ``otool -L
<lib_or_exec>``). For libraries built and installed with CMake, the parent
CMake project must be configured with::
-DBUILD_SHARED_LIBS=ON \
-DCMAKE_INSTALL_RPATH_USE_LINK_PATH=TRUE \
-DCMAKE_MACOSX_RPATH=TRUE \
For other build systems, see their documentation for shared library support on
OSX. To see the proper way to handle RPATH on OSX, just inspect the build and
install commands that CMake generates (e.g. using ``make VERBOSE=1 <target>``)
for shared libraries and then make sure that these other build systems use
equivalent commands. If that is done properly for the chain of all upstream
shared libraries then the behaviors of this <Project> CMake project described
above should hold on OSX systems as well.
Avoiding installing libraries and headers
-----------------------------------------
By default, any libraries and header files defined by in the TriBITS project
<Project> will get installed into the installation directories specified by
``CMAKE_INSTALL_PREFIX``, ``<Project>_INSTALL_INCLUDE_DIR`` and
``<Project>_INSTALL_LIB_DIR``. However, if the primary desire is to install
executables only, then the user can set::
-D <Project>_INSTALL_LIBRARIES_AND_HEADERS=OFF
which, if in addition static libraries are being built
(i.e. ``BUILD_SHARED_LIBS=OFF``), this this option will result in no libraries
or headers being installed into the ``<install>/include/`` and
``<install>/lib/`` directories, respectively. However, if shared libraries
are being built (i.e. ``BUILD_SHARED_LIBS=ON``), they the libraries will be
installed in ``<install>/lib/`` along with the executables because the
executables can't run without the shared libraries being installed.
Installing the software
-----------------------
To install the software, type::
$ make install
Note that by default CMake actually puts in the build dependencies for
installed targets so in some cases you can just type ``make -j<N> install``
and it will also build the software before installing (but this can be
disabled by setting ``-DCMAKE_SKIP_INSTALL_ALL_DEPENDENCY=ON``). It is
advised to always build and test the software first before installing with::
$ make -j<N> && ctest -j<N> && make -j<N> install
This will ensure that everything is built correctly and all tests pass before
installing.
If there are build failures in any packages and one wants to still install the
packages that do build correctly, then configure with::
-DCMAKE_SKIP_INSTALL_ALL_DEPENDENCY=ON
and run the custom install target::
$ make install_package_by_package
This will ensure that every package that builds correctly will get installed.
(The default 'install' target aborts on the first file install failure.)
Using the installed software in downstream CMake projects
---------------------------------------------------------
As described in `Generating export files`_, when ``-D
<Project>_ENABLE_INSTALL_CMAKE_CONFIG_FILES=ON`` is set at configure time, a
``<Project>Config.cmake`` file and a different ``<Package>Config.cmake`` file
for each enabled package is installed into the install tree under ``-D
CMAKE_INSTALL_PREFIX=<upstreamInstallDir>``. A downstream CMake project can
then pull in CMake targets for the installed libraries using
``find_package()`` in the downstream project's ``CMakeLists.txt`` file. All
of the built and installed libraries can be pulled in and built against at the
project level by configuring the downstream CMake project with::
-D CMAKE_PREFIX_PATH=<upstreamInstallDir>
and having the downstream project's ``CMakeLists.txt`` file call, for
example::
find_package(<Project> REQUIRED)
...
target_link_libraries( <downstream-target>
PRIVATE <Project>::all_libs )
This will put the needed include directories and other imported compiler
options on the downstream compile lines as specified through the IMPORTED
library targets and will put the needed libraries on the link line.
To pull in libraries from only a subset of the installed packages ``<pkg0>
<pkg1> ...``, use, for example::
find_package(<Project> REQUIRED COMPONENTS <pkg0> <pkg1> ...)
...
target_link_libraries( <downstream-target>
PRIVATE <Project>::all_selected_libs )
The target ``<Project>::all_selected_libs`` only contains the library targets
for the selected packages (through their ``<Package>::all_libs`` targets) for
the packages requested in the ``COMPONENTS <pkg0> <pkg1> ...`` argument.
(NOTE, the target ``<Project>::all_libs`` is unaffected by the ``COMPONENTS``
argument and always links to all of the enabled package's libraries.)
Downstream projects can also pull in and use installed libraries by finding
individual packages by calling ``find_package(<Package> REQUIRED)`` for each
package ``<Package>`` and then linking against the defined IMPORTED CMake
target ``<Package>::all_libs`` such as::
find_package(<Package1> REQUIRED)
find_package(<Package2> REQUIRED)
...
target_link_libraries( <downstream-target>
PUBLIC <Package1>::all_libs
PRIVATE <Package2>::all_libs
)
Finding and using libraries for packages at the package-level provides better
fine-grained control over internal linking and provides greater flexibility in
case these packages are not all installed in the same upstream CMake project
in the future.
To see an example of all of these use cases being demonstrated, see
`TribitsExampleApp`_ and the `TriBITS TribitsExampleApp Tests`_.
Using packages from the build tree in downstream CMake projects
------------------------------------------------------------------
Note that libraries from enabled and built packages can also be used from the
``<Project>`` build tree without needing to install. Being able to build
against pre-built packages in the build tree can be very useful such as when
the project is part of a CMake super-build where one does not want to install
the intermediate packages.
Let ``<upstreamBuildDir>`` be the build directory for ``<Project>`` that has
already been configured and built (but not necessarily installed). A
downstream CMake project can pull in and link against any of the enabled
libraries in the upstream ``<Project>`` configuring the downstream CMake
project with::
-D CMAKE_PREFIX_PATH=<upstreamBuildDir>/cmake_packages
and then finding the individual packages and linking to them in the downstream
CMake project's ``CMakeLists.txt`` file as usual using, for example::
find_package(<Package1> REQUIRED)
find_package(<Package2> REQUIRED)
...
target_link_libraries( <downstream-target>
PUBLIC <Package1>::all_libs
PRIVATE <Package2>::all_libs
)
Note that in this case, ``target_link_libraries()`` ensures that the include
directories and other imported compiler options from the source tree and the
build tree are automatically injected into the build targets associated with
the ``<downstream-target>`` object compile lines and link lines.
Also note that package config files for all of the enabled external
packages/TPLs will also be written into the build tree under
``<upstreamBuildDir>/external_packages``. These contain modern CMake targets
that are pulled in by the downstream ``<Package>Config.cmake`` files under
``<upstreamBuildDir>/external_packages``. These external package/TPL config
files are placed in a separate directory to avoid being found by accident.
Installation Testing
====================
The CMake project <Project> has built-in support for testing an installation
of itself using its own tests and examples. The way it works is to configure,
build, and install just the libraries and header files using::
$ mkdir BUILD_LIBS
$ cd BUILD_LIBS/
$ cmake \
-DCMAKE_INSTALL_PREFIX=<install-dir> \
-D<Project>_ENABLE_ALL_PACKAGES=ON \
-D<Project>_ENABLE_TESTS=OFF \
[other options] \
<projectDir>
$ make -j16 install # or ninja -j16
and then create a different build directory to configure and build just the
tests and examples (not the libraries) against the pre-installed libraries and
header files using::
$ mkdir BUILD_TESTS
$ cd BUILD_TESTS/
$ cmake \
-D<Project>_ENABLE_ALL_PACKAGES=ON \
-D<Project>_ENABLE_TESTS=ON \
-D<Project>_ENABLE_INSTALLATION_TESTING=ON \
-D<Project>_INSTALLATION_DIR=<install-dir> \
[other options] \
<projectDir>
$ make -j16 # or ninja -j16
$ ctest -j16
If that second project builds and all the tests pass, then the project was
installed correctly. This uses the project's own tests and examples to test
the installation of the project. The library source and header files are
unused in the second project build. In fact, you can delete them and ensure
that they are not used in the build and testing of the tests and examples!
This can also be used for testing backward compatibility of the project (or
perhaps for a subset of packages). In this case, build and install the
libraries and header files for a newer version of the project and then
configure, build, and run the tests and examples for an older version of the
project sources pointing to the installed header files and libraries from the
newer version.
Packaging
=========
Packaged source and binary distributions can also be created using CMake and
CPack.
Creating a tarball of the source tree
-------------------------------------
To create a source tarball of the project, first configure with the list of
desired packages (see `Selecting the list of packages to enable`_) and pass in
::
-D <Project>_ENABLE_CPACK_PACKAGING=ON
To actually generate the distribution files, use::
$ make package_source
The above command will tar up *everything* in the source tree except for files
explicitly excluded in the CMakeLists.txt files and packages that are not
enabled so make sure that you start with a totally clean source tree before
you do this. You can clean the source tree first to remove all ignored files
using::
$ git clean -fd -x
You can include generated files in the tarball, such as Doxygen output files,
by creating them first, then running ``make package_source`` and they will be
included in the distribution (unless there is an internal exclude set).
Disabled subpackages can be included or excluded from the tarball by setting
``<Project>_EXCLUDE_DISABLED_SUBPACKAGES_FROM_DISTRIBUTION`` (the TriBITS
project has its own default, check ``CMakeCache.txt`` to see what the default
is). If ``<Project>_EXCLUDE_DISABLED_SUBPACKAGES_FROM_DISTRIBUTION=ON`` and
but one wants to include some subpackages that are otherwise excluded, just
enable them or their outer package so they will be included in the source
tarball. To get a printout of set regular expressions that will be used to
match files to exclude, set::
-D <Project>_DUMP_CPACK_SOURCE_IGNORE_FILES=ON
While a set of default CPack source generator types is defined for this
project (see the ``CMakeCache.txt`` file), it can be overridden using, for
example::
-D <Project>_CPACK_SOURCE_GENERATOR="TGZ;TBZ2"
(see CMake documentation to find out the types of supported CPack source
generators on your system).
NOTE: When configuring from an untarred source tree that has missing packages,
one must configure with::
-D <Project>_ASSERT_DEFINED_DEPENDENCIES=OFF
Otherwise, TriBITS will error out complaining about missing packages. (Note
that ``<Project>_ASSERT_DEFINED_DEPENDENCIES`` will default to ```OFF``` in
release mode, i.e. ``<Project>_ENABLE_DEVELOPMENT_MODE==OFF``.)
Dashboard submissions
=====================
All TriBITS projects have built-in support for submitting configure, build,
and test results to CDash using the custom ``dashboard`` target. This uses
the `tribits_ctest_driver()`_ function internally set up to work correctly
from an existing binary directory with a valid initial configure. The few of
the advantages of using the custom TriBITS-enabled ``dashboard`` target over
just using the standard ``ctest -D Experimental`` command are:
* The configure, build, and test results are broken down nicely
package-by-package on CDash.
* Additional notes files will be uploaded to the build on CDash.
For more details, see `tribits_ctest_driver()`_.
To use the ``dashboard`` target, first, configure as normal but add cache vars
for the the build and test parallel levels with::
-DCTEST_BUILD_FLAGS=-j4 -DCTEST_PARALLEL_LEVEL=4
(or with some other values ``-j<N>``). Then, invoke the (re)configure, build,
test and submit with::
$ make dashboard
This invokes a ``ctest -S`` script that calls the `tribits_ctest_driver()`_
function to do an experimental build for all of the enabled packages for which
you have enabled tests. (The packages that are implicitly enabled due to
package dependencies are not directly processed and no rows on CDash will be
show up for those packages.)
**NOTE:** This target generates a lot of output, so it is typically better to
pipe this to a file with::
$ make dashboard &> make.dashboard.out
and then watch that file in another terminal with::
$ tail -f make.dashboard.out
**NOTE:** To pass multiple arguments for ``CTEST_BUILD_FLAGS`` (like adding
``-k 99999999`` to tell ninja to continue even if there are build errors),
one must quote the entire argument string as::
"-DCTEST_BUILD_FLAGS=-j4 -k 99999999"
Setting options to change behavior of 'dashboard' target
--------------------------------------------------------
There are a number of options that you can set in the cache and/or in the
environment to control what this script does. Several options must be set in
the cache in the CMake configure of the project such as the CDash sites where
results are submitted to with the vars ``CTEST_DROP_METHOD``,
``CTEST_DROP_SITE``, ``CTEST_DROP_LOCATION``,
``TRIBITS_2ND_CTEST_DROP_LOCATION``, and ``TRIBITS_2ND_CTEST_DROP_SITE``.
Other options that control the behavior of the ``dashboard`` target must be
set in the env when calling ``make dashboard``. For the full set of options
that control the ``dashboard`` target, see `tribits_ctest_driver()`_. To see
the full list of options, and their default values, one can run with::
$ env CTEST_DEPENDENCY_HANDLING_UNIT_TESTING=TRUE \
make dashboard
This will print the options with their default values and then do a sort of
mock running of the CTest driver script and point out what it will do with the
given setup.
Any of the vars that are forwarded to the ``ctest -S`` invocation will be
shown in the STDOUT of the ``make dashboard`` invocation on the line::
Running: env [vars passed through env] <path>/ctest ... -S ...
Any variables passed through the ``env`` command listed there in ``[vars
passed through env ]`` can only be changed by setting cache variables in the
CMake project and can't be overridden in the env when invoking the
``dashboard`` target. For example, the variable ``CTEST_DO_SUBMIT`` is
forwarded to the ``ctest -S`` invocation and can't be overridden with::
$ env CTEST_DO_SUBMIT=OFF make dashboard
Instead, to change this value, one must reconfigure and then run as::
$ cmake CTEST_DO_SUBMIT=OFF .
$ make dashboard
But any variable that is not listed in ``[vars passed through env ]`` in the
printed out ``ctest -S`` command that are read in by `tribits_ctest_driver()`_
can be set in the env by calling::
$ env [other vars read by tribits_ctest_driver()] make dashboard
To know that these vars are picked up, grep the STDOUT from ``make dashboard``
for lines containing::
-- ENV_<var_name>=
That way, you will know the var was pick up and read correctly.
Common options and use cases for the 'dashboard' target
-------------------------------------------------------
What follows are suggestions on how to use the ``dashboard`` target for
different use cases.
One option that is useful to set is the build name on CDash at configure time
with::
-DCTEST_BUILD_NAME=MyBuild
After ``make dashboard`` finishes running, look for the build 'MyBuild' (or
whatever build name you used above) in the <Project> CDash dashboard (the
CDash URL is printed at the end of STDOUT). It is useful to set
``CTEST_BUILD_NAME`` to some unique name to make it easier to find your
results on the CDash dashboard. If one does not set ``CTEST_BUILD_NAME``,
then the name of the binary directory is used instead by default (which may
not be very descriptive if it called something like ``BUILD``).
If there is already a valid configure and build and one does not want to
reconfigure and rebuild or submit configure and build results then one can run
with::
$ env CTEST_DO_CONFIGURE=OFF CTEST_DO_BUILD=OFF \
make dashboard
This will only run the enabled pre-built tests and submit test results to
CDash. (But is usually good to reconfigure and rebuild and submit those
results to CDash as well in order to define more context for the test
results.)
The configure, builds, and submits are either done package-by-package or
all-at-once as controlled by the variable ``<Project>_CTEST_DO_ALL_AT_ONCE``.
This can be set in the CMake cache when configuring the project using::
-D<Project>_CTEST_DO_ALL_AT_ONCE=TRUE
Using the ``dashboard`` target, one can also run coverage and memory testing
and submit to CDash as described below. But to take full advantage of the
all-at-once mode and to have results displayed on CDash broken down
package-by-package, one must be submitting to a newer CDash version 3.0+.
For submitting line coverage results, configure with::
-D<Project>_ENABLE_COVERAGE_TESTING=ON
and the environment variable ``CTEST_DO_COVERAGE_TESTING=TRUE`` is
automatically set by the target ``dashboard`` so you don't have to set this
yourself. Then, when you run the ``dashboard`` target, it will automatically
submit coverage results to CDash as well.
Doing memory checking running the enabled tests with Valgrind requires that
you set ``CTEST_DO_MEMORY_TESTING=TRUE`` with the ``env`` command when running
the ``dashboard`` target as::
$ env CTEST_DO_MEMORY_TESTING=TRUE make dashboard
but also note that you may also need to set the valgrind command and options
with::
$ env CTEST_DO_MEMORY_TESTING=TRUE \
CTEST_MEMORYCHECK_COMMAND=<abs-path-to-valgrind> \
CTEST_MEMORYCHECK_COMMAND_OPTIONS="-q --trace-children=yes --tool=memcheck \
--leak-check=yes --workaround-gcc296-bugs=yes \
--num-callers=50 --suppressions=<abs-path-to-supp-file1> \
... --suppressions=<abs-path-to-supp-fileN>" \
make dashboard
The CMake cache variable ``<Project>_DASHBOARD_CTEST_ARGS`` can be set on the
cmake configure line in order to pass additional arguments to ``ctest -S``
when invoking the package-by-package CTest driver. For example::
-D<Project>_DASHBOARD_CTEST_ARGS="-VV" \
will set very verbose output with CTest that includes the STDOUT for every
test run. (The default args are ``-V`` which shows which tests are run but
not the test STDOUT.)
Changing the CDash sites for the 'dashboard' target
---------------------------------------------------
As described above in `Setting options to change behavior of 'dashboard'
target`_, one can change the location where configure, build, and test results
are submitted to one more two CDash sites. For well-structured TriBITS CMake
projects defining a flexible ``CTestConfig.cmake`` file, the location of the
main CDash site can be changed by configuring with::
-DCTEST_DROP_SITE="some-site.com" \
-DCTEST_DROP_LOCATION="/cdash/submit.php?project=<Project>" \
.. _TRIBITS_2ND_CTEST_DROP_SITE:
.. _TRIBITS_2ND_CTEST_DROP_LOCATION:
Also note that one can submit results to a second CDash site by configuring
with::
-DTRIBITS_2ND_CTEST_DROP_SITE="<second-site>" \
-DTRIBITS_2ND_CTEST_DROP_LOCATION="<second-location>" \
If left the same as ``CTEST_DROP_SITE`` or ``CTEST_DROP_LOCATION``, then
``TRIBITS_2ND_CTEST_DROP_SITE`` and ``TRIBITS_2ND_CTEST_DROP_LOCATION``,
respectively, can be left empty "" and the defaults will be used. For
example, to submit to an experimental CDash site on the same machine, one
would configure with::
-DTRIBITS_2ND_CTEST_DROP_LOCATION="/testing/cdash/submit.php?project=<Project>"
and ``CTEST_DROP_SITE`` would be used for ``TRIBITS_2ND_CTEST_DROP_SITE``
since ``TRIBITS_2ND_CTEST_DROP_SITE`` is empty. This is a common use case
when upgrading to a new CDash installation or testing new features for CDash
before impacting the existing CDash site. (However, the user must set at
least one of these variables to non-empty in order to trigger the second
submit.)
**NOTE:** If the project is already set up to submit to a second CDash site
and one wants to turn that off, one can configure with::
-DTRIBITS_2ND_CTEST_DROP_SITE=OFF \
-DTRIBITS_2ND_CTEST_DROP_LOCATION=OFF \
Configuring from scratch needed if 'dashboard' target aborts early
------------------------------------------------------------------
Finally, note that in package-by-package mode
(i.e. ``<Project>_CTEST_DO_ALL_AT_ONCE=FALSE``) that if one kills the ``make
dashboard`` target before it completes, then one must reconfigure from scratch
in order to get the build directory back into the same state before the
command was run. This is because the ``dashboard`` target in
package-by-package mode must first reconfigure the project with no enabled
packages before it does the package-by-package configure/build/test/submit
which enables each package one at a time. After the package-by-package
configure/build/test/submit cycles are complete, then the project is
reconfigured with the original set of package enables and returned to the
original configure state. Even with the all-at-once mode, if one kills the
``make dashboard`` command before the reconfigure completes, one may be left
with an invalid configuration of the project. In these cases, one may need to
configure from scratch to get back to the original state before calling ``make
dashboard``.
.. _TribitsExampleApp: https://github.com/TriBITSPub/TriBITS/tree/master/tribits/examples/TribitsExampleApp
.. _TriBITS TribitsExampleApp Tests: https://github.com/TriBITSPub/TriBITS/blob/master/test/core/ExamplesUnitTests/TribitsExampleApp_Tests.cmake
.. _xSDK Community Package Policies: https://doi.org/10.6084/m9.figshare.4495136
.. LocalWords: templated instantiation Makefiles CMake