Serial Peripheral Interface Bus (SPI)

The Serial Peripheral Interface bus is a synchronous serial communication interface specification used for short distance communication primary in embedded systems. The interface was originally developed by Motorola and can be used in a master or slave configuration. It is typically used as a 3-wire bus containing the following signals (other than power and ground and chip-selects):

  • SCLK
  • MISO
  • MOSI
  • SS# (Optional Slave Select)

SPI can be a multi-slave bus if chip selects are used which are asserted by the host controller to enable one device at a time.


On-Board SPI controllers

Newport / Ventana

Various Gateworks boards support 3.3V TTL SPI to an off-board expansion connector with a single CS#:

Family Board Device Tree Node Connector
Newport GW630x spi_7_0 J8
GW640x spi_7_0 J8
Ventana GW552x ecspi3 J24
GW54xx-E+ ecspi2 J32

Pinout (all signals are 3.3V TTL):

  • 1 - MOSI
  • 2 - MISO
  • 3 - SCLK
  • 4 - SS0#
  • 5 - 3.3V
  • GND


While SPI is used for on-board FLASH storage for many Laguna boards, it also is brought out to the J9 expansion connector available on the Laguna GW2388:

  • J9.2 - CS#
  • J9.4 - SCLK
  • J9.6 - MISO
  • J9.8 - MOSI
  • J9.10 - GND

For details on adding kernel devices to the Laguna kernel see below

USB SPI controllers

A SPI master can also be added via USB expansion. For example:

Using a Linux SPI kernel driver

device-tree: Newport / Ventana

Linux kernels that utilize the device-tree need to add a device-tree node to the SPI controller to use a kernel driver. Note that you can also access SPI from userspace using spidev (see below)

An example device-tree child node for a m25p32-vme6g nor flash device would be:

flash: m25p80@0 {                         /* "m25p80@0" parameter is the driver responsible for controlling your device */
        compatible = "m25p32";            /* "m25p32" parameter is your device ID string */
        spi-max-frequency = <30000000>;   /* assigned value is your device frequency specified by its datasheet (in decimal) */
        reg = <0>;                        /* is generally 0 for single device connections */
        #device-width = <1>;              /* These next three lines are not comments, but rather specially formatted
        #address-cells = <1>;              * attributes that the driver will receive. The actual values assigned here
        #size-cells = <1>;                 * are device specific, and should be entered after consulting the data sheet
        partition@0 {                     /* The label of the 0th partition; following partitions follow the "partition@X" format */
                label = "data";           /* The name of this partition  */
                reg = <0x0 0x2000000>;    /* The starting address, followed by the length of the partition (in hex) */

Note that this node must appear within the SPI host controller. For a list of device-tree nodes see above.

Note that the values shown in the above node are specific to the m25p32-vme6g, and can vary greatly from the actual device you are using. Replace attribute values as necessary.

When searching for your device ID string or its controlling driver, searching via a Linux LXR site like the one at missing link electronics for your device name can be helpful.

non-device-tree: Laguna

Linux kernels that do not utilize device-tree need to register a spi_board_info struct with the kernel via spi_register_board_info().

To add a SPI device to Laguna you would add it to the spi_board_info array in laguna.c. Be sure to specify chip_select=1 to use CS1 as CS0 is used for the on-board SPI FLASH device.


Linux spidev userspace API

SPI devices have a limited userspace API, supporting basic half-duplex read() and write() access to SPI slave devices referred to as spidev. Using ioctl() requests, full duplex transfers and device I/O configuration are also available.

Some reasons you might want to use this programming interface include:

  • Prototyping in an environment that's not crash-prone; stray pointers in userspace won't normally bring down any Linux system.
  • Developing simple protocols used to talk to micro-controllers acting as SPI slaves, which you may need to change quite often.

Of course there are drivers that can never be written in userspace, because they need to access kernel interfaces (such as IRQ handlers or other layers of the driver stack) that are not accessible to userspace.

Userspace access to SPI devices is done through the /dev/spidev<bus>.<chip-select> device interface. In order to use this you must have spidev enabled in the kernel (CONFIG_SPI_SPIDEV) and have a spidev node defined under the SPI controller in the device-tree.

In order to support spidev a spidev child node needs to be present in the device-tree under the SPI host controller. For example, a GW54xx which brings out the ecspi2 host interface:

&ecspi2 {
        cs-gpios = <&gpio2 26 GPIO_ACTIVE_HIGH>;
        pinctrl-names = "default";
        pinctrl-0 = <&pinctrl_ecspi2>;
        status = "okay";

        spidev0: spidev@0 {
                compatible = "rohm,dh2228fv";
                reg = <0>;
                spi-max-frequency = <60000000>;

For instructions on compiling and updating a device-tree see linux/devicetree

An application using spidev would include the <linux/spi/spidev.h> header file.

For more info see:

Last modified 4 months ago Last modified on 11/30/2017 02:22:37 PM

Attachments (2)

Download all attachments as: .zip