| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482 |
- /* SDIO example, host (uses sdmmc_host/sdspi_host driver)
- This example code is in the Public Domain (or CC0 licensed, at your option.)
- Unless required by applicable law or agreed to in writing, this
- software is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
- CONDITIONS OF ANY KIND, either express or implied.
- */
- #include <stdio.h>
- #include <stdint.h>
- #include <stddef.h>
- #include <string.h>
- #include "freertos/FreeRTOS.h"
- #include "freertos/task.h"
- #include "freertos/semphr.h"
- #include "freertos/queue.h"
- #include "soc/sdmmc_periph.h"
- #include "soc/sdio_slave_periph.h"
- #include "esp_log.h"
- #include "esp_attr.h"
- #include "esp_serial_slave_link/essl_sdio.h"
- #include "sdkconfig.h"
- #include "driver/sdmmc_host.h"
- #include "driver/sdspi_host.h"
- #define TIMEOUT_MAX UINT32_MAX
- #define GPIO_B1 21
- #if CONFIG_EXAMPLE_SLAVE_B1
- #define SLAVE_PWR_GPIO GPIO_B1
- #endif
- /*
- sdio host example.
- This example is supposed to work together with the sdio slave example. It uses the pins as follows:
- * Host Slave
- * IO14 CLK
- * IO15 CMD
- * IO2 D0
- * IO4 D1
- * IO12 D2
- * IO13 D3
- This is the only pins that can be used in standard ESP modules. The other set of pins (6, 11, 7, 8, 9, 10)
- are occupied by the spi bus communicating with the flash.
- Protocol Above the ESP slave service:
- - Interrupts:
- 0 is used to notify the slave to read the register 0.
- - Registers:
- - 0 is the register to hold tasks. Bits:
- - 0: the slave should reset.
- - 1: the slave should send interrupts.
- - 2: the slave should write the shared registers acoording to the value in register 1.
- - 1 is the register to hold test value.
- - other registers will be written by the slave for testing.
- - FIFO:
- The receving FIFO is size of 256 bytes.
- When the host writes something to slave recv FIFO, the slave should return it as is to the sending FIFO.
- The example works as following process:
- 1. reset the slave.
- 2. tell the slave to write registers and read them back.
- 3. tell the slave to send interrupts to the host.
- 4. send data to slave FIFO and read them back.
- 5. loop step 4.
- */
- #define WRITE_BUFFER_LEN 4096
- #define READ_BUFFER_LEN 4096
- #define SLAVE_BUFFER_SIZE 128
- static const char TAG[] = "example_host";
- #define SDIO_INTERRUPT_LINE GPIO_NUM_4 //DATA1
- #define SLAVE_INTR_NOTIFY 0
- #define SLAVE_REG_JOB 0
- #define SLAVE_REG_VALUE 1
- typedef enum {
- JOB_IDLE = 0,
- JOB_RESET = 1,
- JOB_SEND_INT = 2,
- JOB_WRITE_REG = 4,
- } example_job_t;
- //host use this to inform the slave it should reset its counters
- esp_err_t slave_reset(essl_handle_t handle)
- {
- esp_err_t ret;
- ESP_LOGI(TAG, "send reset to slave...");
- ret = essl_write_reg(handle, 0, JOB_RESET, NULL, TIMEOUT_MAX);
- if (ret != ESP_OK) {
- return ret;
- }
- ret = essl_send_slave_intr(handle, BIT(SLAVE_INTR_NOTIFY), TIMEOUT_MAX);
- if (ret != ESP_OK) {
- return ret;
- }
- vTaskDelay(500 / portTICK_PERIOD_MS);
- ret = essl_wait_for_ready(handle, TIMEOUT_MAX);
- ESP_LOGI(TAG, "slave io ready");
- return ret;
- }
- #ifdef CONFIG_EXAMPLE_SDIO_OVER_SPI
- static void gpio_d2_set_high(void)
- {
- gpio_config_t d2_config = {
- .pin_bit_mask = BIT64(SDIO_SLAVE_SLOT1_IOMUX_PIN_NUM_D2),
- .mode = GPIO_MODE_OUTPUT,
- .pull_up_en = true,
- };
- gpio_config(&d2_config);
- gpio_set_level(SDIO_SLAVE_SLOT1_IOMUX_PIN_NUM_D2, 1);
- }
- #endif
- static esp_err_t print_sdio_cis_information(sdmmc_card_t* card)
- {
- const size_t cis_buffer_size = 256;
- uint8_t cis_buffer[cis_buffer_size];
- size_t cis_data_len = 1024; //specify maximum searching range to avoid infinite loop
- esp_err_t ret = ESP_OK;
- ret = sdmmc_io_get_cis_data(card, cis_buffer, cis_buffer_size, &cis_data_len);
- if (ret == ESP_ERR_INVALID_SIZE) {
- int temp_buf_size = cis_data_len;
- uint8_t* temp_buf = malloc(temp_buf_size);
- assert(temp_buf);
- ESP_LOGW(TAG, "CIS data longer than expected, temporary buffer allocated.");
- ret = sdmmc_io_get_cis_data(card, temp_buf, temp_buf_size, &cis_data_len);
- ESP_ERROR_CHECK(ret);
- sdmmc_io_print_cis_info(temp_buf, cis_data_len, NULL);
- free(temp_buf);
- } else if (ret == ESP_OK) {
- sdmmc_io_print_cis_info(cis_buffer, cis_data_len, NULL);
- } else {
- //including ESP_ERR_NOT_FOUND
- ESP_LOGE(TAG, "failed to get the entire CIS data.");
- abort();
- }
- return ESP_OK;
- }
- //host use this to initialize the slave card as well as SDIO registers
- esp_err_t slave_init(essl_handle_t* handle)
- {
- esp_err_t err;
- /* Probe */
- #ifndef CONFIG_EXAMPLE_SDIO_OVER_SPI
- sdmmc_host_t config = SDMMC_HOST_DEFAULT();
- #ifdef CONFIG_EXAMPLE_SDIO_4BIT
- ESP_LOGI(TAG, "Probe using SD 4-bit...\n");
- config.flags = SDMMC_HOST_FLAG_4BIT;
- #else
- ESP_LOGI(TAG, "Probe using SD 1-bit...\n");
- config.flags = SDMMC_HOST_FLAG_1BIT;
- #endif
- #ifdef CONFIG_EXAMPLE_SDIO_HIGHSPEED
- config.max_freq_khz = SDMMC_FREQ_HIGHSPEED;
- #else
- config.max_freq_khz = SDMMC_FREQ_DEFAULT;
- #endif
- sdmmc_slot_config_t slot_config = SDMMC_SLOT_CONFIG_DEFAULT();
- /* Note: For small devkits there may be no pullups on the board.
- This enables the internal pullups to help evaluate the driver quickly.
- However the internal pullups are not sufficient and not reliable,
- please make sure external pullups are connected to the bus in your
- real design.
- */
- //slot_config.flags = SDMMC_SLOT_FLAG_INTERNAL_PULLUP;
- err = sdmmc_host_init();
- ESP_ERROR_CHECK(err);
- err = sdmmc_host_init_slot(SDMMC_HOST_SLOT_1, &slot_config);
- ESP_ERROR_CHECK(err);
- #else //over SPI
- sdspi_device_config_t dev_config = SDSPI_DEVICE_CONFIG_DEFAULT();
- dev_config.gpio_cs = SDIO_SLAVE_SLOT1_IOMUX_PIN_NUM_D3;
- dev_config.gpio_int = SDIO_SLAVE_SLOT1_IOMUX_PIN_NUM_D1;
- err = gpio_install_isr_service(0);
- ESP_ERROR_CHECK(err);
- spi_bus_config_t bus_config = {
- .mosi_io_num = SDIO_SLAVE_SLOT1_IOMUX_PIN_NUM_CMD,
- .miso_io_num = SDIO_SLAVE_SLOT1_IOMUX_PIN_NUM_D0,
- .sclk_io_num = SDIO_SLAVE_SLOT1_IOMUX_PIN_NUM_CLK,
- .quadwp_io_num = -1,
- .quadhd_io_num = -1,
- .max_transfer_sz = 4000,
- };
- err = spi_bus_initialize(dev_config.host_id, &bus_config, 1);
- ESP_ERROR_CHECK(err);
- sdspi_device_handle_t sdspi_handle;
- err = sdspi_host_init();
- ESP_ERROR_CHECK(err);
- err = sdspi_host_init_device(&slot_config, &sdspi_handle);
- ESP_ERROR_CHECK(err);
- ESP_LOGI(TAG, "Probe using SPI...\n");
- sdmmc_host_t config = SDSPI_HOST_DEFAULT();
- config.slot = sdspi_handle;
- //we have to pull up all the slave pins even when the pin is not used
- gpio_d2_set_high();
- #endif //over SPI
- sdmmc_card_t *card = (sdmmc_card_t *)malloc(sizeof(sdmmc_card_t));
- if (card == NULL) {
- return ESP_ERR_NO_MEM;
- }
- for (;;) {
- if (sdmmc_card_init(&config, card) == ESP_OK) {
- break;
- }
- ESP_LOGW(TAG, "slave init failed, retry...");
- vTaskDelay(1000 / portTICK_PERIOD_MS);
- }
- sdmmc_card_print_info(stdout, card);
- gpio_pullup_en(14);
- gpio_pulldown_dis(14);
- gpio_pullup_en(15);
- gpio_pulldown_dis(15);
- gpio_pullup_en(2);
- gpio_pulldown_dis(2);
- gpio_pullup_en(4);
- gpio_pulldown_dis(4);
- gpio_pullup_en(12);
- gpio_pulldown_dis(12);
- gpio_pullup_en(13);
- gpio_pulldown_dis(13);
- essl_sdio_config_t ser_config = {
- .card = card,
- .recv_buffer_size = SLAVE_BUFFER_SIZE,
- };
- err = essl_sdio_init_dev(handle, &ser_config);
- ESP_ERROR_CHECK(err);
- esp_err_t ret = essl_init(*handle, TIMEOUT_MAX);
- ESP_ERROR_CHECK(ret);
- ret = print_sdio_cis_information(card);
- ESP_ERROR_CHECK(ret);
- return ret;
- }
- void slave_power_on(void)
- {
- #ifdef SLAVE_PWR_GPIO
- int level_active;
- #ifdef CONFIG_EXAMPLE_SLAVE_PWR_NEGTIVE_ACTIVE
- level_active = 0;
- #else
- level_active = 1;
- #endif
- gpio_config_t cfg = {
- .pin_bit_mask = BIT64(GPIO_B1),
- .mode = GPIO_MODE_OUTPUT,
- .pull_up_en = false,
- .pull_down_en = false,
- .intr_type = GPIO_INTR_DISABLE,
- };
- gpio_config(&cfg);
- gpio_set_level(GPIO_B1, !level_active);
- vTaskDelay(100);
- gpio_set_level(SLAVE_PWR_GPIO, level_active);
- vTaskDelay(100);
- #endif
- }
- DMA_ATTR uint8_t rcv_buffer[READ_BUFFER_LEN];
- static esp_err_t get_intr(essl_handle_t handle, uint32_t* out_raw, uint32_t* out_st)
- {
- esp_err_t ret = ESP_OK;
- #ifndef CONFIG_EXAMPLE_NO_INTR_LINE
- ret = essl_wait_int(handle, 0);
- if (ret != ESP_OK) {
- return ret;
- }
- #endif
- ret = essl_get_intr(handle, out_raw, out_st, TIMEOUT_MAX);
- if (ret != ESP_OK) return ret;
- ret = essl_clear_intr(handle, *out_raw, TIMEOUT_MAX);
- if (ret != ESP_OK) return ret;
- ESP_LOGD(TAG, "intr: %08"PRIX32, *out_raw);
- return ESP_OK;
- }
- //try to get an interrupt from the slave and handle it, return if none.
- esp_err_t process_event(essl_handle_t handle)
- {
- uint32_t intr_raw, intr_st;
- esp_err_t ret = get_intr(handle, &intr_raw, &intr_st);
- if (ret == ESP_ERR_TIMEOUT) {
- return ret;
- }
- ESP_ERROR_CHECK(ret);
- for (int i = 0; i < 8; i++) {
- if (intr_raw & BIT(i)) {
- ESP_LOGI(TAG, "host int: %d", i);
- }
- }
- const int wait_ms = 50;
- if (intr_raw & HOST_SLC0_RX_NEW_PACKET_INT_ST) {
- ESP_LOGD(TAG, "new packet coming");
- while (1) {
- size_t size_read = READ_BUFFER_LEN;
- ret = essl_get_packet(handle, rcv_buffer, READ_BUFFER_LEN, &size_read, wait_ms);
- if (ret == ESP_ERR_NOT_FOUND) {
- ESP_LOGE(TAG, "interrupt but no data can be read");
- break;
- } else if (ret != ESP_OK && ret != ESP_ERR_NOT_FINISHED) {
- ESP_LOGE(TAG, "rx packet error: %08X", ret);
- return ret;
- }
- ESP_LOGI(TAG, "receive data, size: %d", size_read);
- ESP_LOG_BUFFER_HEXDUMP(TAG, rcv_buffer, size_read, ESP_LOG_INFO);
- if (ret == ESP_OK) {
- break;
- }
- }
- }
- return ESP_OK;
- }
- //tell the slave to do a job
- static inline esp_err_t slave_inform_job(essl_handle_t handle, example_job_t job)
- {
- esp_err_t ret;
- ret = essl_write_reg(handle, SLAVE_REG_JOB, job, NULL, TIMEOUT_MAX);
- ESP_ERROR_CHECK(ret);
- ret = essl_send_slave_intr(handle, BIT(SLAVE_INTR_NOTIFY), TIMEOUT_MAX);
- ESP_ERROR_CHECK(ret);
- return ret;
- }
- //tell the slave to write registers by write one of them, and read them back
- void job_write_reg(essl_handle_t handle, int value)
- {
- esp_err_t ret;
- uint8_t reg_read[60];
- ESP_LOGI(TAG, "========JOB: write slave reg========");
- ret = essl_write_reg(handle, SLAVE_REG_VALUE, value, NULL, TIMEOUT_MAX);
- ESP_ERROR_CHECK(ret);
- ret = slave_inform_job(handle, JOB_WRITE_REG);
- ESP_ERROR_CHECK(ret);
- vTaskDelay(10);
- for (int i = 0; i < 60; i++) {
- ESP_LOGD(TAG, "reading register %d", i);
- ret = essl_read_reg(handle, i, ®_read[i], TIMEOUT_MAX);
- ESP_ERROR_CHECK(ret);
- }
- ESP_LOGI(TAG, "read registers:");
- ESP_LOG_BUFFER_HEXDUMP(TAG, reg_read, 64, ESP_LOG_INFO);
- }
- //the slave only load 16 buffers a time
- //so first 5 packets (use 1+1+8+4+1=15 buffers) are sent, the others (513, 517) failed (timeout)
- int packet_len[] = {6, 12, 1024, 512, 3, 513, 517};
- //the sending buffer should be word aligned
- DMA_ATTR uint8_t send_buffer[READ_BUFFER_LEN];
- //send packets to the slave (they will return and be handled by the interrupt handler)
- void job_fifo(essl_handle_t handle)
- {
- for (int i = 0; i < READ_BUFFER_LEN; i++) {
- send_buffer[i] = 0x46 + i * 5;
- }
- esp_err_t ret;
- int pointer = 0;
- ESP_LOGI(TAG, "========JOB: send fifos========");
- /* CAUTION: This example shows that we can send random length of packet to the slave.
- * However it takes time of two transactions if the length is not multiples of 4 bytes.
- * e.g. sending 6 bytes is done by sending 4 + 2 bytes each transaction.
- * Try to avoid unaligned packets if possible to get higher effeciency.
- */
- for (int i = 0; i < sizeof(packet_len) / sizeof(int); i++) {
- const int wait_ms = 50;
- int length = packet_len[i];
- ret = essl_send_packet(handle, send_buffer + pointer, length, wait_ms);
- if (ret == ESP_ERR_TIMEOUT) {
- ESP_LOGD(TAG, "several packets are expected to timeout.");
- } else {
- ESP_ERROR_CHECK(ret);
- ESP_LOGI(TAG, "send packet length: %d", length);
- }
- pointer += (length + 3) & (~3); //the length can be random, but data should start at the 32-bit boundary.
- }
- }
- //inform the slave to send interrupts to host (the interrupts will be handled in the interrupt handler)
- void job_getint(essl_handle_t handle)
- {
- ESP_LOGI(TAG, "========JOB: get interrupts from slave========");
- slave_inform_job(handle, JOB_SEND_INT);
- }
- void app_main(void)
- {
- essl_handle_t handle;
- esp_err_t err;
- //enable the power if on espressif SDIO master-slave board
- slave_power_on();
- ESP_LOGI(TAG, "host ready, start initializing slave...");
- err = slave_init(&handle);
- ESP_ERROR_CHECK(err);
- err = slave_reset(handle);
- ESP_ERROR_CHECK(err);
- uint32_t start, end;
- job_write_reg(handle, 10);
- int times = 2;
- while (1) {
- job_getint(handle);
- start = xTaskGetTickCount();
- while (1) {
- process_event(handle);
- vTaskDelay(1);
- end = xTaskGetTickCount();
- if ((end - start) * 1000 / CONFIG_FREERTOS_HZ > 5000) {
- break;
- }
- }
- if (--times == 0) {
- break;
- }
- };
- while (1) {
- job_fifo(handle);
- start = xTaskGetTickCount();
- while (1) {
- process_event(handle);
- vTaskDelay(1);
- end = xTaskGetTickCount();
- if ((end - start) * 1000 / CONFIG_FREERTOS_HZ > 2000) {
- break;
- }
- }
- }
- }
|