Merge branch 'feat/sdmmc_spi_no_crc' into 'master'

feat(sdmmc_io): support sending CMD53 with fixed address, bypass sdspi crc check

Closes IDF-9387

See merge request espressif/esp-idf!30643
This commit is contained in:
Michael (XIAO Xufeng) 2024-05-24 20:27:28 +08:00
commit 3e877842cd
4 changed files with 138 additions and 20 deletions

View File

@ -14,6 +14,7 @@
#include "freertos/task.h" #include "freertos/task.h"
#include "esp_timer.h" #include "esp_timer.h"
#include "ccomp_timer.h" #include "ccomp_timer.h"
#include "string.h"
#include "sdmmc_cmd.h" #include "sdmmc_cmd.h"
#include "driver/sdmmc_host.h" #include "driver/sdmmc_host.h"
@ -43,7 +44,7 @@ typedef struct {
---------------------------------------------------------------*/ ---------------------------------------------------------------*/
static sdmmc_card_t s_card; static sdmmc_card_t s_card;
static void s_master_init(test_sdio_param_t *host_param, essl_handle_t *out_handle) static void s_master_init(test_sdio_param_t *host_param, essl_handle_t *out_handle, sdmmc_card_t** out_card)
{ {
sdmmc_host_t host_config = (sdmmc_host_t)SDMMC_HOST_DEFAULT(); sdmmc_host_t host_config = (sdmmc_host_t)SDMMC_HOST_DEFAULT();
host_config.flags = host_param->host_flags; host_config.flags = host_param->host_flags;
@ -81,6 +82,10 @@ static void s_master_init(test_sdio_param_t *host_param, essl_handle_t *out_hand
TEST_ESP_OK(essl_sdio_init_dev(out_handle, &essl_sdio_config)); TEST_ESP_OK(essl_sdio_init_dev(out_handle, &essl_sdio_config));
TEST_ESP_OK(essl_init(*out_handle, TEST_TIMEOUT_MAX)); TEST_ESP_OK(essl_init(*out_handle, TEST_TIMEOUT_MAX));
if (out_card) {
*out_card = card;
}
} }
static void s_master_deinit(void) static void s_master_deinit(void)
@ -114,7 +119,7 @@ TEST_CASE("SDIO_SDMMC: test interrupt", "[sdio]")
.max_freq_khz = SDMMC_FREQ_HIGHSPEED, .max_freq_khz = SDMMC_FREQ_HIGHSPEED,
}; };
//essl init and sdmmc init //essl init and sdmmc init
s_master_init(&test_param, &handle); s_master_init(&test_param, &handle, NULL);
TEST_ESP_OK(essl_set_intr_ena(handle, TEST_INT_MASK_ALL, TEST_TIMEOUT_MAX)); TEST_ESP_OK(essl_set_intr_ena(handle, TEST_INT_MASK_ALL, TEST_TIMEOUT_MAX));
ret = essl_wait_int(handle, 0); ret = essl_wait_int(handle, 0);
@ -149,7 +154,7 @@ TEST_CASE("SDIO_SDMMC: test register", "[sdio]")
.max_freq_khz = SDMMC_FREQ_HIGHSPEED, .max_freq_khz = SDMMC_FREQ_HIGHSPEED,
}; };
//essl init and sdmmc init //essl init and sdmmc init
s_master_init(&test_param, &handle); s_master_init(&test_param, &handle, NULL);
uint32_t init_val = 30; uint32_t init_val = 30;
srand(850); srand(850);
@ -183,7 +188,7 @@ TEST_CASE("SDIO_SDMMC: test reset", "[sdio]")
.max_freq_khz = SDMMC_FREQ_HIGHSPEED, .max_freq_khz = SDMMC_FREQ_HIGHSPEED,
}; };
//essl init and sdmmc init //essl init and sdmmc init
s_master_init(&test_param, &handle); s_master_init(&test_param, &handle, NULL);
//wait for the slave to stop, reset and start again //wait for the slave to stop, reset and start again
vTaskDelay(10); vTaskDelay(10);
@ -215,6 +220,66 @@ TEST_CASE("SDIO_SDMMC: test reset", "[sdio]")
s_master_deinit(); s_master_deinit();
} }
/*---------------------------------------------------------------
SDMMC_SDIO: test fixed addr
---------------------------------------------------------------*/
#include "soc/soc.h"
#define HOST_SLCHOST_CONF_W0_REG (DR_REG_SLCHOST_BASE + 0x6C)
TEST_CASE("SDIO_SDMMC: test fixed addr", "[sdio]")
{
essl_handle_t handle = NULL;
sdmmc_card_t* card;
test_sdio_param_t test_param = {
.host_flags = SDMMC_HOST_FLAG_4BIT | SDMMC_HOST_FLAG_ALLOC_ALIGNED_BUF,
.max_freq_khz = SDMMC_FREQ_HIGHSPEED,
};
//essl init and sdmmc init
s_master_init(&test_param, &handle, &card);
vTaskDelay(10);
const int test_size = 128;
const int write_addr = 6;
uint8_t buf[test_size] = {};
srand(850);
for (int i = 0; i < test_size; i++) {
buf[i] = rand();
}
ESP_LOG_BUFFER_HEX("write_val", buf, test_size);
TEST_ESP_OK(sdmmc_io_write_bytes(card, 1, ((HOST_SLCHOST_CONF_W0_REG + write_addr) & 0x3FF) | SDMMC_IO_FIXED_ADDR, buf, test_size));
const int max_size = 64;
uint8_t read_buf[max_size] = {};
TEST_ESP_OK(sdmmc_io_read_bytes(card, 1, HOST_SLCHOST_CONF_W0_REG & 0x3FF, read_buf, max_size));
ESP_LOG_BUFFER_HEX("read_all", read_buf, max_size);
for (int i = 0; i < max_size; i++) {
if (i >= 24 && i < 28) {
continue;
}
if (i >= 32 && i < 48) {
continue;
}
if (i == write_addr) {
TEST_ASSERT_EQUAL_HEX8(buf[test_size - 1], read_buf[i]);
} else {
TEST_ASSERT_EQUAL_HEX8(0xcc, read_buf[i]);
}
}
const int read_size = (test_size > max_size ? max_size : test_size);
memset(read_buf, 0, read_size);
TEST_ESP_OK(sdmmc_io_read_bytes(card, 1, ((HOST_SLCHOST_CONF_W0_REG + write_addr) & 0x3FF) | SDMMC_IO_FIXED_ADDR, read_buf, read_size));
ESP_LOG_BUFFER_HEX("read_fixed", read_buf, read_size);
for (int i = 0; i < read_size; i++) {
TEST_ASSERT_EQUAL_HEX8(buf[test_size - 1], read_buf[i]);
}
s_send_finish_test(handle);
s_master_deinit();
}
/*--------------------------------------------------------------- /*---------------------------------------------------------------
Transaction Tests Transaction Tests
---------------------------------------------------------------*/ ---------------------------------------------------------------*/
@ -241,7 +306,7 @@ static void test_from_host(bool check_data)
ESP_LOGI(TAG, "host speed: %"PRIu32" kHz", test_param_lists[i].max_freq_khz); ESP_LOGI(TAG, "host speed: %"PRIu32" kHz", test_param_lists[i].max_freq_khz);
essl_handle_t handle = NULL; essl_handle_t handle = NULL;
s_master_init(&test_param_lists[i], &handle); s_master_init(&test_param_lists[i], &handle, NULL);
// Two counters are used. The `esp_timer_get_time()` is for the typical time, and the // Two counters are used. The `esp_timer_get_time()` is for the typical time, and the
// `ccomp_timer` is for performance test to reduce influence caused by cache miss. // `ccomp_timer` is for performance test to reduce influence caused by cache miss.
@ -298,7 +363,7 @@ static void test_to_host(bool check_data)
ESP_LOGI(TAG, "host speed: %"PRIu32" kHz", test_param_lists[i].max_freq_khz); ESP_LOGI(TAG, "host speed: %"PRIu32" kHz", test_param_lists[i].max_freq_khz);
essl_handle_t handle = NULL; essl_handle_t handle = NULL;
s_master_init(&test_param_lists[i], &handle); s_master_init(&test_param_lists[i], &handle, NULL);
esp_err_t ret; esp_err_t ret;
int offset = 0; int offset = 0;

