2021-05-24 02:09:38 +02:00
/*
* SPDX - FileCopyrightText : 2015 - 2021 Espressif Systems ( Shanghai ) CO LTD
*
* SPDX - License - Identifier : Apache - 2.0
*/
2017-03-31 15:05:25 +08:00
# include <string.h>
2019-06-06 10:57:29 +08:00
# include "sdkconfig.h"
2017-03-31 15:05:25 +08:00
# include "driver/spi_master.h"
2018-06-08 15:32:43 +08:00
# include "soc/spi_periph.h"
2017-03-31 15:05:25 +08:00
# include "esp_types.h"
# include "esp_attr.h"
# include "esp_log.h"
# include "esp_err.h"
# include "soc/soc.h"
2020-09-08 17:05:49 +08:00
# include "soc/soc_caps.h"
2021-07-09 16:46:27 +08:00
# include "soc/soc_pins.h"
2019-01-23 17:07:03 +08:00
# include "soc/lldesc.h"
2017-03-31 15:05:25 +08:00
# include "driver/gpio.h"
# include "driver/periph_ctrl.h"
2017-05-03 18:03:28 +10:00
# include "esp_heap_caps.h"
2019-09-03 14:06:26 +08:00
# include "driver/spi_common_internal.h"
2018-01-31 11:15:23 +08:00
# include "stdatomic.h"
2019-01-23 17:07:03 +08:00
# include "hal/spi_hal.h"
2021-03-16 10:55:05 +08:00
# include "hal/gpio_hal.h"
2020-06-19 12:00:58 +08:00
# include "esp_rom_gpio.h"
2020-11-26 19:56:13 +11:00
# if CONFIG_IDF_TARGET_ESP32
# include "soc/dport_reg.h"
# endif
2020-09-08 17:05:49 +08:00
# if SOC_GDMA_SUPPORTED
2021-01-27 21:56:16 +08:00
# include "esp_private/gdma.h"
2020-09-08 17:05:49 +08:00
# endif
2019-10-24 19:00:26 +08:00
2017-03-31 15:05:25 +08:00
static const char * SPI_TAG = " spi " ;
2019-02-11 14:17:31 +08:00
# define SPI_CHECK(a, str, ret_val) do { \
2017-03-31 15:05:25 +08:00
if ( ! ( a ) ) { \
ESP_LOGE ( SPI_TAG , " %s(%d): %s " , __FUNCTION__ , __LINE__ , str ) ; \
return ( ret_val ) ; \
2019-02-11 14:17:31 +08:00
} \
} while ( 0 )
# define SPI_CHECK_PIN(pin_num, pin_name, check_output) if (check_output) { \
SPI_CHECK ( GPIO_IS_VALID_OUTPUT_GPIO ( pin_num ) , pin_name " not valid " , ESP_ERR_INVALID_ARG ) ; \
} else { \
SPI_CHECK ( GPIO_IS_VALID_GPIO ( pin_num ) , pin_name " not valid " , ESP_ERR_INVALID_ARG ) ; \
}
2017-03-31 15:05:25 +08:00
2021-02-07 14:39:07 +08:00
# define SPI_MAIN_BUS_DEFAULT() { \
2021-02-03 15:14:17 +08:00
. host_id = 0 , \
. bus_attr = { \
. tx_dma_chan = 0 , \
. rx_dma_chan = 0 , \
. max_transfer_sz = SOC_SPI_MAXIMUM_BUFFER_SIZE , \
. dma_desc_num = 0 , \
} , \
}
2017-03-31 15:05:25 +08:00
2018-04-24 15:34:07 +08:00
# define FUNC_GPIO PIN_FUNC_GPIO
2017-08-31 19:59:30 +08:00
2019-10-24 19:00:26 +08:00
typedef struct {
int host_id ;
spi_destroy_func_t destroy_func ;
void * destroy_arg ;
spi_bus_attr_t bus_attr ;
2021-01-27 21:56:16 +08:00
# if SOC_GDMA_SUPPORTED
gdma_channel_handle_t tx_channel ;
gdma_channel_handle_t rx_channel ;
# endif
2019-10-24 19:00:26 +08:00
} spicommon_bus_context_t ;
2017-03-31 15:05:25 +08:00
//Periph 1 is 'claimed' by SPI flash code.
2021-01-27 21:56:16 +08:00
static atomic_bool spi_periph_claimed [ SOC_SPI_PERIPH_NUM ] = { ATOMIC_VAR_INIT ( true ) , ATOMIC_VAR_INIT ( false ) ,
# if (SOC_SPI_PERIPH_NUM >= 3)
ATOMIC_VAR_INIT ( false ) ,
# endif
# if (SOC_SPI_PERIPH_NUM >= 4)
2021-02-03 15:14:17 +08:00
ATOMIC_VAR_INIT ( false ) ,
2019-06-13 14:19:31 +08:00
# endif
2019-06-13 14:12:54 +08:00
} ;
2021-02-03 15:14:17 +08:00
2018-10-05 15:39:32 +08:00
static const char * spi_claiming_func [ 3 ] = { NULL , NULL , NULL } ;
2021-02-07 14:39:07 +08:00
static spicommon_bus_context_t s_mainbus = SPI_MAIN_BUS_DEFAULT ( ) ;
2021-02-03 15:14:17 +08:00
static spicommon_bus_context_t * bus_ctx [ SOC_SPI_PERIPH_NUM ] = { & s_mainbus } ;
2021-01-27 21:56:16 +08:00
# if !SOC_GDMA_SUPPORTED
2021-02-03 15:14:17 +08:00
//Each bit stands for 1 dma channel, BIT(0) should be used for SPI1
2017-08-31 19:59:30 +08:00
static uint8_t spi_dma_chan_enabled = 0 ;
static portMUX_TYPE spi_dma_spinlock = portMUX_INITIALIZER_UNLOCKED ;
2021-02-03 15:14:17 +08:00
# endif //#if !SOC_GDMA_SUPPORTED
2017-08-31 19:59:30 +08:00
2019-10-24 19:00:26 +08:00
2021-02-03 15:14:17 +08:00
static inline bool is_valid_host ( spi_host_device_t host )
{
# if (SOC_SPI_PERIPH_NUM == 2)
return host > = SPI1_HOST & & host < = SPI2_HOST ;
# elif (SOC_SPI_PERIPH_NUM == 3)
return host > = SPI1_HOST & & host < = SPI3_HOST ;
2021-01-26 20:18:52 +08:00
# endif
2021-02-03 15:14:17 +08:00
}
2021-01-26 20:18:52 +08:00
2021-01-27 21:56:16 +08:00
//----------------------------------------------------------alloc spi periph-------------------------------------------------------//
2017-03-31 15:05:25 +08:00
//Returns true if this peripheral is successfully claimed, false if otherwise.
2018-10-05 15:39:32 +08:00
bool spicommon_periph_claim ( spi_host_device_t host , const char * source )
2017-04-27 11:24:44 +08:00
{
2018-01-31 11:15:23 +08:00
bool false_var = false ;
bool ret = atomic_compare_exchange_strong ( & spi_periph_claimed [ host ] , & false_var , true ) ;
2018-10-05 15:39:32 +08:00
if ( ret ) {
spi_claiming_func [ host ] = source ;
periph_module_enable ( spi_periph_signal [ host ] . module ) ;
} else {
ESP_EARLY_LOGE ( SPI_TAG , " SPI%d already claimed by %s. " , host + 1 , spi_claiming_func [ host ] ) ;
}
2017-03-31 15:05:25 +08:00
return ret ;
}
2018-10-05 15:39:32 +08:00
bool spicommon_periph_in_use ( spi_host_device_t host )
{
return atomic_load ( & spi_periph_claimed [ host ] ) ;
}
2017-03-31 15:05:25 +08:00
//Returns true if this peripheral is successfully freed, false if otherwise.
2017-04-27 11:24:44 +08:00
bool spicommon_periph_free ( spi_host_device_t host )
{
2018-01-31 11:15:23 +08:00
bool true_var = true ;
bool ret = atomic_compare_exchange_strong ( & spi_periph_claimed [ host ] , & true_var , false ) ;
2018-06-08 15:32:43 +08:00
if ( ret ) periph_module_disable ( spi_periph_signal [ host ] . module ) ;
2017-03-31 15:05:25 +08:00
return ret ;
}
2017-04-27 11:24:44 +08:00
int spicommon_irqsource_for_host ( spi_host_device_t host )
{
2018-06-08 15:32:43 +08:00
return spi_periph_signal [ host ] . irq ;
2017-03-31 15:05:25 +08:00
}
2019-06-13 14:12:54 +08:00
int spicommon_irqdma_source_for_host ( spi_host_device_t host )
{
return spi_periph_signal [ host ] . irq_dma ;
}
2021-02-03 15:14:17 +08:00
//----------------------------------------------------------alloc dma periph-------------------------------------------------------//
2021-01-27 21:56:16 +08:00
# if !SOC_GDMA_SUPPORTED
2020-11-26 19:56:13 +11:00
static inline periph_module_t get_dma_periph ( int dma_chan )
2019-06-13 14:12:54 +08:00
{
2021-02-03 15:14:17 +08:00
assert ( dma_chan > = 1 & & dma_chan < = SOC_SPI_DMA_CHAN_NUM ) ;
2020-01-17 11:47:08 +08:00
# if CONFIG_IDF_TARGET_ESP32S2
2019-12-26 15:25:24 +08:00
if ( dma_chan = = 1 ) {
2019-06-13 14:19:31 +08:00
return PERIPH_SPI2_DMA_MODULE ;
2021-02-03 15:14:17 +08:00
} else if ( dma_chan = = 2 ) {
2019-06-13 14:19:31 +08:00
return PERIPH_SPI3_DMA_MODULE ;
} else {
abort ( ) ;
}
2019-12-26 15:25:24 +08:00
# elif CONFIG_IDF_TARGET_ESP32
2019-06-13 14:12:54 +08:00
return PERIPH_SPI_DMA_MODULE ;
2019-06-13 14:19:31 +08:00
# endif
2019-06-13 14:12:54 +08:00
}
2021-12-22 11:17:42 +08:00
static bool claim_dma_chan ( int dma_chan , uint32_t * out_actual_dma_chan )
2017-08-31 19:59:30 +08:00
{
bool ret = false ;
portENTER_CRITICAL ( & spi_dma_spinlock ) ;
2021-02-03 15:14:17 +08:00
bool is_used = ( BIT ( dma_chan ) & spi_dma_chan_enabled ) ;
if ( ! is_used ) {
spi_dma_chan_enabled | = BIT ( dma_chan ) ;
periph_module_enable ( get_dma_periph ( dma_chan ) ) ;
2021-02-20 11:03:28 +08:00
* out_actual_dma_chan = dma_chan ;
2017-08-31 19:59:30 +08:00
ret = true ;
}
portEXIT_CRITICAL ( & spi_dma_spinlock ) ;
return ret ;
}
2021-12-22 11:17:42 +08:00
static void connect_spi_and_dma ( spi_host_device_t host , int dma_chan )
2020-09-08 17:05:49 +08:00
{
# if CONFIG_IDF_TARGET_ESP32
DPORT_SET_PERI_REG_BITS ( DPORT_SPI_DMA_CHAN_SEL_REG , 3 , dma_chan , ( host * 2 ) ) ;
# elif CONFIG_IDF_TARGET_ESP32S2
//On ESP32S2, each SPI controller has its own DMA channel. So there is no need to connect them.
2020-11-26 19:56:13 +11:00
# endif
2021-01-27 21:56:16 +08:00
}
2021-12-22 11:17:42 +08:00
static esp_err_t alloc_dma_chan ( spi_host_device_t host_id , spi_dma_chan_t dma_chan , uint32_t * out_actual_tx_dma_chan , uint32_t * out_actual_rx_dma_chan )
2021-01-27 21:56:16 +08:00
{
2021-02-03 15:14:17 +08:00
assert ( is_valid_host ( host_id ) ) ;
# if CONFIG_IDF_TARGET_ESP32
2021-02-20 11:03:28 +08:00
assert ( dma_chan > SPI_DMA_DISABLED & & dma_chan < = SPI_DMA_CH_AUTO ) ;
2021-02-03 15:14:17 +08:00
# elif CONFIG_IDF_TARGET_ESP32S2
2021-02-20 11:03:28 +08:00
assert ( dma_chan = = ( int ) host_id | | dma_chan = = SPI_DMA_CH_AUTO ) ;
2021-01-27 21:56:16 +08:00
# endif
2020-09-08 17:05:49 +08:00
2021-01-27 21:56:16 +08:00
esp_err_t ret = ESP_OK ;
2021-02-03 15:14:17 +08:00
bool success = false ;
2021-02-20 11:03:28 +08:00
uint32_t actual_dma_chan = 0 ;
2021-01-26 20:18:52 +08:00
2021-02-20 11:03:28 +08:00
if ( dma_chan = = SPI_DMA_CH_AUTO ) {
2021-01-26 20:18:52 +08:00
# if CONFIG_IDF_TARGET_ESP32
2021-02-03 15:14:17 +08:00
for ( int i = 1 ; i < SOC_SPI_DMA_CHAN_NUM + 1 ; i + + ) {
2021-12-22 11:17:42 +08:00
success = claim_dma_chan ( i , & actual_dma_chan ) ;
2021-02-03 15:14:17 +08:00
if ( success ) {
2021-01-26 20:18:52 +08:00
break ;
}
}
# elif CONFIG_IDF_TARGET_ESP32S2
2021-02-03 15:14:17 +08:00
//On ESP32S2, each SPI controller has its own DMA channel
2021-12-22 11:17:42 +08:00
success = claim_dma_chan ( host_id , & actual_dma_chan ) ;
2021-01-26 20:18:52 +08:00
# endif //#if CONFIG_IDF_TARGET_XXX
2021-02-20 11:03:28 +08:00
} else {
2021-12-22 11:17:42 +08:00
success = claim_dma_chan ( ( int ) dma_chan , & actual_dma_chan ) ;
2021-01-26 20:18:52 +08:00
}
2021-02-20 11:03:28 +08:00
//On ESP32 and ESP32S2, actual_tx_dma_chan and actual_rx_dma_chan are always same
* out_actual_tx_dma_chan = actual_dma_chan ;
* out_actual_rx_dma_chan = actual_dma_chan ;
2021-02-03 15:14:17 +08:00
if ( ! success ) {
SPI_CHECK ( false , " no available dma channel " , ESP_ERR_NOT_FOUND ) ;
2021-01-26 20:18:52 +08:00
}
2021-12-22 11:17:42 +08:00
connect_spi_and_dma ( host_id , * out_actual_tx_dma_chan ) ;
2021-02-03 15:14:17 +08:00
return ret ;
}
2021-01-26 20:18:52 +08:00
2021-01-27 21:56:16 +08:00
# else //SOC_GDMA_SUPPORTED
2021-12-22 11:17:42 +08:00
static esp_err_t alloc_dma_chan ( spi_host_device_t host_id , spi_dma_chan_t dma_chan , uint32_t * out_actual_tx_dma_chan , uint32_t * out_actual_rx_dma_chan )
2021-02-03 15:14:17 +08:00
{
assert ( is_valid_host ( host_id ) ) ;
2021-02-20 11:03:28 +08:00
assert ( dma_chan = = SPI_DMA_CH_AUTO ) ;
2021-01-27 21:56:16 +08:00
2021-02-03 15:14:17 +08:00
esp_err_t ret = ESP_OK ;
2021-01-27 21:56:16 +08:00
spicommon_bus_context_t * ctx = bus_ctx [ host_id ] ;
2021-02-20 11:03:28 +08:00
if ( dma_chan = = SPI_DMA_CH_AUTO ) {
2021-01-27 21:56:16 +08:00
gdma_channel_alloc_config_t tx_alloc_config = {
. flags . reserve_sibling = 1 ,
. direction = GDMA_CHANNEL_DIRECTION_TX ,
} ;
ret = gdma_new_channel ( & tx_alloc_config , & ctx - > tx_channel ) ;
if ( ret ! = ESP_OK ) {
return ret ;
}
gdma_channel_alloc_config_t rx_alloc_config = {
. direction = GDMA_CHANNEL_DIRECTION_RX ,
. sibling_chan = ctx - > tx_channel ,
} ;
ret = gdma_new_channel ( & rx_alloc_config , & ctx - > rx_channel ) ;
if ( ret ! = ESP_OK ) {
return ret ;
}
2021-02-03 15:14:17 +08:00
if ( host_id = = SPI2_HOST ) {
2021-01-27 21:56:16 +08:00
gdma_connect ( ctx - > rx_channel , GDMA_MAKE_TRIGGER ( GDMA_TRIG_PERIPH_SPI , 2 ) ) ;
gdma_connect ( ctx - > tx_channel , GDMA_MAKE_TRIGGER ( GDMA_TRIG_PERIPH_SPI , 2 ) ) ;
}
# if (SOC_SPI_PERIPH_NUM >= 3)
2021-02-03 15:14:17 +08:00
else if ( host_id = = SPI3_HOST ) {
2021-01-27 21:56:16 +08:00
gdma_connect ( ctx - > rx_channel , GDMA_MAKE_TRIGGER ( GDMA_TRIG_PERIPH_SPI , 3 ) ) ;
gdma_connect ( ctx - > tx_channel , GDMA_MAKE_TRIGGER ( GDMA_TRIG_PERIPH_SPI , 3 ) ) ;
}
# endif
2021-02-03 15:14:17 +08:00
gdma_get_channel_id ( ctx - > tx_channel , ( int * ) out_actual_tx_dma_chan ) ;
gdma_get_channel_id ( ctx - > rx_channel , ( int * ) out_actual_rx_dma_chan ) ;
2021-01-27 21:56:16 +08:00
}
return ret ;
}
2021-02-03 15:14:17 +08:00
# endif //#if !SOC_GDMA_SUPPORTED
2021-01-27 21:56:16 +08:00
2021-12-22 11:17:42 +08:00
esp_err_t spicommon_dma_chan_alloc ( spi_host_device_t host_id , spi_dma_chan_t dma_chan , uint32_t * out_actual_tx_dma_chan , uint32_t * out_actual_rx_dma_chan )
2021-01-27 21:56:16 +08:00
{
2021-02-03 15:14:17 +08:00
assert ( is_valid_host ( host_id ) ) ;
2021-02-20 11:03:28 +08:00
# if CONFIG_IDF_TARGET_ESP32
assert ( dma_chan > SPI_DMA_DISABLED & & dma_chan < = SPI_DMA_CH_AUTO ) ;
# elif CONFIG_IDF_TARGET_ESP32S2
assert ( dma_chan = = ( int ) host_id | | dma_chan = = SPI_DMA_CH_AUTO ) ;
# endif
2021-02-03 15:14:17 +08:00
2021-01-27 21:56:16 +08:00
esp_err_t ret = ESP_OK ;
2021-02-03 15:14:17 +08:00
uint32_t actual_tx_dma_chan = 0 ;
uint32_t actual_rx_dma_chan = 0 ;
2021-01-27 21:56:16 +08:00
spicommon_bus_context_t * ctx = ( spicommon_bus_context_t * ) calloc ( 1 , sizeof ( spicommon_bus_context_t ) ) ;
if ( ! ctx ) {
ret = ESP_ERR_NO_MEM ;
goto cleanup ;
}
bus_ctx [ host_id ] = ctx ;
ctx - > host_id = host_id ;
2021-12-22 11:17:42 +08:00
ret = alloc_dma_chan ( host_id , dma_chan , & actual_tx_dma_chan , & actual_rx_dma_chan ) ;
2021-01-27 21:56:16 +08:00
if ( ret ! = ESP_OK ) {
goto cleanup ;
}
2021-02-03 15:14:17 +08:00
ctx - > bus_attr . tx_dma_chan = actual_tx_dma_chan ;
ctx - > bus_attr . rx_dma_chan = actual_rx_dma_chan ;
* out_actual_tx_dma_chan = actual_tx_dma_chan ;
* out_actual_rx_dma_chan = actual_rx_dma_chan ;
2021-01-27 21:56:16 +08:00
return ret ;
cleanup :
free ( ctx ) ;
ctx = NULL ;
return ret ;
}
//----------------------------------------------------------free dma periph-------------------------------------------------------//
2021-12-22 11:17:42 +08:00
static esp_err_t dma_chan_free ( spi_host_device_t host_id )
2021-01-27 21:56:16 +08:00
{
2021-02-03 15:14:17 +08:00
assert ( is_valid_host ( host_id ) ) ;
spicommon_bus_context_t * ctx = bus_ctx [ host_id ] ;
2021-01-27 21:56:16 +08:00
# if !SOC_GDMA_SUPPORTED
2021-02-03 15:14:17 +08:00
//On ESP32S2, each SPI controller has its own DMA channel
int dma_chan = ctx - > bus_attr . tx_dma_chan ;
assert ( spi_dma_chan_enabled & BIT ( dma_chan ) ) ;
2021-01-27 21:56:16 +08:00
portENTER_CRITICAL ( & spi_dma_spinlock ) ;
2021-02-03 15:14:17 +08:00
spi_dma_chan_enabled & = ~ BIT ( dma_chan ) ;
2021-01-27 21:56:16 +08:00
periph_module_disable ( get_dma_periph ( dma_chan ) ) ;
portEXIT_CRITICAL ( & spi_dma_spinlock ) ;
# else //SOC_GDMA_SUPPORTED
if ( ctx - > rx_channel ) {
gdma_disconnect ( ctx - > rx_channel ) ;
gdma_del_channel ( ctx - > rx_channel ) ;
}
if ( ctx - > tx_channel ) {
gdma_disconnect ( ctx - > tx_channel ) ;
gdma_del_channel ( ctx - > tx_channel ) ;
}
2021-01-26 20:18:52 +08:00
# endif
return ESP_OK ;
}
2021-12-22 11:17:42 +08:00
esp_err_t spicommon_dma_chan_free ( spi_host_device_t host_id )
2021-01-27 21:56:16 +08:00
{
2021-02-03 15:14:17 +08:00
assert ( is_valid_host ( host_id ) ) ;
2021-01-27 21:56:16 +08:00
2021-12-22 11:17:42 +08:00
esp_err_t ret = dma_chan_free ( host_id ) ;
2021-01-27 21:56:16 +08:00
free ( bus_ctx [ host_id ] ) ;
bus_ctx [ host_id ] = NULL ;
return ret ;
}
//----------------------------------------------------------IO general-------------------------------------------------------//
2021-07-09 16:46:27 +08:00
# if SOC_SPI_SUPPORT_OCT
static bool check_iomux_pins_oct ( spi_host_device_t host , const spi_bus_config_t * bus_config )
2019-02-11 14:17:31 +08:00
{
2021-08-18 19:19:55 +08:00
if ( host ! = SPI2_HOST ) {
return false ;
}
2021-07-09 16:46:27 +08:00
int io_nums [ ] = { bus_config - > data0_io_num , bus_config - > data1_io_num , bus_config - > data2_io_num , bus_config - > data3_io_num ,
bus_config - > sclk_io_num , bus_config - > data4_io_num , bus_config - > data5_io_num , bus_config - > data6_io_num , bus_config - > data7_io_num } ;
int io_mux_nums [ ] = { SPI2_IOMUX_PIN_NUM_MOSI_OCT , SPI2_IOMUX_PIN_NUM_MISO_OCT , SPI2_IOMUX_PIN_NUM_WP_OCT , SPI2_IOMUX_PIN_NUM_HD_OCT ,
SPI2_IOMUX_PIN_NUM_CLK_OCT , SPI2_IOMUX_PIN_NUM_IO4_OCT , SPI2_IOMUX_PIN_NUM_IO5_OCT , SPI2_IOMUX_PIN_NUM_IO6_OCT , SPI2_IOMUX_PIN_NUM_IO7_OCT } ;
for ( size_t i = 0 ; i < sizeof ( io_nums ) / sizeof ( io_nums [ 0 ] ) ; i + + ) {
if ( io_nums [ i ] > = 0 & & io_nums [ i ] ! = io_mux_nums [ i ] ) {
2020-11-26 19:56:13 +11:00
return false ;
}
2021-07-09 16:46:27 +08:00
}
return true ;
}
# endif
static bool check_iomux_pins_quad ( spi_host_device_t host , const spi_bus_config_t * bus_config )
{
if ( bus_config - > sclk_io_num > = 0 & &
bus_config - > sclk_io_num ! = spi_periph_signal [ host ] . spiclk_iomux_pin ) {
return false ;
}
2019-02-11 14:17:31 +08:00
if ( bus_config - > quadwp_io_num > = 0 & &
2020-11-26 19:56:13 +11:00
bus_config - > quadwp_io_num ! = spi_periph_signal [ host ] . spiwp_iomux_pin ) {
2021-07-09 16:46:27 +08:00
return false ;
}
2019-02-11 14:17:31 +08:00
if ( bus_config - > quadhd_io_num > = 0 & &
2020-11-26 19:56:13 +11:00
bus_config - > quadhd_io_num ! = spi_periph_signal [ host ] . spihd_iomux_pin ) {
2021-07-09 16:46:27 +08:00
return false ;
}
2019-02-11 14:17:31 +08:00
if ( bus_config - > mosi_io_num > = 0 & &
2020-11-26 19:56:13 +11:00
bus_config - > mosi_io_num ! = spi_periph_signal [ host ] . spid_iomux_pin ) {
2021-07-09 16:46:27 +08:00
return false ;
}
2019-02-11 14:17:31 +08:00
if ( bus_config - > miso_io_num > = 0 & &
2020-11-26 19:56:13 +11:00
bus_config - > miso_io_num ! = spi_periph_signal [ host ] . spiq_iomux_pin ) {
2021-07-09 16:46:27 +08:00
return false ;
}
return true ;
}
static bool bus_uses_iomux_pins ( spi_host_device_t host , const spi_bus_config_t * bus_config )
{
//Check if SPI pins could be routed to iomux.
# if SOC_SPI_SUPPORT_OCT
//The io mux pins available for Octal mode is not the same as the ones we use for non-Octal mode.
if ( ( bus_config - > flags & SPICOMMON_BUSFLAG_OCTAL ) = = SPICOMMON_BUSFLAG_OCTAL ) {
return check_iomux_pins_oct ( host , bus_config ) ;
}
# endif
return check_iomux_pins_quad ( host , bus_config ) ;
}
# if SOC_SPI_SUPPORT_OCT
static void bus_iomux_pins_set_oct ( spi_host_device_t host , const spi_bus_config_t * bus_config )
{
assert ( host = = SPI2_HOST ) ;
int io_nums [ ] = { bus_config - > data0_io_num , bus_config - > data1_io_num , bus_config - > data2_io_num , bus_config - > data3_io_num ,
bus_config - > sclk_io_num , bus_config - > data4_io_num , bus_config - > data5_io_num , bus_config - > data6_io_num , bus_config - > data7_io_num } ;
int io_signals [ ] = { spi_periph_signal [ host ] . spid_in , spi_periph_signal [ host ] . spiq_in , spi_periph_signal [ host ] . spiwp_in ,
spi_periph_signal [ host ] . spihd_in , spi_periph_signal [ host ] . spiclk_in , spi_periph_signal [ host ] . spid4_out ,
spi_periph_signal [ host ] . spid5_out , spi_periph_signal [ host ] . spid6_out , spi_periph_signal [ host ] . spid7_out } ;
for ( size_t i = 0 ; i < sizeof ( io_nums ) / sizeof ( io_nums [ 0 ] ) ; i + + ) {
if ( io_nums [ i ] > 0 ) {
gpio_iomux_in ( io_nums [ i ] , io_signals [ i ] ) ;
// In Octal mode use function channel 2
gpio_iomux_out ( io_nums [ i ] , SPI2_FUNC_NUM_OCT , false ) ;
2020-11-26 19:56:13 +11:00
}
2021-07-09 16:46:27 +08:00
}
}
# endif //SOC_SPI_SUPPORT_OCT
2019-02-11 14:17:31 +08:00
2021-07-09 16:46:27 +08:00
static void bus_iomux_pins_set_quad ( spi_host_device_t host , const spi_bus_config_t * bus_config )
{
if ( bus_config - > mosi_io_num > = 0 ) {
gpio_iomux_in ( bus_config - > mosi_io_num , spi_periph_signal [ host ] . spid_in ) ;
gpio_iomux_out ( bus_config - > mosi_io_num , spi_periph_signal [ host ] . func , false ) ;
}
if ( bus_config - > miso_io_num > = 0 ) {
gpio_iomux_in ( bus_config - > miso_io_num , spi_periph_signal [ host ] . spiq_in ) ;
gpio_iomux_out ( bus_config - > miso_io_num , spi_periph_signal [ host ] . func , false ) ;
}
if ( bus_config - > quadwp_io_num > = 0 ) {
gpio_iomux_in ( bus_config - > quadwp_io_num , spi_periph_signal [ host ] . spiwp_in ) ;
gpio_iomux_out ( bus_config - > quadwp_io_num , spi_periph_signal [ host ] . func , false ) ;
}
if ( bus_config - > quadhd_io_num > = 0 ) {
gpio_iomux_in ( bus_config - > quadhd_io_num , spi_periph_signal [ host ] . spihd_in ) ;
gpio_iomux_out ( bus_config - > quadhd_io_num , spi_periph_signal [ host ] . func , false ) ;
}
if ( bus_config - > sclk_io_num > = 0 ) {
gpio_iomux_in ( bus_config - > sclk_io_num , spi_periph_signal [ host ] . spiclk_in ) ;
gpio_iomux_out ( bus_config - > sclk_io_num , spi_periph_signal [ host ] . func , false ) ;
}
}
static void bus_iomux_pins_set ( spi_host_device_t host , const spi_bus_config_t * bus_config )
{
# if SOC_SPI_SUPPORT_OCT
if ( ( bus_config - > flags & SPICOMMON_BUSFLAG_OCTAL ) = = SPICOMMON_BUSFLAG_OCTAL ) {
bus_iomux_pins_set_oct ( host , bus_config ) ;
return ;
}
# endif
bus_iomux_pins_set_quad ( host , bus_config ) ;
2019-02-11 14:17:31 +08:00
}
2017-03-31 15:05:25 +08:00
/*
Do the common stuff to hook up a SPI host to a bus defined by a bunch of GPIO pins . Feed it a host number and a
2018-04-24 15:34:07 +08:00
bus config struct and it ' ll set up the GPIO matrix and enable the device . If a pin is set to non - negative value ,
2018-03-21 20:42:45 +08:00
it should be able to be initialized .
2017-03-31 15:05:25 +08:00
*/
2021-01-27 21:56:16 +08:00
esp_err_t spicommon_bus_initialize_io ( spi_host_device_t host , const spi_bus_config_t * bus_config , uint32_t flags , uint32_t * flags_o )
2017-03-31 15:05:25 +08:00
{
2021-07-09 16:46:27 +08:00
# if SOC_SPI_SUPPORT_OCT
// In the driver of previous version, spi data4 ~ spi data7 are not in spi_bus_config_t struct. So the new-added pins come as 0
// if they are not really set. Add this boolean variable to check if the user has set spi data4 ~spi data7 pins .
bool io4_7_is_blank = ! bus_config - > data4_io_num & & ! bus_config - > data5_io_num & & ! bus_config - > data6_io_num & & ! bus_config - > data7_io_num ;
// This boolean variable specifies if user sets pins used for octal mode (users can set spi data4 ~ spi data7 to -1).
bool io4_7_enabled = ! io4_7_is_blank & & bus_config - > data4_io_num > = 0 & & bus_config - > data5_io_num > = 0 & &
bus_config - > data6_io_num > = 0 & & bus_config - > data7_io_num > = 0 ;
SPI_CHECK ( ( flags & SPICOMMON_BUSFLAG_MASTER ) | | ! ( ( flags & SPICOMMON_BUSFLAG_OCTAL ) = = SPICOMMON_BUSFLAG_OCTAL ) , " Octal SPI mode / OPI mode only works when SPI is used as Master " , ESP_ERR_INVALID_ARG ) ;
SPI_CHECK ( host = = SPI2_HOST | | ! ( ( flags & SPICOMMON_BUSFLAG_OCTAL ) = = SPICOMMON_BUSFLAG_OCTAL ) , " Only SPI2 supports Octal SPI mode / OPI mode " , ESP_ERR_INVALID_ARG ) ;
# endif //SOC_SPI_SUPPORT_OCT
2020-09-08 17:05:49 +08:00
uint32_t temp_flag = 0 ;
2018-03-21 20:42:45 +08:00
2019-02-11 14:17:31 +08:00
bool miso_need_output ;
bool mosi_need_output ;
bool sclk_need_output ;
if ( ( flags & SPICOMMON_BUSFLAG_MASTER ) ! = 0 ) {
//initial for master
miso_need_output = ( ( flags & SPICOMMON_BUSFLAG_DUAL ) ! = 0 ) ? true : false ;
mosi_need_output = true ;
sclk_need_output = true ;
} else {
//initial for slave
miso_need_output = true ;
mosi_need_output = ( ( flags & SPICOMMON_BUSFLAG_DUAL ) ! = 0 ) ? true : false ;
sclk_need_output = false ;
}
const bool wp_need_output = true ;
const bool hd_need_output = true ;
//check pin capabilities
2018-03-21 20:42:45 +08:00
if ( bus_config - > sclk_io_num > = 0 ) {
temp_flag | = SPICOMMON_BUSFLAG_SCLK ;
2019-02-11 14:17:31 +08:00
SPI_CHECK_PIN ( bus_config - > sclk_io_num , " sclk " , sclk_need_output ) ;
2017-03-31 15:05:25 +08:00
}
2018-03-21 20:42:45 +08:00
if ( bus_config - > quadwp_io_num > = 0 ) {
2019-02-11 14:17:31 +08:00
SPI_CHECK_PIN ( bus_config - > quadwp_io_num , " wp " , wp_need_output ) ;
2017-03-31 15:05:25 +08:00
}
2018-03-21 20:42:45 +08:00
if ( bus_config - > quadhd_io_num > = 0 ) {
2019-02-11 14:17:31 +08:00
SPI_CHECK_PIN ( bus_config - > quadhd_io_num , " hd " , hd_need_output ) ;
2017-12-28 10:31:23 +08:00
}
2021-07-09 16:46:27 +08:00
# if SOC_SPI_SUPPORT_OCT
const bool io4_need_output = true ;
const bool io5_need_output = true ;
const bool io6_need_output = true ;
const bool io7_need_output = true ;
// set flags for OCTAL mode according to the existence of spi data4 ~ spi data7
if ( io4_7_enabled ) {
temp_flag | = SPICOMMON_BUSFLAG_IO4_IO7 ;
if ( bus_config - > data4_io_num > = 0 ) {
SPI_CHECK_PIN ( bus_config - > data4_io_num , " spi data4 " , io4_need_output ) ;
}
if ( bus_config - > data5_io_num > = 0 ) {
SPI_CHECK_PIN ( bus_config - > data5_io_num , " spi data5 " , io5_need_output ) ;
}
if ( bus_config - > data6_io_num > = 0 ) {
SPI_CHECK_PIN ( bus_config - > data6_io_num , " spi data6 " , io6_need_output ) ;
}
if ( bus_config - > data7_io_num > = 0 ) {
SPI_CHECK_PIN ( bus_config - > data7_io_num , " spi data7 " , io7_need_output ) ;
}
}
# endif //SOC_SPI_SUPPORT_OCT
2019-02-11 14:17:31 +08:00
//set flags for QUAD mode according to the existence of wp and hd
if ( bus_config - > quadhd_io_num > = 0 & & bus_config - > quadwp_io_num > = 0 ) temp_flag | = SPICOMMON_BUSFLAG_WPHD ;
2018-03-21 20:42:45 +08:00
if ( bus_config - > mosi_io_num > = 0 ) {
temp_flag | = SPICOMMON_BUSFLAG_MOSI ;
2019-02-11 14:17:31 +08:00
SPI_CHECK_PIN ( bus_config - > mosi_io_num , " mosi " , mosi_need_output ) ;
2018-03-21 20:42:45 +08:00
}
2021-07-09 16:46:27 +08:00
if ( bus_config - > miso_io_num > = 0 ) {
2018-03-21 20:42:45 +08:00
temp_flag | = SPICOMMON_BUSFLAG_MISO ;
2019-02-11 14:17:31 +08:00
SPI_CHECK_PIN ( bus_config - > miso_io_num , " miso " , miso_need_output ) ;
2018-03-21 20:42:45 +08:00
}
2018-04-24 15:34:07 +08:00
//set flags for DUAL mode according to output-capability of MOSI and MISO pins.
if ( ( bus_config - > mosi_io_num < 0 | | GPIO_IS_VALID_OUTPUT_GPIO ( bus_config - > mosi_io_num ) ) & &
2018-03-21 20:42:45 +08:00
( bus_config - > miso_io_num < 0 | | GPIO_IS_VALID_OUTPUT_GPIO ( bus_config - > miso_io_num ) ) ) {
temp_flag | = SPICOMMON_BUSFLAG_DUAL ;
2018-04-24 15:34:07 +08:00
}
2019-02-11 14:17:31 +08:00
//check if the selected pins correspond to the iomux pins of the peripheral
2020-04-29 16:18:25 +08:00
bool use_iomux = ! ( flags & SPICOMMON_BUSFLAG_GPIO_PINS ) & & bus_uses_iomux_pins ( host , bus_config ) ;
if ( use_iomux ) {
temp_flag | = SPICOMMON_BUSFLAG_IOMUX_PINS ;
} else {
temp_flag | = SPICOMMON_BUSFLAG_GPIO_PINS ;
}
2019-02-11 14:17:31 +08:00
uint32_t missing_flag = flags & ~ temp_flag ;
missing_flag & = ~ SPICOMMON_BUSFLAG_MASTER ; //don't check this flag
if ( missing_flag ! = 0 ) {
//check pins existence
2021-07-09 16:46:27 +08:00
if ( missing_flag & SPICOMMON_BUSFLAG_SCLK ) {
ESP_LOGE ( SPI_TAG , " sclk pin required. " ) ;
}
if ( missing_flag & SPICOMMON_BUSFLAG_MOSI ) {
ESP_LOGE ( SPI_TAG , " mosi pin required. " ) ;
}
if ( missing_flag & SPICOMMON_BUSFLAG_MISO ) {
ESP_LOGE ( SPI_TAG , " miso pin required. " ) ;
}
if ( missing_flag & SPICOMMON_BUSFLAG_DUAL ) {
ESP_LOGE ( SPI_TAG , " not both mosi and miso output capable " ) ;
}
if ( missing_flag & SPICOMMON_BUSFLAG_WPHD ) {
ESP_LOGE ( SPI_TAG , " both wp and hd required. " ) ;
}
if ( missing_flag & SPICOMMON_BUSFLAG_IOMUX_PINS ) {
ESP_LOGE ( SPI_TAG , " not using iomux pins " ) ;
}
# if SOC_SPI_SUPPORT_OCT
if ( missing_flag & SPICOMMON_BUSFLAG_IO4_IO7 ) {
ESP_LOGE ( SPI_TAG , " spi data4 ~ spi data7 are required. " ) ;
}
# endif
2019-02-11 14:17:31 +08:00
SPI_CHECK ( missing_flag = = 0 , " not all required capabilities satisfied. " , ESP_ERR_INVALID_ARG ) ;
}
2017-03-31 15:05:25 +08:00
2018-06-06 01:36:01 +08:00
if ( use_iomux ) {
//All SPI iomux pin selections resolve to 1, so we put that here instead of trying to figure
2017-03-31 15:05:25 +08:00
//out which FUNC_GPIOx_xSPIxx to grab; they all are defined to 1 anyway.
2018-10-05 15:39:32 +08:00
ESP_LOGD ( SPI_TAG , " SPI%d use iomux pins. " , host + 1 ) ;
2021-07-09 16:46:27 +08:00
bus_iomux_pins_set ( host , bus_config ) ;
2017-03-31 15:05:25 +08:00
} else {
2018-03-21 20:42:45 +08:00
//Use GPIO matrix
2018-10-05 15:39:32 +08:00
ESP_LOGD ( SPI_TAG , " SPI%d use gpio matrix. " , host + 1 ) ;
2017-12-12 17:00:02 +08:00
if ( bus_config - > mosi_io_num > = 0 ) {
2019-02-11 14:17:31 +08:00
if ( mosi_need_output | | ( temp_flag & SPICOMMON_BUSFLAG_DUAL ) ) {
2018-03-21 20:42:45 +08:00
gpio_set_direction ( bus_config - > mosi_io_num , GPIO_MODE_INPUT_OUTPUT ) ;
2020-06-19 12:00:58 +08:00
esp_rom_gpio_connect_out_signal ( bus_config - > mosi_io_num , spi_periph_signal [ host ] . spid_out , false , false ) ;
2018-03-21 20:42:45 +08:00
} else {
gpio_set_direction ( bus_config - > mosi_io_num , GPIO_MODE_INPUT ) ;
}
2020-06-19 12:00:58 +08:00
esp_rom_gpio_connect_in_signal ( bus_config - > mosi_io_num , spi_periph_signal [ host ] . spid_in , false ) ;
2020-01-17 11:47:08 +08:00
# if CONFIG_IDF_TARGET_ESP32S2
2019-06-06 10:57:29 +08:00
PIN_INPUT_ENABLE ( GPIO_PIN_MUX_REG [ bus_config - > mosi_io_num ] ) ;
# endif
2021-03-16 10:55:05 +08:00
gpio_hal_iomux_func_sel ( GPIO_PIN_MUX_REG [ bus_config - > mosi_io_num ] , FUNC_GPIO ) ;
2017-03-31 15:05:25 +08:00
}
2017-12-12 17:00:02 +08:00
if ( bus_config - > miso_io_num > = 0 ) {
2019-02-11 14:17:31 +08:00
if ( miso_need_output | | ( temp_flag & SPICOMMON_BUSFLAG_DUAL ) ) {
2018-03-21 20:42:45 +08:00
gpio_set_direction ( bus_config - > miso_io_num , GPIO_MODE_INPUT_OUTPUT ) ;
2020-06-19 12:00:58 +08:00
esp_rom_gpio_connect_out_signal ( bus_config - > miso_io_num , spi_periph_signal [ host ] . spiq_out , false , false ) ;
2018-03-21 20:42:45 +08:00
} else {
gpio_set_direction ( bus_config - > miso_io_num , GPIO_MODE_INPUT ) ;
}
2020-06-19 12:00:58 +08:00
esp_rom_gpio_connect_in_signal ( bus_config - > miso_io_num , spi_periph_signal [ host ] . spiq_in , false ) ;
2020-01-17 11:47:08 +08:00
# if CONFIG_IDF_TARGET_ESP32S2
2019-06-06 10:57:29 +08:00
PIN_INPUT_ENABLE ( GPIO_PIN_MUX_REG [ bus_config - > miso_io_num ] ) ;
# endif
2021-03-16 10:55:05 +08:00
gpio_hal_iomux_func_sel ( GPIO_PIN_MUX_REG [ bus_config - > miso_io_num ] , FUNC_GPIO ) ;
2017-03-31 15:05:25 +08:00
}
2018-03-21 20:42:45 +08:00
if ( bus_config - > quadwp_io_num > = 0 ) {
2017-04-24 16:10:37 +08:00
gpio_set_direction ( bus_config - > quadwp_io_num , GPIO_MODE_INPUT_OUTPUT ) ;
2020-06-19 12:00:58 +08:00
esp_rom_gpio_connect_out_signal ( bus_config - > quadwp_io_num , spi_periph_signal [ host ] . spiwp_out , false , false ) ;
esp_rom_gpio_connect_in_signal ( bus_config - > quadwp_io_num , spi_periph_signal [ host ] . spiwp_in , false ) ;
2020-01-17 11:47:08 +08:00
# if CONFIG_IDF_TARGET_ESP32S2
2019-06-06 10:57:29 +08:00
PIN_INPUT_ENABLE ( GPIO_PIN_MUX_REG [ bus_config - > quadwp_io_num ] ) ;
# endif
2021-03-16 10:55:05 +08:00
gpio_hal_iomux_func_sel ( GPIO_PIN_MUX_REG [ bus_config - > quadwp_io_num ] , FUNC_GPIO ) ;
2017-03-31 15:05:25 +08:00
}
2018-03-21 20:42:45 +08:00
if ( bus_config - > quadhd_io_num > = 0 ) {
2017-04-24 16:10:37 +08:00
gpio_set_direction ( bus_config - > quadhd_io_num , GPIO_MODE_INPUT_OUTPUT ) ;
2020-06-19 12:00:58 +08:00
esp_rom_gpio_connect_out_signal ( bus_config - > quadhd_io_num , spi_periph_signal [ host ] . spihd_out , false , false ) ;
esp_rom_gpio_connect_in_signal ( bus_config - > quadhd_io_num , spi_periph_signal [ host ] . spihd_in , false ) ;
2020-01-17 11:47:08 +08:00
# if CONFIG_IDF_TARGET_ESP32S2
2019-06-06 10:57:29 +08:00
PIN_INPUT_ENABLE ( GPIO_PIN_MUX_REG [ bus_config - > quadhd_io_num ] ) ;
# endif
2021-03-16 10:55:05 +08:00
gpio_hal_iomux_func_sel ( GPIO_PIN_MUX_REG [ bus_config - > quadhd_io_num ] , FUNC_GPIO ) ;
2017-03-31 15:05:25 +08:00
}
2017-12-12 17:00:02 +08:00
if ( bus_config - > sclk_io_num > = 0 ) {
2019-02-11 14:17:31 +08:00
if ( sclk_need_output ) {
gpio_set_direction ( bus_config - > sclk_io_num , GPIO_MODE_INPUT_OUTPUT ) ;
2020-06-19 12:00:58 +08:00
esp_rom_gpio_connect_out_signal ( bus_config - > sclk_io_num , spi_periph_signal [ host ] . spiclk_out , false , false ) ;
2019-02-11 14:17:31 +08:00
} else {
gpio_set_direction ( bus_config - > sclk_io_num , GPIO_MODE_INPUT ) ;
}
2020-06-19 12:00:58 +08:00
esp_rom_gpio_connect_in_signal ( bus_config - > sclk_io_num , spi_periph_signal [ host ] . spiclk_in , false ) ;
2020-01-17 11:47:08 +08:00
# if CONFIG_IDF_TARGET_ESP32S2
2019-06-06 10:57:29 +08:00
PIN_INPUT_ENABLE ( GPIO_PIN_MUX_REG [ bus_config - > sclk_io_num ] ) ;
# endif
2021-03-16 10:55:05 +08:00
gpio_hal_iomux_func_sel ( GPIO_PIN_MUX_REG [ bus_config - > sclk_io_num ] , FUNC_GPIO ) ;
2017-03-31 15:05:25 +08:00
}
2021-07-09 16:46:27 +08:00
# if SOC_SPI_SUPPORT_OCT
2022-06-09 17:20:30 +02:00
if ( ( flags & SPICOMMON_BUSFLAG_OCTAL ) = = SPICOMMON_BUSFLAG_OCTAL ) {
2021-07-09 16:46:27 +08:00
int io_nums [ ] = { bus_config - > data4_io_num , bus_config - > data5_io_num , bus_config - > data6_io_num , bus_config - > data7_io_num } ;
uint8_t io_signals [ 4 ] [ 2 ] = { { spi_periph_signal [ host ] . spid4_out , spi_periph_signal [ host ] . spid4_in } ,
{ spi_periph_signal [ host ] . spid5_out , spi_periph_signal [ host ] . spid5_in } ,
{ spi_periph_signal [ host ] . spid6_out , spi_periph_signal [ host ] . spid6_in } ,
{ spi_periph_signal [ host ] . spid7_out , spi_periph_signal [ host ] . spid7_in } } ;
for ( size_t i = 0 ; i < sizeof ( io_nums ) / sizeof ( io_nums [ 0 ] ) ; i + + ) {
if ( io_nums [ i ] > = 0 ) {
gpio_set_direction ( io_nums [ i ] , GPIO_MODE_INPUT_OUTPUT ) ;
esp_rom_gpio_connect_out_signal ( io_nums [ i ] , io_signals [ i ] [ 0 ] , false , false ) ;
esp_rom_gpio_connect_in_signal ( io_nums [ i ] , io_signals [ i ] [ 1 ] , false ) ;
# if CONFIG_IDF_TARGET_ESP32S2
PIN_INPUT_ENABLE ( GPIO_PIN_MUX_REG [ io_nums [ i ] ] ) ;
# endif
gpio_hal_iomux_func_sel ( GPIO_PIN_MUX_REG [ io_nums [ i ] ] , FUNC_GPIO ) ;
}
}
}
# endif //SOC_SPI_SUPPORT_OCT
2017-03-31 15:05:25 +08:00
}
2018-03-21 20:42:45 +08:00
if ( flags_o ) * flags_o = temp_flag ;
2017-03-31 15:05:25 +08:00
return ESP_OK ;
}
2018-06-08 15:33:04 +08:00
esp_err_t spicommon_bus_free_io_cfg ( const spi_bus_config_t * bus_cfg )
{
int pin_array [ ] = {
bus_cfg - > mosi_io_num ,
bus_cfg - > miso_io_num ,
bus_cfg - > sclk_io_num ,
bus_cfg - > quadwp_io_num ,
bus_cfg - > quadhd_io_num ,
} ;
for ( int i = 0 ; i < sizeof ( pin_array ) / sizeof ( int ) ; i + + ) {
const int io = pin_array [ i ] ;
if ( io > = 0 & & GPIO_IS_VALID_GPIO ( io ) ) gpio_reset_pin ( io ) ;
}
return ESP_OK ;
}
2017-04-27 11:24:44 +08:00
void spicommon_cs_initialize ( spi_host_device_t host , int cs_io_num , int cs_num , int force_gpio_matrix )
{
2018-06-08 15:32:43 +08:00
if ( ! force_gpio_matrix & & cs_io_num = = spi_periph_signal [ host ] . spics0_iomux_pin & & cs_num = = 0 ) {
2017-03-31 15:05:25 +08:00
//The cs0s for all SPI peripherals map to pin mux source 1, so we use that instead of a define.
2018-06-08 15:32:43 +08:00
gpio_iomux_in ( cs_io_num , spi_periph_signal [ host ] . spics_in ) ;
2019-06-13 14:12:54 +08:00
gpio_iomux_out ( cs_io_num , spi_periph_signal [ host ] . func , false ) ;
2017-03-31 15:05:25 +08:00
} else {
//Use GPIO matrix
2019-02-11 14:17:31 +08:00
if ( GPIO_IS_VALID_OUTPUT_GPIO ( cs_io_num ) ) {
gpio_set_direction ( cs_io_num , GPIO_MODE_INPUT_OUTPUT ) ;
2020-06-19 12:00:58 +08:00
esp_rom_gpio_connect_out_signal ( cs_io_num , spi_periph_signal [ host ] . spics_out [ cs_num ] , false , false ) ;
2019-02-11 14:17:31 +08:00
} else {
gpio_set_direction ( cs_io_num , GPIO_MODE_INPUT ) ;
}
2020-06-19 12:00:58 +08:00
if ( cs_num = = 0 ) esp_rom_gpio_connect_in_signal ( cs_io_num , spi_periph_signal [ host ] . spics_in , false ) ;
2019-06-06 10:57:29 +08:00
PIN_INPUT_ENABLE ( GPIO_PIN_MUX_REG [ cs_io_num ] ) ;
2021-03-16 10:55:05 +08:00
gpio_hal_iomux_func_sel ( GPIO_PIN_MUX_REG [ cs_io_num ] , FUNC_GPIO ) ;
2017-03-31 15:05:25 +08:00
}
}
2018-06-08 15:33:04 +08:00
void spicommon_cs_free_io ( int cs_gpio_num )
{
assert ( cs_gpio_num > = 0 & & GPIO_IS_VALID_GPIO ( cs_gpio_num ) ) ;
gpio_reset_pin ( cs_gpio_num ) ;
}
2019-06-26 11:42:02 +08:00
bool spicommon_bus_using_iomux ( spi_host_device_t host )
2017-04-27 11:24:44 +08:00
{
2019-06-26 11:42:02 +08:00
# define CHECK_IOMUX_PIN(HOST, PIN_NAME) if (GPIO.func_in_sel_cfg[spi_periph_signal[(HOST)].PIN_NAME##_in].sig_in_sel) return false
2017-03-31 15:05:25 +08:00
2019-06-26 11:42:02 +08:00
CHECK_IOMUX_PIN ( host , spid ) ;
CHECK_IOMUX_PIN ( host , spiq ) ;
CHECK_IOMUX_PIN ( host , spiwp ) ;
CHECK_IOMUX_PIN ( host , spihd ) ;
return true ;
}
2017-03-31 15:05:25 +08:00
2019-10-24 19:00:26 +08:00
void spi_bus_main_set_lock ( spi_bus_lock_handle_t lock )
{
bus_ctx [ 0 ] - > bus_attr . lock = lock ;
}
spi_bus_lock_handle_t spi_bus_lock_get_by_id ( spi_host_device_t host_id )
{
return bus_ctx [ host_id ] - > bus_attr . lock ;
}
2021-01-27 21:56:16 +08:00
//----------------------------------------------------------master bus init-------------------------------------------------------//
2021-02-20 11:03:28 +08:00
esp_err_t spi_bus_initialize ( spi_host_device_t host_id , const spi_bus_config_t * bus_config , spi_dma_chan_t dma_chan )
2019-10-24 19:00:26 +08:00
{
esp_err_t err = ESP_OK ;
spicommon_bus_context_t * ctx = NULL ;
spi_bus_attr_t * bus_attr = NULL ;
2021-01-27 21:56:16 +08:00
uint32_t actual_tx_dma_chan = 0 ;
uint32_t actual_rx_dma_chan = 0 ;
2021-01-26 20:18:52 +08:00
2019-10-24 19:00:26 +08:00
SPI_CHECK ( is_valid_host ( host_id ) , " invalid host_id " , ESP_ERR_INVALID_ARG ) ;
SPI_CHECK ( bus_ctx [ host_id ] = = NULL , " SPI bus already initialized. " , ESP_ERR_INVALID_STATE ) ;
# ifdef CONFIG_IDF_TARGET_ESP32
2021-02-20 11:03:28 +08:00
SPI_CHECK ( dma_chan > = SPI_DMA_DISABLED & & dma_chan < = SPI_DMA_CH_AUTO , " invalid dma channel " , ESP_ERR_INVALID_ARG ) ;
2019-10-24 19:00:26 +08:00
# elif CONFIG_IDF_TARGET_ESP32S2
2021-02-20 11:03:28 +08:00
SPI_CHECK ( dma_chan = = SPI_DMA_DISABLED | | dma_chan = = ( int ) host_id | | dma_chan = = SPI_DMA_CH_AUTO , " invalid dma channel " , ESP_ERR_INVALID_ARG ) ;
2021-01-26 20:18:52 +08:00
# elif SOC_GDMA_SUPPORTED
2021-02-20 11:03:28 +08:00
SPI_CHECK ( dma_chan = = SPI_DMA_DISABLED | | dma_chan = = SPI_DMA_CH_AUTO , " invalid dma channel, chip only support spi dma channel auto-alloc " , ESP_ERR_INVALID_ARG ) ;
2019-10-24 19:00:26 +08:00
# endif
SPI_CHECK ( ( bus_config - > intr_flags & ( ESP_INTR_FLAG_HIGH | ESP_INTR_FLAG_EDGE | ESP_INTR_FLAG_INTRDISABLED ) ) = = 0 , " intr flag not allowed " , ESP_ERR_INVALID_ARG ) ;
# ifndef CONFIG_SPI_MASTER_ISR_IN_IRAM
SPI_CHECK ( ( bus_config - > intr_flags & ESP_INTR_FLAG_IRAM ) = = 0 , " ESP_INTR_FLAG_IRAM should be disabled when CONFIG_SPI_MASTER_ISR_IN_IRAM is not set. " , ESP_ERR_INVALID_ARG ) ;
# endif
bool spi_chan_claimed = spicommon_periph_claim ( host_id , " spi master " ) ;
SPI_CHECK ( spi_chan_claimed , " host_id already in use " , ESP_ERR_INVALID_STATE ) ;
//clean and initialize the context
2021-01-26 20:18:52 +08:00
ctx = ( spicommon_bus_context_t * ) calloc ( 1 , sizeof ( spicommon_bus_context_t ) ) ;
2019-10-24 19:00:26 +08:00
if ( ! ctx ) {
err = ESP_ERR_NO_MEM ;
goto cleanup ;
}
2021-01-27 21:56:16 +08:00
bus_ctx [ host_id ] = ctx ;
2021-01-26 20:18:52 +08:00
ctx - > host_id = host_id ;
2019-10-24 19:00:26 +08:00
bus_attr = & ctx - > bus_attr ;
2021-01-26 20:18:52 +08:00
bus_attr - > bus_cfg = * bus_config ;
2021-02-20 11:03:28 +08:00
if ( dma_chan ! = SPI_DMA_DISABLED ) {
2021-01-27 21:56:16 +08:00
bus_attr - > dma_enabled = 1 ;
2021-12-22 11:17:42 +08:00
err = alloc_dma_chan ( host_id , dma_chan , & actual_tx_dma_chan , & actual_rx_dma_chan ) ;
2021-01-26 20:18:52 +08:00
if ( err ! = ESP_OK ) {
2021-01-27 21:56:16 +08:00
goto cleanup ;
2021-01-26 20:18:52 +08:00
}
2021-01-27 21:56:16 +08:00
bus_attr - > tx_dma_chan = actual_tx_dma_chan ;
bus_attr - > rx_dma_chan = actual_rx_dma_chan ;
2021-01-26 20:18:52 +08:00
2019-10-24 19:00:26 +08:00
int dma_desc_ct = lldesc_get_required_num ( bus_config - > max_transfer_sz ) ;
if ( dma_desc_ct = = 0 ) dma_desc_ct = 1 ; //default to 4k when max is not given
bus_attr - > max_transfer_sz = dma_desc_ct * LLDESC_MAX_NUM_PER_DESC ;
bus_attr - > dmadesc_tx = heap_caps_malloc ( sizeof ( lldesc_t ) * dma_desc_ct , MALLOC_CAP_DMA ) ;
bus_attr - > dmadesc_rx = heap_caps_malloc ( sizeof ( lldesc_t ) * dma_desc_ct , MALLOC_CAP_DMA ) ;
if ( bus_attr - > dmadesc_tx = = NULL | | bus_attr - > dmadesc_rx = = NULL ) {
err = ESP_ERR_NO_MEM ;
goto cleanup ;
}
bus_attr - > dma_desc_num = dma_desc_ct ;
2021-01-26 20:18:52 +08:00
} else {
2021-01-27 21:56:16 +08:00
bus_attr - > dma_enabled = 0 ;
2021-01-26 20:18:52 +08:00
bus_attr - > max_transfer_sz = SOC_SPI_MAXIMUM_BUFFER_SIZE ;
bus_attr - > dma_desc_num = 0 ;
2019-10-24 19:00:26 +08:00
}
spi_bus_lock_config_t lock_config = {
. host_id = host_id ,
. cs_num = SOC_SPI_PERIPH_CS_NUM ( host_id ) ,
} ;
err = spi_bus_init_lock ( & bus_attr - > lock , & lock_config ) ;
if ( err ! = ESP_OK ) {
goto cleanup ;
}
# ifdef CONFIG_PM_ENABLE
err = esp_pm_lock_create ( ESP_PM_APB_FREQ_MAX , 0 , " spi_master " ,
& bus_attr - > pm_lock ) ;
if ( err ! = ESP_OK ) {
goto cleanup ;
}
# endif //CONFIG_PM_ENABLE
2021-01-27 21:56:16 +08:00
err = spicommon_bus_initialize_io ( host_id , bus_config , SPICOMMON_BUSFLAG_MASTER | bus_config - > flags , & bus_attr - > flags ) ;
2019-10-24 19:00:26 +08:00
if ( err ! = ESP_OK ) {
goto cleanup ;
}
return ESP_OK ;
cleanup :
if ( bus_attr ) {
# ifdef CONFIG_PM_ENABLE
esp_pm_lock_delete ( bus_attr - > pm_lock ) ;
# endif
if ( bus_attr - > lock ) {
spi_bus_deinit_lock ( bus_attr - > lock ) ;
}
free ( bus_attr - > dmadesc_tx ) ;
free ( bus_attr - > dmadesc_rx ) ;
2021-02-03 15:14:17 +08:00
bus_attr - > dmadesc_tx = NULL ;
bus_attr - > dmadesc_rx = NULL ;
2021-01-27 21:56:16 +08:00
if ( bus_attr - > dma_enabled ) {
2021-12-22 11:17:42 +08:00
dma_chan_free ( host_id ) ;
2021-01-27 21:56:16 +08:00
}
2019-10-24 19:00:26 +08:00
}
spicommon_periph_free ( host_id ) ;
2021-01-27 21:56:16 +08:00
free ( bus_ctx [ host_id ] ) ;
bus_ctx [ host_id ] = NULL ;
2019-10-24 19:00:26 +08:00
return err ;
}
const spi_bus_attr_t * spi_bus_get_attr ( spi_host_device_t host_id )
{
if ( bus_ctx [ host_id ] = = NULL ) return NULL ;
return & bus_ctx [ host_id ] - > bus_attr ;
}
esp_err_t spi_bus_free ( spi_host_device_t host_id )
{
2022-03-24 16:28:37 +08:00
if ( bus_ctx [ host_id ] = = NULL ) {
return ESP_ERR_INVALID_STATE ;
}
2019-10-24 19:00:26 +08:00
esp_err_t err = ESP_OK ;
spicommon_bus_context_t * ctx = bus_ctx [ host_id ] ;
spi_bus_attr_t * bus_attr = & ctx - > bus_attr ;
if ( ctx - > destroy_func ) {
err = ctx - > destroy_func ( ctx - > destroy_arg ) ;
}
spicommon_bus_free_io_cfg ( & bus_attr - > bus_cfg ) ;
# ifdef CONFIG_PM_ENABLE
esp_pm_lock_delete ( bus_attr - > pm_lock ) ;
# endif
spi_bus_deinit_lock ( bus_attr - > lock ) ;
free ( bus_attr - > dmadesc_rx ) ;
free ( bus_attr - > dmadesc_tx ) ;
2021-02-03 15:14:17 +08:00
bus_attr - > dmadesc_tx = NULL ;
bus_attr - > dmadesc_rx = NULL ;
2021-01-27 21:56:16 +08:00
if ( bus_attr - > dma_enabled > 0 ) {
2021-12-22 11:17:42 +08:00
dma_chan_free ( host_id ) ;
2019-10-24 19:00:26 +08:00
}
spicommon_periph_free ( host_id ) ;
free ( ctx ) ;
bus_ctx [ host_id ] = NULL ;
return err ;
}
esp_err_t spi_bus_register_destroy_func ( spi_host_device_t host_id ,
spi_destroy_func_t f , void * arg )
{
bus_ctx [ host_id ] - > destroy_func = f ;
bus_ctx [ host_id ] - > destroy_arg = arg ;
return ESP_OK ;
}
2017-03-31 15:05:25 +08:00
/*
Code for workaround for DMA issue in ESP32 v0 / v1 silicon
*/
2019-06-13 14:19:31 +08:00
# if CONFIG_IDF_TARGET_ESP32
2017-04-27 11:24:44 +08:00
static volatile int dmaworkaround_channels_busy [ 2 ] = { 0 , 0 } ;
2017-03-31 15:05:25 +08:00
static dmaworkaround_cb_t dmaworkaround_cb ;
static void * dmaworkaround_cb_arg ;
2017-04-27 11:24:44 +08:00
static portMUX_TYPE dmaworkaround_mux = portMUX_INITIALIZER_UNLOCKED ;
static int dmaworkaround_waiting_for_chan = 0 ;
2017-03-31 15:05:25 +08:00
2017-04-27 11:24:44 +08:00
bool IRAM_ATTR spicommon_dmaworkaround_req_reset ( int dmachan , dmaworkaround_cb_t cb , void * arg )
2017-03-31 15:05:25 +08:00
{
2022-09-15 18:26:12 +08:00
2017-04-27 11:24:44 +08:00
int otherchan = ( dmachan = = 1 ) ? 2 : 1 ;
2017-03-31 15:05:25 +08:00
bool ret ;
2018-03-23 19:28:03 +05:30
portENTER_CRITICAL_ISR ( & dmaworkaround_mux ) ;
2017-05-08 16:11:46 +08:00
if ( dmaworkaround_channels_busy [ otherchan - 1 ] ) {
2017-03-31 15:05:25 +08:00
//Other channel is busy. Call back when it's done.
2017-04-27 11:24:44 +08:00
dmaworkaround_cb = cb ;
dmaworkaround_cb_arg = arg ;
dmaworkaround_waiting_for_chan = otherchan ;
ret = false ;
2017-03-31 15:05:25 +08:00
} else {
//Reset DMA
2017-10-02 17:48:16 +11:00
periph_module_reset ( PERIPH_SPI_DMA_MODULE ) ;
2017-04-27 11:24:44 +08:00
ret = true ;
2017-03-31 15:05:25 +08:00
}
2018-03-23 19:28:03 +05:30
portEXIT_CRITICAL_ISR ( & dmaworkaround_mux ) ;
2017-03-31 15:05:25 +08:00
return ret ;
}
2019-07-16 16:33:30 +07:00
bool IRAM_ATTR spicommon_dmaworkaround_reset_in_progress ( void )
2017-03-31 15:05:25 +08:00
{
2017-04-27 11:24:44 +08:00
return ( dmaworkaround_waiting_for_chan ! = 0 ) ;
2017-03-31 15:05:25 +08:00
}
2017-04-27 11:24:44 +08:00
void IRAM_ATTR spicommon_dmaworkaround_idle ( int dmachan )
{
2018-03-23 19:28:03 +05:30
portENTER_CRITICAL_ISR ( & dmaworkaround_mux ) ;
2017-05-08 16:11:46 +08:00
dmaworkaround_channels_busy [ dmachan - 1 ] = 0 ;
2017-03-31 15:05:25 +08:00
if ( dmaworkaround_waiting_for_chan = = dmachan ) {
//Reset DMA
2017-10-02 17:48:16 +11:00
periph_module_reset ( PERIPH_SPI_DMA_MODULE ) ;
2017-04-27 11:24:44 +08:00
dmaworkaround_waiting_for_chan = 0 ;
2017-03-31 15:05:25 +08:00
//Call callback
dmaworkaround_cb ( dmaworkaround_cb_arg ) ;
}
2018-03-23 19:28:03 +05:30
portEXIT_CRITICAL_ISR ( & dmaworkaround_mux ) ;
2017-03-31 15:05:25 +08:00
}
2017-04-27 11:24:44 +08:00
void IRAM_ATTR spicommon_dmaworkaround_transfer_active ( int dmachan )
{
2018-03-23 19:28:03 +05:30
portENTER_CRITICAL_ISR ( & dmaworkaround_mux ) ;
2017-05-08 16:11:46 +08:00
dmaworkaround_channels_busy [ dmachan - 1 ] = 1 ;
2018-03-23 19:28:03 +05:30
portEXIT_CRITICAL_ISR ( & dmaworkaround_mux ) ;
2017-03-31 15:05:25 +08:00
}
2022-09-15 18:26:12 +08:00
# endif //#if CONFIG_IDF_TARGET_ESP32