2021-05-24 01:50:04 +02:00
/*
* SPDX - FileCopyrightText : 2010 - 2021 Espressif Systems ( Shanghai ) CO LTD
*
* SPDX - License - Identifier : Apache - 2.0
*/
2020-04-29 16:20:40 +08:00
# pragma once
# include "esp_types.h"
2020-09-10 10:37:58 +08:00
# include "soc/soc_caps.h"
2020-04-29 16:20:40 +08:00
# include "freertos/FreeRTOS.h"
# include "hal/spi_types.h"
# include "driver/spi_common.h"
# include "sdkconfig.h"
# ifdef __cplusplus
extern " C "
{
# endif
# if !SOC_SPI_SUPPORT_SLAVE_HD_VER2 && !CI_HEADER_CHECK
# error The SPI peripheral does not support this feature
# endif
2023-09-14 16:52:12 +08:00
# define SPI_SLAVE_HD_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 lose some memory and performance
2020-04-29 16:20:40 +08:00
/// Descriptor of data to send/receive
typedef struct {
2021-01-18 17:16:52 +08:00
uint8_t * data ; ///< Buffer to send, must be DMA capable
size_t len ; ///< Len of data to send/receive. For receiving the buffer length should be multiples of 4 bytes, otherwise the extra part will be truncated.
size_t trans_len ; ///< For RX direction, it indicates the data actually received. For TX direction, it is meaningless.
2023-09-14 16:52:12 +08:00
uint32_t flags ; ///< Bitwise OR of SPI_SLAVE_HD_TRANS_* flags
2021-01-18 17:16:52 +08:00
void * arg ; ///< Extra argument indiciating this data
2020-04-29 16:20:40 +08:00
} spi_slave_hd_data_t ;
/// Information of SPI Slave HD event
typedef struct {
2021-01-18 17:16:52 +08:00
spi_event_t event ; ///< Event type
spi_slave_hd_data_t * trans ; ///< Corresponding transaction for SPI_EV_SEND and SPI_EV_RECV events
2020-04-29 16:20:40 +08:00
} spi_slave_hd_event_t ;
/// Callback for SPI Slave HD
typedef bool ( * slave_cb_t ) ( void * arg , spi_slave_hd_event_t * event , BaseType_t * awoken ) ;
/// Channel of SPI Slave HD to do data transaction
typedef enum {
2021-01-18 17:16:52 +08:00
SPI_SLAVE_CHAN_TX = 0 , ///< The output channel (RDDMA)
SPI_SLAVE_CHAN_RX = 1 , ///< The input channel (WRDMA)
2020-04-29 16:20:40 +08:00
} spi_slave_chan_t ;
/// Callback configuration structure for SPI Slave HD
typedef struct {
2021-01-18 17:16:52 +08:00
slave_cb_t cb_buffer_tx ; ///< Callback when master reads from shared buffer
slave_cb_t cb_buffer_rx ; ///< Callback when master writes to shared buffer
2021-04-16 21:31:55 +08:00
slave_cb_t cb_send_dma_ready ; ///< Callback when TX data buffer is loaded to the hardware (DMA)
2021-01-18 17:16:52 +08:00
slave_cb_t cb_sent ; ///< Callback when data are sent
2021-04-16 21:31:55 +08:00
slave_cb_t cb_recv_dma_ready ; ///< Callback when RX data buffer is loaded to the hardware (DMA)
2021-01-18 17:16:52 +08:00
slave_cb_t cb_recv ; ///< Callback when data are received
slave_cb_t cb_cmd9 ; ///< Callback when CMD9 received
slave_cb_t cb_cmdA ; ///< Callback when CMDA received
void * arg ; ///< Argument indicating this SPI Slave HD peripheral instance
2020-04-29 16:20:40 +08:00
} spi_slave_hd_callback_config_t ;
2021-01-18 17:16:52 +08:00
//flags for ``spi_slave_hd_slot_config_t`` to use
# define SPI_SLAVE_HD_TXBIT_LSBFIRST (1<<0) ///< Transmit command/address/data LSB first instead of the default MSB first
# define SPI_SLAVE_HD_RXBIT_LSBFIRST (1<<1) ///< Receive data LSB first instead of the default MSB first
# define SPI_SLAVE_HD_BIT_LSBFIRST (SPI_SLAVE_HD_TXBIT_LSBFIRST|SPI_SLAVE_HD_RXBIT_LSBFIRST) ///< Transmit and receive LSB first
# define SPI_SLAVE_HD_APPEND_MODE (1<<2) ///< Adopt DMA append mode for transactions. In this mode, users can load(append) DMA descriptors without stopping the DMA
2020-04-29 16:20:40 +08:00
2021-01-18 17:16:52 +08:00
/// Configuration structure for the SPI Slave HD driver
typedef struct {
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 )
*/
2021-01-18 17:16:52 +08:00
uint32_t spics_io_num ; ///< CS GPIO pin for this device
uint32_t flags ; ///< Bitwise OR of SPI_SLAVE_HD_* flags
uint32_t command_bits ; ///< command field bits, multiples of 8 and at least 8.
uint32_t address_bits ; ///< address field bits, multiples of 8 and at least 8.
uint32_t dummy_bits ; ///< dummy field bits, multiples of 8 and at least 8.
uint32_t queue_size ; ///< Transaction queue size. This sets how many transactions can be 'in the air' (queued using spi_slave_hd_queue_trans but not yet finished using spi_slave_hd_get_trans_result) at the same time
2021-02-20 11:03:28 +08:00
spi_dma_chan_t dma_chan ; ///< DMA channel to used.
2021-01-18 17:16:52 +08:00
spi_slave_hd_callback_config_t cb_config ; ///< Callback configuration
2020-04-29 16:20:40 +08:00
} spi_slave_hd_slot_config_t ;
/**
* @ brief Initialize the SPI Slave HD driver .
*
* @ param host_id The host to use
* @ param bus_config Bus configuration for the bus used
* @ param config Configuration for the SPI Slave HD driver
* @ return
2021-02-03 15:14:17 +08:00
* - ESP_OK : on success
* - ESP_ERR_INVALID_ARG : invalid argument given
2020-04-29 16:20:40 +08:00
* - ESP_ERR_INVALID_STATE : function called in invalid state , may be some resources are already in use
2021-02-03 15:14:17 +08:00
* - ESP_ERR_NOT_FOUND if there is no available DMA channel
* - ESP_ERR_NO_MEM : memory allocation failed
2020-04-29 16:20:40 +08:00
* - or other return value from ` esp_intr_alloc `
*/
esp_err_t spi_slave_hd_init ( spi_host_device_t host_id , const spi_bus_config_t * bus_config ,
const spi_slave_hd_slot_config_t * config ) ;
/**
* @ brief Deinitialize the SPI Slave HD driver
*
* @ param host_id The host to deinitialize the driver
* @ return
* - ESP_OK : on success
* - ESP_ERR_INVALID_ARG : if the host_id is not correct
*/
esp_err_t spi_slave_hd_deinit ( spi_host_device_t host_id ) ;
/**
2021-01-18 17:16:52 +08:00
* @ brief Queue transactions ( segment mode )
2020-04-29 16:20:40 +08:00
*
* @ param host_id Host to queue the transaction
2021-01-18 17:16:52 +08:00
* @ param chan SPI_SLAVE_CHAN_TX or SPI_SLAVE_CHAN_RX
* @ param trans Transaction descriptors
2020-04-29 16:20:40 +08:00
* @ param timeout Timeout before the data is queued
* @ return
* - ESP_OK : on success
* - ESP_ERR_INVALID_ARG : The input argument is invalid . Can be the following reason :
* - The buffer given is not DMA capable
* - The length of data is invalid ( not larger than 0 , or exceed the max transfer length )
2021-01-18 17:16:52 +08:00
* - The transaction direction is invalid
* - ESP_ERR_TIMEOUT : Cannot queue the data before timeout . Master is still processing previous transaction .
* - ESP_ERR_INVALID_STATE : Function called in invalid state . This API should be called under segment mode .
2020-04-29 16:20:40 +08:00
*/
esp_err_t spi_slave_hd_queue_trans ( spi_host_device_t host_id , spi_slave_chan_t chan , spi_slave_hd_data_t * trans , TickType_t timeout ) ;
/**
2021-01-18 17:16:52 +08:00
* @ brief Get the result of a data transaction ( segment mode )
*
* @ note This API should be called successfully the same times as the ` ` spi_slave_hd_queue_trans ` ` .
2020-04-29 16:20:40 +08:00
*
* @ param host_id Host to queue the transaction
* @ param chan Channel to get the result , SPI_SLAVE_CHAN_TX or SPI_SLAVE_CHAN_RX
2021-01-18 17:16:52 +08:00
* @ param [ out ] out_trans Pointer to the transaction descriptor ( ` ` spi_slave_hd_data_t ` ` ) passed to the driver before . Hardware has finished this transaction . Member ` ` trans_len ` ` indicates the actual number of bytes of received data , it ' s meaningless for TX .
2020-04-29 16:20:40 +08:00
* @ param timeout Timeout before the result is got
* @ return
* - ESP_OK : on success
* - ESP_ERR_INVALID_ARG : Function is not valid
* - ESP_ERR_TIMEOUT : There ' s no transaction done before timeout
2021-01-18 17:16:52 +08:00
* - ESP_ERR_INVALID_STATE : Function called in invalid state . This API should be called under segment mode .
2020-04-29 16:20:40 +08:00
*/
2021-01-18 17:16:52 +08:00
esp_err_t spi_slave_hd_get_trans_res ( spi_host_device_t host_id , spi_slave_chan_t chan , spi_slave_hd_data_t * * out_trans , TickType_t timeout ) ;
2020-04-29 16:20:40 +08:00
/**
* @ brief Read the shared registers
*
* @ param host_id Host to read the shared registers
* @ param addr Address of register to read , 0 to ` ` SOC_SPI_MAXIMUM_BUFFER_SIZE - 1 ` `
* @ param [ out ] out_data Output buffer to store the read data
* @ param len Length to read , not larger than ` ` SOC_SPI_MAXIMUM_BUFFER_SIZE - addr ` `
*/
void spi_slave_hd_read_buffer ( spi_host_device_t host_id , int addr , uint8_t * out_data , size_t len ) ;
/**
* @ brief Write the shared registers
*
* @ param host_id Host to write the shared registers
* @ param addr Address of register to write , 0 to ` ` SOC_SPI_MAXIMUM_BUFFER_SIZE - 1 ` `
* @ param data Buffer holding the data to write
* @ param len Length to write , ` ` SOC_SPI_MAXIMUM_BUFFER_SIZE - addr ` `
*/
void spi_slave_hd_write_buffer ( spi_host_device_t host_id , int addr , uint8_t * data , size_t len ) ;
2021-01-18 17:16:52 +08:00
/**
* @ brief Load transactions ( append mode )
*
* @ note In this mode , user transaction descriptors will be appended to the DMA and the DMA will keep processing the data without stopping
*
* @ param host_id Host to load transactions
* @ param chan SPI_SLAVE_CHAN_TX or SPI_SLAVE_CHAN_RX
* @ param trans Transaction descriptor
* @ param timeout Timeout before the transaction is loaded
* @ return
* - ESP_OK : on success
* - ESP_ERR_INVALID_ARG : The input argument is invalid . Can be the following reason :
* - The buffer given is not DMA capable
* - The length of data is invalid ( not larger than 0 , or exceed the max transfer length )
* - The transaction direction is invalid
* - ESP_ERR_TIMEOUT : Master is still processing previous transaction . There is no available transaction for slave to load
* - ESP_ERR_INVALID_STATE : Function called in invalid state . This API should be called under append mode .
*/
esp_err_t spi_slave_hd_append_trans ( spi_host_device_t host_id , spi_slave_chan_t chan , spi_slave_hd_data_t * trans , TickType_t timeout ) ;
/**
* @ brief Get the result of a data transaction ( append mode )
*
* @ note This API should be called the same times as the ` ` spi_slave_hd_append_trans ` `
*
* @ param host_id Host to load the transaction
* @ param chan SPI_SLAVE_CHAN_TX or SPI_SLAVE_CHAN_RX
* @ param [ out ] out_trans Pointer to the transaction descriptor ( ` ` spi_slave_hd_data_t ` ` ) passed to the driver before . Hardware has finished this transaction . Member ` ` trans_len ` ` indicates the actual number of bytes of received data , it ' s meaningless for TX .
* @ param timeout Timeout before the result is got
* @ return
* - ESP_OK : on success
* - ESP_ERR_INVALID_ARG : Function is not valid
* - ESP_ERR_TIMEOUT : There ' s no transaction done before timeout
* - ESP_ERR_INVALID_STATE : Function called in invalid state . This API should be called under append mode .
*/
esp_err_t spi_slave_hd_get_append_trans_res ( spi_host_device_t host_id , spi_slave_chan_t chan , spi_slave_hd_data_t * * out_trans , TickType_t timeout ) ;
2020-04-29 16:20:40 +08:00
# ifdef __cplusplus
}
2020-11-10 18:40:01 +11:00
# endif