View File

@ -155,6 +155,24 @@ TEST_CASE("SDIO_Slave: test reset", "[sdio]")
sdio_slave_deinit(); sdio_slave_deinit();
} }
/*---------------------------------------------------------------
SDMMC_SDIO: test fixed addr
---------------------------------------------------------------*/
TEST_CASE("SDIO_Slave: test fixed addr", "[sdio]")
{
s_slave_init(SDIO_SLAVE_SEND_PACKET);
TEST_ESP_OK(sdio_slave_start());
ESP_LOGI(TAG, "slave ready");
for (int i = 0; i < 64; i++) {
sdio_slave_write_reg(i, 0xcc);
}
wait_for_finish(&s_test_slv_ctx);
sdio_slave_stop();
sdio_slave_deinit();
}
/*--------------------------------------------------------------- /*---------------------------------------------------------------
Transaction Tests Transaction Tests
---------------------------------------------------------------*/ ---------------------------------------------------------------*/

View File

@ -1,5 +1,5 @@
/* /*
* SPDX-FileCopyrightText: 2015-2021 Espressif Systems (Shanghai) CO LTD * SPDX-FileCopyrightText: 2015-2024 Espressif Systems (Shanghai) CO LTD
* *
* SPDX-License-Identifier: Apache-2.0 * SPDX-License-Identifier: Apache-2.0
*/ */
@ -14,6 +14,10 @@
extern "C" { extern "C" {
#endif #endif
/** Call `sdmmc_io_read_bytes`, `sdmmc_io_write_bytes`, `sdmmc_io_read_blocks` or `sdmmc_io_write_bocks` APIs with
* address ORed by this flag to send CMD53 with OP Code clear (fixed address) */
#define SDMMC_IO_FIXED_ADDR BIT(31)
/** /**
* Probe and initialize SD/MMC card using given host * Probe and initialize SD/MMC card using given host
* *
@ -196,6 +200,9 @@ esp_err_t sdmmc_io_write_byte(sdmmc_card_t* card, uint32_t function,
* This function performs read operation using CMD53 in byte mode. * This function performs read operation using CMD53 in byte mode.
* For block mode, see sdmmc_io_read_blocks. * For block mode, see sdmmc_io_read_blocks.
* *
* By default OP Code is set (incrementing address). To send CMD53 without this bit, OR the argument `addr` with
* `SDMMC_IO_FIXED_ADDR`.
*
* @param card pointer to card information structure previously initialized * @param card pointer to card information structure previously initialized
* using sdmmc_card_init * using sdmmc_card_init
* @param function IO function number * @param function IO function number
@ -218,6 +225,9 @@ esp_err_t sdmmc_io_read_bytes(sdmmc_card_t* card, uint32_t function,
* This function performs write operation using CMD53 in byte mode. * This function performs write operation using CMD53 in byte mode.
* For block mode, see sdmmc_io_write_blocks. * For block mode, see sdmmc_io_write_blocks.
* *
* By default OP Code is set (incrementing address). To send CMD53 without this bit, OR the argument `addr` with
* `SDMMC_IO_FIXED_ADDR`.
*
* @param card pointer to card information structure previously initialized * @param card pointer to card information structure previously initialized
* using sdmmc_card_init * using sdmmc_card_init
* @param function IO function number * @param function IO function number
@ -239,6 +249,9 @@ esp_err_t sdmmc_io_write_bytes(sdmmc_card_t* card, uint32_t function,
* This function performs read operation using CMD53 in block mode. * This function performs read operation using CMD53 in block mode.
* For byte mode, see sdmmc_io_read_bytes. * For byte mode, see sdmmc_io_read_bytes.
* *
* By default OP Code is set (incrementing address). To send CMD53 without this bit, OR the argument `addr` with
* `SDMMC_IO_FIXED_ADDR`.
*
* @param card pointer to card information structure previously initialized * @param card pointer to card information structure previously initialized
* using sdmmc_card_init * using sdmmc_card_init
* @param function IO function number * @param function IO function number
@ -261,6 +274,9 @@ esp_err_t sdmmc_io_read_blocks(sdmmc_card_t* card, uint32_t function,
* This function performs write operation using CMD53 in block mode. * This function performs write operation using CMD53 in block mode.
* For byte mode, see sdmmc_io_write_bytes. * For byte mode, see sdmmc_io_write_bytes.
* *
* By default OP Code is set (incrementing address). To send CMD53 without this bit, OR the argument `addr` with
* `SDMMC_IO_FIXED_ADDR`.
*
* @param card pointer to card information structure previously initialized * @param card pointer to card information structure previously initialized
* using sdmmc_card_init * using sdmmc_card_init
* @param function IO function number * @param function IO function number

View File

@ -336,6 +336,13 @@ esp_err_t sdmmc_io_rw_extended(sdmmc_card_t* card, int func,
esp_err_t sdmmc_io_read_bytes(sdmmc_card_t* card, uint32_t function, esp_err_t sdmmc_io_read_bytes(sdmmc_card_t* card, uint32_t function,
uint32_t addr, void* dst, size_t size) uint32_t addr, void* dst, size_t size)
{ {
uint32_t arg = SD_ARG_CMD53_READ | SD_ARG_CMD53_INCREMENT;
//Extract and unset the bit used to indicate the OP Code (inverted logic)
if (addr & SDMMC_IO_FIXED_ADDR) {
arg &= ~SD_ARG_CMD53_INCREMENT;
addr &= ~SDMMC_IO_FIXED_ADDR;
}
/* host quirk: SDIO transfer with length not divisible by 4 bytes /* host quirk: SDIO transfer with length not divisible by 4 bytes
* has to be split into two transfers: one with aligned length, * has to be split into two transfers: one with aligned length,
* the other one for the remaining 1-3 bytes. * the other one for the remaining 1-3 bytes.
@ -347,9 +354,7 @@ esp_err_t sdmmc_io_read_bytes(sdmmc_card_t* card, uint32_t function,
// Note: sdmmc_io_rw_extended has an internal timeout, // Note: sdmmc_io_rw_extended has an internal timeout,
// typically SDMMC_DEFAULT_CMD_TIMEOUT_MS // typically SDMMC_DEFAULT_CMD_TIMEOUT_MS
esp_err_t err = sdmmc_io_rw_extended(card, function, addr, esp_err_t err = sdmmc_io_rw_extended(card, function, addr, arg, pc_dst, will_transfer);
SD_ARG_CMD53_READ | SD_ARG_CMD53_INCREMENT,
pc_dst, will_transfer);
if (unlikely(err != ESP_OK)) { if (unlikely(err != ESP_OK)) {
return err; return err;
} }
@ -363,18 +368,22 @@ esp_err_t sdmmc_io_read_bytes(sdmmc_card_t* card, uint32_t function,
esp_err_t sdmmc_io_write_bytes(sdmmc_card_t* card, uint32_t function, esp_err_t sdmmc_io_write_bytes(sdmmc_card_t* card, uint32_t function,
uint32_t addr, const void* src, size_t size) uint32_t addr, const void* src, size_t size)
{ {
uint32_t arg = SD_ARG_CMD53_WRITE | SD_ARG_CMD53_INCREMENT;
//Extract and unset the bit used to indicate the OP Code (inverted logic)
if (addr & SDMMC_IO_FIXED_ADDR) {
arg &= ~SD_ARG_CMD53_INCREMENT;
addr &= ~SDMMC_IO_FIXED_ADDR;
}
/* same host quirk as in sdmmc_io_read_bytes */ /* same host quirk as in sdmmc_io_read_bytes */
const uint8_t *pc_src = (const uint8_t*) src; const uint8_t *pc_src = (const uint8_t*) src;
while (size > 0) { while (size > 0) {
size_t size_aligned = size & (~3); size_t size_aligned = size & (~3);
size_t will_transfer = size_aligned > 0 ? size_aligned : size; size_t will_transfer = size_aligned > 0 ? size_aligned : size;
// Note: sdmmc_io_rw_extended has an internal timeout, // Note: sdmmc_io_rw_extended has an internal timeout,
// typically SDMMC_DEFAULT_CMD_TIMEOUT_MS // typically SDMMC_DEFAULT_CMD_TIMEOUT_MS
esp_err_t err = sdmmc_io_rw_extended(card, function, addr, esp_err_t err = sdmmc_io_rw_extended(card, function, addr, arg, (void*) pc_src, will_transfer);
SD_ARG_CMD53_WRITE | SD_ARG_CMD53_INCREMENT,
(void*) pc_src, will_transfer);
if (unlikely(err != ESP_OK)) { if (unlikely(err != ESP_OK)) {
return err; return err;
} }
@ -388,27 +397,37 @@ esp_err_t sdmmc_io_write_bytes(sdmmc_card_t* card, uint32_t function,
esp_err_t sdmmc_io_read_blocks(sdmmc_card_t* card, uint32_t function, esp_err_t sdmmc_io_read_blocks(sdmmc_card_t* card, uint32_t function,
uint32_t addr, void* dst, size_t size) uint32_t addr, void* dst, size_t size)
{ {
uint32_t arg = SD_ARG_CMD53_READ | SD_ARG_CMD53_INCREMENT | SD_ARG_CMD53_BLOCK_MODE;
//Extract and unset the bit used to indicate the OP Code (inverted logic)
if (addr & SDMMC_IO_FIXED_ADDR) {
arg &= ~SD_ARG_CMD53_INCREMENT;
addr &= ~SDMMC_IO_FIXED_ADDR;
}
esp_dma_mem_info_t dma_mem_info; esp_dma_mem_info_t dma_mem_info;
card->host.get_dma_info(card->host.slot, &dma_mem_info); card->host.get_dma_info(card->host.slot, &dma_mem_info);
if (unlikely(!esp_dma_is_buffer_alignment_satisfied(dst, size, dma_mem_info))) { if (unlikely(!esp_dma_is_buffer_alignment_satisfied(dst, size, dma_mem_info))) {
return ESP_ERR_INVALID_ARG; return ESP_ERR_INVALID_ARG;
} }
return sdmmc_io_rw_extended(card, function, addr, return sdmmc_io_rw_extended(card, function, addr, arg, dst, size);
SD_ARG_CMD53_READ | SD_ARG_CMD53_INCREMENT | SD_ARG_CMD53_BLOCK_MODE,
dst, size);
} }
esp_err_t sdmmc_io_write_blocks(sdmmc_card_t* card, uint32_t function, esp_err_t sdmmc_io_write_blocks(sdmmc_card_t* card, uint32_t function,
uint32_t addr, const void* src, size_t size) uint32_t addr, const void* src, size_t size)
{ {
uint32_t arg = SD_ARG_CMD53_WRITE | SD_ARG_CMD53_INCREMENT | SD_ARG_CMD53_BLOCK_MODE;
//Extract and unset the bit used to indicate the OP Code (inverted logic)
if (addr & SDMMC_IO_FIXED_ADDR) {
arg &= ~SD_ARG_CMD53_INCREMENT;
addr &= ~SDMMC_IO_FIXED_ADDR;
}
esp_dma_mem_info_t dma_mem_info; esp_dma_mem_info_t dma_mem_info;
card->host.get_dma_info(card->host.slot, &dma_mem_info); card->host.get_dma_info(card->host.slot, &dma_mem_info);
if (unlikely(!esp_dma_is_buffer_alignment_satisfied(src, size, dma_mem_info))) { if (unlikely(!esp_dma_is_buffer_alignment_satisfied(src, size, dma_mem_info))) {
return ESP_ERR_INVALID_ARG; return ESP_ERR_INVALID_ARG;
} }
return sdmmc_io_rw_extended(card, function, addr, return sdmmc_io_rw_extended(card, function, addr, arg, (void*) src, size);
SD_ARG_CMD53_WRITE | SD_ARG_CMD53_INCREMENT | SD_ARG_CMD53_BLOCK_MODE,
(void*) src, size);
} }
esp_err_t sdmmc_io_enable_int(sdmmc_card_t* card) esp_err_t sdmmc_io_enable_int(sdmmc_card_t* card)