#include "storage.hpp" #include "gpio-expander.hpp" #include #include #include #include "diskio_impl.h" #include "diskio_sdmmc.h" #include "driver/gpio.h" #include "driver/sdmmc_types.h" #include "driver/sdspi_host.h" #include "esp_check.h" #include "esp_err.h" #include "esp_vfs_fat.h" #include "ff.h" #include "hal/gpio_types.h" #include "hal/spi_types.h" #include "sdmmc_cmd.h" static const char* kTag = "SDSTORAGE"; static const uint8_t kMaxOpenFiles = 8; namespace gay_ipod { const char* kStoragePath = "/sdcard"; // Static functions for interrop with the ESP IDF API, which requires a // function pointer. namespace callback { static std::atomic instance = nullptr; static std::atomic bootstrap = nullptr; static esp_err_t do_transaction(sdspi_dev_handle_t handle, sdmmc_command_t* cmdinfo) { auto bootstrap_fn = bootstrap.load(); if (bootstrap_fn != nullptr) { return bootstrap_fn(handle, cmdinfo); } auto instance_unwrapped = instance.load(); if (instance_unwrapped == nullptr) { ESP_LOGW(kTag, "uncaught sdspi transaction"); return ESP_OK; } // TODO: what if a transaction comes in right now? return instance_unwrapped->HandleTransaction(handle, cmdinfo); } } // namespace callback auto SdStorage::create(GpioExpander* gpio) -> cpp::result, Error> { // Acquiring the bus will also flush the mux switch change. gpio->set_pin(GpioExpander::SD_MUX_SWITCH, GpioExpander::SD_MUX_ESP); sdspi_dev_handle_t handle; std::unique_ptr host; std::unique_ptr card; FATFS* fs = nullptr; // Now we can init the driver and set up the SD card into SPI mode. sdspi_host_init(); sdspi_device_config_t config = { .host_id = VSPI_HOST, // CS handled manually bc it's on the GPIO expander .gpio_cs = GPIO_NUM_2, .gpio_cd = SDSPI_SLOT_NO_CD, .gpio_wp = SDSPI_SLOT_NO_WP, .gpio_int = GPIO_NUM_NC, }; if (esp_err_t err = sdspi_host_init_device(&config, &handle) != ESP_OK) { ESP_LOGE(kTag, "Failed to init, err %d", err); return cpp::fail(Error::FAILED_TO_INIT); } host = std::make_unique(sdmmc_host_t SDSPI_HOST_DEFAULT()); card = std::make_unique(); // We manage the CS pin ourselves via the GPIO expander. To do this safely in // a multithreaded environment, we wrap the ESP IDF do_transaction function // with our own that acquires the CS mutex for the duration of the SPI // transaction. auto do_transaction = host->do_transaction; host->do_transaction = &callback::do_transaction; host->slot = handle; callback::bootstrap = do_transaction; auto lock = gpio->AcquireSpiBus(GpioExpander::SD_CARD); // Will return ESP_ERR_INVALID_RESPONSE if there is no card esp_err_t err = sdmmc_card_init(host.get(), card.get()); if (err != ESP_OK) { ESP_LOGW(kTag, "Failed to read, err: %d", err); return cpp::fail(Error::FAILED_TO_READ); } ESP_ERROR_CHECK(esp_vfs_fat_register(kStoragePath, "", kMaxOpenFiles, &fs)); ff_diskio_register_sdmmc(fs->pdrv, card.get()); // Mount right now, not on first operation. FRESULT ferr = f_mount(fs, "", 1); if (ferr != FR_OK) { ESP_LOGW(kTag, "Failed to mount, err: %d", ferr); return cpp::fail(Error::FAILED_TO_MOUNT); } return std::make_unique(gpio, do_transaction, handle, host, card, fs); } SdStorage::SdStorage(GpioExpander* gpio, esp_err_t (*do_transaction)(sdspi_dev_handle_t, sdmmc_command_t*), sdspi_dev_handle_t handle, std::unique_ptr& host, std::unique_ptr& card, FATFS* fs) : gpio_(gpio), do_transaction_(do_transaction), handle_(handle), host_(std::move(host)), card_(std::move(card)), fs_(fs) { callback::instance = this; callback::bootstrap = nullptr; } SdStorage::~SdStorage() { // Unmount and unregister the filesystem f_unmount(""); ff_diskio_register(fs_->pdrv, NULL); esp_vfs_fat_unregister_path(kStoragePath); fs_ = nullptr; callback::instance = nullptr; // Uninstall the SPI driver sdspi_host_remove_device(this->handle_); sdspi_host_deinit(); } auto SdStorage::HandleTransaction(sdspi_dev_handle_t handle, sdmmc_command_t* cmdinfo) -> esp_err_t { auto lock = gpio_->AcquireSpiBus(GpioExpander::SD_CARD); return do_transaction_(handle, cmdinfo); } } // namespace gay_ipod