SPI Device API
To create an SPI device, first call spi_init, passing in an spi_config_t struct. This struct defines the clock and MOSI/MISO pins, the SPI mode, and the done callback.
spi_dev_t spi_init(spi_config_t *config)
Initializes an SPI device interface. The config argument defines the pins, mode, and callbacks for the SPI device. It contains the following fields:
| Field | Type | Description |
|---|---|---|
sck | pin_t | The clock pin |
mosi | pin_t | The MOSI data pin (or NO_PIN to disable MOSI) |
miso | pin_t | The MISO data pin (or NO_PIN to disable MISO) |
mode | uint32_t | SPI mode: 0, 1, 2, or 3 (default: 0) |
done | callback | Called when an SPI transaction finishes (see below) |
user_data | void * | Data that will be passed in the first argument of the done callback |
The API does not support a CS/SS pin: it is up to the user to select/deselect the SPI interface by calling spi_start() and spi_stop().
Note: spi_init can only be called from chip_init(). Do not call it at a later time.
Example:
const spi_config_t spi1 = {
.sck = pin_init("SCK", INPUT),
.mosi = pin_init("MOSI", INPUT),
.miso = pin_init("MISO", INPUT),
.mode = 0,
.done = chip_spi_done, // See the example below
.user_data = chip,
};
void spi_start(spi_dev_t spi, uint8_t *buffer, uint32_t count)
Starts an SPI transaction, sending and receiving count bytes to/from the given buffer.
You will usually listen for the CS (chip select) pin with pin_watch. Call spi_start() when the CS pin goes low, and spi_stop() when the CS pin goes high.
When creating a device that transfers large amounts of data (e.g. an LCD display), it's recommended to use a large buffer size (few kilobytes). The simulator can use the larger buffer to optimize DMA-controlled SPI transfer and speed up the simulation.
For simple devices that transfer small amounts of data, you can use a single-byte buffer, and process each byte as it arrives in the done callback.
void spi_stop(spi_dev_t spi)
Stops the SPI interface. Usually, you'd call this method when the CS pin goes high.
The done callback
The signature for the done callback is as follows:
static void chip_spi_done(void *user_data, uint8_t *buffer, uint32_t count) {
// 1. process the received data (optional)
// 2. if the CS pin is still low, schedule the next SPI transaction using `spi_start`
}
The done callback runs when an SPI transaction finishes: either when the buffer provided to spi_start is full, or when spi_stop was called. The buffer contains the data received (it is the same buffer given to spi_start), and count is the number of bytes that have been transferred (or 0 if spi_stop was called before a complete byte has been transferred).
Your done callback should check the status of the CS pin, and if it is still low, it should call spi_start() again to receive the next chunk of data from the microcontroller.