// SPDX-License-Identifier: Apache-2.0 #include #include #include #include #include "tactility/drivers/gpio_descriptor.h" #include #include #include #define TAG "esp32_sdmmc" #define GET_CONFIG(device) ((const struct Esp32SdmmcConfig*)device->config) #define GET_DATA(device) ((struct Esp32SdmmcInternal*)device_get_driver_data(device)) extern "C" { error_t mount(Device* device) { return ERROR_NONE; } error_t unmount(Device* device) { return ERROR_NONE; } bool is_mounted(Device* device) { return true; } error_t get_mount_path(Device*, char* out_path) { return ERROR_NONE; } static const FileSystemApi sdmmc_filesystem_api = { .mount = mount, .unmount = unmount, .is_mounted = is_mounted, .get_mount_path = get_mount_path }; struct Esp32SdmmcInternal { RecursiveMutex mutex = {}; bool initialized = false; // Pin descriptors GpioDescriptor* pin_clk_descriptor = nullptr; GpioDescriptor* pin_cmd_descriptor = nullptr; GpioDescriptor* pin_d0_descriptor = nullptr; GpioDescriptor* pin_d1_descriptor = nullptr; GpioDescriptor* pin_d2_descriptor = nullptr; GpioDescriptor* pin_d3_descriptor = nullptr; GpioDescriptor* pin_d4_descriptor = nullptr; GpioDescriptor* pin_d5_descriptor = nullptr; GpioDescriptor* pin_d6_descriptor = nullptr; GpioDescriptor* pin_d7_descriptor = nullptr; GpioDescriptor* pin_cd_descriptor = nullptr; GpioDescriptor* pin_wp_descriptor = nullptr; explicit Esp32SdmmcInternal() { recursive_mutex_construct(&mutex); } ~Esp32SdmmcInternal() { cleanup_pins(); recursive_mutex_destruct(&mutex); } void cleanup_pins() { release_pin(&pin_clk_descriptor); release_pin(&pin_cmd_descriptor); release_pin(&pin_d0_descriptor); release_pin(&pin_d1_descriptor); release_pin(&pin_d2_descriptor); release_pin(&pin_d3_descriptor); release_pin(&pin_d4_descriptor); release_pin(&pin_d5_descriptor); release_pin(&pin_d6_descriptor); release_pin(&pin_d7_descriptor); release_pin(&pin_cd_descriptor); release_pin(&pin_wp_descriptor); } void lock() { recursive_mutex_lock(&mutex); } error_t try_lock(TickType_t timeout) { return recursive_mutex_try_lock(&mutex, timeout); } void unlock() { recursive_mutex_unlock(&mutex); } }; static error_t start(Device* device) { LOG_I(TAG, "start %s", device->name); auto* data = new (std::nothrow) Esp32SdmmcInternal(); if (!data) return ERROR_OUT_OF_MEMORY; device_set_driver_data(device, data); auto* sdmmc_config = GET_CONFIG(device); // Acquire pins from the specified GPIO pin specs. Optional pins are allowed. bool pins_ok = acquire_pin_or_set_null(sdmmc_config->pin_clk, &data->pin_clk_descriptor) && acquire_pin_or_set_null(sdmmc_config->pin_cmd, &data->pin_cmd_descriptor) && acquire_pin_or_set_null(sdmmc_config->pin_d0, &data->pin_d0_descriptor) && acquire_pin_or_set_null(sdmmc_config->pin_d1, &data->pin_d1_descriptor) && acquire_pin_or_set_null(sdmmc_config->pin_d2, &data->pin_d2_descriptor) && acquire_pin_or_set_null(sdmmc_config->pin_d3, &data->pin_d3_descriptor) && acquire_pin_or_set_null(sdmmc_config->pin_d4, &data->pin_d4_descriptor) && acquire_pin_or_set_null(sdmmc_config->pin_d5, &data->pin_d5_descriptor) && acquire_pin_or_set_null(sdmmc_config->pin_d6, &data->pin_d6_descriptor) && acquire_pin_or_set_null(sdmmc_config->pin_d7, &data->pin_d7_descriptor) && acquire_pin_or_set_null(sdmmc_config->pin_cd, &data->pin_cd_descriptor) && acquire_pin_or_set_null(sdmmc_config->pin_wp, &data->pin_wp_descriptor); if (!pins_ok) { LOG_E(TAG, "Failed to acquire required one or more pins"); data->cleanup_pins(); device_set_driver_data(device, nullptr); delete data; return ERROR_RESOURCE; } // TODO: filesystem data->initialized = true; return ERROR_NONE; } static error_t stop(Device* device) { ESP_LOGI(TAG, "stop %s", device->name); auto* driver_data = GET_DATA(device); auto* dts_config = GET_CONFIG(device); // TODO: filesystem driver_data->cleanup_pins(); device_set_driver_data(device, nullptr); delete driver_data; return ERROR_NONE; } extern Module platform_esp32_module; Driver esp32_spi_driver = { .name = "esp32_sdmmc", .compatible = (const char*[]) { "espressif,esp32-sdmmc", nullptr }, .start_device = start, .stop_device = stop, .api = nullptr, .device_type = nullptr, .owner = &platform_esp32_module, .internal = nullptr }; } // extern "C"