1.. _cmake-details:
2
3Build System (CMake)
4********************
5
6
7CMake is used to build your application together with the Zephyr kernel. A
8CMake build is done in two stages. The first stage is called
9**configuration**. During configuration, the CMakeLists.txt build scripts are
10executed. After configuration is finished, CMake has an internal model of the
11Zephyr build, and can generate build scripts that are native to the host
12platform.
13
14CMake supports generating scripts for several build systems, but only Ninja and
15Make are tested and supported by Zephyr. After configuration, you begin the
16**build** stage by executing the generated build scripts. These build scripts
17can recompile the application without involving CMake following
18most code changes. However, after certain changes, the configuration step must
19be executed again before building. The build scripts can detect some of these
20situations and reconfigure automatically, but there are cases when this must be
21done manually.
22
23Zephyr uses CMake's concept of a 'target' to organize the build. A
24target can be an executable, a library, or a generated file. For
25application developers, the library target is the most important to
26understand. All source code that goes into a Zephyr build does so by
27being included in a library target, even application code.
28
29Library targets have source code, that is added through CMakeLists.txt
30build scripts like this:
31
32.. code-block:: cmake
33
34   target_sources(app PRIVATE src/main.c)
35
36In the above :file:`CMakeLists.txt`, an existing library target named ``app``
37is configured to include the source file :file:`src/main.c`. The ``PRIVATE``
38keyword indicates that we are modifying the internals of how the library is
39being built. Using the keyword ``PUBLIC`` would modify how other
40libraries that link with app are built. In this case, using ``PUBLIC``
41would cause libraries that link with ``app`` to also include the
42source file :file:`src/main.c`, behavior that we surely do not want. The
43``PUBLIC`` keyword could however be useful when modifying the include
44paths of a target library.
45
46
47Build and Configuration Phases
48==============================
49
50The Zephyr build process can be divided into two main phases: a configuration
51phase (driven by CMake) and a build phase (driven by Make or Ninja).
52
53.. _build_configuration_phase:
54
55Configuration Phase
56-------------------
57
58The configuration phase begins when the user invokes *CMake* to generate a
59build system, specifying a source application directory and a board target.
60
61.. figure:: build-config-phase.svg
62    :align: center
63    :alt: Zephyr's build configuration phase
64    :figclass: align-center
65    :width: 80%
66
67CMake begins by processing the :file:`CMakeLists.txt` file in the application
68directory, which refers to the :file:`CMakeLists.txt` file in the Zephyr
69top-level directory, which in turn refers to :file:`CMakeLists.txt` files
70throughout the build tree (directly and indirectly). Its primary output is a
71set of Makefiles or Ninja files to drive the build process, but the CMake
72scripts also do some processing of their own, which is explained here.
73
74Note that paths beginning with :file:`build/` below refer to the build
75directory you create when running CMake.
76
77Devicetree
78   :file:`*.dts` (*devicetree source*) and :file:`*.dtsi` (*devicetree source
79   include*) files are collected from the target's architecture, SoC, board,
80   and application directories.
81
82   :file:`*.dtsi` files are included by :file:`*.dts` files via the C
83   preprocessor (often abbreviated *cpp*, which should not be confused with
84   C++). The C preprocessor is also used to merge in any devicetree
85   :file:`*.overlay` files, and to expand macros in :file:`*.dts`,
86   :file:`*.dtsi`, and :file:`*.overlay` files. The preprocessor output is
87   placed in :file:`build/zephyr/zephyr.dts.pre`.
88
89   The preprocessed devicetree sources are parsed by
90   :zephyr_file:`gen_defines.py <scripts/dts/gen_defines.py>` to generate a
91   :file:`build/zephyr/include/generated/zephyr/devicetree_generated.h` header with
92   preprocessor macros.
93
94   Source code should access preprocessor macros generated from devicetree by
95   including the :zephyr_file:`devicetree.h <include/zephyr/devicetree.h>` header,
96   which includes :file:`devicetree_generated.h`.
97
98   :file:`gen_defines.py` also writes the final devicetree to
99   :file:`build/zephyr/zephyr.dts` in the build directory. This file's contents
100   may be useful for debugging.
101
102   If the devicetree compiler ``dtc`` is installed, it is run on
103   :file:`build/zephyr/zephyr.dts` to catch any extra warnings and errors
104   generated by this tool. The output from ``dtc`` is unused otherwise, and
105   this step is skipped if ``dtc`` is not installed.
106
107   The above is just a brief overview. For more information on devicetree, see
108   :ref:`dt-guide`.
109
110Kconfig
111   :file:`Kconfig` files define available configuration options for the
112   target architecture, SoC, board, and application, as well as dependencies
113   between options.
114
115   Kconfig configurations are stored in *configuration files*. The initial
116   configuration is generated by merging configuration fragments from the board
117   and application (e.g. :file:`prj.conf`).
118
119   The output from Kconfig is an :file:`autoconf.h` header with preprocessor
120   assignments, and a :file:`.config` file that acts both as a saved
121   configuration and as configuration output (used by CMake). The definitions in
122   :file:`autoconf.h` are automatically exposed at compile time, so there is no
123   need to include this header.
124
125   Information from devicetree is available to Kconfig, through the functions
126   defined in :zephyr_file:`kconfigfunctions.py
127   <scripts/kconfig/kconfigfunctions.py>`.
128
129   See :ref:`the Kconfig section of the manual <kconfig>` for more information.
130
131Build Phase
132-----------
133
134The build phase begins when the user invokes ``make`` or ``ninja``. Its
135ultimate output is a complete Zephyr application in a format suitable for
136loading/flashing on the desired target board (:file:`zephyr.elf`,
137:file:`zephyr.hex`, etc.) The build phase can be broken down, conceptually,
138into four stages: the pre-build, first-pass binary, final binary, and
139post-processing.
140
141Pre-build
142+++++++++
143
144Pre-build occurs before any source files are compiled, because during
145this phase header files used by the source files are generated.
146
147Offset generation
148   Access to high-level data structures and members is sometimes
149   required when the definitions of those structures is not
150   immediately accessible (e.g., assembly language). The generation of
151   *offsets.h* (by *gen_offset_header.py*) facilitates this.
152
153System call boilerplate
154   The *gen_syscall.py* and *parse_syscalls.py*  scripts work
155   together to bind potential system call functions with their
156   implementations.
157
158.. figure:: build-build-phase-1.svg
159    :align: center
160    :alt: Zephyr's build stage I
161    :figclass: align-center
162    :width: 80%
163
164Intermediate binaries
165+++++++++++++++++++++
166
167Compilation proper begins with the first intermediate binary. Source files (C
168and assembly) are collected from various subsystems (which ones is
169decided during the configuration phase), and compiled into archives
170(with reference to header files in the tree, as well as those
171generated during the configuration phase and the pre-build stage(s)).
172
173.. figure:: build-build-phase-2.svg
174    :align: center
175    :alt: Zephyr's build stage II
176    :figclass: align-center
177    :width: 80%
178
179The exact number of intermediate binaries is decided during the configuration
180phase.
181
182If memory protection is enabled, then:
183
184Partition grouping
185   The *gen_app_partitions.py* script scans all the
186   generated archives and outputs linker scripts to ensure that
187   application partitions are properly grouped and aligned for the
188   target’s memory protection hardware.
189
190Then *cpp* is used to combine linker script fragments from the target’s
191architecture/SoC, the kernel tree, optionally the partition output if
192memory protection is enabled, and any other fragments selected during
193the configuration process, into a *linker.cmd* file. The compiled
194archives are then linked with *ld* as specified in the
195*linker.cmd*.
196
197Unfixed size binary
198   The unfixed size intermediate binary is produced when :ref:`usermode_api`
199   is enabled or :ref:`devicetree` is in use.
200   It produces a binary where sizes are not fixed and thus it may be used
201   by post-process steps that will impact the size of the final binary.
202
203.. figure:: build-build-phase-3.svg
204    :align: center
205    :alt: Zephyr's build stage III
206    :figclass: align-center
207    :width: 80%
208
209Fixed size binary
210   The fixed size intermediate binary is produced when :ref:`usermode_api`
211   is enabled or when generated IRQ tables are used,
212   :kconfig:option:`CONFIG_GEN_ISR_TABLES`
213   It produces a binary where sizes are fixed and thus the size must not change
214   between the intermediate binary and the final binary.
215
216.. figure:: build-build-phase-4.svg
217    :align: center
218    :alt: Zephyr's build stage IV
219    :figclass: align-center
220    :width: 80%
221
222Intermediate binaries post-processing
223+++++++++++++++++++++++++++++++++++++
224
225The binaries from the previous stage are incomplete, with empty and/or
226placeholder sections that must be filled in by, essentially, reflection.
227
228To complete the build procedure the following scripts are executed on the
229intermediate binaries to produce the missing pieces needed for the final
230binary.
231
232When :ref:`usermode_api` is enabled:
233
234Partition alignment
235   The *gen_app_partitions.py* script scans the unfixed size binary and
236   generates an app shared memory aligned linker script snippet where the
237   partitions are sorted in descending order.
238
239.. figure:: build-postprocess-1.svg
240    :align: center
241    :alt: Zephyr's intermediate binary post-process I
242    :figclass: align-center
243    :width: 80%
244
245When :ref:`devicetree` is used:
246
247Device dependencies
248   The *gen_device_deps.py* script scans the unfixed size binary to determine
249   relationships between devices that were recorded from devicetree data,
250   and replaces the encoded relationships with values that are optimized to
251   locate the devices actually present in the application.
252
253.. figure:: build-postprocess-2.svg
254    :align: center
255    :alt: Zephyr's intermediate binary post-process II
256    :figclass: align-center
257    :width: 80%
258
259When :kconfig:option:`CONFIG_GEN_ISR_TABLES` is enabled:
260   The *gen_isr_tables.py* script scans the fixed size binary and creates
261   an isr_tables.c source file with a hardware vector table and/or software
262   IRQ table.
263
264.. figure:: build-postprocess-3.svg
265    :align: center
266    :alt: Zephyr's intermediate binary post-process III
267    :figclass: align-center
268    :width: 80%
269
270When :ref:`usermode_api` is enabled:
271
272Kernel object hashing
273   The *gen_kobject_list.py* scans the *ELF DWARF*
274   debug data to find the address of the all kernel objects. This
275   list is passed to *gperf*, which generates a perfect hash function and
276   table of those addresses, then that output is optimized by
277   *process_gperf.py*, using known properties of our special case.
278
279.. figure:: build-postprocess-4.svg
280    :align: center
281    :alt: Zephyr's intermediate binary post-process IV
282    :figclass: align-center
283    :width: 80%
284
285When no intermediate binary post-processing is required then the first
286intermediate binary will be directly used as the final binary.
287
288Final binary
289++++++++++++
290
291The binary from the previous stage is incomplete, with empty and/or
292placeholder sections that must be filled in by, essentially, reflection.
293
294The link from the previous stage is repeated, this time with the missing
295pieces populated.
296
297.. figure:: build-build-phase-5.svg
298    :align: center
299    :alt: Zephyr's build final stage
300    :figclass: align-center
301    :width: 80%
302
303Post processing
304+++++++++++++++
305
306Finally, if necessary, the completed kernel is converted from *ELF* to
307the format expected by the loader and/or flash tool required by the
308target. This is accomplished in a straightforward manner with *objdump*.
309
310.. figure:: build-build-phase-6.svg
311    :align: center
312    :alt: Zephyr's build final stage post-process
313    :figclass: align-center
314    :width: 80%
315
316
317.. _build_system_scripts:
318
319Supporting Scripts and Tools
320============================
321
322The following is a detailed description of the scripts used during the build process.
323
324.. _gen_syscalls.py:
325
326:zephyr_file:`scripts/build/gen_syscalls.py`
327--------------------------------------------
328
329.. include:: ../../../scripts/build/gen_syscalls.py
330   :start-after: """
331   :end-before: """
332
333.. _gen_device_deps.py:
334
335:zephyr_file:`scripts/build/gen_device_deps.py`
336-----------------------------------------------
337
338.. include:: ../../../scripts/build/gen_device_deps.py
339   :start-after: """
340   :end-before: """
341
342.. _gen_kobject_list.py:
343
344:zephyr_file:`scripts/build/gen_kobject_list.py`
345------------------------------------------------
346
347.. include:: ../../../scripts/build/gen_kobject_list.py
348   :start-after: """
349   :end-before: """
350
351.. _gen_offset_header.py:
352
353:zephyr_file:`scripts/build/gen_offset_header.py`
354-------------------------------------------------
355
356.. include:: ../../../scripts/build/gen_offset_header.py
357   :start-after: """
358   :end-before: """
359
360.. _parse_syscalls.py:
361
362:zephyr_file:`scripts/build/parse_syscalls.py`
363----------------------------------------------
364
365
366.. include:: ../../../scripts/build/parse_syscalls.py
367   :start-after: """
368   :end-before: """
369
370.. _gen_idt.py:
371
372:zephyr_file:`arch/x86/gen_idt.py`
373----------------------------------
374
375.. include:: ../../../arch/x86/gen_idt.py
376   :start-after: """
377   :end-before: """
378
379.. _gen_gdt.py:
380
381:zephyr_file:`arch/x86/gen_gdt.py`
382----------------------------------
383
384.. include:: ../../../arch/x86/gen_gdt.py
385   :start-after: """
386   :end-before: """
387
388.. _gen_relocate_app.py:
389
390:zephyr_file:`scripts/build/gen_relocate_app.py`
391------------------------------------------------
392
393.. include:: ../../../scripts/build/gen_relocate_app.py
394   :start-after: """
395   :end-before: """
396
397.. _process_gperf.py:
398
399:zephyr_file:`scripts/build/process_gperf.py`
400---------------------------------------------
401
402.. include:: ../../../scripts/build/process_gperf.py
403   :start-after: """
404   :end-before: """
405
406:zephyr_file:`scripts/build/gen_app_partitions.py`
407--------------------------------------------------
408
409.. include:: ../../../scripts/build/gen_app_partitions.py
410   :start-after: """
411   :end-before: """
412
413.. _check_init_priorities.py:
414
415:zephyr_file:`scripts/build/check_init_priorities.py`
416-----------------------------------------------------
417
418.. include:: ../../../scripts/build/check_init_priorities.py
419   :start-after: """
420   :end-before: """
421