// SPDX-License-Identifier: Apache-2.0 #include #include #include #include #include #include #include #include #define TAG "esp32_gpio" #define GET_CONFIG(device) ((struct Esp32GpioConfig*)device->config) extern "C" { static error_t set_level(GpioDescriptor* descriptor, bool high) { auto esp_error = gpio_set_level(static_cast(descriptor->pin), high); return esp_err_to_error(esp_error); } static error_t get_level(GpioDescriptor* descriptor, bool* high) { *high = gpio_get_level(static_cast(descriptor->pin)) != 0; return ERROR_NONE; } static error_t set_flags(GpioDescriptor* descriptor, gpio_flags_t flags) { const Esp32GpioConfig* config = GET_CONFIG(descriptor->controller); if (descriptor->pin >= config->gpioCount) { return ERROR_INVALID_ARGUMENT; } gpio_mode_t mode; if ((flags & GPIO_FLAG_DIRECTION_INPUT_OUTPUT) == GPIO_FLAG_DIRECTION_INPUT_OUTPUT) { mode = GPIO_MODE_INPUT_OUTPUT; } else if (flags & GPIO_FLAG_DIRECTION_INPUT) { mode = GPIO_MODE_INPUT; } else if (flags & GPIO_FLAG_DIRECTION_OUTPUT) { mode = GPIO_MODE_OUTPUT; } else { return ERROR_INVALID_ARGUMENT; } const gpio_config_t esp_config = { .pin_bit_mask = 1ULL << descriptor->pin, .mode = mode, .pull_up_en = (flags & GPIO_FLAG_PULL_UP) ? GPIO_PULLUP_ENABLE : GPIO_PULLUP_DISABLE, .pull_down_en = (flags & GPIO_FLAG_PULL_DOWN) ? GPIO_PULLDOWN_ENABLE : GPIO_PULLDOWN_DISABLE, .intr_type = GPIO_FLAG_INTERRUPT_FROM_OPTIONS(flags), #if SOC_GPIO_SUPPORT_PIN_HYS_FILTER .hys_ctrl_mode = GPIO_HYS_SOFT_DISABLE #endif }; auto esp_error = gpio_config(&esp_config); return esp_err_to_error(esp_error); } static error_t get_flags(GpioDescriptor* descriptor, gpio_flags_t* flags) { gpio_io_config_t esp_config; if (gpio_get_io_config(static_cast(descriptor->pin), &esp_config) != ESP_OK) { return ERROR_RESOURCE; } gpio_flags_t output = 0; if (esp_config.pu) { output |= GPIO_FLAG_PULL_UP; } if (esp_config.pd) { output |= GPIO_FLAG_PULL_DOWN; } if (esp_config.ie) { output |= GPIO_FLAG_DIRECTION_INPUT; } if (esp_config.oe) { output |= GPIO_FLAG_DIRECTION_OUTPUT; } if (esp_config.oe_inv) { output |= GPIO_FLAG_ACTIVE_LOW; } *flags = output; return ERROR_NONE; } static error_t get_native_pin_number(GpioDescriptor* descriptor, void* pin_number) { auto* esp_pin_number = reinterpret_cast(pin_number); *esp_pin_number = static_cast(descriptor->pin); return ERROR_NONE; } static error_t start(Device* device) { ESP_LOGI(TAG, "start %s", device->name); auto pin_count = GET_CONFIG(device)->gpioCount; return gpio_controller_init_descriptors(device, pin_count, nullptr); } static error_t stop(Device* device) { ESP_LOGI(TAG, "stop %s", device->name); return gpio_controller_deinit_descriptors(device); } const static GpioControllerApi esp32_gpio_api = { .set_level = set_level, .get_level = get_level, .set_flags = set_flags, .get_flags = get_flags, .get_native_pin_number = get_native_pin_number }; extern struct Module platform_module; Driver esp32_gpio_driver = { .name = "esp32_gpio", .compatible = (const char*[]) { "espressif,esp32-gpio", nullptr }, .start_device = start, .stop_device = stop, .api = (void*)&esp32_gpio_api, .device_type = &GPIO_CONTROLLER_TYPE, .owner = &platform_module, .internal = nullptr }; } // extern "C"