|  |  | 
Serial Peripheral Interface (SPI) Driver Interface.
The Serial Peripheral Interface (SPI) driver is a generic, full-duplex driver that transmits and receives data on a SPI bus. SPI is sometimes called SSI (Synchronous Serial Interface). The SPI protocol defines the format of a data transfer over the SPI bus, but it leaves flow control, data formatting, and handshaking mechanisms to higher-level software layers.
The SPI driver operates on some key definitions and assumptions:
To use the SPI driver to send data over the SPI bus, the application calls the following APIs:
The following code example opens a SPI instance as a SPI controller, and issues a transaction.
More details on usage are provided in the following subsections.
SPI_init() must be called before any other SPI APIs. This function iterates through the elements of the SPI_config[] array, calling the element's device implementation SPI initialization function.
After initializing the SPI driver by calling SPI_init(), the application can open a SPI instance by calling SPI_open(). This function takes an index into the SPI_config[] array, and a SPI parameters data structure. The SPI instance is specified by the index of the SPI in SPI_config[]. Calling SPI_open() a second time with the same index previously passed to SPI_open() will result in an error. You can, though, re-use the index if the instance is closed via SPI_close().
If no SPI_Params structure is passed to SPI_open(), default values are used. If the open call is successful, it returns a non-NULL value.
Example opening a SPI driver instance in blocking mode:
Example opening a SPI driver instance in callback mode:
The SPI_Params structure is passed to the SPI_open() call. If NULL is passed for the parameters, SPI_open() uses default parameters. A SPI_Params structure is initialized with default values by passing it to SPI_Params_init(). Some of the SPI parameters are described below. To see brief descriptions of all the parameters, see SPI_Params.
The SPI driver operates in both SPI controller and SPI peripheral modes. Logically, the implementation is identical, however the difference between these two modes is driven by hardware. The default mode is SPI_CONTROLLER, but can be set to peripheral mode by setting SPI_Params.mode to SPI_PERIPHERAL in the parameters passed to SPI_open(). See Controller/Peripheral Modes for further details.
The SPI driver supports two transfer modes of operation: blocking and callback. The transfer mode is determined by the transferMode parameter in the SPI_Params data structure. The SPI driver defaults to blocking mode, if the application does not set it. Once a SPI driver is opened, the only way to change the operation mode is to close and re-open the SPI instance with the new transfer mode.
In blocking mode, a task's code execution is blocked until a SPI transaction has completed or a timeout has occurred. This ensures that only one SPI transfer operates at a given time. Other tasks requesting SPI transfers while a transfer is currently taking place will receive a FALSE return value. If a timeout occurs the transfer is canceled, the task is unblocked & will receive a FALSE return value. The transaction count field will have the amount of frames which were transferred successfully before the timeout. In blocking mode, transfers cannot be performed in software or hardware ISR context.
In callback mode, a SPI transaction functions asynchronously, which means that it does not block code execution. After a SPI transaction has been completed, the SPI driver calls a user-provided hook function. Callback mode is supported in the execution context of tasks and hardware interrupt routines.
The SPI driver can configure the device's SPI peripheral to transfer data in several SPI format options: SPI (with various polarity and phase settings), TI, and Micro-wire. The frame format is set with SPI_Params.frameFormat. Some SPI implementations may not support all frame formats & the SPI driver will fail to opened. Refer to the device specific implementation documentation for details on which frame formats are supported.
The smallest single unit of data transmitted onto the SPI bus is called a SPI frame and is of size SPI_Params.dataSize. A series of SPI frames transmitted/received on a SPI bus is referred to as a SPI transaction.
A SPI transaction consists of a series of SPI frames transmitted/received on a SPI bus. A SPI transaction is performed using SPI_transfer(). SPI_transfer() accepts a pointer to a SPI_Transaction structure that dictates the quantity of data to be sent and received. The SPI_Transaction.txBuf and SPI_Transaction.rxBuf are both pointers to data buffers. If txBuf is NULL, the driver sends SPI frames with all data set to the default value specified in the hardware attributes. If rxBuf is NULL, the driver discards all SPI frames received. SPI_transfer() of a SPI transaction is performed atomically.
When the SPI is opened, the dataSize value determines the element types of txBuf and rxBuf. If the dataSize is from 4 to 8 bits, the driver assumes the data buffers are of type uint8_t (unsigned char). If the dataSize is from 8 to 16 bits, the driver assumes the data buffers are of type uint16_t (unsigned short). If the dataSize is greater than 16 bits, the driver assumes the data buffers are uint32_t (unsigned long). Some SPI driver implementations may not support all data sizes; refer to device specific SPI implementation documentation for details on what data sizes are supported.
The optional SPI_Transaction.arg variable can only be used when the SPI driver has been opened in callback mode. This variable is used to pass a user-defined value into the user-defined callback function.
SPI_transfer() always performs full-duplex SPI transactions. This means the SPI simultaneously receives data as it transmits data. The application is responsible for formatting the data to be transmitted as well as determining whether the data received is meaningful. Specifics about SPI frame formatting and data sizes are provided in device-specific data sheets and technical reference manuals.
The following code snippets perform SPI transactions.
Example transferring 6-bit SPI frames. The transmit and receive buffers are of type uint8_t.
Example transferring 12-bit SPI frames. The transmit and receive buffers are of type uint16_t.
The following example shows an example of a callback function that utilizes the arg parameter.
SPI_transferCancel() is used to cancel a SPI transaction when the driver is used in SPI_MODE_CALLBACK mode.
Calling this API while no transfer is in progress has no effect. If a transfer is in progress, it is canceled and the callback functions is called. The SPI_Status status field in the SPI_Transaction structure can be examined within the callback to determine if the transaction succeeded.
Example:
This SPI driver functions in both SPI controller and SPI peripheral modes. Logically, the implementation is identical, however the difference between these two modes is driven by hardware. As a SPI controller, the peripheral is in control of the clock signal and therefore will commence communications to the SPI peripheral immediately. As a SPI peripheral, the SPI driver prepares the peripheral to transmit and receive data in a way such that the peripheral is ready to transfer data when the SPI controller initiates a transaction.
The SPI protocol requires that the SPI controller asserts a SPI peripheral's chip select pin prior to starting a SPI transaction. While this protocol is generally followed, various types of SPI peripherals have different timing requirements as to when and for how long the chip select pin must remain asserted for a SPI transaction.
Commonly, the SPI controller uses a hardware chip select to assert and de-assert the SPI peripheral for every data frame. In other cases, a SPI peripheral imposes the requirement of asserting the chip select over several SPI data frames. This is generally accomplished by using a regular, general-purpose output pin. Due to the complexity of such SPI peripheral implementations, this SPI driver has been designed to operate transparently to the SPI chip select. When the hardware chip select is used, the peripheral automatically selects/enables the peripheral. When using a software chip select, the application needs to handle the proper chip select and pin configuration. Chip select support will vary per SPI peripheral, refer to the device specific implementation documentation for details on chip select support.
In order to use the SPI APIs, the application is required to provide device-specific SPI configuration in the ti_drivers_config.c file. The SPI driver interface defines a configuration data structure:
The application must declare an array of SPI_Config elements, named SPI_config[]. Each element of SPI_config[] must be populated with pointers to a device specific SPI driver implementation's function table, driver object, and hardware attributes. The hardware attributes define properties such as the SPI peripheral's base address, and the PICO and POCI pins. Each element in SPI_config[] corresponds to a SPI instance, and none of the elements should have NULL pointers. There is no correlation between the index and the peripheral designation (such as SPI0 or SPI1). For example, it is possible to use SPI_config[0] for SPI1.
Because the SPI configuration is very device dependent, you will need to check the doxygen for the device specific SPI implementation. There you will find a description of the SPI hardware attributes. Please also refer to the ti_drivers_config.c file of any of your examples to see the SPI configuration.
#include <stdbool.h>#include <stddef.h>#include <stdint.h>

