mirror of
https://github.com/espressif/esp-idf.git
synced 2025-08-05 05:34:32 +02:00
fix(driver_spi): master driver change esp32p4 default src to pll
This commit is contained in:
@@ -89,6 +89,7 @@
|
||||
#define WIRE_DELAY 12.5
|
||||
|
||||
#else
|
||||
#define UNCONNECTED_PIN 8
|
||||
#define GPIO_DELAY 0
|
||||
#define ESP_SPI_SLAVE_TV 0
|
||||
#define WIRE_DELAY 12.5
|
||||
|
@@ -115,6 +115,7 @@ We have two bits to control the interrupt:
|
||||
#include "esp_private/periph_ctrl.h"
|
||||
#include "esp_private/spi_common_internal.h"
|
||||
#include "esp_private/spi_master_internal.h"
|
||||
#include "esp_private/esp_clk_tree_common.h"
|
||||
#include "driver/spi_master.h"
|
||||
#include "esp_clk_tree.h"
|
||||
#include "clk_ctrl_os.h"
|
||||
@@ -139,6 +140,7 @@ We have two bits to control the interrupt:
|
||||
#define SPI_MASTER_MALLOC_CAPS (MALLOC_CAP_DEFAULT)
|
||||
#endif
|
||||
|
||||
#define SPI_PERIPH_SRC_FREQ_MAX (80*1000*1000) //peripheral hardware limitation for clock source into peripheral
|
||||
typedef struct spi_device_t spi_device_t;
|
||||
|
||||
/// struct to hold private transaction data (like tx and rx buffer for DMA).
|
||||
@@ -383,6 +385,25 @@ int spi_get_freq_limit(bool gpio_is_used, int input_delay_ns)
|
||||
#endif
|
||||
}
|
||||
|
||||
#if SPI_LL_SUPPORT_CLK_SRC_PRE_DIV
|
||||
static uint32_t s_spi_find_clock_src_pre_div(uint32_t src_freq, uint32_t target_freq)
|
||||
{
|
||||
// pre division must be even and at least 2
|
||||
uint32_t min_div = ((src_freq / SPI_PERIPH_SRC_FREQ_MAX) + 1) & (~0x01UL);
|
||||
min_div = min_div < 2 ? 2 : min_div;
|
||||
|
||||
uint32_t total_div = src_freq / target_freq;
|
||||
// Loop the `div` to find a divisible value of `total_div`
|
||||
for (uint32_t pre_div = min_div; pre_div <= total_div; pre_div += 2) {
|
||||
if ((total_div % pre_div) || (total_div / pre_div) > SPI_LL_PERIPH_CLK_DIV_MAX) {
|
||||
continue;
|
||||
}
|
||||
return pre_div;
|
||||
}
|
||||
return min_div;
|
||||
}
|
||||
#endif //SPI_LL_SUPPORT_CLK_SRC_PRE_DIV
|
||||
|
||||
/*
|
||||
Add a device. This allocates a CS line for the device, allocates memory for the device structure and hooks
|
||||
up the CS pin to whatever is specified.
|
||||
@@ -404,36 +425,21 @@ esp_err_t spi_bus_add_device(spi_host_device_t host_id, const spi_device_interfa
|
||||
spi_host_t *host = bus_driver_ctx[host_id];
|
||||
const spi_bus_attr_t* bus_attr = host->bus_attr;
|
||||
SPI_CHECK(dev_config->spics_io_num < 0 || GPIO_IS_VALID_OUTPUT_GPIO(dev_config->spics_io_num), "spics pin invalid", ESP_ERR_INVALID_ARG);
|
||||
SPI_CHECK(dev_config->clock_speed_hz > 0, "invalid sclk speed", ESP_ERR_INVALID_ARG);
|
||||
#if SOC_SPI_SUPPORT_CLK_RC_FAST
|
||||
if (dev_config->clock_source == SPI_CLK_SRC_RC_FAST) {
|
||||
SPI_CHECK(periph_rtc_dig_clk8m_enable(), "the selected clock not available", ESP_ERR_INVALID_STATE);
|
||||
}
|
||||
#endif
|
||||
spi_clock_source_t clk_src = SPI_CLK_SRC_DEFAULT;
|
||||
uint32_t clock_source_hz = 0;
|
||||
uint32_t clock_source_div = 1;
|
||||
if (dev_config->clock_source) {
|
||||
clk_src = dev_config->clock_source;
|
||||
}
|
||||
spi_clock_source_t clk_src = dev_config->clock_source ? dev_config->clock_source : SPI_CLK_SRC_DEFAULT;
|
||||
esp_clk_tree_src_get_freq_hz(clk_src, ESP_CLK_TREE_SRC_FREQ_PRECISION_CACHED, &clock_source_hz);
|
||||
#if SPI_LL_SUPPORT_CLK_SRC_PRE_DIV
|
||||
SPI_CHECK((dev_config->clock_speed_hz > 0) && (dev_config->clock_speed_hz <= MIN(clock_source_hz / 2, (80 * 1000000))), "invalid sclk speed", ESP_ERR_INVALID_ARG);
|
||||
|
||||
if (clock_source_hz / 2 > (80 * 1000000)) { //clock_source_hz beyond peripheral HW limitation, calc pre-divider
|
||||
hal_utils_clk_info_t clk_cfg = {
|
||||
.src_freq_hz = clock_source_hz,
|
||||
.exp_freq_hz = dev_config->clock_speed_hz * 2, //we have (hs_clk = 2*mst_clk), calc hs_clk first
|
||||
.round_opt = HAL_DIV_ROUND,
|
||||
.min_integ = 1,
|
||||
.max_integ = SPI_LL_CLK_SRC_PRE_DIV_MAX / 2,
|
||||
};
|
||||
hal_utils_calc_clk_div_integer(&clk_cfg, &clock_source_div);
|
||||
}
|
||||
clock_source_div *= 2; //convert to mst_clk function divider
|
||||
clock_source_div = s_spi_find_clock_src_pre_div(clock_source_hz, dev_config->clock_speed_hz);
|
||||
clock_source_hz /= clock_source_div; //actual freq enter to SPI peripheral
|
||||
#else
|
||||
SPI_CHECK((dev_config->clock_speed_hz > 0) && (dev_config->clock_speed_hz <= clock_source_hz), "invalid sclk speed", ESP_ERR_INVALID_ARG);
|
||||
#endif
|
||||
SPI_CHECK(dev_config->clock_speed_hz <= clock_source_hz, "invalid sclk speed", ESP_ERR_INVALID_ARG);
|
||||
#ifdef CONFIG_IDF_TARGET_ESP32
|
||||
//The hardware looks like it would support this, but actually setting cs_ena_pretrans when transferring in full
|
||||
//duplex mode does absolutely nothing on the ESP32.
|
||||
|
@@ -68,7 +68,7 @@
|
||||
#define IDF_PERFORMANCE_MAX_SPI_PER_TRANS_NO_POLLING_NO_DMA 54
|
||||
|
||||
#elif CONFIG_IDF_TARGET_ESP32P4
|
||||
#define IDF_PERFORMANCE_MAX_SPI_CLK_FREQ 26*1000*1000
|
||||
#define IDF_PERFORMANCE_MAX_SPI_CLK_FREQ 40*1000*1000
|
||||
#define IDF_PERFORMANCE_MAX_SPI_PER_TRANS_NO_POLLING 44
|
||||
#define IDF_PERFORMANCE_MAX_SPI_PER_TRANS_POLLING 28
|
||||
#define IDF_PERFORMANCE_MAX_SPI_PER_TRANS_NO_POLLING_NO_DMA 26
|
||||
|
@@ -150,7 +150,7 @@ TEST_CASE("SPI Master clk_source and divider accuracy", "[spi]")
|
||||
int real_freq_khz;
|
||||
spi_device_get_actual_freq(handle, &real_freq_khz);
|
||||
// (byte_len * 8 / real_freq_hz) * 1000 000, (unit)us
|
||||
int trans_cost_us_predict = (float)TEST_CLK_BYTE_LEN * 8 * 1000 / real_freq_khz;
|
||||
int trans_cost_us_predict = (float)TEST_CLK_BYTE_LEN * 8 * 1000 / real_freq_khz + IDF_PERFORMANCE_MAX_SPI_PER_TRANS_POLLING;
|
||||
|
||||
// transaction and measure time
|
||||
start = esp_timer_get_time();
|
||||
|
@@ -3,6 +3,7 @@
|
||||
*
|
||||
* SPDX-License-Identifier: Apache-2.0
|
||||
*/
|
||||
#include <sys/param.h>
|
||||
#include "esp_log.h"
|
||||
#include "esp_attr.h"
|
||||
#include "soc/spi_periph.h"
|
||||
@@ -21,10 +22,6 @@
|
||||
#if (TEST_SPI_PERIPH_NUM >= 2)
|
||||
//These will only be enabled on chips with 2 or more SPI peripherals
|
||||
|
||||
#ifndef MIN
|
||||
#define MIN(a, b)((a) > (b)? (b): (a))
|
||||
#endif
|
||||
|
||||
/********************************************************************************
|
||||
* Test By Internal Connections
|
||||
********************************************************************************/
|
||||
@@ -109,10 +106,6 @@ static void local_test_start(spi_device_handle_t *spi, int freq, const spitest_p
|
||||
devcfg.flags |= SPI_DEVICE_NO_DUMMY;
|
||||
}
|
||||
|
||||
#if CONFIG_IDF_TARGET_ESP32P4 //TODO: IDF-8313, update P4 defaulte clock source
|
||||
devcfg.clock_source = SPI_CLK_SRC_SPLL;
|
||||
#endif
|
||||
|
||||
//slave config
|
||||
slvcfg.mode = pset->mode;
|
||||
slave_pull_up(&buscfg, slvcfg.spics_io_num);
|
||||
@@ -666,20 +659,10 @@ TEST_CASE("Slave receive correct data", "[spi]")
|
||||
}
|
||||
}
|
||||
|
||||
#if !TEMPORARY_DISABLED_FOR_TARGETS(ESP32S2, ESP32S3, ESP32C3, ESP32C2)
|
||||
//These tests are ESP32 only due to lack of runners
|
||||
#else // #if (TEST_SPI_PERIPH_NUM >= 2)
|
||||
|
||||
/********************************************************************************
|
||||
* Test By Master & Slave (2 boards)
|
||||
*
|
||||
* Wiring:
|
||||
* | Master | Slave |
|
||||
* | ------ | ----- |
|
||||
* | 12 | 19 |
|
||||
* | 13 | 23 |
|
||||
* | 14 | 18 |
|
||||
* | 15 | 5 |
|
||||
* | GND | GND |
|
||||
*
|
||||
* Test By Master & Slave (2 boards) using burger runner
|
||||
********************************************************************************/
|
||||
static void test_master_init(void **context);
|
||||
static void test_master_deinit(void *context);
|
||||
@@ -738,7 +721,7 @@ static void test_master_start(spi_device_handle_t *spi, int freq, const spitest_
|
||||
buspset.quadhd_io_num = UNCONNECTED_PIN;
|
||||
}
|
||||
spi_device_interface_config_t devpset = SPI_DEVICE_TEST_DEFAULT_CONFIG();
|
||||
devpset.spics_io_num = SPI2_IOMUX_PIN_NUM_CS;
|
||||
devpset.spics_io_num = PIN_NUM_CS;
|
||||
devpset.mode = pset->mode;
|
||||
const int cs_pretrans_max = 15;
|
||||
if (pset->dup == HALF_DUPLEX_MISO) {
|
||||
@@ -882,7 +865,7 @@ static void timing_slave_start(int speed, const spitest_param_set_t *pset, spite
|
||||
slv_buscfg.quadhd_io_num = UNCONNECTED_PIN;
|
||||
}
|
||||
spi_slave_interface_config_t slvcfg = SPI_SLAVE_TEST_DEFAULT_CONFIG();
|
||||
slvcfg.spics_io_num = SPI2_IOMUX_PIN_NUM_CS;
|
||||
slvcfg.spics_io_num = PIN_NUM_CS;
|
||||
slvcfg.mode = pset->mode;
|
||||
//Enable pull-ups on SPI lines so we don't detect rogue pulses when no master is connected.
|
||||
slave_pull_up(&slv_buscfg, slvcfg.spics_io_num);
|
||||
@@ -1267,9 +1250,6 @@ spitest_param_set_t mode_conf[] = {
|
||||
},
|
||||
};
|
||||
TEST_SPI_MASTER_SLAVE(MODE, mode_conf, "")
|
||||
|
||||
#endif // !TEMPORARY_DISABLED_FOR_TARGETS(ESP32S2, ESP32S3, ESP32C3, ESP32C2)
|
||||
|
||||
#endif // #if (TEST_SPI_PERIPH_NUM >= 2)
|
||||
|
||||
#define TEST_STEP_LEN 96
|
||||
@@ -1279,9 +1259,7 @@ static int s_spi_bus_freq[] = {
|
||||
IDF_PERFORMANCE_MAX_SPI_CLK_FREQ / 7,
|
||||
IDF_PERFORMANCE_MAX_SPI_CLK_FREQ / 4,
|
||||
IDF_PERFORMANCE_MAX_SPI_CLK_FREQ / 2,
|
||||
#if !CONFIG_IDF_TARGET_ESP32P4 //TODO: IDF-8313, update P4 defaulte clock source
|
||||
IDF_PERFORMANCE_MAX_SPI_CLK_FREQ,
|
||||
#endif
|
||||
};
|
||||
|
||||
//------------------------------------------- Full Duplex with DMA Freq test --------------------------------------
|
||||
|
@@ -38,10 +38,10 @@ extern "C" {
|
||||
#define HAL_SPI_SWAP_DATA_TX(data, len) HAL_SWAP32((uint32_t)(data) << (32 - len))
|
||||
#define SPI_LL_GET_HW(ID) (((ID)==1) ? &GPSPI2 : (((ID)==2) ? &GPSPI3 : NULL))
|
||||
|
||||
#define SPI_LL_DMA_MAX_BIT_LEN (1 << 18) //reg len: 18 bits
|
||||
#define SPI_LL_DMA_MAX_BIT_LEN SPI_MS_DATA_BITLEN
|
||||
#define SPI_LL_CPU_MAX_BIT_LEN (16 * 32) //Fifo len: 16 words
|
||||
#define SPI_LL_SUPPORT_CLK_SRC_PRE_DIV 1 //clock source have divider before peripheral
|
||||
#define SPI_LL_CLK_SRC_PRE_DIV_MAX 512//div1(8bit) * div2(8bit but set const 2)
|
||||
#define SPI_LL_PERIPH_CLK_DIV_MAX ((SPI_CLKCNT_N + 1) * (SPI_CLKDIV_PRE + 1)) //peripheral internal maxmum clock divider
|
||||
#define SPI_LL_MOSI_FREE_LEVEL 1 //Default level after bus initialized
|
||||
|
||||
/**
|
||||
|
Reference in New Issue
Block a user