Difference between revisions of "Cmake"

From Wiki at Neela Nurseries
Jump to navigation Jump to search
m (→‎^ Cmake Functions: - cmake functions ExternalProject_Add() and find_package())
 
(55 intermediate revisions by the same user not shown)
Line 2: Line 2:
 
[[Zephyr_building_blocks|Zephyr RTOS building blocks]]   ::   [[Device_tree|Device Tree Source]]   ::   [[Kconfig|Kconfig]]   ::   [[cmake|cmake]]   ::   [[west_manifest_files|`west` manifest files]]
 
[[Zephyr_building_blocks|Zephyr RTOS building blocks]]   ::   [[Device_tree|Device Tree Source]]   ::   [[Kconfig|Kconfig]]   ::   [[cmake|cmake]]   ::   [[west_manifest_files|`west` manifest files]]
 
</center>
 
</center>
 +
 +
Keywords and noteworthy:  cmake key concepts : commands : objects : targets : properties : variables
  
 
<!-- comment -->
 
<!-- comment -->
  
 +
<span style="font-size: 22px">Overview</span>
 +
 +
This page a starting point for cmake notes and links to references about [https://cmake.org/cmake/help/book/mastering-cmake/index.html cmake].  There is a lot of functionality provided by cmake.  Further, it is not obvious from looking at larger projects involving cmake, such as Zephyr RTOS, how for example cmake assigns and tracks which compiler to use for source files written in a particular language.
 +
 +
Some highlights from Cmake's online "Getting started" book:
 +
 +
(1)  Inception of cmake, language, and important elements of cmake
 +
 +
* Cmake as a project was started in 1999.
 +
 +
* Cmake implements its own language.
 +
 +
* Cmake language is composed of comments, commands and variables.
 +
 +
(2) Cmake variables come in three types
 +
 +
*  ${variable}
 +
 +
*  $ENV{variable}
 +
 +
*  $CACHE{variable}
 +
 +
<!-- spacer -->
 +
 +
To sort elsewhere:
 +
 +
(1)  [[#nn_anchor_zephyr_app_common_code|Zephyr app common code]]
 +
 +
(2)  [https://cmake.org/cmake/help/latest/variable/CMAKE_MODULE_PATH.html CMAKE_MODULE_PATH] used often in project top level cmake script.
 +
 +
<!-- spacer -->
 +
 +
<!-- comentario -->
 +
 +
== [[#top|^]] Configuration Stage and Build Stage of Cmake ==
 +
 +
*  https://docs.zephyrproject.org/3.1.0/build/cmake/index.html
 +
 +
<!-- comentario -->
 +
 +
== [[#top|^]] Important Reading ==
 +
 +
Following sections link to and add notes about some higher level areas of cmake.
 +
 +
=== [[#top|^]] cmake Policies ===
 +
 +
https://cmake.org/cmake/help/v2.8.12/cmake.html#section_Policies
 +
 +
<!-- odne komentar -->
 +
 +
=== [[#top|^]] cmake Commands ===
 +
 +
Cmake provides commands, most of them to configure a build system or "recipe".  The following link goes to cmake documentation on cmake's standard commands.
 +
 +
*  https://cmake.org/cmake/help/v2.8.12/cmake.html#section_Commands
 +
 +
Cmake also supports the writing of customized commands.  Here is a link to a discussion about customizing a new command to report a project's set of include paths at configuration time.  Note this is "configuration time" and not yet "build time", the latter of which is when more information is known to the build system.
 +
 +
*  https://discourse.cmake.org/t/how-to-find-all-the-include-folders-for-a-target-for-a-static-analyzer/3233/3
 +
 +
A forum post discussing how to make cmake generate info to go into a file at build time:
 +
 +
*  https://stackoverflow.com/questions/13419255/how-to-get-cmake-to-create-timestamp-file-after-an-actual-build-link-do-nothin
 +
 +
<!-- odne komentar -->
 +
 +
== [[#top|^]] Project Organization and Cmake ==
 +
 +
In the [https://cmake.org/cmake/help/book/mastering-cmake/ "Mastering Cmake"] online book Cmake team speaks to Cmake's strategy to create a dedicated build directory at the root of a given project.  This dedicated build directory keeps all build artifacts separate from project sources, and allows for a simple and safe way to deleting build generated files without impacting project sources.  This strategy can be achieved with other build systems like GNU `make`, but cmake operates on projects this way by default.
 +
 +
There's more to project organization than choosing an area for build output (artifacts).  In particular when a final compiled image involves two or more applications, a project's organization becomes more complex.  This so whether projects utilize cmake or another build system.
 +
 +
Some links to project organization forum posts follow here:
 +
 +
*  https://softwareengineering.stackexchange.com/questions/289348/directory-organization-of-a-cmake-c-repository-containing-several-projects
 +
 +
. . . though this post is ten years old, may yet hold useful information.
 +
 +
Questions outstanding (from this section):
 +
 +
(Q1)  Why in cmake projects can there end up being so many distinctly named "projects"?
 +
 +
== [[#top|^]] To Create Libraries In A Build ==
 +
 +
Cmake supports the specification and building of library object files as part of a project.  Further, cmake supports the defining and declaring of where and how these libraries will be linked to other targets in the given project.  To create a library, a typical pattern of cmake function calls includes:
 +
 +
<pre>
 +
(1)  add_library(<name_of_library> OBJECT)
 +
 +
(2)  target_include_directories(<name_of_library>
 +
          PUBLIC include_path
 +
          PRIVATE include_path_private
 +
      )
 +
 +
(3)  target_sources(<name_of_library PRIVATE
 +
          source_file_1.c
 +
          source_file_2.c
 +
          . . .
 +
      )
 +
 +
(4)  target_link_libraries(<name_of_library> PUBLIC|INTERFACE|PRIVATE
 +
</pre>
 +
 +
Online cmake documentation on the topic of libraries:
 +
 +
*  https://cmake.org/cmake/help/latest/manual/cmake-buildsystem.7.html#target-usage-requirements
 +
 +
*  https://cmake.org/cmake/help/latest/prop_tgt/INTERFACE_LINK_LIBRARIES.html#prop_tgt:INTERFACE_LINK_LIBRARIES
 +
 +
*  https://cmake.org/cmake/help/latest/command/target_compile_definitions.html
  
2021-08-06
+
== DESPUES DE ESTO REVISTAR TODO - 2025 Q3 ==
  
Starting a page for <code>cmake</code> notes here.  Links to cmake site:
+
<!-- odne komentar -->
<ul>
 
*  https://cmake.org/cmake/help/latest/
 
  
*  https://cmake.org/overview/
+
== [[#top|^]] Scope of Variables in Cmake ==
*  https://cmake.org/examples/  . . . very limited, need other resources
 
</ul>
 
Longer guides:
 
<ul>
 
*  https://cmake.org/cmake/help/v3.21/
 
</ul>
 
  
 +
Matthew Gaudet cmake variable scoping:
  
A Nordic Semi document on `cmake` and its role in the build process of Zephyr based applications:
+
*  https://www.mgaudet.ca/technical/2017/8/31/some-notes-on-cmake-variables-and-scopes
*  https://developer.nordicsemi.com/nRF_Connect_SDK/doc/latest/zephyr/guides/build/index.html
 
  
 +
<!-- odne komentar -->
  
2021-08-17 - search for cmake / Zephyr means to add sources to a Zephyr based project:
+
== [[#top|^]] Practical Things to Know and Do ==
<ul>
 
*  https://github.com/zephyrproject-rtos/zephyr/blob/main/cmake/extensions.cmake
 
*  https://cmake.org/cmake/help/latest/manual/cmake-buildsystem.7.html
 
</ul>
 
  
 +
<i>stub section intro</i>
 
<!-- comentario -->
 
<!-- comentario -->
 +
=== [[#top|^]] To Debug Cmake Scripts ===
 +
 +
A gitlab dot io cmake resource:
 +
 +
*  https://cliutils.gitlab.io/modern-cmake/chapters/features/debug.html
  
== [[#top|^]] To print variables in cmake ==
 
 
Note that the following post has a cool answer which involves a cmake macro to print all variables known to a given cmake invocation or use, and also a more simple answer to print one specific variable:
 
Note that the following post has a cool answer which involves a cmake macro to print all variables known to a given cmake invocation or use, and also a more simple answer to print one specific variable:
 
*  https://stackoverflow.com/questions/31343813/displaying-cmake-variables
 
*  https://stackoverflow.com/questions/31343813/displaying-cmake-variables
Line 135: Line 241:
 
</pre>
 
</pre>
  
== [[#top|^]] Source and Library Path Setting ==
+
=== [[#top|^]] Source and Library Path Setting ===
 +
 
 +
Cmake has at least as of 2020 a notion of "translation map" which relates to the ways in which cmake determines file paths.  One introduction to this concept arises at the forum post:
 +
 
 +
*  https://discourse.cmake.org/t/when-does-cmake-current-binary-dir-resolve-symlinks/809
 +
*  https://gitlab.kitware.com/cmake/cmake/-/issues/16228
 +
 
 +
There is comment in cmake documentation that use of symbolic links can lead to subtle errors.  Need to locate this reference.
 +
 
 +
2025-04-16
 +
 
 +
In Zephyr 3.7.0 CMAKE_CURRENT_BINARY_DIR is set in file `zephyr/cmake/package_helper.cmake`:
 +
 
 +
105:set(CMAKE_CURRENT_BINARY_DIR ${CMAKE_BINARY_DIR})`.
 +
 
 +
The variable referenced here is also set in `package_helper.cmake`:
 +
 
 +
./cmake/package_helper.cmake:75:    set(CMAKE_BINARY_DIR ${argB})
 +
./cmake/package_helper.cmake:79:    set(CMAKE_BINARY_DIR ${CMAKE_ARGV${argB_value_index}})
 +
./cmake/package_helper.cmake:104:cmake_path(ABSOLUTE_PATH CMAKE_BINARY_DIR NORMALIZE)
  
 +
- - - - - - - - - -
  
 +
Earlier references:
  
 
*  https://cmake.org/cmake/help/latest/command/link_directories.html
 
*  https://cmake.org/cmake/help/latest/command/link_directories.html
Line 145: Line 272:
 
*  https://cmake.org/cmake/help/latest/variable/CMAKE_BINARY_DIR.html?highlight=cmake_binary_dir
 
*  https://cmake.org/cmake/help/latest/variable/CMAKE_BINARY_DIR.html?highlight=cmake_binary_dir
  
 +
<!-- comment -->
  
<!-- comment -->
+
=== [[#top|^]] cmake Lists ===
 +
 
 +
Look up `append_if_path_exists()` . . .
 +
 
 +
<!-- odne komentar -->
  
== [[#top|^]] Cmake Functions ==
+
=== [[#top|^]] cmake Functions ===
  
 
An interesting pattern search in Zephyr 3.2.0 source tree and associated third party libraries is `$ grep -nr ExternalProject_Add ./*`, which reveals which projects and locations in Zephyr's source tree this cmake function is called.  This function is needed when building multicore Zephyr apps.
 
An interesting pattern search in Zephyr 3.2.0 source tree and associated third party libraries is `$ grep -nr ExternalProject_Add ./*`, which reveals which projects and locations in Zephyr's source tree this cmake function is called.  This function is needed when building multicore Zephyr apps.
Line 154: Line 286:
 
There is a detailed document on `ExternalProject_Add()` at the Kitware public github repository:
 
There is a detailed document on `ExternalProject_Add()` at the Kitware public github repository:
  
=== [[#top|^]] ExternalProject ===
+
<ul><!-- SUB-SECTION INDENT BEGIN -->
 +
==== [[#top|^]] ExternalProject_Add ====
  
 
<!-- comment -->
 
<!-- comment -->
  
 
Cmake function `ExternalProject_Add()` is used in multi-core Zephyr sample projects.  Some refs to this function include:
 
Cmake function `ExternalProject_Add()` is used in multi-core Zephyr sample projects.  Some refs to this function include:
 +
 
*  https://gitlab.kitware.com/cmake/cmake/blob/master/Modules/ExternalProject.cmake
 
*  https://gitlab.kitware.com/cmake/cmake/blob/master/Modules/ExternalProject.cmake
  
=== [[#top|^]] find_package ===
+
*  https://cmake.org/cmake/help/latest/module/ExternalProject.html?highlight=externalproject_add#command:externalproject_add
 +
 
 +
*  https://github.com/pyke369/sffmpeg/blob/master/CMakeLists.txt n  . . . example of `INCLUDE_DIRECTORIES(BEFORE ${CMAKE_BINARY_DIR}/include)`
 +
 
 +
==== [[#top|^]] find_package ====
  
 
<!-- comment -->
 
<!-- comment -->
Line 170: Line 308:
  
 
<!-- comment -->
 
<!-- comment -->
 +
 +
==== [[#top|^]] add_subdirectory_ifdef() ====
 +
 +
<span id="nn_anchor_zephyr_app_common_code"></span>
 +
Really good blog post at blog.golioth.io regarding how to configure a "super project" containing multiple Zephyr based apps with shared, select-able at build time common code:
 +
 +
<ul>
 +
*  https://blog.golioth.io/cmake-and-kconfig-tricks-for-common-code-in-zephyr-apps/
 +
</ul>
 +
<!-- odne komentar -->
 +
==== [[#top|^]] function() ====
 +
 +
- stub -
 +
<!-- odne komentar -->
 +
==== [[#top|^]] execute_process() ====
 +
 +
- stub -
 +
<!-- odne komentar -->
 +
==== [[#top|^]] add_custom_command() ====
 +
 +
- stub -
 +
<!-- odne komentar -->
  
 
== [[#top|^]] Install latest version of cmake ==
 
== [[#top|^]] Install latest version of cmake ==
Line 295: Line 455:
 
-->
 
-->
  
== [[#top|^]] cmake Package Configuration Files ==
+
== [[#top|^]] cmake Package Configuration Files - What Are They? ==
 +
 
 +
QUESTION (2022-12-29):  what are cmake package configuration files?
  
 
When attempting to switch Zephyr sdk of original latest 2021 kionix-driver-demo, from Nordic ncs SDK to Zephyr RTOS Project itself, and a more recent release of Zephyr RTOS, get following error early early in build process:
 
When attempting to switch Zephyr sdk of original latest 2021 kionix-driver-demo, from Nordic ncs SDK to Zephyr RTOS Project itself, and a more recent release of Zephyr RTOS, get following error early early in build process:
Line 374: Line 536:
 
-- Configuring incomplete, errors occurred!
 
-- Configuring incomplete, errors occurred!
 
FATAL ERROR: command exited with status 1: /usr/bin/cmake -DWEST_PYTHON=/usr/bin/python3 -B/home/ted/projects/zephyr-based/out-of-tree/kionix-driver-demo/build -GNinja -DBOARD=lpcxpresso55s69_cpu0 -S/home/ted/projects/zephyr-based/out-of-tree/kionix-driver-demo
 
FATAL ERROR: command exited with status 1: /usr/bin/cmake -DWEST_PYTHON=/usr/bin/python3 -B/home/ted/projects/zephyr-based/out-of-tree/kionix-driver-demo/build -GNinja -DBOARD=lpcxpresso55s69_cpu0 -S/home/ted/projects/zephyr-based/out-of-tree/kionix-driver-demo
</pre>
+
</pre>comentario
  
  
Line 409: Line 571:
 
</pre>
 
</pre>
  
 +
<!-- comentario -->
  
<!-- comentario -->
+
== [[#top|^]] Cmake and Zephyr Builds ==
 +
 
 +
<i>* * * Stub section * * *</i>
 +
 
 +
The previous section of this nn page segways into questions about how exactly cmake functions as a build system for Zephyr RTOS based applications.  Typically, a call to the Python based script `west` starts a build.  Zephyr RTOS project adds functionality to `west`, including the `west` command `build`.  (That's not available when invoking `west` outside of a Zephyr project workspace.
 +
 
 +
This section intended as a landing point for notes and findings about the steps cmake commneces during a Zephyr project build.
 +
 
 +
Some files, some build artifacts which may be of interest in this search:
 +
 
 +
(1) $workspace/build/cpu0/zephyr/linker_zephyr_pre0.cmd.dep
 +
 
 +
== ----- ----- ----- ----- ----- ----- ----- ----- ----- ----- ----- ----- ----- ----- ----- ----- ==
 +
 
 +
<!-- odne komentar -->
 +
 
 +
== SORT SECTION BEGIN ==
 +
 
 +
<!-- odne komentar -->
  
 
== [[#top|^]] Zephyr Threads Tutorial by Maksim Drachov ==
 
== [[#top|^]] Zephyr Threads Tutorial by Maksim Drachov ==
Line 450: Line 631:
 
</pre>
 
</pre>
  
 +
<!-- comentario -->
 +
 +
== [[#top|^]] Cmake links to sort ==
 +
 +
2021-08-06
 +
 +
Starting a page for <code>cmake</code> notes here.  Links to cmake site:
 +
<ul>
 +
*  https://cmake.org/cmake/help/latest/
 +
 +
*  https://cmake.org/overview/
 +
*  https://cmake.org/examples/  . . . very limited, need other resources
 +
</ul>
 +
Longer guides:
 +
<ul>
 +
*  https://cmake.org/cmake/help/v3.21/
 +
 +
*  https://cmake.org/getting-started/
 +
</ul>
 +
 +
 +
A Nordic Semi document on `cmake` and its role in the build process of Zephyr based applications:
 +
*  https://developer.nordicsemi.com/nRF_Connect_SDK/doc/latest/zephyr/guides/build/index.html
 +
 +
 +
2021-08-17 - search for cmake / Zephyr means to add sources to a Zephyr based project:
 +
<ul>
 +
*  https://github.com/zephyrproject-rtos/zephyr/blob/main/cmake/extensions.cmake
 +
*  https://cmake.org/cmake/help/latest/manual/cmake-buildsystem.7.html
 +
</ul>
 +
== [[#top|^]] Topics To Follow Up ==
 +
 +
Who accepts the `--extra-args=CONFIG_*` option plus value?  From `zephyr/scripts/pylib/twister/twisterlib/testinstance.py`:
 +
 +
<pre>
 +
225    def create_overlay(self, platform, enable_asan=False, enable_ubsan=False, enable_coverage=False, coverage_platform=[]):
 +
226        # Create this in a "twister/" subdirectory otherwise this
 +
227        # will pass this overlay to kconfig.py *twice* and kconfig.cmake
 +
228        # will silently give that second time precedence over any
 +
229        # --extra-args=CONFIG_*
 +
230        subdir = os.path.join(self.build_dir, "twister")
 +
231
 +
232        content = ""
 +
</pre>
 +
 +
 +
In Zephyr 3.2.0, `zephyr/cmake/modules/configuration_files.cmake` looks promising in terms of revealing where and how Kconfig processing is called during build of a Zephyr based app.  An excerpt from this file:
 +
 +
<pre>
 +
  1 # SPDX-License-Identifier: Apache-2.0
 +
  2 #
 +
  3 # Copyright (c) 2021, Nordic Semiconductor ASA
 +
  4
 +
  5 # Zephyr build system configuration files.
 +
  6 #
 +
  7 # Locate the Kconfig and DT config files that are to be used.
 +
  8 # Also, locate the appropriate application config directory.
 +
  9 #
 +
10 # Outcome:
 +
11 # The following variables will be defined when this CMake module completes:
 +
12 #
 +
13 # - CONF_FILE:              List of Kconfig fragments
 +
14 # - DTC_OVERLAY_FILE:      List of devicetree overlay files
 +
15 # - APPLICATION_CONFIG_DIR: Root folder for application configuration
 +
16 #
 +
17 # If any of the above variables are already set when this CMake module is
 +
18 # loaded, then no changes to the variable will happen.
 +
19 #
 +
20 # Variables set by this module and not mentioned above are considered internal
 +
21 # use only and may be removed, renamed, or re-purposed without prior notice.
 +
</pre>
 +
 +
The cmake variable CONF_FILE can be set by:
 +
 +
  set(CONF_FILE prj-fragment-1.conf prj-fragment-2.conf other-configuration.conf)
 +
 +
<!-- comentario -->
 +
 +
== ----- ----- ----- ----- ----- ----- ----- ----- ----- ----- ----- ----- ----- ----- ----- ----- ==
 +
 +
<!-- odne komentar -->
 +
 +
== [[#top|^]] A Cmake Exploration Effort ==
 +
 +
Overview:  cmake tests to see where certain libraries are getting pulled across one application to a second app.
 +
 +
In a multi-application cmake project, cmake files typically populate multiple directories.  The build process commences with one cmake file being processed first.  To better understand what happens next, cmake's `message()` function or command acts like a C printf() statement and can reveal build state at various stages of this process . . .
 +
 +
Zephyr RTOS adds cmake scripts as part of its build process.  When we build one or more Zephyr based apps we see various messages from cmake during these builds.  For multi-app builds we see near the start of each build:
 +
 +
"Loading Zephyr default modules . . ."
 +
 +
This message comes from:
 +
 +
<pre>
 +
$ grep -nr 'Loading Zephyr default' --exclude-dir=build --exclude-dir=docker_cache ./*
 +
./share/zephyr-package/cmake/ZephyrConfig.cmake:65:      zephyr_package_message(NOTICE "Loading Zephyr default modules (${location}).")
 +
$
 +
</pre>
 +
 +
In builds of Zephyr 3.7.0 based apps, first build system message to standard out appears to be "making build dir " . . .
 +
 +
<pre>
 +
ted@localhost1:~/projects-sandbox/workspace-for-nexus/zephyr/scripts/west_commands$ !grep
 +
grep -nr 'making build dir' ./*
 +
./build.py:601:        _banner('making build dir {} pristine'.format(self.build_dir))
 +
</pre>
 +
 +
The definition of `_banner()` is at:
 +
 +
<pre>
 +
./zephyr/scripts/west_commands/run_common.py:55:def _banner(msg):
 +
</pre>
 +
 +
=== [[#top|^]] cmake command include_directories() ===
 +
 +
Some related commands from cmake:  include_directories(), target_include_directories(), and from Zephyr's additions
 +
 +
*  https://cmake.org/cmake/help/latest/command/include_directories.html
 +
 +
Note that ``target_include_directories()`` expects at minimum a couple of arguments before the include paths themselves.  One example of its practical use is
 +
 +
<pre>
 +
target_include_directories(app PRIVATE
 +
    include
 +
)
 +
</pre>
 +
 +
Visit cmake documentation to learn which other qualifying arguments may be given in place of ``app`` and ``PRIVATE``, which are specific values.
  
 +
<!-- odne komentar -->
  
 
<center>
 
<center>

Latest revision as of 06:50, 29 September 2025

Zephyr RTOS building blocks   ::   Device Tree Source   ::   Kconfig   ::   cmake   ::   `west` manifest files

Keywords and noteworthy: cmake key concepts : commands : objects : targets : properties : variables


Overview

This page a starting point for cmake notes and links to references about cmake. There is a lot of functionality provided by cmake. Further, it is not obvious from looking at larger projects involving cmake, such as Zephyr RTOS, how for example cmake assigns and tracks which compiler to use for source files written in a particular language.

Some highlights from Cmake's online "Getting started" book:

(1) Inception of cmake, language, and important elements of cmake

  • Cmake as a project was started in 1999.
  • Cmake implements its own language.
  • Cmake language is composed of comments, commands and variables.

(2) Cmake variables come in three types

  • ${variable}
  • $ENV{variable}
  • $CACHE{variable}


To sort elsewhere:

(1) Zephyr app common code

(2) CMAKE_MODULE_PATH used often in project top level cmake script.


^ Configuration Stage and Build Stage of Cmake


^ Important Reading

Following sections link to and add notes about some higher level areas of cmake.

^ cmake Policies

https://cmake.org/cmake/help/v2.8.12/cmake.html#section_Policies


^ cmake Commands

Cmake provides commands, most of them to configure a build system or "recipe". The following link goes to cmake documentation on cmake's standard commands.

Cmake also supports the writing of customized commands. Here is a link to a discussion about customizing a new command to report a project's set of include paths at configuration time. Note this is "configuration time" and not yet "build time", the latter of which is when more information is known to the build system.

A forum post discussing how to make cmake generate info to go into a file at build time:


^ Project Organization and Cmake

In the "Mastering Cmake" online book Cmake team speaks to Cmake's strategy to create a dedicated build directory at the root of a given project. This dedicated build directory keeps all build artifacts separate from project sources, and allows for a simple and safe way to deleting build generated files without impacting project sources. This strategy can be achieved with other build systems like GNU `make`, but cmake operates on projects this way by default.

There's more to project organization than choosing an area for build output (artifacts). In particular when a final compiled image involves two or more applications, a project's organization becomes more complex. This so whether projects utilize cmake or another build system.

Some links to project organization forum posts follow here:

. . . though this post is ten years old, may yet hold useful information.

Questions outstanding (from this section):

(Q1) Why in cmake projects can there end up being so many distinctly named "projects"?

^ To Create Libraries In A Build

Cmake supports the specification and building of library object files as part of a project. Further, cmake supports the defining and declaring of where and how these libraries will be linked to other targets in the given project. To create a library, a typical pattern of cmake function calls includes:

 (1)  add_library(<name_of_library> OBJECT)

 (2)  target_include_directories(<name_of_library>
          PUBLIC include_path
          PRIVATE include_path_private
      )

 (3)  target_sources(<name_of_library PRIVATE
          source_file_1.c
          source_file_2.c
           . . .
      )

 (4)  target_link_libraries(<name_of_library> PUBLIC|INTERFACE|PRIVATE 

Online cmake documentation on the topic of libraries:

DESPUES DE ESTO REVISTAR TODO - 2025 Q3

^ Scope of Variables in Cmake

Matthew Gaudet cmake variable scoping:


^ Practical Things to Know and Do

stub section intro

^ To Debug Cmake Scripts

A gitlab dot io cmake resource:

Note that the following post has a cool answer which involves a cmake macro to print all variables known to a given cmake invocation or use, and also a more simple answer to print one specific variable:


A modified generic.cmake file, changes address one or more unset variables and solved build error on newest Zephyr work station:

# SPDX-License-Identifier: Apache-2.0

set_ifndef(CC gcc)

## 2022-03-03 - Ted moving this cmake function call beyond locally
##  added variable assignements:
##
## find_program(CMAKE_C_COMPILER ${CROSS_COMPILE}${CC}   PATHS ${TOOLCHAIN_HOME} NO_DEFAULT_PATH)

#
## - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
## 2022-02-08 Reference https://cmake.org/cmake/help/latest/command/message.html
## - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
#
message(NOTICE "*** zztop 1 *** building with user configuration var ZEPHYR_TOOLCHAIN_VARIANT set to '${ZEPHYR_TOOLCHAIN_VARIANT}'")
message(NOTICE "*** zztop 2 *** internal variable CROSS_COMPILE set to '${CROSS_COMPILE}'")
message(NOTICE "*** zztop 3 *** internal variable TOOLCHAIN_HOME set to '${TOOLCHAIN_HOME}'")

## 2022-03-03 THU - Not ***ing sure why we need to set these today:
set(ZEPHYR_TOOLCHAIN_VARIANT gnuarmemb)
set(GNUARMEMB_TOOLCHAIN_PATH /opt/gcc-arm-none-eabi-10.3-2021.10)
set(TOOLCHAIN_HOME /opt/gcc-arm-none-eabi-10.3-2021.10)
## ^^^ These three cmake variable settings not sufficient, trying another based on working but different
##  toolchain on host ind:
set(CROSS_COMPILE /opt/gcc-arm-none-eabi-10.3-2021.10/bin/arm-none-eabi-)
##
## $ ls /opt/gcc-arm-none-eabi-10.3-2021.10/bin/arm-none-eabi-*
##

message(NOTICE "*** zztop 4 *** building with user configuration var ZEPHYR_TOOLCHAIN_VARIANT set to '${ZEPHYR_TOOLCHAIN_VARIANT}'")
message(NOTICE "*** zztop 5 *** internal variable CROSS_COMPILE set to '${CROSS_COMPILE}'")
message(NOTICE "*** zztop 6 *** internal variable TOOLCHAIN_HOME set to '${TOOLCHAIN_HOME}'")


find_program(CMAKE_C_COMPILER ${CROSS_COMPILE}${CC}   PATHS ${TOOLCHAIN_HOME} NO_DEFAULT_PATH)

if(CMAKE_C_COMPILER STREQUAL CMAKE_C_COMPILER-NOTFOUND)
  message(FATAL_ERROR "Zephyr was unable to find the toolchain. Is the environment misconfigured?
User-configuration:
ZEPHYR_TOOLCHAIN_VARIANT: ${ZEPHYR_TOOLCHAIN_VARIANT}
Internal variables:
CROSS_COMPILE: ${CROSS_COMPILE}
TOOLCHAIN_HOME: ${TOOLCHAIN_HOME}
")
endif()

execute_process(
  COMMAND ${CMAKE_C_COMPILER} --version
  RESULT_VARIABLE ret
  OUTPUT_QUIET
  ERROR_QUIET
  )
if(ret)
  message(FATAL_ERROR "Executing the below command failed. Are permissions set correctly?
'${CMAKE_C_COMPILER} --version'
"
    )
endif()

^ Source and Library Path Setting

Cmake has at least as of 2020 a notion of "translation map" which relates to the ways in which cmake determines file paths. One introduction to this concept arises at the forum post:

There is comment in cmake documentation that use of symbolic links can lead to subtle errors. Need to locate this reference.

2025-04-16

In Zephyr 3.7.0 CMAKE_CURRENT_BINARY_DIR is set in file `zephyr/cmake/package_helper.cmake`:

105:set(CMAKE_CURRENT_BINARY_DIR ${CMAKE_BINARY_DIR})`.

The variable referenced here is also set in `package_helper.cmake`:

./cmake/package_helper.cmake:75:    set(CMAKE_BINARY_DIR ${argB})
./cmake/package_helper.cmake:79:    set(CMAKE_BINARY_DIR ${CMAKE_ARGV${argB_value_index}})
./cmake/package_helper.cmake:104:cmake_path(ABSOLUTE_PATH CMAKE_BINARY_DIR NORMALIZE)

- - - - - - - - - -

Earlier references:

Cmake has a notion of a binary or binaries directory, and has a built-in variable CMAKE_BINARY_DIR as described here:


^ cmake Lists

Look up `append_if_path_exists()` . . .


^ cmake Functions

An interesting pattern search in Zephyr 3.2.0 source tree and associated third party libraries is `$ grep -nr ExternalProject_Add ./*`, which reveals which projects and locations in Zephyr's source tree this cmake function is called. This function is needed when building multicore Zephyr apps.

There is a detailed document on `ExternalProject_Add()` at the Kitware public github repository:

    ^ ExternalProject_Add

    Cmake function `ExternalProject_Add()` is used in multi-core Zephyr sample projects. Some refs to this function include:

    ^ find_package


    ^ add_subdirectory_ifdef()

    Really good blog post at blog.golioth.io regarding how to configure a "super project" containing multiple Zephyr based apps with shared, select-able at build time common code:

    ^ function()

    - stub -

    ^ execute_process()

    - stub -

    ^ add_custom_command()

    - stub -

    ^ Install latest version of cmake


    ^ cmake Package Configuration Files - What Are They?

    QUESTION (2022-12-29): what are cmake package configuration files?

    When attempting to switch Zephyr sdk of original latest 2021 kionix-driver-demo, from Nordic ncs SDK to Zephyr RTOS Project itself, and a more recent release of Zephyr RTOS, get following error early early in build process:

    :~/projects-sandbox/workspace-out-of-tree/kionix-driver-demo$ west build -b lpcxpresso55s69_cpu0 -p
    -- west build: making build dir /home/ted/projects-sandbox/workspace-out-of-tree/kionix-driver-demo/build pristine
    -- west build: generating a build system
    -- Build type: debug
    Loading Zephyr default modules (Zephyr base).
    -- Application: /home/ted/projects-sandbox/workspace-out-of-tree/kionix-driver-demo
    -- Found Python3: /usr/bin/python3.8 (found suitable exact version "3.8.10") found components: Interpreter 
    -- Cache files will be written to: /home/ted/.cache/zephyr
    -- Zephyr version: 3.1.0 (/home/ted/projects-sandbox/workspace-out-of-tree/zephyr)
    -- Found west (found suitable version "0.12.0", minimum required is "0.7.1")
    -- Board: lpcxpresso55s69_cpu0
    CMake Error at /home/ted/projects-sandbox/workspace-out-of-tree/zephyr/cmake/modules/verify-toolchain.cmake:79 (find_package):
      Could not find a package configuration file provided by "Zephyr-sdk"
      (requested version 0.13.1) with any of the following names:
    
        Zephyr-sdkConfig.cmake
        zephyr-sdk-config.cmake
    
      Add the installation prefix of "Zephyr-sdk" to CMAKE_PREFIX_PATH or set
      "Zephyr-sdk_DIR" to a directory containing one of the above files.  If
      "Zephyr-sdk" provides a separate development package or SDK, be sure it has
      been installed.
    Call Stack (most recent call first):
      /home/ted/projects-sandbox/workspace-out-of-tree/zephyr/cmake/modules/zephyr_default.cmake:121 (include)
      /home/ted/projects-sandbox/workspace-out-of-tree/zephyr/share/zephyr-package/cmake/ZephyrConfig.cmake:51 (include)
      /home/ted/projects-sandbox/workspace-out-of-tree/zephyr/share/zephyr-package/cmake/ZephyrConfig.cmake:76 (include_boilerplate)
      CMakeLists.txt:51 (find_package)
    
    
    -- Configuring incomplete, errors occurred!
    FATAL ERROR: command exited with status 1: /usr/bin/cmake -DWEST_PYTHON=/usr/bin/python3 -B/home/ted/projects-sandbox/workspace-out-of-tree/kionix-driver-demo/build -S/home/ted/projects-sandbox/workspace-out-of-tree/kionix-driver-demo -GNinja -DBOARD=lpcxpresso55s69_cpu0
    

    So at time of above error we didn't know how to trace that problem, though there is a call stack reported by cmake which tells us some cmake files we may review for better understanding. There are notes on cmake's documentation pages as of 2022-12-27 which mention the naming conventions of files which cmake's find_package() will look for when it is invoked.

    NEED to add link to cmake's find_package() documentation.

    2022-12-28 - some new cmake build errors:

    (1)

    ted@localhost:~/projects/zephyr-based/out-of-tree/kionix-driver-demo$ west build -b lpcxpresso55s69_cpu0 -p
    -- west build: making build dir /home/ted/projects/zephyr-based/out-of-tree/kionix-driver-demo/build pristine
    -- west build: generating a build system
    Loading Zephyr default modules (Zephyr base).
    -- Application: /home/ted/projects/zephyr-based/out-of-tree/kionix-driver-demo
    -- Found Python3: /usr/bin/python3.8 (found suitable exact version "3.8.10") found components: Interpreter 
    -- Cache files will be written to: /home/ted/.cache/zephyr
    -- Zephyr version: 3.2.0 (/home/ted/projects/zephyr-based/out-of-tree/zephyr)
    -- Found west (found suitable version "0.11.0", minimum required is "0.7.1")
    -- Board: lpcxpresso55s69_cpu0
    -- ZEPHYR_TOOLCHAIN_VARIANT not set, trying to locate Zephyr SDK
    CMake Error at /home/ted/projects/zephyr-based/out-of-tree/zephyr/cmake/modules/FindZephyr-sdk.cmake:63 (find_package):
      Could not find a configuration file for package "Zephyr-sdk" that is
      compatible with requested version "0.15".
    
      The following configuration files were considered but not accepted:
    
        /opt/zephyr-sdk-0.12.4/cmake/Zephyr-sdkConfig.cmake, version: 0.12.4
        /opt/zephyr-sdk-0.14.2/cmake/Zephyr-sdkConfig.cmake, version: 0.14.2
        /opt/zephyr-sdk-0.13.0/cmake/Zephyr-sdkConfig.cmake, version: 0.13.0
    
    Call Stack (most recent call first):
      /home/ted/projects/zephyr-based/out-of-tree/zephyr/cmake/modules/FindHostTools.cmake:53 (find_package)
      /home/ted/projects/zephyr-based/out-of-tree/zephyr/cmake/modules/dts.cmake:8 (find_package)
      /home/ted/projects/zephyr-based/out-of-tree/zephyr/cmake/modules/zephyr_default.cmake:108 (include)
      /home/ted/projects/zephyr-based/out-of-tree/zephyr/share/zephyr-package/cmake/ZephyrConfig.cmake:66 (include)
      /home/ted/projects/zephyr-based/out-of-tree/zephyr/share/zephyr-package/cmake/ZephyrConfig.cmake:92 (include_boilerplate)
      CMakeLists.txt:15 (find_package)
    
    
    -- Configuring incomplete, errors occurred!
    FATAL ERROR: command exited with status 1: /usr/bin/cmake -DWEST_PYTHON=/usr/bin/python3 -B/home/ted/projects/zephyr-based/out-of-tree/kionix-driver-demo/build -GNinja -DBOARD=lpcxpresso55s69_cpu0 -S/home/ted/projects/zephyr-based/out-of-tree/kionix-driver-demo
    
    comentario


    After setting `ZEPHYR_TOOLCHAIN_VARIANT` in the current shell, get these cmake errors:

    (2)

    ted@localhost:~/projects/zephyr-based/out-of-tree/kionix-driver-demo$ export ZEPHYR_TOOLCHAIN_VARIANT=gnuarmemb
    ted@localhost:~/projects/zephyr-based/out-of-tree/kionix-driver-demo$ west build -b lpcxpresso55s69_cpu0 -p
    -- west build: making build dir /home/ted/projects/zephyr-based/out-of-tree/kionix-driver-demo/build pristine
    -- west build: generating a build system
    Loading Zephyr default modules (Zephyr base).
    -- Application: /home/ted/projects/zephyr-based/out-of-tree/kionix-driver-demo
    -- Found Python3: /usr/bin/python3.8 (found suitable exact version "3.8.10") found components: Interpreter 
    -- Cache files will be written to: /home/ted/.cache/zephyr
    -- Zephyr version: 3.2.0 (/home/ted/projects/zephyr-based/out-of-tree/zephyr)
    -- Found west (found suitable version "0.11.0", minimum required is "0.7.1")
    -- Board: lpcxpresso55s69_cpu0
    CMake Error at /home/ted/projects/zephyr-based/out-of-tree/zephyr/cmake/modules/extensions.cmake:2071 (message):
      Assertion failed: GNUARMEMB_TOOLCHAIN_PATH is not set
    Call Stack (most recent call first):
      /home/ted/projects/zephyr-based/out-of-tree/zephyr/cmake/toolchain/gnuarmemb/generic.cmake:4 (assert)
      /home/ted/projects/zephyr-based/out-of-tree/zephyr/cmake/modules/FindHostTools.cmake:103 (include)
      /home/ted/projects/zephyr-based/out-of-tree/zephyr/cmake/modules/dts.cmake:8 (find_package)
      /home/ted/projects/zephyr-based/out-of-tree/zephyr/cmake/modules/zephyr_default.cmake:108 (include)
      /home/ted/projects/zephyr-based/out-of-tree/zephyr/share/zephyr-package/cmake/ZephyrConfig.cmake:66 (include)
      /home/ted/projects/zephyr-based/out-of-tree/zephyr/share/zephyr-package/cmake/ZephyrConfig.cmake:92 (include_boilerplate)
      CMakeLists.txt:15 (find_package)
    
    
    -- Configuring incomplete, errors occurred!
    FATAL ERROR: command exited with status 1: /usr/bin/cmake -DWEST_PYTHON=/usr/bin/python3 -B/home/ted/projects/zephyr-based/out-of-tree/kionix-driver-demo/build -GNinja -DBOARD=lpcxpresso55s69_cpu0 -S/home/ted/projects/zephyr-based/out-of-tree/kionix-driver-demo
    


    ^ Cmake and Zephyr Builds

    * * * Stub section * * *

    The previous section of this nn page segways into questions about how exactly cmake functions as a build system for Zephyr RTOS based applications. Typically, a call to the Python based script `west` starts a build. Zephyr RTOS project adds functionality to `west`, including the `west` command `build`. (That's not available when invoking `west` outside of a Zephyr project workspace.

    This section intended as a landing point for notes and findings about the steps cmake commneces during a Zephyr project build.

    Some files, some build artifacts which may be of interest in this search:

    (1) $workspace/build/cpu0/zephyr/linker_zephyr_pre0.cmd.dep

    ----- ----- ----- ----- ----- ----- ----- ----- ----- ----- ----- ----- ----- ----- ----- -----

    SORT SECTION BEGIN

    ^ Zephyr Threads Tutorial by Maksim Drachov

    Developer Maksim Drachov some helpful Zephyr build system and cmake tutorials, or explanatory documents on github. While not everything is explained, one of his example projects which builds a TCP/IP networking stack for a posix_native targeted board offers a number of more advanced cmake functions and features as part of this example project.

    Earliest Maksim github link noted here is:


    ^ cmake Debugging

    -- Application: /home/ted/projects/zephyr-based/kionix-driver-demo
    -- Zephyr version: 2.6.0-rc1 (/home/ted/projects/zephyr-based/zephyr), build: v2.6.0-rc1-ncs1
    -- Found Python3: /usr/bin/python3.8 (found suitable exact version "3.8.10") found components: Interpreter 
    -- Found west (found suitable version "0.12.0", minimum required is "0.7.1")
    -- Board: sparkfun_thing_plus_nrf9160
    -- Cache files will be written to: /home/ted/.cache/zephyr
    -- Found dtc: /usr/bin/dtc (found suitable version "1.5.0", minimum required is "1.4.6")
    CMake Error at /home/ted/projects/zephyr-based/zephyr/cmake/toolchain/zephyr/generic.cmake:3 (if):
      if given arguments:
    
        "VERSION_LESS_EQUAL" "0.11.2"
    
      Unknown arguments specified
    Call Stack (most recent call first):
      /home/ted/projects/zephyr-based/zephyr/cmake/generic_toolchain.cmake:36 (include)
      /home/ted/projects/zephyr-based/zephyr/cmake/app/boilerplate.cmake:553 (include)
      /home/ted/projects/zephyr-based/zephyr/share/zephyr-package/cmake/ZephyrConfig.cmake:24 (include)
      /home/ted/projects/zephyr-based/zephyr/share/zephyr-package/cmake/ZephyrConfig.cmake:35 (include_boilerplate)
      CMakeLists.txt:40 (find_package)
    
    
    -- Configuring incomplete, errors occurred!
    FATAL ERROR: command exited with status 1: /usr/bin/cmake -DWEST_PYTHON=/usr/bin/python3 -B/home/ted/projects/zephyr-based/kionix-driver-demo/build -S/home/ted/projects/zephyr-based/kionix-driver-demo -GNinja -DBOARD=sparkfun_thing_plus_nrf9160
    


    ^ Cmake links to sort

    2021-08-06

    Starting a page for cmake notes here. Links to cmake site:

    Longer guides:


    A Nordic Semi document on `cmake` and its role in the build process of Zephyr based applications:


    2021-08-17 - search for cmake / Zephyr means to add sources to a Zephyr based project:

    ^ Topics To Follow Up

    Who accepts the `--extra-args=CONFIG_*` option plus value? From `zephyr/scripts/pylib/twister/twisterlib/testinstance.py`:

    225     def create_overlay(self, platform, enable_asan=False, enable_ubsan=False, enable_coverage=False, coverage_platform=[]):
    226         # Create this in a "twister/" subdirectory otherwise this
    227         # will pass this overlay to kconfig.py *twice* and kconfig.cmake
    228         # will silently give that second time precedence over any
    229         # --extra-args=CONFIG_*
    230         subdir = os.path.join(self.build_dir, "twister")
    231 
    232         content = ""
    


    In Zephyr 3.2.0, `zephyr/cmake/modules/configuration_files.cmake` looks promising in terms of revealing where and how Kconfig processing is called during build of a Zephyr based app. An excerpt from this file:

      1 # SPDX-License-Identifier: Apache-2.0
      2 #
      3 # Copyright (c) 2021, Nordic Semiconductor ASA
      4 
      5 # Zephyr build system configuration files.
      6 #
      7 # Locate the Kconfig and DT config files that are to be used.
      8 # Also, locate the appropriate application config directory.
      9 #
     10 # Outcome:
     11 # The following variables will be defined when this CMake module completes:
     12 #
     13 # - CONF_FILE:              List of Kconfig fragments
     14 # - DTC_OVERLAY_FILE:       List of devicetree overlay files
     15 # - APPLICATION_CONFIG_DIR: Root folder for application configuration
     16 #
     17 # If any of the above variables are already set when this CMake module is
     18 # loaded, then no changes to the variable will happen.
     19 #
     20 # Variables set by this module and not mentioned above are considered internal
     21 # use only and may be removed, renamed, or re-purposed without prior notice.
    

    The cmake variable CONF_FILE can be set by:

     set(CONF_FILE prj-fragment-1.conf prj-fragment-2.conf other-configuration.conf)
    


    ----- ----- ----- ----- ----- ----- ----- ----- ----- ----- ----- ----- ----- ----- ----- -----

    ^ A Cmake Exploration Effort

    Overview: cmake tests to see where certain libraries are getting pulled across one application to a second app.

    In a multi-application cmake project, cmake files typically populate multiple directories. The build process commences with one cmake file being processed first. To better understand what happens next, cmake's `message()` function or command acts like a C printf() statement and can reveal build state at various stages of this process . . .

    Zephyr RTOS adds cmake scripts as part of its build process. When we build one or more Zephyr based apps we see various messages from cmake during these builds. For multi-app builds we see near the start of each build:

    "Loading Zephyr default modules . . ."

    This message comes from:

    $ grep -nr 'Loading Zephyr default' --exclude-dir=build --exclude-dir=docker_cache ./*
    ./share/zephyr-package/cmake/ZephyrConfig.cmake:65:      zephyr_package_message(NOTICE "Loading Zephyr default modules (${location}).")
    $
    

    In builds of Zephyr 3.7.0 based apps, first build system message to standard out appears to be "making build dir " . . .

    ted@localhost1:~/projects-sandbox/workspace-for-nexus/zephyr/scripts/west_commands$ !grep
    grep -nr 'making build dir' ./*
    ./build.py:601:        _banner('making build dir {} pristine'.format(self.build_dir))
    

    The definition of `_banner()` is at:

    ./zephyr/scripts/west_commands/run_common.py:55:def _banner(msg):
    

    ^ cmake command include_directories()

    Some related commands from cmake: include_directories(), target_include_directories(), and from Zephyr's additions

    Note that ``target_include_directories()`` expects at minimum a couple of arguments before the include paths themselves. One example of its practical use is

    target_include_directories(app PRIVATE
        include
    )
    

    Visit cmake documentation to learn which other qualifying arguments may be given in place of ``app`` and ``PRIVATE``, which are specific values.


    - - - top of page - - -