blob: 176c9bf5ede3563a98002dcd21008c57befe57ee [file] [log] [blame]
.. _mspi_api:
Multi-bit SPI Bus
#################
The MSPI (multi-bit SPI) is provided as a generic API to accommodate
advanced SPI peripherals and devices that typically require command,
address and data phases, and multiple signal lines during these phases.
While the API supports advanced features such as :term:`XIP` and scrambling,
it is also compatible with generic SPI.
.. contents::
:local:
:depth: 2
.. _mspi-controller-api:
MSPI Controller API
*******************
Zephyr's MSPI controller API may be used when a multi-bit SPI controller
is present. E.g. Ambiq MSPI, QSPI, OSPI, Flexspi, etc.
The API supports single to hex SDR/DDR IO with variable latency and advanced
features such as :term:`XIP` and scrambling. Applicable devices include but
not limited to high-speed, high density flash/psram memory devices, displays
and sensors.
The MSPI interface contains controller drivers that are SoC platform specific
and implement the MSPI APIs, and device drivers that reference these APIs.
The relationship between the controller and device drivers is many-to-many to
allow for easy switching between platforms.
Here is a list of generic steps for initializing the MSPI controller and the
MSPI bus inside the device driver initialization function:
#. Initialize the data structure of the MSPI controller driver instance.
The usual device defining macros such as :c:macro:`DEVICE_DT_INST_DEFINE`
can be used, and the initialization function, config and data provided
as a parameter to the macro.
#. Initialize the hardware, including but not limited to:
* Check :c:struct:`mspi_cfg` against hardware's own capabilities to prevent
incorrect usages.
* Setup default pinmux.
* Setup the clock for the controller.
* Power on the hardware.
* Configure the hardware using :c:struct:`mspi_cfg` and possibly more
platform specific settings.
* Usually, the :c:struct:`mspi_cfg` is filled from device tree and contains
static, boot time parameters. However, if needed, one can use :c:func:`mspi_config`
to re-initialize the hardware with new parameters during runtime.
* Release any lock if applicable.
#. Perform device driver initialization. As usually, :c:macro:`DEVICE_DT_INST_DEFINE`
can be used. Inside device driver initialization function, perform the following
required steps.
#. Call :c:func:`mspi_dev_config` with device specific hardware settings obtained
from device datasheets.
* The :c:struct:`mspi_dev_cfg` should be filled by device tree and helper macro
:c:macro:`MSPI_DEVICE_CONFIG_DT` can be used.
* The controller driver should then validate the members of :c:struct:`mspi_dev_cfg`
to prevent incorrect usage.
* The controller driver should implement a mutex to protect from accidental access.
* The controller driver may also switch between different devices based on
:c:struct:`mspi_dev_id`.
#. Call API for additional setups if supported by hardware
* :c:func:`mspi_xip_config` for :term:`XIP` feature
* :c:func:`mspi_scramble_config` for scrambling feature
* :c:func:`mspi_timing_config` for platform specific timing setup.
#. Register any callback with :c:func:`mspi_register_callback` if needed.
#. Release the controller mutex lock.
Transceive
==========
The transceive request is of type :c:struct:`mspi_xfer` which allows dynamic change to
the transfer related settings once the mode of operation is determined and configured
by :c:func:`mspi_dev_config`.
The API also supports bulk transfers with different starting addresses and sizes with
:c:struct:`mspi_xfer_packet`. However, it is up to the controller implementation
whether to support scatter IO and callback management. The controller can determine
which user callback to trigger based on :c:enum:`mspi_bus_event_cb_mask` upon completion
of each async/sync transfer if the callback had been registered using
:c:func:`mspi_register_callback`. Or not to trigger any callback at all with
:c:enum:`MSPI_BUS_NO_CB` even if the callbacks are already registered.
In which case that a controller supports hardware command queue, user could take full
advantage of the hardware performance if scatter IO and callback management are supported
by the driver implementation.
Device Tree
===========
Here is an example for defining an MSPI controller in device tree:
The mspi controller's bindings should reference mspi-controller.yaml as one of the base.
.. code-block:: devicetree
mspi0: mspi@400 {
status = "okay";
compatible = "zephyr,mspi-emul-controller";
reg = < 0x400 0x4 >;
#address-cells = < 0x1 >;
#size-cells = < 0x0 >;
clock-frequency = < 0x17d7840 >;
op-mode = "MSPI_CONTROLLER";
duplex = "MSPI_HALF_DUPLEX";
ce-gpios = < &gpio0 0x5 0x1 >, < &gpio0 0x12 0x1 >;
dqs-support;
pinctrl-0 = < &pinmux-mspi0 >;
pinctrl-names = "default";
};
Here is an example for defining an MSPI device in device tree:
The mspi device's bindings should reference mspi-device.yaml as one of the base.
.. code-block:: devicetree
&mspi0 {
mspi_dev0: mspi_dev0@0 {
status = "okay";
compatible = "zephyr,mspi-emul-device";
reg = < 0x0 >;
size = < 0x10000 >;
mspi-max-frequency = < 0x2dc6c00 >;
mspi-io-mode = "MSPI_IO_MODE_QUAD";
mspi-data-rate = "MSPI_DATA_RATE_SINGLE";
mspi-hardware-ce-num = < 0x0 >;
read-instruction = < 0xb >;
write-instruction = < 0x2 >;
instruction-length = "INSTR_1_BYTE";
address-length = "ADDR_4_BYTE";
rx-dummy = < 0x8 >;
tx-dummy = < 0x0 >;
xip-config = < 0x0 0x0 0x0 0x0 >;
ce-break-config = < 0x0 0x0 >;
};
};
User should specify target operating parameters in the DTS such as ``mspi-max-frequency``,
``mspi-io-mode`` and ``mspi-data-rate`` even though they may subject to change during runtime.
It should represent the typical configuration of the device during normal operations.
Multi Peripheral
================
With :c:struct:`mspi_dev_id` defined as collection of the device index and CE GPIO from
device tree, the API supports multiple devices on the same controller instance.
The controller driver implementation may or may not support device switching,
which can be performed either by software or by hardware. If the switching is handled
by software, it should be performed in :c:func:`mspi_dev_config` call.
The device driver should record the current operating conditions of the device to support
software controlled device switching by saving and updating :c:struct:`mspi_dev_cfg` and
other relevant mspi struct or private data structures. In particular, :c:struct:`mspi_dev_id`
which contains the identity of the device needs to be used for every API call.
Configuration Options
*********************
Related configuration options:
* :kconfig:option:`CONFIG_MSPI`
* :kconfig:option:`CONFIG_MSPI_ASYNC`
* :kconfig:option:`CONFIG_MSPI_PERIPHERAL`
* :kconfig:option:`CONFIG_MSPI_XIP`
* :kconfig:option:`CONFIG_MSPI_SCRAMBLE`
* :kconfig:option:`CONFIG_MSPI_TIMING`
* :kconfig:option:`CONFIG_MSPI_INIT_PRIORITY`
* :kconfig:option:`CONFIG_MSPI_COMPLETION_TIMEOUT_TOLERANCE`
API Reference
*************
.. doxygengroup:: mspi_interface