2021-05-24 01:50:04 +02:00
/*
2023-09-01 17:51:54 +08:00
* SPDX - FileCopyrightText : 2010 - 2023 Espressif Systems ( Shanghai ) CO LTD
2021-05-24 01:50:04 +02:00
*
* SPDX - License - Identifier : Apache - 2.0
*/
2017-03-31 15:05:25 +08:00
# ifndef _DRIVER_SPI_SLAVE_H_
# define _DRIVER_SPI_SLAVE_H_
# include "esp_err.h"
# include "freertos/FreeRTOS.h"
# include "freertos/semphr.h"
# include "driver/spi_common.h"
# ifdef __cplusplus
extern " C "
{
# endif
# define SPI_SLAVE_TXBIT_LSBFIRST (1<<0) ///< Transmit command/address/data LSB first instead of the default MSB first
# define SPI_SLAVE_RXBIT_LSBFIRST (1<<1) ///< Receive data LSB first instead of the default MSB first
2018-01-15 11:46:33 +08:00
# define SPI_SLAVE_BIT_LSBFIRST (SPI_SLAVE_TXBIT_LSBFIRST|SPI_SLAVE_RXBIT_LSBFIRST) ///< Transmit and receive LSB first
2022-09-15 15:56:25 +08:00
# define SPI_SLAVE_NO_RETURN_RESULT (1<<2) ///< Don't return the descriptor to the host on completion (use `post_trans_cb` to notify instead)
2017-03-31 15:05:25 +08:00
2022-11-24 15:54:44 +08:00
/** @cond */
2017-03-31 15:05:25 +08:00
typedef struct spi_slave_transaction_t spi_slave_transaction_t ;
2022-11-24 15:54:44 +08:00
/** @endcond */
2017-03-31 15:05:25 +08:00
typedef void ( * slave_transaction_cb_t ) ( spi_slave_transaction_t * trans ) ;
/**
* @ brief This is a configuration for a SPI host acting as a slave device .
*/
typedef struct {
2017-04-13 11:14:35 +08:00
int spics_io_num ; ///< CS GPIO pin for this device
uint32_t flags ; ///< Bitwise OR of SPI_SLAVE_* flags
int queue_size ; ///< Transaction queue size. This sets how many transactions can be 'in the air' (queued using spi_slave_queue_trans but not yet finished using spi_slave_get_trans_result) at the same time
2021-02-03 21:56:53 +08:00
uint8_t mode ; /**< SPI mode, representing a pair of (CPOL, CPHA) configuration:
- 0 : ( 0 , 0 )
- 1 : ( 0 , 1 )
- 2 : ( 1 , 0 )
- 3 : ( 1 , 1 )
*/
2018-10-23 16:57:32 +08:00
slave_transaction_cb_t post_setup_cb ; /**< Callback called after the SPI registers are loaded with new data.
*
* This callback is called within interrupt
* context should be in IRAM for best
* performance , see " Transferring Speed "
* section in the SPI Master documentation for
* full details . If not , the callback may crash
* during flash operation when the driver is
* initialized with ESP_INTR_FLAG_IRAM .
*/
slave_transaction_cb_t post_trans_cb ; /**< Callback called after a transaction is done.
*
* This callback is called within interrupt
* context should be in IRAM for best
* performance , see " Transferring Speed "
* section in the SPI Master documentation for
* full details . If not , the callback may crash
* during flash operation when the driver is
* initialized with ESP_INTR_FLAG_IRAM .
*/
2017-03-31 15:05:25 +08:00
} spi_slave_interface_config_t ;
2023-09-01 17:51:54 +08:00
# define SPI_SLAVE_TRANS_DMA_BUFFER_ALIGN_AUTO (1<<0) ///< Automatically re-malloc dma buffer if user buffer doesn't meet hardware alignment or dma_capable, this process may loss some memory and performance
2017-03-31 15:05:25 +08:00
/**
* This structure describes one SPI transaction
*/
struct spi_slave_transaction_t {
2023-09-01 17:51:54 +08:00
uint32_t flags ; ///< Bitwise OR of SPI_SLAVE_TRANS_* flags
2017-03-31 15:05:25 +08:00
size_t length ; ///< Total data length, in bits
2017-09-30 19:59:05 +08:00
size_t trans_len ; ///< Transaction data length, in bits
2017-03-31 15:05:25 +08:00
const void * tx_buffer ; ///< Pointer to transmit buffer, or NULL for no MOSI phase
2018-11-20 11:13:44 +08:00
void * rx_buffer ; /**< Pointer to receive buffer, or NULL for no MISO phase.
2023-09-01 17:51:54 +08:00
* When the DMA is enabled , must start at WORD boundary ( ` ` rx_buffer % 4 = = 0 ` ` ) ,
2018-11-20 11:13:44 +08:00
* and has length of a multiple of 4 bytes .
*/
2017-04-13 11:14:35 +08:00
void * user ; ///< User-defined variable. Can be used to store eg transaction ID.
2017-03-31 15:05:25 +08:00
} ;
/**
2017-04-27 11:24:44 +08:00
* @ brief Initialize a SPI bus as a slave interface
2017-03-31 15:05:25 +08:00
*
2021-03-05 16:20:33 +08:00
* @ warning SPI0 / 1 is not supported
2017-03-31 15:05:25 +08:00
*
2021-02-03 15:14:17 +08:00
* @ param host SPI peripheral to use as a SPI slave interface
* @ param bus_config Pointer to a spi_bus_config_t struct specifying how the host should be initialized
* @ param slave_config Pointer to a spi_slave_interface_config_t struct specifying the details for the slave interface
2021-02-20 11:03:28 +08:00
* @ param dma_chan - Selecting a DMA channel for an SPI bus allows transactions on the bus with size only limited by the amount of internal memory .
* - Selecting SPI_DMA_DISABLED limits the size of transactions .
* - Set to SPI_DMA_DISABLED if only the SPI flash uses this bus .
* - Set to SPI_DMA_CH_AUTO to let the driver to allocate the DMA channel .
2017-05-05 11:55:19 +08:00
*
2018-11-13 11:09:13 +08:00
* @ warning If a DMA channel is selected , any transmit and receive buffer used should be allocated in
2017-05-05 11:55:19 +08:00
* DMA - capable memory .
*
2018-11-13 11:09:13 +08:00
* @ warning The ISR of SPI is always executed on the core which calls this
* function . Never starve the ISR on this core or the SPI transactions will not
* be handled .
*
* @ return
2017-03-31 15:05:25 +08:00
* - ESP_ERR_INVALID_ARG if configuration is invalid
2021-02-03 15:14:17 +08:00
* - ESP_ERR_INVALID_STATE if host already is in use
* - ESP_ERR_NOT_FOUND if there is no available DMA channel
2017-03-31 15:05:25 +08:00
* - ESP_ERR_NO_MEM if out of memory
* - ESP_OK on success
*/
2021-02-20 11:03:28 +08:00
esp_err_t spi_slave_initialize ( spi_host_device_t host , const spi_bus_config_t * bus_config , const spi_slave_interface_config_t * slave_config , spi_dma_chan_t dma_chan ) ;
2017-03-31 15:05:25 +08:00
/**
* @ brief Free a SPI bus claimed as a SPI slave interface
*
* @ param host SPI peripheral to free
2018-11-13 11:09:13 +08:00
* @ return
2017-03-31 15:05:25 +08:00
* - ESP_ERR_INVALID_ARG if parameter is invalid
* - ESP_ERR_INVALID_STATE if not all devices on the bus are freed
* - ESP_OK on success
*/
esp_err_t spi_slave_free ( spi_host_device_t host ) ;
/**
* @ brief Queue a SPI transaction for execution
*
2023-11-27 12:29:45 +08:00
* @ note On esp32 , if trans length not WORD aligned , the rx buffer last word memory will still overwritten by DMA HW
*
2017-04-27 11:24:44 +08:00
* Queues a SPI transaction to be executed by this slave device . ( The transaction queue size was specified when the slave
* device was initialised via spi_slave_initialize . ) This function may block if the queue is full ( depending on the
* ticks_to_wait parameter ) . No SPI operation is directly initiated by this function , the next queued transaction
* will happen when the master initiates a SPI transaction by pulling down CS and sending out clock signals .
*
* This function hands over ownership of the buffers in ` ` trans_desc ` ` to the SPI slave driver ; the application is
* not to access this memory until ` ` spi_slave_queue_trans ` ` is called to hand ownership back to the application .
2017-03-31 15:05:25 +08:00
*
2017-04-13 11:14:35 +08:00
* @ param host SPI peripheral that is acting as a slave
* @ param trans_desc Description of transaction to execute . Not const because we may want to write status back
* into the transaction description .
2017-03-31 15:05:25 +08:00
* @ param ticks_to_wait Ticks to wait until there ' s room in the queue ; use portMAX_DELAY to
* never time out .
2018-11-13 11:09:13 +08:00
* @ return
2017-03-31 15:05:25 +08:00
* - ESP_ERR_INVALID_ARG if parameter is invalid
2023-09-01 17:51:54 +08:00
* - ESP_ERR_NO_MEM if set flag ` SPI_SLAVE_TRANS_DMA_BUFFER_ALIGN_AUTO ` but there is no free memory
* - ESP_ERR_INVALID_STATE if sync data between Cache and memory failed
2017-03-31 15:05:25 +08:00
* - ESP_OK on success
*/
2017-04-13 11:14:35 +08:00
esp_err_t spi_slave_queue_trans ( spi_host_device_t host , const spi_slave_transaction_t * trans_desc , TickType_t ticks_to_wait ) ;
2017-03-31 15:05:25 +08:00
/**
* @ brief Get the result of a SPI transaction queued earlier
*
2018-11-13 11:09:13 +08:00
* This routine will wait until a transaction to the given device ( queued earlier with
2017-04-13 11:14:35 +08:00
* spi_slave_queue_trans ) has succesfully completed . It will then return the description of the
2018-11-13 11:09:13 +08:00
* completed transaction so software can inspect the result and e . g . free the memory or
2017-03-31 15:05:25 +08:00
* re - use the buffers .
*
2017-04-27 11:24:44 +08:00
* It is mandatory to eventually use this function for any transaction queued by ` ` spi_slave_queue_trans ` ` .
*
2017-04-13 11:14:35 +08:00
* @ param host SPI peripheral to that is acting as a slave
2018-11-13 11:09:13 +08:00
* @ param [ out ] trans_desc Pointer to variable able to contain a pointer to the description of the
2017-03-31 15:05:25 +08:00
* transaction that is executed
* @ param ticks_to_wait Ticks to wait until there ' s a returned item ; use portMAX_DELAY to never time
* out .
2018-11-13 11:09:13 +08:00
* @ return
2017-03-31 15:05:25 +08:00
* - ESP_ERR_INVALID_ARG if parameter is invalid
2022-09-15 15:56:25 +08:00
* - ESP_ERR_NOT_SUPPORTED if flag ` SPI_SLAVE_NO_RETURN_RESULT ` is set
2017-03-31 15:05:25 +08:00
* - ESP_OK on success
*/
esp_err_t spi_slave_get_trans_result ( spi_host_device_t host , spi_slave_transaction_t * * trans_desc , TickType_t ticks_to_wait ) ;
/**
* @ brief Do a SPI transaction
*
2017-04-13 11:14:35 +08:00
* Essentially does the same as spi_slave_queue_trans followed by spi_slave_get_trans_result . Do
2018-11-13 11:09:13 +08:00
* not use this when there is still a transaction queued that hasn ' t been finalized
2017-04-13 11:14:35 +08:00
* using spi_slave_get_trans_result .
2017-03-31 15:05:25 +08:00
*
2017-04-13 11:14:35 +08:00
* @ param host SPI peripheral to that is acting as a slave
2018-11-13 11:09:13 +08:00
* @ param trans_desc Pointer to variable able to contain a pointer to the description of the
2017-04-13 11:14:35 +08:00
* transaction that is executed . Not const because we may want to write status back
* into the transaction description .
2017-03-31 15:05:25 +08:00
* @ param ticks_to_wait Ticks to wait until there ' s a returned item ; use portMAX_DELAY to never time
* out .
2018-11-13 11:09:13 +08:00
* @ return
2017-03-31 15:05:25 +08:00
* - ESP_ERR_INVALID_ARG if parameter is invalid
* - ESP_OK on success
*/
esp_err_t spi_slave_transmit ( spi_host_device_t host , spi_slave_transaction_t * trans_desc , TickType_t ticks_to_wait ) ;
# ifdef __cplusplus
}
# endif
2018-01-15 11:46:33 +08:00
# endif