1.. _dt-trouble:
2
3Troubleshooting devicetree
4##########################
5
6Here are some tips for fixing misbehaving devicetree related code.
7
8See :ref:`dt-howtos` for other "HOWTO" style information.
9
10.. _dt-trouble-try-pristine:
11
12Try again with a pristine build directory
13*****************************************
14
15.. important:: Try this first, before doing anything else.
16
17See :ref:`west-building-pristine` for examples, or just delete the build
18directory completely and retry.
19
20This is general advice which is especially applicable to debugging devicetree
21issues, because the outputs are created during the CMake configuration phase,
22and are not always regenerated when one of their inputs changes.
23
24Make sure <devicetree.h> is included
25************************************
26
27Unlike Kconfig symbols, the :file:`devicetree.h` header must be included
28explicitly.
29
30Many Zephyr header files rely on information from devicetree, so including some
31other API may transitively include :file:`devicetree.h`, but that's not
32guaranteed.
33
34undefined reference to ``__device_dts_ord_<N>``
35***********************************************
36
37This usually happens on a line like this:
38
39.. code-block:: c
40
41   const struct device *dev = DEVICE_DT_GET(NODE_ID);
42
43where ``NODE_ID`` is a valid :ref:`node identifier <dt-node-identifiers>`, but
44no device driver has allocated a ``struct device`` for this devicetree node.
45You thus get a linker error, because you're asking for a pointer to a device
46that isn't defined.
47
48To fix it, you need to make sure that:
49
501. The node is enabled: the node must have ``status = "okay";``.
51
52   (Recall that a missing ``status`` property means the same thing as ``status
53   = "okay";``; see :ref:`dt-important-props` for more information about
54   ``status``).
55
562. A device driver responsible for allocating the ``struct device`` is enabled.
57   That is, the Kconfig option which makes the build system compile the driver
58   sources into your application needs to be set to ``y``.
59
60   (See :ref:`setting_configuration_values` for more information on setting
61   Kconfig options.)
62
63Below, ``<build>`` means your build directory.
64
65**Making sure the node is enabled**:
66
67To find the devicetree node you need to check, use the number ``<N>`` from the
68linker error. Look for this number in the list of nodes at the top of
69:file:`<build>/zephyr/include/generated/zephyr/devicetree_generated.h`. For example, if
70``<N>`` is 15, and your :file:`devicetree_generated.h` file looks like this,
71the node you are interested in is ``/soc/i2c@deadbeef``:
72
73.. code-block:: none
74
75   /*
76    * Generated by gen_defines.py
77    *
78    * DTS input file:
79    *   <build>/zephyr/zephyr.dts.pre
80    *
81    * Directories with bindings:
82    *   $ZEPHYR_BASE/dts/bindings
83    *
84    * Node dependency ordering (ordinal and path):
85    *   0   /
86    *   1   /aliases
87   [...]
88    *   15  /soc/i2c@deadbeef
89   [...]
90
91Now look for this node in :file:`<build>/zephyr/zephyr.dts`, which is the final
92devicetree for your application build. (See :ref:`get-devicetree-outputs` for
93information and examples.)
94
95If the node has ``status = "disabled";`` in :file:`zephyr.dts`, then you need
96to enable it by setting ``status = "okay";``, probably by using a devicetree
97:ref:`overlay <set-devicetree-overlays>`. For example, if :file:`zephyr.dts`
98looks like this:
99
100.. code-block:: DTS
101
102   i2c0: i2c@deadbeef {
103           status = "disabled";
104   };
105
106Then you should put this into your devicetree overlay and
107:ref:`dt-trouble-try-pristine`:
108
109.. code-block:: DTS
110
111   &i2c0 {
112           status = "okay";
113   };
114
115Make sure that you see ``status = "okay";`` in :file:`zephyr.dts` after you
116rebuild.
117
118**Making sure the device driver is enabled**:
119
120The first step is to figure out which device driver is responsible for handling
121your devicetree node and allocating devices for it. To do this, you need to
122start with the ``compatible`` property in your devicetree node, and find the
123driver that allocates ``struct device`` instances for that compatible.
124
125If you're not familiar with how devices are allocated from devicetree nodes
126based on compatible properties, the ZDS 2021 talk `A deep dive into the Zephyr
1272.5 device model`_ may be a useful place to start, along with the
128:ref:`device_model_api` pages. See :ref:`dt-important-props` and the Devicetree
129specification for more information about ``compatible``.
130
131.. _A deep dive into the Zephyr 2.5 device model:
132   https://www.youtube.com/watch?v=sWaxQyIgEBY
133
134There is currently no documentation for what device drivers exist and which
135devicetree compatibles they are associated with. You will have to figure this
136out by reading the source code:
137
138- Look in :zephyr_file:`drivers` for the appropriate subdirectory that
139  corresponds to the API your device implements
140- Look inside that directory for relevant files until you figure out what the
141  driver is, or realize there is no such driver.
142
143Often, but not always, you can find the driver by looking for a file that sets
144the ``DT_DRV_COMPAT`` macro to match your node's ``compatible`` property,
145except lowercased and with special characters converted to underscores. For
146example, if your node's compatible is ``vnd,foo-device``, look for a file with this
147line:
148
149.. code-block:: C
150
151   #define DT_DRV_COMPAT vnd_foo_device
152
153.. important::
154
155   This **does not always work** since not all drivers use ``DT_DRV_COMPAT``.
156
157If you find a driver, you next need to make sure the Kconfig option that
158compiles it is enabled. (If you don't find a driver, and you are sure the
159compatible property is correct, then you need to write a driver. Writing
160drivers is outside the scope of this documentation page.)
161
162Continuing the above example, if your devicetree node looks like this now:
163
164.. code-block:: DTS
165
166   i2c0: i2c@deadbeef {
167           compatible = "nordic,nrf-twim";
168           status = "okay";
169   };
170
171Then you would look inside of :zephyr_file:`drivers/i2c` for the driver file
172that handles the compatible ``nordic,nrf-twim``. In this case, that is
173:zephyr_file:`drivers/i2c/i2c_nrfx_twim.c`. Notice how even in cases where
174``DT_DRV_COMPAT`` is not set, you can use information like driver file names as
175clues.
176
177Once you know the driver you want to enable, you need to make sure its Kconfig
178option is set to ``y``. You can figure out which Kconfig option is needed by
179looking for a line similar to this one in the :file:`CMakeLists.txt` file in
180the drivers subdirectory. Continuing the above example,
181:zephyr_file:`drivers/i2c/CMakeLists.txt` has a line that looks like this:
182
183.. code-block:: cmake
184
185   zephyr_library_sources_ifdef(CONFIG_NRFX_TWIM       i2c_nrfx_twim.c)
186
187This means that :kconfig:option:`CONFIG_NRFX_TWIM` must be set to ``y`` in
188:file:`<build>/zephyr/.config` file.
189
190If your driver's Kconfig is not set to ``y``, you need to figure out what you
191need to do to make that happen. Often, this will happen automatically as soon
192as you enable the devicetree node. Otherwise, it is sometimes as simple as
193adding a line like this to your application's :file:`prj.conf` file and then
194making sure to :ref:`dt-trouble-try-pristine`:
195
196.. code-block:: cfg
197
198   CONFIG_FOO=y
199
200where ``CONFIG_FOO`` is the option that :file:`CMakeLists.txt` uses to decide
201whether or not to compile the driver.
202
203However, there may be other problems in your way, such as unmet Kconfig
204dependencies that you also have to enable before you can enable your driver.
205
206Consult the Kconfig file that defines ``CONFIG_FOO`` (for your value of
207``FOO``) for more information.
208
209.. _dt-use-the-right-names:
210
211Make sure you're using the right names
212**************************************
213
214Remember that:
215
216- In C/C++, devicetree names must be lowercased and special characters must be
217  converted to underscores. Zephyr's generated devicetree header has DTS names
218  converted in this way into the C tokens used by the preprocessor-based
219  ``<devicetree.h>`` API.
220- In overlays, use devicetree node and property names the same way they
221  would appear in any DTS file. Zephyr overlays are just DTS fragments.
222
223For example, if you're trying to **get** the ``clock-frequency`` property of a
224node with path ``/soc/i2c@12340000`` in a C/C++ file:
225
226.. code-block:: c
227
228   /*
229    * foo.c: lowercase-and-underscores names
230    */
231
232   /* Don't do this: */
233   #define MY_CLOCK_FREQ DT_PROP(DT_PATH(soc, i2c@1234000), clock-frequency)
234   /*                                           ^               ^
235    *                                        @ should be _     - should be _  */
236
237   /* Do this instead: */
238   #define MY_CLOCK_FREQ DT_PROP(DT_PATH(soc, i2c_1234000), clock_frequency)
239   /*                                           ^               ^           */
240
241And if you're trying to **set** that property in a devicetree overlay:
242
243.. code-block:: none
244
245   /*
246    * foo.overlay: DTS names with special characters, etc.
247    */
248
249   /* Don't do this; you'll get devicetree errors. */
250   &{/soc/i2c_12340000/} {
251   	clock_frequency = <115200>;
252   };
253
254   /* Do this instead. Overlays are just DTS fragments. */
255   &{/soc/i2c@12340000/} {
256   	clock-frequency = <115200>;
257   };
258
259Look at the preprocessor output
260*******************************
261
262To save preprocessor output files, enable the
263:kconfig:option:`CONFIG_COMPILER_SAVE_TEMPS` option. For example, to build
264:zephyr:code-sample:`hello_world` with west with this option set, use:
265
266.. code-block:: sh
267
268   west build -b BOARD samples/hello_world -- -DCONFIG_COMPILER_SAVE_TEMPS=y
269
270This will create a preprocessor output file named :file:`foo.c.i` in the build
271directory for each source file :file:`foo.c`.
272
273You can then search for the file in the build directory to see what your
274devicetree macros expanded to. For example, on macOS and Linux, using ``find``
275to find :file:`main.c.i`:
276
277.. code-block:: sh
278
279   $ find build -name main.c.i
280   build/CMakeFiles/app.dir/src/main.c.i
281
282It's usually easiest to run a style formatter on the results before opening
283them. For example, to use ``clang-format`` to reformat the file in place:
284
285.. code-block:: sh
286
287   clang-format -i build/CMakeFiles/app.dir/src/main.c.i
288
289You can then open the file in your favorite editor to view the final C results
290after preprocessing.
291
292Do not track macro expansion
293****************************
294
295Compiler messages for devicetree errors can sometimes be very long. This
296typically happens when the compiler prints a message for every step of a
297complex macro expansion that has several intermediate expansion steps.
298
299To prevent the compiler from doing this, you can disable the
300:kconfig:option:`CONFIG_COMPILER_TRACK_MACRO_EXPANSION` option. This typically
301reduces the output to one message per error.
302
303For example, to build :zephyr:code-sample:`hello_world` with west and this option disabled,
304use:
305
306.. code-block:: sh
307
308   west build -b BOARD samples/hello_world -- -DCONFIG_COMPILER_TRACK_MACRO_EXPANSION=n
309
310Validate properties
311*******************
312
313If you're getting a compile error reading a node property, check your node
314identifier and property. For example, if you get a build error on a line that
315looks like this:
316
317.. code-block:: c
318
319   int baud_rate = DT_PROP(DT_NODELABEL(my_serial), current_speed);
320
321Try checking the node by adding this to the file and recompiling:
322
323.. code-block:: c
324
325   #if !DT_NODE_EXISTS(DT_NODELABEL(my_serial))
326   #error "whoops"
327   #endif
328
329If you see the "whoops" error message when you rebuild, the node identifier
330isn't referring to a valid node. :ref:`get-devicetree-outputs` and debug from
331there.
332
333Some hints for what to check next if you don't see the "whoops" error message:
334
335- did you :ref:`dt-use-the-right-names`?
336- does the :ref:`property exist <dt-checking-property-exists>`?
337- does the node have a :ref:`matching binding <dt-bindings>`?
338- does the binding define the property?
339
340.. _missing-dt-binding:
341
342Check for missing bindings
343**************************
344
345See :ref:`dt-bindings` for information about bindings, and
346:ref:`devicetree_binding_index` for information on bindings built into Zephyr.
347
348If the build fails to :ref:`dts-find-binding` for a node, then either the
349node's ``compatible`` property is not defined, or its value has no matching
350binding. If the property is set, check for typos in its name. In a devicetree
351source file, ``compatible`` should look like ``"vnd,some-device"`` --
352:ref:`dt-use-the-right-names`.
353
354If your binding file is not under :file:`zephyr/dts`, you may need to set
355:ref:`DTS_ROOT <dts_root>`; see :ref:`dt-where-bindings-are-located`.
356
357Errors with DT_INST_() APIs
358***************************
359
360If you're using an API like :c:func:`DT_INST_PROP`, you must define
361``DT_DRV_COMPAT`` to the lowercase-and-underscores version of the compatible
362you are interested in. See :ref:`dt-create-devices-inst`.
363