From 2699bb49aea59f2145a426db69f901f117cf2ef0 Mon Sep 17 00:00:00 2001 From: Ivan Grokhotkov Date: Mon, 23 Apr 2018 20:10:38 +0800 Subject: [PATCH 1/2] sdspi: handle delayed responses for data write commands Command response tokens can be delayed from the original command by 1 to 8 bytes. In 4a2489b9, handling for delayed tokens was added for normal (no data) commands, and data read commands. This adds handling for delayed commands for data write commands. --- components/driver/sdspi_host.c | 73 +++++++++++++++++++++------------- 1 file changed, 45 insertions(+), 28 deletions(-) diff --git a/components/driver/sdspi_host.c b/components/driver/sdspi_host.c index e0faa73a68..ab9420eb12 100644 --- a/components/driver/sdspi_host.c +++ b/components/driver/sdspi_host.c @@ -67,6 +67,8 @@ static esp_err_t start_command_write_blocks(int slot, sdspi_hw_cmd_t *cmd, static esp_err_t start_command_default(int slot, int flags, sdspi_hw_cmd_t *cmd); +static esp_err_t poll_cmd_response(int slot, sdspi_hw_cmd_t *cmd); + /// A few helper functions /// Set CS high for given slot @@ -429,32 +431,21 @@ static esp_err_t start_command_default(int slot, int flags, sdspi_hw_cmd_t *cmd) /* response is a stuff byte from previous transfer, ignore it */ cmd->r1 = 0xff; } + if (ret != ESP_OK) { + ESP_LOGD(TAG, "%s: spi_device_transmit returned 0x%x", __func__, ret); + return ret; + } if (flags & SDSPI_CMD_FLAG_NORSP) { /* no (correct) response expected from the card, so skip polling loop */ ESP_LOGV(TAG, "%s: ignoring response byte", __func__); cmd->r1 = 0x00; } - int response_delay_bytes = SDSPI_RESPONSE_MAX_DELAY; - while ((cmd->r1 & SD_SPI_R1_NO_RESPONSE) != 0 && response_delay_bytes-- > 0) { - spi_transaction_t* t = get_transaction(slot); - *t = (spi_transaction_t) { - .flags = SPI_TRANS_USE_RXDATA | SPI_TRANS_USE_TXDATA, - .length = 8, - }; - t->tx_data[0] = 0xff; - ret = spi_device_transmit(spi_handle(slot), t); - uint8_t r1 = t->rx_data[0]; - release_transaction(slot); - if (ret != ESP_OK) { - return ret; - } - cmd->r1 = r1; + ret = poll_cmd_response(slot, cmd); + if (ret != ESP_OK) { + ESP_LOGD(TAG, "%s: poll_cmd_response returned 0x%x", __func__, ret); + return ret; } - if (cmd->r1 & SD_SPI_R1_NO_RESPONSE) { - ESP_LOGD(TAG, "%s: no response token found", __func__); - return ESP_ERR_TIMEOUT; - } - return ret; + return ESP_OK; } // Wait until MISO goes high @@ -563,6 +554,30 @@ static esp_err_t poll_data_token(int slot, spi_transaction_t* t, return ESP_ERR_TIMEOUT; } +static esp_err_t poll_cmd_response(int slot, sdspi_hw_cmd_t *cmd) +{ + int response_delay_bytes = SDSPI_RESPONSE_MAX_DELAY; + while ((cmd->r1 & SD_SPI_R1_NO_RESPONSE) != 0 && response_delay_bytes-- > 0) { + spi_transaction_t* t = get_transaction(slot); + *t = (spi_transaction_t) { + .flags = SPI_TRANS_USE_RXDATA | SPI_TRANS_USE_TXDATA, + .length = 8, + }; + t->tx_data[0] = 0xff; + esp_err_t ret = spi_device_transmit(spi_handle(slot), t); + uint8_t r1 = t->rx_data[0]; + release_transaction(slot); + if (ret != ESP_OK) { + return ret; + } + cmd->r1 = r1; + } + if (cmd->r1 & SD_SPI_R1_NO_RESPONSE) { + return ESP_ERR_TIMEOUT; + } + return ESP_OK; +} + /** * Receiving one or more blocks of data happens as follows: @@ -759,9 +774,17 @@ static esp_err_t start_command_write_blocks(int slot, sdspi_hw_cmd_t *cmd, if (ret != ESP_OK) { return ret; } + wait_for_transactions(slot); + + // Poll for command response which may be delayed up to 8 bytes + ret = poll_cmd_response(slot, cmd); + if (ret != ESP_OK) { + ESP_LOGD(TAG, "%s: poll_cmd_response returned 0x%x", __func__, ret); + return ret; + } + uint8_t start_token = tx_length <= SDSPI_MAX_DATA_LEN ? TOKEN_BLOCK_START : TOKEN_BLOCK_START_WRITE_MULTI; - wait_for_transactions(slot); while (tx_length > 0) { @@ -771,7 +794,7 @@ static esp_err_t start_command_write_blocks(int slot, sdspi_hw_cmd_t *cmd, .length = sizeof(start_token) * 8, .tx_buffer = &start_token }; - esp_err_t ret = spi_device_queue_trans(spi_handle(slot), t_start_token, 0); + ret = spi_device_queue_trans(spi_handle(slot), t_start_token, 0); if (ret != ESP_OK) { return ret; } @@ -816,12 +839,6 @@ static esp_err_t start_command_write_blocks(int slot, sdspi_hw_cmd_t *cmd, // Wait for data to be sent wait_for_transactions(slot); - // Check if R1 response for the command was correct - if (cmd->r1 != 0) { - ESP_LOGD(TAG, "%s: invalid R1 response: 0x%02x", __func__, cmd->r1); - return ESP_ERR_INVALID_RESPONSE; - } - // Poll for response spi_transaction_t* t_poll = get_transaction(slot); ret = poll_response_token(slot, t_poll, cmd->timeout_ms); From 5a63faa516a4ef169c4652667e801771a03e770d Mon Sep 17 00:00:00 2001 From: Ivan Grokhotkov Date: Mon, 23 Apr 2018 20:14:23 +0800 Subject: [PATCH 2/2] sdmmc: enable more test cases Some test cases did not run automatically in CI, this change enables them. --- components/sdmmc/test/test_sd.c | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/components/sdmmc/test/test_sd.c b/components/sdmmc/test/test_sd.c index 33b3890ccd..1808a0cdd0 100644 --- a/components/sdmmc/test/test_sd.c +++ b/components/sdmmc/test/test_sd.c @@ -177,7 +177,7 @@ static void read_write_test(sdmmc_card_t* card) do_single_write_read_test(card, card->csd.capacity/2, 128, 1); } -TEST_CASE("can write and read back blocks", "[sd][test_env=UT_T1_SDMODE][ignore]") +TEST_CASE("can write and read back blocks", "[sd][test_env=UT_T1_SDMODE]") { sdmmc_host_t config = SDMMC_HOST_DEFAULT(); config.max_freq_khz = SDMMC_FREQ_HIGHSPEED; @@ -192,7 +192,7 @@ TEST_CASE("can write and read back blocks", "[sd][test_env=UT_T1_SDMODE][ignore] TEST_ESP_OK(sdmmc_host_deinit()); } -TEST_CASE("can write and read back blocks(using SPI)", "[sdspi][test_env=UT_T1_SPIMODE][ignore]") +TEST_CASE("can write and read back blocks(using SPI)", "[sdspi][test_env=UT_T1_SPIMODE]") { sdmmc_host_t config = SDSPI_HOST_DEFAULT(); sdspi_slot_config_t slot_config = SDSPI_SLOT_CONFIG_DEFAULT(); @@ -206,7 +206,7 @@ TEST_CASE("can write and read back blocks(using SPI)", "[sdspi][test_env=UT_T1_S TEST_ESP_OK(sdspi_host_deinit()); } -TEST_CASE("reads and writes with an unaligned buffer", "[sd][test_env=UT_T1_SDMODE][ignore]") +TEST_CASE("reads and writes with an unaligned buffer", "[sd][test_env=UT_T1_SDMODE]") { sdmmc_host_t config = SDMMC_HOST_DEFAULT(); sdmmc_slot_config_t slot_config = SDMMC_SLOT_CONFIG_DEFAULT();