Go to the source code of this file.
| Data Structures | |
| struct | SPI_Transaction | 
| A SPI_Transaction data structure is used with SPI_transfer(). It indicates how many SPI_FrameFormat frames are sent and received from the buffers pointed to txBuf and rxBuf. The arg variable is an user-definable argument which gets passed to the SPI_CallbackFxn when the SPI driver is in SPI_MODE_CALLBACK.  More... | |
| struct | SPI_Params | 
| SPI Parameters.  More... | |
| struct | SPI_FxnTable | 
| The definition of a SPI function table that contains the required set of functions to control a specific SPI driver implementation.  More... | |
| struct | SPI_Config_ | 
| SPI Global configuration.  More... | |
| Macros | |
| #define | SPI_CMD_RESERVED (32) | 
| #define | SPI_STATUS_RESERVED (-32) | 
| #define | SPI_STATUS_SUCCESS (0) | 
| Successful status code returned by SPI_control().  More... | |
| #define | SPI_STATUS_ERROR (-1) | 
| Generic error status code returned by SPI_control().  More... | |
| #define | SPI_STATUS_UNDEFINEDCMD (-2) | 
| An error status code returned by SPI_control() for undefined command codes.  More... | |
| #define | SPI_WAIT_FOREVER (~(0U)) | 
| Wait forever define used to specify timeouts.  More... | |
| Typedefs | |
| typedef struct SPI_Config_ * | SPI_Handle | 
| A handle that is returned from a SPI_open() call.  More... | |
| typedef void(* | SPI_CallbackFxn) (SPI_Handle handle, SPI_Transaction *transaction) | 
| The definition of a callback function used by the SPI driver when used in SPI_MODE_CALLBACK.  More... | |
| typedef void(* | SPI_CloseFxn) (SPI_Handle handle) | 
| A function pointer to a driver specific implementation of SPI_close().  More... | |
| typedef int_fast16_t(* | SPI_ControlFxn) (SPI_Handle handle, uint_fast16_t cmd, void *arg) | 
| A function pointer to a driver specific implementation of SPI_control().  More... | |
| typedef void(* | SPI_InitFxn) (SPI_Handle handle) | 
| A function pointer to a driver specific implementation of SPI_init().  More... | |
| typedef SPI_Handle(* | SPI_OpenFxn) (SPI_Handle handle, SPI_Params *params) | 
| A function pointer to a driver specific implementation of SPI_open().  More... | |
| typedef bool(* | SPI_TransferFxn) (SPI_Handle handle, SPI_Transaction *transaction) | 
| A function pointer to a driver specific implementation of SPI_transfer().  More... | |
| typedef void(* | SPI_TransferCancelFxn) (SPI_Handle handle) | 
| A function pointer to a driver specific implementation of SPI_transferCancel().  More... | |
| typedef struct SPI_Config_ | SPI_Config | 
| SPI Global configuration.  More... | |
| Enumerations | |
| enum | SPI_Status { SPI_TRANSFER_COMPLETED = 0, SPI_TRANSFER_STARTED, SPI_TRANSFER_CANCELED, SPI_TRANSFER_FAILED, SPI_TRANSFER_CSN_DEASSERT, SPI_TRANSFER_PEND_CSN_ASSERT, SPI_TRANSFER_QUEUED } | 
| Status codes that are set by the SPI driver.  More... | |
| enum | SPI_Mode { SPI_CONTROLLER = 0, SPI_PERIPHERAL = 1 } | 
| Definitions for various SPI modes of operation.  More... | |
| enum | SPI_FrameFormat { SPI_POL0_PHA0 = 0, SPI_POL0_PHA1 = 1, SPI_POL1_PHA0 = 2, SPI_POL1_PHA1 = 3, SPI_TI = 4, SPI_MW = 5 } | 
| Definitions for various SPI data frame formats.  More... | |
| enum | SPI_TransferMode { SPI_MODE_BLOCKING, SPI_MODE_CALLBACK } | 
| SPI transfer mode determines the whether the SPI controller operates synchronously or asynchronously. In SPI_MODE_BLOCKING mode SPI_transfer() blocks code execution until the SPI transaction has completed. In SPI_MODE_CALLBACK SPI_transfer() does not block code execution and instead calls a SPI_CallbackFxn callback function when the transaction has completed (successfully or not).  More... | |
| Functions | |
| void | SPI_close (SPI_Handle handle) | 
| Function to close a SPI peripheral specified by the SPI handle.  More... | |
| int_fast16_t | SPI_control (SPI_Handle handle, uint_fast16_t cmd, void *controlArg) | 
| Function performs implementation specific features on a given SPI_Handle.  More... | |
| void | SPI_init (void) | 
| This function initializes the SPI module.  More... | |
| SPI_Handle | SPI_open (uint_least8_t index, SPI_Params *params) | 
| This function opens a given SPI peripheral.  More... | |
| void | SPI_Params_init (SPI_Params *params) | 
| Function to initialize the SPI_Params struct to its defaults.  More... | |
| bool | SPI_transfer (SPI_Handle handle, SPI_Transaction *transaction) | 
| Function to perform SPI transactions.  More... | |
| void | SPI_transferCancel (SPI_Handle handle) | 
| Function to cancel SPI transactions.  More... | |
| #define SPI_WAIT_FOREVER (~(0U)) | 
Wait forever define used to specify timeouts.
| typedef struct SPI_Config_* SPI_Handle | 
A handle that is returned from a SPI_open() call.
| typedef void(* SPI_CallbackFxn) (SPI_Handle handle, SPI_Transaction *transaction) | 
The definition of a callback function used by the SPI driver when used in SPI_MODE_CALLBACK.
| SPI_Handle | A SPI_Handle | 
| SPI_Transaction* | Pointer to a SPI_Transaction | 
| typedef void(* SPI_CloseFxn) (SPI_Handle handle) | 
A function pointer to a driver specific implementation of SPI_close().
| typedef int_fast16_t(* SPI_ControlFxn) (SPI_Handle handle, uint_fast16_t cmd, void *arg) | 
A function pointer to a driver specific implementation of SPI_control().
| typedef void(* SPI_InitFxn) (SPI_Handle handle) | 
A function pointer to a driver specific implementation of SPI_init().
| typedef SPI_Handle(* SPI_OpenFxn) (SPI_Handle handle, SPI_Params *params) | 
A function pointer to a driver specific implementation of SPI_open().
| typedef bool(* SPI_TransferFxn) (SPI_Handle handle, SPI_Transaction *transaction) | 
A function pointer to a driver specific implementation of SPI_transfer().
| typedef void(* SPI_TransferCancelFxn) (SPI_Handle handle) | 
A function pointer to a driver specific implementation of SPI_transferCancel().
| typedef struct SPI_Config_ SPI_Config | 
SPI Global configuration.
The SPI_Config structure contains a set of pointers used to characterize the SPI driver implementation.
This structure needs to be defined before calling SPI_init() and it must not be changed thereafter.
| enum SPI_Status | 
Status codes that are set by the SPI driver.
| enum SPI_Mode | 
| enum SPI_FrameFormat | 
Definitions for various SPI data frame formats.
| enum SPI_TransferMode | 
SPI transfer mode determines the whether the SPI controller operates synchronously or asynchronously. In SPI_MODE_BLOCKING mode SPI_transfer() blocks code execution until the SPI transaction has completed. In SPI_MODE_CALLBACK SPI_transfer() does not block code execution and instead calls a SPI_CallbackFxn callback function when the transaction has completed (successfully or not).
| Enumerator | |
|---|---|
| SPI_MODE_BLOCKING | SPI_transfer() blocks execution. This mode can only be used when called within a Task context | 
| SPI_MODE_CALLBACK | SPI_transfer() does not block code execution and will call a SPI_CallbackFxn. This mode can be used in a Task, software or hardware interrupt context. | 
| void SPI_close | ( | SPI_Handle | handle | ) | 
Function to close a SPI peripheral specified by the SPI handle.
| handle | A SPI_Handle returned from SPI_open() | 
| int_fast16_t SPI_control | ( | SPI_Handle | handle, | 
| uint_fast16_t | cmd, | ||
| void * | controlArg | ||
| ) | 
Function performs implementation specific features on a given SPI_Handle.
Commands for SPI_control can originate from SPI.h or from implementation specific SPI*.h (SPICC26X2DMA.h, SPIMSP432DMA.h, etc.. ) files. While commands from SPI.h are API portable across driver implementations, not all implementations may support all these commands. Conversely, commands from driver implementation specific SPI*.h files add unique driver capabilities but are not API portable across all SPI driver implementations.
Commands supported by SPI.h follow a SPI_CMD_<cmd> naming convention.
 Commands supported by SPI*.h follow a SPI*_CMD_<cmd> naming convention.
 Each control command defines arg differently. The types of arg are documented with each command.
