zephyr/doc/build/dts/phandles.rst
Marti Bolivar b141c72a2a doc: dts: add guide to phandles
Phandles, specifier spaces, and cell names are simultaneously
extremely common and woefully underdocumented. Address that by:

- reworking our existing documentation on these subjects in
  bindings-syntax.rst, fixing missing information in the
  property syntax template as well

- adding a standalone guide which describes how all the pieces
  fit together, providing a bridge for the gap between
  DTS/bindings and C APIs

My goal is not to eventually make this a comprehensive place
where *all* specifier spaces are documented. It would be better (more
scalable, more discoverable) to improve the individual API pages to
cover the devicetree-related conventions that apply in each case.
That's a problem for someone else and another day, but we do need a
few concrete references in the DTS guides to keep the motivation
clear.

Signed-off-by: Marti Bolivar <marti.bolivar@nordicsemi.no>
2023-01-23 07:16:04 -08:00

374 lines
12 KiB
ReStructuredText

.. _dt-phandles:
Phandles
########
The devicetree concept of a *phandle* is very similar to pointers in
C. You can use phandles to refer to nodes in devicetree similarly to the way
you can use pointers to refer to structures in C.
.. contents:: Contents
:local:
Getting phandles
****************
The usual way to get a phandle for a devicetree node is from one of its node
labels. For example, with this devicetree:
.. code-block:: DTS
/ {
lbl_a: node-1 {};
lbl_b: lbl_c: node-2 {};
};
You can write the phandle for:
- ``/node-1`` as ``&lbl_a``
- ``/node-2`` as either ``&lbl_b`` or ``&lbl_c``
Notice how the ``&nodelabel`` devicetree syntax is similar to the "address of"
C syntax.
Using phandles
**************
.. note::
"Type" in this section refers to one of the type names documented in
:ref:`dt-bindings-properties` in the devicetree bindings documentation.
Here are the main ways you will use phandles.
One node: phandle type
======================
You can use phandles to refer to ``node-b`` from ``node-a``, where ``node-b``
is related to ``node-a`` in some way.
One common example is when ``node-a`` represents some hardware that
generates an interrupt, and ``node-b`` represents the interrupt
controller that receives the asserted interrupt. In this case, you could
write:
.. code-block:: DTS
node_b: node-b {
interrupt-controller;
};
node-a {
interrupt-parent = <&node_b>;
};
This uses the standard ``interrupt-parent`` property defined in the
devicetree specification to capture the relationship between the two nodes.
These properties have type ``phandle``.
Zero or more nodes: phandles type
=================================
You can use phandles to make an array of references to other nodes.
One common example occurs in :ref:`pin control <pinctrl-guide>`. Pin control
properties like ``pinctrl-0``, ``pinctrl-1`` etc. may contain multiple
phandles, each of which "points" to a node containing information related to
pin configuration for that hardware peripheral. Here's an example of six
phandles in a single property:
.. code-block:: DTS
pinctrl-0 = <&quadspi_clk_pe10 &quadspi_ncs_pe11
&quadspi_bk1_io0_pe12 &quadspi_bk1_io1_pe13
&quadspi_bk1_io2_pe14 &quadspi_bk1_io3_pe15>;
These properties have type ``phandles``.
Zero or more nodes with metadata: phandle-array type
====================================================
You can use phandles to refer to and configure one or more resources that are
"owned" by some other node.
This is the most complex case. There are examples and more details in the
next section.
These properties have type ``phandle-array``.
.. _dt-phandle-arrays:
phandle-array properties
************************
These properties are commonly used to specify a resource that is owned by
another node along with additional metadata about the resource.
High level description
======================
Usually, properties with this type are written like ``phandle-array-prop`` in
this example:
.. code-block:: dts
node {
phandle-array-prop = <&foo 1 2>, <&bar 3>, <&baz 4 5>;
};
That is, the property's value is written as a comma-separated sequence of
"groups", where each "group" is written inside of angle brackets (``< ... >``).
Each "group" starts with a phandle (``&foo``, ``&bar``, ``&baz``). The values
that follow the phandle in each "group" are called *specifiers*. There are
three specifiers in the above example:
#. ``1 2``
#. ``3``
#. ``4 5``
The phandle in each "group" is used to "point" to the hardware that controls
the resource you are interested in. The specifier describes the resource
itself, along with any additional necessary metadata.
The rest of this section describes a common example. Subsequent sections
document more rules about how to use phandle-array properties in practice.
Example phandle-arrays: GPIOs
=============================
Perhaps the most common use case for phandle-array properties is specifying one
or more GPIOs on your SoC that another chip on your board connects to. For that
reason, we'll focus on that use case here. However, there are **many other use
cases** that are handled in devicetree with phandle-array properties.
For example, consider an external chip with an interrupt pin that is connected
to a GPIO on your SoC. You will typically need to provide that GPIO's
information (GPIO controller and pin number) to the :ref:`device driver
<device_model_api>` for that chip. You usually also need to provide other
metadata about the GPIO, like whether it is active low or high, what kind of
internal pull resistor within the SoC should be enabled in order to communicate
with the device, etc., to the driver.
In the devicetree, there will be a node that represents the GPIO controller
that controls a group of pins. This reflects the way GPIO IP blocks are usually
developed in hardware. Therefore, there is no single node in the devicetree
that represents a GPIO pin, and you can't use a single phandle to represent it.
Instead, you would use a phandle-array property, like this:
.. code-block::
my-external-ic {
irq-gpios = <&gpioX pin flags>;
};
In this example, ``irq-gpios`` is a phandle-array property with just one
"group" in its value. ``&gpioX`` is the phandle for the GPIO controller node
that controls the pin. ``pin`` is the pin number (0, 1, 2, ...). ``flags`` is a
bit mask describing pin metadata (for example ``(GPIO_ACTIVE_LOW |
GPIO_PULL_UP)``); see :zephyr_file:`include/zephyr/dt-bindings/gpio/gpio.h` for
more details.
The device driver handling the ``my-external-ic`` node can then use the
``irq-gpios`` property's value to set up interrupt handling for the chip as it
is used on your board. This lets you configure the device driver in devicetree,
without changing the driver's source code.
Such properties can contain multiple values as well:
.. code-block::
my-other-external-ic {
handshake-gpios = <&gpioX pinX flagsX>, <&gpioY pinY flagsY>;
};
The above example specifies two pins:
- ``pinX`` on the GPIO controller with phandle ``&gpioX``, flags ``flagsX``
- ``pinY`` on ``&gpioY``, flags ``flagsY``
You may be wondering how the "pin and flags" convention is established and
enforced. To answer this question, we'll need to introduce a concept called
specifier spaces before moving on to some information about devicetree
bindings.
.. _dt-specifier-spaces:
Specifier spaces
****************
*Specifier spaces* are a way to allow nodes to describe how you should
use them in phandle-array properties.
We'll start with an abstract, high level description of how specifier spaces
work in DTS files, before moving on to a concrete example and providing
references to further reading for how this all works in practice using DTS
files and bindings files.
High level description
======================
As described above, a phandle-array property is a sequence of "groups" of
phandles followed by some number of cells:
.. code-block:: dts
node {
phandle-array-prop = <&foo 1 2>, <&bar 3>;
};
The cells that follow each phandle are called a *specifier*. In this example,
there are two specifiers:
#. ``1 2``: two cells
#. ``3``: one cell
Every phandle-array property has an associated *specifier space*. This sounds
complex, but it's really just a way to assign a meaning to the cells that
follow each phandle in a hardware specific way. Every specifier space has a
unique name. There are a few "standard" names for commonly used hardware, but
you can create your own as well.
Devicetree nodes encode the number of cells that must appear in a specifier, by
name, using the ``#SPACE_NAME-cells`` property. For example, let's assume that
``phandle-array-prop``\ 's specifier space is named ``baz``. Then we would need
the ``foo`` and ``bar`` nodes to have the following ``#baz-cells`` properties:
.. code-block:: DTS
foo: node@1000 {
#baz-cells = <2>;
};
bar: node@2000 {
#baz-cells = <1>;
};
Without the ``#baz-cells`` property, the devicetree tooling would not be able
to validate the number of cells in each specifier in ``phandle-array-prop``.
This flexibility allows you to write down an array of hardware resources in a
single devicetree property, even though the amount of metadata you need to
describe each resource might be different for different nodes.
A single node can also have different numbers of cells in different specifier
spaces. For example, we might have:
.. code-block:: DTS
foo: node@1000 {
#baz-cells = <2>;
#bob-cells = <1>;
};
With that, if ``phandle-array-prop-2`` has specifier space ``bob``, we could
write:
.. code-block:: DTS
node {
phandle-array-prop = <&foo 1 2>, <&bar 3>;
phandle-array-prop-2 = <&foo 4>;
};
This flexibility allows you to have a node that manages multiple different
kinds of resources at the same time. The node describes the amount of metadata
needed to describe each kind of resource (how many cells are needed in each
case) using different ``#SPACE_NAME-cells`` properties.
Example specifier space: gpio
=============================
From the above example, you're already familiar with how one specifier space
works: in the "gpio" space, specifiers almost always have two cells:
#. a pin number
#. a bit mask of flags related to the pin
Therefore, almost all GPIO controller nodes you will see in practice will look
like this:
.. code-block:: DTS
gpioX: gpio-controller@deadbeef {
gpio-controller;
#gpio-cells = <2>;
};
Associating properties with specifier spaces
********************************************
Above, we have described that:
- each phandle-array property has an associated specifier space
- specifier spaces are identified by name
- devicetree nodes use ``#SPECIFIER_NAME-cells`` properties to
configure the number of cells which must appear in a specifier
In this section, we explain how phandle-array properties get their specifier
spaces.
High level description
======================
In general, a ``phandle-array`` property named ``foos`` implicitly has
specifier space ``foo``. For example:
.. code-block:: YAML
properties:
dmas:
type: phandle-array
pwms:
type: phandle-array
The ``dmas`` property's specifier space is "dma". The ``pwm`` property's
specifier space is ``pwm``.
Special case: GPIO
==================
``*-gpios`` properties are special-cased so that e.g. ``foo-gpios`` resolves to
``#gpio-cells`` rather than ``#foo-gpio-cells``.
Manually specifying a space
===========================
You can manually specify the specifier space for any ``phandle-array``
property. See :ref:`dt-bindings-specifier-space`.
Naming the cells in a specifier
*******************************
You should name the cells in each specifier space your hardware supports when
writing bindings. For details on how to do this, see :ref:`dt-bindings-cells`.
This allows C code to query information about and retrieve the values of cells
in a specifier by name using devicetree APIs like these:
- :c:macro:`DT_PHA_BY_IDX`
- :c:macro:`DT_PHA_BY_NAME`
This feature and these macros are used internally by numerous hardware-specific
APIs. Here are a few examples:
- :c:macro:`DT_GPIO_PIN_BY_IDX`
- :c:macro:`DT_PWMS_CHANNEL_BY_IDX`
- :c:macro:`DT_DMAS_CELL_BY_NAME`
- :c:macro:`DT_IO_CHANNELS_INPUT_BY_IDX`
- :c:macro:`DT_CLOCKS_CELL_BY_NAME`
See also
********
- :ref:`dt-writing-property-values`: how to write phandles in devicetree
properties
- :ref:`dt-bindings-properties`: how to write bindings for properties with
phandle types (``phandle``, ``phandles``, ``phandle-array``)
- :ref:`dt-bindings-specifier-space`: how to manually specify a phandle-array
property's specifier space