cd2a0389ed
Fixes #56200 Add the ZephyrAppConfiguration package and update the corresponding documentation. This adds flexibility to the CMake build configuration by providing a workspace configuration package and an application package, which only applies to the current application. The workspace package stays the same as before, but the application package is new and lives, per default, inside the application folder. Signed-off-by: Nico Lüthi <nylnx@outlook.com>
449 lines
16 KiB
ReStructuredText
449 lines
16 KiB
ReStructuredText
.. _cmake_pkg:
|
|
|
|
Zephyr CMake Package
|
|
####################
|
|
|
|
The Zephyr `CMake package`_ is a convenient way to create a Zephyr-based application.
|
|
|
|
.. note::
|
|
The :ref:`zephyr-app-types` section introduces the application types
|
|
used in this page.
|
|
|
|
The Zephyr CMake package ensures that CMake can automatically select a Zephyr installation to use for building
|
|
the application, whether it is a :ref:`Zephyr repository application <zephyr-repo-app>`,
|
|
a :ref:`Zephyr workspace application <zephyr-workspace-app>`, or a
|
|
:ref:`Zephyr freestanding application <zephyr-freestanding-app>`.
|
|
|
|
When developing a Zephyr-based application, then a developer simply needs to write
|
|
``find_package(Zephyr)`` in the beginning of the application :file:`CMakeLists.txt` file.
|
|
|
|
To use the Zephyr CMake package it must first be exported to the `CMake user package registry`_.
|
|
This is means creating a reference to the current Zephyr installation inside the
|
|
CMake user package registry.
|
|
|
|
|
|
.. tabs::
|
|
|
|
.. group-tab:: Ubuntu
|
|
|
|
In Linux, the CMake user package registry is found in:
|
|
|
|
``~/.cmake/packages/Zephyr``
|
|
|
|
.. group-tab:: macOS
|
|
|
|
In macOS, the CMake user package registry is found in:
|
|
|
|
``~/.cmake/packages/Zephyr``
|
|
|
|
.. group-tab:: Windows
|
|
|
|
In Windows, the CMake user package registry is found in:
|
|
|
|
``HKEY_CURRENT_USER\Software\Kitware\CMake\Packages\Zephyr``
|
|
|
|
|
|
The Zephyr CMake package allows CMake to automatically find a Zephyr base.
|
|
One or more Zephyr installations must be exported.
|
|
Exporting multiple Zephyr installations may be useful when developing or testing
|
|
Zephyr freestanding applications, Zephyr workspace application with vendor forks, etc..
|
|
|
|
|
|
Zephyr CMake package export (west)
|
|
**********************************
|
|
|
|
When installing Zephyr using :ref:`west <get_the_code>` then it is recommended
|
|
to export Zephyr using ``west zephyr-export``.
|
|
|
|
.. _zephyr_cmake_package_export:
|
|
|
|
Zephyr CMake package export (without west)
|
|
******************************************
|
|
|
|
Zephyr CMake package is exported to the CMake user package registry using the following commands:
|
|
|
|
.. code-block:: bash
|
|
|
|
cmake -P <PATH-TO-ZEPHYR>/share/zephyr-package/cmake/zephyr_export.cmake
|
|
|
|
This will export the current Zephyr to the CMake user package registry.
|
|
|
|
To also export the Zephyr Unittest CMake package, run the following command in addition:
|
|
|
|
.. code-block:: bash
|
|
|
|
cmake -P <PATH-TO-ZEPHYR>/share/zephyrunittest-package/cmake/zephyr_export.cmake
|
|
|
|
.. _zephyr_cmake_package_zephyr_base:
|
|
|
|
Zephyr Base Environment Setting
|
|
*******************************
|
|
|
|
The Zephyr CMake package search functionality allows for explicitly specifying
|
|
a Zephyr base using an environment variable.
|
|
|
|
To do this, use the following ``find_package()`` syntax:
|
|
|
|
.. code-block:: cmake
|
|
|
|
find_package(Zephyr REQUIRED HINTS $ENV{ZEPHYR_BASE})
|
|
|
|
This syntax instructs CMake to first search for Zephyr using the Zephyr base environment setting
|
|
:envvar:`ZEPHYR_BASE` and then use the normal search paths.
|
|
|
|
.. _zephyr_cmake_search_order:
|
|
|
|
Zephyr CMake Package Search Order
|
|
*********************************
|
|
|
|
When Zephyr base environment setting is not used for searching, the Zephyr installation matching
|
|
the following criteria will be used:
|
|
|
|
* A Zephyr repository application will use the Zephyr in which it is located.
|
|
For example:
|
|
|
|
.. code-block:: none
|
|
|
|
<projects>/zephyr-workspace/zephyr
|
|
└── samples
|
|
└── hello_world
|
|
|
|
in this example, ``hello_world`` will use ``<projects>/zephyr-workspace/zephyr``.
|
|
|
|
|
|
* Zephyr workspace application will use the Zephyr that share the same workspace.
|
|
For example:
|
|
|
|
.. code-block:: none
|
|
|
|
<projects>/zephyr-workspace
|
|
├── zephyr
|
|
├── ...
|
|
└── my_applications
|
|
└── my_first_app
|
|
|
|
in this example, ``my_first_app`` will use ``<projects>/zephyr-workspace/zephyr`` as this Zephyr
|
|
is located in the same workspace as the Zephyr workspace application.
|
|
|
|
.. note::
|
|
The root of a Zephyr workspace is identical to ``west topdir`` if the workspace was
|
|
installed using ``west``
|
|
|
|
* Zephyr freestanding application will use the Zephyr registered in the CMake user package registry.
|
|
For example:
|
|
|
|
.. code-block:: none
|
|
|
|
<projects>/zephyr-workspace-1
|
|
└── zephyr (Not exported to CMake)
|
|
|
|
<projects>/zephyr-workspace-2
|
|
└── zephyr (Exported to CMake)
|
|
|
|
<home>/app
|
|
├── CMakeLists.txt
|
|
├── prj.conf
|
|
└── src
|
|
└── main.c
|
|
|
|
in this example, only ``<projects>/zephyr-workspace-2/zephyr`` is exported to the CMake package
|
|
registry and therefore this Zephyr will be used by the Zephyr freestanding application
|
|
``<home>/app``.
|
|
|
|
If user wants to test the application with ``<projects>/zephyr-workspace-1/zephyr``, this can be
|
|
done by using the Zephyr Base environment setting, meaning set
|
|
``ZEPHYR_BASE=<projects>/zephyr-workspace-1/zephyr``, before
|
|
running CMake.
|
|
|
|
.. note::
|
|
|
|
The Zephyr package selected on the first CMake invocation will be used for all subsequent
|
|
builds. To change the Zephyr package, for example to test the application using Zephyr base
|
|
environment setting, then it is necessary to do a pristine build first
|
|
(See :ref:`application_rebuild`).
|
|
|
|
Zephyr CMake Package Version
|
|
****************************
|
|
|
|
When writing an application then it is possible to specify a Zephyr version number ``x.y.z`` that
|
|
must be used in order to build the application.
|
|
|
|
Specifying a version is especially useful for a Zephyr freestanding application as it ensures the
|
|
application is built with a minimal Zephyr version.
|
|
|
|
It also helps CMake to select the correct Zephyr to use for building, when there are multiple
|
|
Zephyr installations in the system.
|
|
|
|
For example:
|
|
|
|
.. code-block:: cmake
|
|
|
|
find_package(Zephyr 2.2.0)
|
|
project(app)
|
|
|
|
will require ``app`` to be built with Zephyr 2.2.0 as minimum.
|
|
CMake will search all exported candidates to find a Zephyr installation which matches this version
|
|
criteria.
|
|
|
|
Thus it is possible to have multiple Zephyr installations and have CMake automatically select
|
|
between them based on the version number provided, see `CMake package version`_ for details.
|
|
|
|
For example:
|
|
|
|
.. code-block:: none
|
|
|
|
<projects>/zephyr-workspace-2.a
|
|
└── zephyr (Exported to CMake)
|
|
|
|
<projects>/zephyr-workspace-2.b
|
|
└── zephyr (Exported to CMake)
|
|
|
|
<home>/app
|
|
├── CMakeLists.txt
|
|
├── prj.conf
|
|
└── src
|
|
└── main.c
|
|
|
|
in this case, there are two released versions of Zephyr installed at their own workspaces.
|
|
Workspace 2.a and 2.b, corresponding to the Zephyr version.
|
|
|
|
To ensure ``app`` is built with minimum version ``2.a`` the following ``find_package``
|
|
syntax may be used:
|
|
|
|
.. code-block:: cmake
|
|
|
|
find_package(Zephyr 2.a)
|
|
project(app)
|
|
|
|
|
|
Note that both ``2.a`` and ``2.b`` fulfill this requirement.
|
|
|
|
CMake also supports the keyword ``EXACT``, to ensure an exact version is used, if that is required.
|
|
In this case, the application CMakeLists.txt could be written as:
|
|
|
|
.. code-block:: cmake
|
|
|
|
find_package(Zephyr 2.a EXACT)
|
|
project(app)
|
|
|
|
In case no Zephyr is found which satisfies the version required, as example, the application specifies
|
|
|
|
.. code-block:: cmake
|
|
|
|
find_package(Zephyr 2.z)
|
|
project(app)
|
|
|
|
then an error similar to below will be printed::
|
|
|
|
Could not find a configuration file for package "Zephyr" that is compatible
|
|
with requested version "2.z".
|
|
|
|
The following configuration files were considered but not accepted:
|
|
|
|
<projects>/zephyr-workspace-2.a/zephyr/share/zephyr-package/cmake/ZephyrConfig.cmake, version: 2.a.0
|
|
<projects>/zephyr-workspace-2.b/zephyr/share/zephyr-package/cmake/ZephyrConfig.cmake, version: 2.b.0
|
|
|
|
|
|
.. note:: It can also be beneficial to specify a version number for Zephyr repository applications
|
|
and Zephyr workspace applications. Specifying a version in those cases ensures the
|
|
application will only build if the Zephyr repository or workspace is matching.
|
|
This can be useful to avoid accidental builds when only part of a workspace has been
|
|
updated.
|
|
|
|
|
|
Multiple Zephyr Installations (Zephyr workspace)
|
|
************************************************
|
|
|
|
Testing out a new Zephyr version, while at the same time keeping the existing Zephyr in the
|
|
workspace untouched is sometimes beneficial.
|
|
|
|
Or having both an upstream Zephyr, Vendor specific, and a custom Zephyr in same workspace.
|
|
|
|
For example:
|
|
|
|
.. code-block:: none
|
|
|
|
<projects>/zephyr-workspace
|
|
├── zephyr
|
|
├── zephyr-vendor
|
|
├── zephyr-custom
|
|
├── ...
|
|
└── my_applications
|
|
└── my_first_app
|
|
|
|
|
|
in this setup, ``find_package(Zephyr)`` has the following order of precedence for selecting
|
|
which Zephyr to use:
|
|
|
|
* Project name: ``zephyr``
|
|
* First project, when Zephyr projects are ordered lexicographical, in this case.
|
|
|
|
* ``zephyr-custom``
|
|
* ``zephyr-vendor``
|
|
|
|
This means that ``my_first_app`` will use ``<projects>/zephyr-workspace/zephyr``.
|
|
|
|
It is possible to specify a Zephyr preference list in the application.
|
|
|
|
A Zephyr preference list can be specified as:
|
|
|
|
.. code-block:: cmake
|
|
|
|
set(ZEPHYR_PREFER "zephyr-custom" "zephyr-vendor")
|
|
find_package(Zephyr)
|
|
|
|
project(my_first_app)
|
|
|
|
|
|
the ``ZEPHYR_PREFER`` is a list, allowing for multiple Zephyrs.
|
|
If a Zephyr is specified in the list, but not found in the system, it is simply ignored and
|
|
``find_package(Zephyr)`` will continue to the next candidate.
|
|
|
|
|
|
This allows for temporary creation of a new Zephyr release to be tested, without touching current
|
|
Zephyr. When testing is done, the ``zephyr-test`` folder can simply be removed.
|
|
Such a CMakeLists.txt could look as:
|
|
|
|
.. code-block:: cmake
|
|
|
|
set(ZEPHYR_PREFER "zephyr-test")
|
|
find_package(Zephyr)
|
|
|
|
project(my_first_app)
|
|
|
|
.. _cmake_build_config_package:
|
|
|
|
Zephyr Build Configuration CMake packages
|
|
*****************************************
|
|
|
|
There are two Zephyr Build configuration packages which provide control over the build
|
|
settings in Zephyr in a more generic way. These packages are:
|
|
|
|
* **ZephyrBuildConfiguration**: Applies to all Zephyr applications in the workspace
|
|
* **ZephyrAppConfiguration**: Applies only to the application you are currently building
|
|
|
|
They are similar to the per-user :file:`.zephyrrc` file that can be used to set :ref:`env_vars`,
|
|
but they set CMake variables instead. They also allow you to automatically share the build
|
|
configuration among all users through the project repository. They also allow more advanced use
|
|
cases, such as loading of additional CMake boilerplate code.
|
|
|
|
The Zephyr Build Configuration CMake packages will be loaded in the Zephyr boilerplate code after
|
|
initial properties and ``ZEPHYR_BASE`` has been defined, but before CMake code execution. The
|
|
ZephyrBuildConfiguration is included first and ZephyrAppConfiguration afterwards. That means the
|
|
application-specific package could override the workspace settings, if needed.
|
|
This allows the Zephyr Build Configuration CMake packages to setup or extend properties such as:
|
|
``DTS_ROOT``, ``BOARD_ROOT``, ``TOOLCHAIN_ROOT`` / other toolchain setup, fixed overlays, and any
|
|
other property that can be controlled. It also allows inclusion of additional boilerplate code.
|
|
|
|
To provide a ZephyrBuildConfiguration or ZephyrAppConfiguration, create
|
|
:file:`ZephyrBuildConfig.cmake` and/or :file:`ZephyrAppConfig.cmake` respectively and place them
|
|
in the appropriate location. The CMake ``find_package`` mechanism will search for these files with
|
|
the steps below. Other default CMake package search paths and hints are disabled and there is no
|
|
version checking implemented for these packages. This also means that these packages cannot be
|
|
installed in the CMake package registry. The search steps are:
|
|
|
|
1. If ``ZephyrBuildConfiguration_ROOT``, or ``ZephyrAppConfiguration_ROOT`` respectively, is set,
|
|
search within this prefix path. If a matching file is found, execute this file. If no matching
|
|
file is found, go to step 2.
|
|
2. Search within ``${ZEPHYR_BASE}/../*``, or ``${APPLICATION_SOURCE_DIR}`` respectively. If a
|
|
matching file is found, execute this file. If no matching file is found, abort the search.
|
|
|
|
It is recommended to place the files in the default paths from step 2, but with the
|
|
``<PackageName>_ROOT`` variables you have the flexibility to place them anywhere. This is
|
|
especially necessary for freestanding applications, for which the default path to
|
|
ZephyrBuildConfiguration usually does not work. In this case the ``<PackageName>_ROOT`` variables
|
|
can be set on the CMake command line, **before** ``find_package(Zephyr ...)``, as environment
|
|
variable or from a CMake cache initialization file with the ``-C`` command line option.
|
|
|
|
.. note:: The ``<PackageName>_ROOT`` variables, as well as the default paths, are just the prefixes
|
|
to the search path. These prefixes get combined with additional path suffixes, which together
|
|
form the actual search path. Any combination that honors the
|
|
`CMake package search procedure`_ is valid and will work.
|
|
|
|
If you want to completely disable the search for these packages, you can use the special CMake
|
|
``CMAKE_DISABLE_FIND_PACKAGE_<PackageName>`` variable for that. Just set
|
|
``CMAKE_DISABLE_FIND_PACKAGE_ZephyrBuildConfiguration`` or
|
|
``CMAKE_DISABLE_FIND_PACKAGE_ZephyrAppConfiguration`` to ``TRUE`` to disable the package.
|
|
|
|
An example folder structure could look like this:
|
|
|
|
.. code-block:: none
|
|
|
|
<projects>/zephyr-workspace
|
|
├── zephyr
|
|
├── ...
|
|
├── manifest repo (can be named anything)
|
|
│ └── cmake/ZephyrBuildConfig.cmake
|
|
├── ...
|
|
└── zephyr application
|
|
└── share/zephyrapp-package/cmake/ZephyrAppConfig.cmake
|
|
|
|
A sample :file:`ZephyrBuildConfig.cmake` can be seen below.
|
|
|
|
.. code-block:: cmake
|
|
|
|
# ZephyrBuildConfig.cmake sample code
|
|
|
|
# To ensure final path is absolute and does not contain ../.. in variable.
|
|
get_filename_component(APPLICATION_PROJECT_DIR
|
|
${CMAKE_CURRENT_LIST_DIR}/../../..
|
|
ABSOLUTE
|
|
)
|
|
|
|
# Add this project to list of board roots
|
|
list(APPEND BOARD_ROOT ${APPLICATION_PROJECT_DIR})
|
|
|
|
# Default to GNU Arm Embedded toolchain if no toolchain is set
|
|
if(NOT ENV{ZEPHYR_TOOLCHAIN_VARIANT})
|
|
set(ZEPHYR_TOOLCHAIN_VARIANT gnuarmemb)
|
|
find_program(GNU_ARM_GCC arm-none-eabi-gcc)
|
|
if(NOT ${GNU_ARM_GCC} STREQUAL GNU_ARM_GCC-NOTFOUND)
|
|
# The toolchain root is located above the path to the compiler.
|
|
get_filename_component(GNUARMEMB_TOOLCHAIN_PATH ${GNU_ARM_GCC}/../.. ABSOLUTE)
|
|
endif()
|
|
endif()
|
|
|
|
Zephyr CMake package source code
|
|
********************************
|
|
|
|
The Zephyr CMake package source code in
|
|
:zephyr_file:`share/zephyr-package/cmake` and
|
|
:zephyr_file:`share/zephyrunittest-package/cmake` contains the CMake config
|
|
package which is used by the CMake ``find_package`` function.
|
|
|
|
It also contains code for exporting Zephyr as a CMake config package.
|
|
|
|
The following is an overview of the files in these directories:
|
|
|
|
:file:`ZephyrConfigVersion.cmake`
|
|
The Zephyr package version file. This file is called by CMake to determine
|
|
if this installation fulfils the requirements specified by user when calling
|
|
``find_package(Zephyr ...)``. It is also responsible for detection of Zephyr
|
|
repository or workspace only installations.
|
|
|
|
:file:`ZephyrUnittestConfigVersion.cmake`
|
|
Same responsibility as ``ZephyrConfigVersion.cmake``, but for unit tests.
|
|
Includes ``ZephyrConfigVersion.cmake``.
|
|
|
|
:file:`ZephyrConfig.cmake`
|
|
The Zephyr package file. This file is called by CMake to for the package
|
|
meeting which fulfils the requirements specified by user when calling
|
|
``find_package(Zephyr ...)``. This file is responsible for sourcing of
|
|
boilerplate code.
|
|
|
|
:file:`ZephyrUnittestConfig.cmake`
|
|
Same responsibility as ``ZephyrConfig.cmake``, but for unit tests.
|
|
Includes ``ZephyrConfig.cmake``.
|
|
|
|
:file:`zephyr_package_search.cmake`
|
|
Common file used for detection of Zephyr repository and workspace candidates.
|
|
Used by ``ZephyrConfigVersion.cmake`` and ``ZephyrConfig.cmake`` for common code.
|
|
|
|
:file:`zephyr_export.cmake`
|
|
See :ref:`zephyr_cmake_package_export`.
|
|
|
|
.. _CMake package: https://cmake.org/cmake/help/latest/manual/cmake-packages.7.html
|
|
.. _CMake user package registry: https://cmake.org/cmake/help/latest/manual/cmake-packages.7.html#user-package-registry
|
|
.. _CMake package version: https://cmake.org/cmake/help/latest/command/find_package.html#version-selection
|
|
.. _CMake package search procedure: https://cmake.org/cmake/help/latest/command/find_package.html#search-procedure
|