See SPI_control command codes for command codes.
See SPI_control return status codes for status codes.
| handle | A SPI_Handle returned from SPI_open() | 
| cmd | SPI.h or SPI*.h commands. | 
| controlArg | An optional R/W (read/write) command argument accompanied with cmd | 
| void SPI_init | ( | void | ) | 
This function initializes the SPI module.
| SPI_Handle SPI_open | ( | uint_least8_t | index, | 
| SPI_Params * | params | ||
| ) | 
This function opens a given SPI peripheral.
| index | Index of config to use in the SPI_config array | 
| params | Pointer to an parameter block, if NULL it will use default values. All the fields in this structure are RO (read-only). | 
| void SPI_Params_init | ( | SPI_Params * | params | ) | 
Function to initialize the SPI_Params struct to its defaults.
| params | An pointer to SPI_Params structure for initialization | 
Defaults values are:
| bool SPI_transfer | ( | SPI_Handle | handle, | 
| SPI_Transaction * | transaction | ||
| ) | 
Function to perform SPI transactions.
If the SPI is in SPI_CONTROLLER mode, it will immediately start the transaction. If the SPI is in SPI_PERIPHERAL mode, it prepares the driver for a transaction with a SPI controller device. The device will then wait until the controller begins the transfer.
In SPI_MODE_BLOCKING, SPI_transfer() will block task execution until the transaction has completed or a timeout has occurred.
In SPI_MODE_CALLBACK, SPI_transfer() does not block task execution, but calls a SPI_CallbackFxn once the transfer has finished. This makes SPI_tranfer() safe to be used within a Task, software or hardware interrupt context. If queued transactions are supported SPI_Transfer may be called multiple times to queue multiple transactions. If the driver does not support this functionality additional calls will return false. Refer to device specific SPI driver documentation for support information.
From calling SPI_transfer() until transfer completion, the SPI_Transaction structure must stay persistent and must not be altered by application code. It is also forbidden to modify the content of the SPI_Transaction.txBuf during a transaction, even though the physical transfer might not have started yet. Doing this can result in data corruption. This is especially important for peripheral operations where SPI_transfer() might be called a long time before the actual data transfer begins.
| handle | A SPI_Handle | 
| transaction | A pointer to a SPI_Transaction. All of the fields within transaction except SPI_Transaction.count and SPI_Transaction.status are WO (write-only) unless otherwise noted in the driver implementations. If a transaction timeout has occurred, SPI_Transaction.count will contain the number of frames that were transferred. Neither is it allowed to modify the transaction object nor the content of SPI_Transaction.txBuf until the transfer has completed. | 
true if started successfully; else false | void SPI_transferCancel | ( | SPI_Handle | handle | ) | 
Function to cancel SPI transactions.
In SPI_MODE_BLOCKING, SPI_transferCancel has no effect.
In SPI_MODE_CALLBACK, SPI_transferCancel() will stop an SPI transfer if if one is in progress. If a transaction was in progress, its callback function will be called in context from which this API is called from. The SPI_CallbackFxn function can determine if the transaction was successful or not by reading the SPI_Status status value in the SPI_Transaction structure.
| handle | A SPI_Handle |