mirror of
https://github.com/espressif/esp-idf.git
synced 2025-10-24 03:03:25 +00:00
211 lines
4.8 KiB
C++
211 lines
4.8 KiB
C++
/*
|
|
* SPDX-FileCopyrightText: 2021-2022 Espressif Systems (Shanghai) CO LTD
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
#if __cpp_exceptions
|
|
|
|
#include <array>
|
|
#include "driver/gpio.h"
|
|
#include "gpio_cxx.hpp"
|
|
|
|
namespace idf {
|
|
|
|
#define GPIO_CHECK_THROW(err) CHECK_THROW_SPECIFIC((err), GPIOException)
|
|
|
|
namespace {
|
|
#if CONFIG_IDF_TARGET_LINUX
|
|
constexpr std::array<uint32_t, 1> INVALID_GPIOS = {24};
|
|
#elif CONFIG_IDF_TARGET_ESP32
|
|
constexpr std::array<uint32_t, 1> INVALID_GPIOS = {24};
|
|
#elif CONFIG_IDF_TARGET_ESP32S2
|
|
constexpr std::array<uint32_t, 4> INVALID_GPIOS = {22, 23, 24, 25};
|
|
#elif CONFIG_IDF_TARGET_ESP32S3
|
|
constexpr std::array<uint32_t, 4> INVALID_GPIOS = {22, 23, 24, 25};
|
|
#elif CONFIG_IDF_TARGET_ESP32C3
|
|
constexpr std::array<uint32_t, 0> INVALID_GPIOS = {};
|
|
#elif CONFIG_IDF_TARGET_ESP32C2
|
|
constexpr std::array<uint32_t, 0> INVALID_GPIOS = {};
|
|
#else
|
|
#error "No GPIOs defined for the current target"
|
|
#endif
|
|
|
|
gpio_num_t gpio_to_driver_type(const GPIONum &gpio_num)
|
|
{
|
|
return static_cast<gpio_num_t>(gpio_num.get_num());
|
|
}
|
|
|
|
}
|
|
|
|
GPIOException::GPIOException(esp_err_t error) : ESPException(error) { }
|
|
|
|
esp_err_t check_gpio_pin_num(uint32_t pin_num) noexcept
|
|
{
|
|
if (pin_num >= GPIO_NUM_MAX) {
|
|
return ESP_ERR_INVALID_ARG;
|
|
}
|
|
|
|
for (auto num: INVALID_GPIOS)
|
|
{
|
|
if (pin_num == num) {
|
|
return ESP_ERR_INVALID_ARG;
|
|
}
|
|
}
|
|
|
|
return ESP_OK;
|
|
}
|
|
|
|
esp_err_t check_gpio_drive_strength(uint32_t strength) noexcept
|
|
{
|
|
if (strength >= GPIO_DRIVE_CAP_MAX) {
|
|
return ESP_ERR_INVALID_ARG;
|
|
}
|
|
|
|
return ESP_OK;
|
|
}
|
|
|
|
GPIOPullMode GPIOPullMode::FLOATING()
|
|
{
|
|
return GPIOPullMode(GPIO_FLOATING);
|
|
}
|
|
|
|
GPIOPullMode GPIOPullMode::PULLUP()
|
|
{
|
|
return GPIOPullMode(GPIO_PULLUP_ONLY);
|
|
}
|
|
|
|
GPIOPullMode GPIOPullMode::PULLDOWN()
|
|
{
|
|
return GPIOPullMode(GPIO_PULLDOWN_ONLY);
|
|
}
|
|
|
|
GPIOWakeupIntrType GPIOWakeupIntrType::LOW_LEVEL()
|
|
{
|
|
return GPIOWakeupIntrType(GPIO_INTR_LOW_LEVEL);
|
|
}
|
|
|
|
GPIOWakeupIntrType GPIOWakeupIntrType::HIGH_LEVEL()
|
|
{
|
|
return GPIOWakeupIntrType(GPIO_INTR_HIGH_LEVEL);
|
|
}
|
|
|
|
GPIODriveStrength GPIODriveStrength::DEFAULT()
|
|
{
|
|
return MEDIUM();
|
|
}
|
|
|
|
GPIODriveStrength GPIODriveStrength::WEAK()
|
|
{
|
|
return GPIODriveStrength(GPIO_DRIVE_CAP_0);
|
|
}
|
|
|
|
GPIODriveStrength GPIODriveStrength::LESS_WEAK()
|
|
{
|
|
return GPIODriveStrength(GPIO_DRIVE_CAP_1);
|
|
}
|
|
|
|
GPIODriveStrength GPIODriveStrength::MEDIUM()
|
|
{
|
|
return GPIODriveStrength(GPIO_DRIVE_CAP_2);
|
|
}
|
|
|
|
GPIODriveStrength GPIODriveStrength::STRONGEST()
|
|
{
|
|
return GPIODriveStrength(GPIO_DRIVE_CAP_3);
|
|
}
|
|
|
|
GPIOBase::GPIOBase(GPIONum num) : gpio_num(num)
|
|
{
|
|
GPIO_CHECK_THROW(gpio_reset_pin(gpio_to_driver_type(gpio_num)));
|
|
}
|
|
|
|
void GPIOBase::hold_en()
|
|
{
|
|
GPIO_CHECK_THROW(gpio_hold_en(gpio_to_driver_type(gpio_num)));
|
|
}
|
|
|
|
void GPIOBase::hold_dis()
|
|
{
|
|
GPIO_CHECK_THROW(gpio_hold_dis(gpio_to_driver_type(gpio_num)));
|
|
}
|
|
|
|
void GPIOBase::set_drive_strength(GPIODriveStrength strength)
|
|
{
|
|
GPIO_CHECK_THROW(gpio_set_drive_capability(gpio_to_driver_type(gpio_num),
|
|
static_cast<gpio_drive_cap_t>(strength.get_strength())));
|
|
}
|
|
|
|
GPIO_Output::GPIO_Output(GPIONum num) : GPIOBase(num)
|
|
{
|
|
GPIO_CHECK_THROW(gpio_set_direction(gpio_to_driver_type(gpio_num), GPIO_MODE_OUTPUT));
|
|
}
|
|
|
|
void GPIO_Output::set_high()
|
|
{
|
|
GPIO_CHECK_THROW(gpio_set_level(gpio_to_driver_type(gpio_num), 1));
|
|
}
|
|
|
|
void GPIO_Output::set_low()
|
|
{
|
|
GPIO_CHECK_THROW(gpio_set_level(gpio_to_driver_type(gpio_num), 0));
|
|
}
|
|
|
|
GPIODriveStrength GPIOBase::get_drive_strength()
|
|
{
|
|
gpio_drive_cap_t strength;
|
|
GPIO_CHECK_THROW(gpio_get_drive_capability(gpio_to_driver_type(gpio_num), &strength));
|
|
return GPIODriveStrength(static_cast<uint32_t>(strength));
|
|
}
|
|
|
|
GPIOInput::GPIOInput(GPIONum num) : GPIOBase(num)
|
|
{
|
|
GPIO_CHECK_THROW(gpio_set_direction(gpio_to_driver_type(gpio_num), GPIO_MODE_INPUT));
|
|
}
|
|
|
|
GPIOLevel GPIOInput::get_level() const noexcept
|
|
{
|
|
int level = gpio_get_level(gpio_to_driver_type(gpio_num));
|
|
if (level) {
|
|
return GPIOLevel::HIGH;
|
|
} else {
|
|
return GPIOLevel::LOW;
|
|
}
|
|
}
|
|
|
|
void GPIOInput::set_pull_mode(GPIOPullMode mode)
|
|
{
|
|
GPIO_CHECK_THROW(gpio_set_pull_mode(gpio_to_driver_type(gpio_num),
|
|
static_cast<gpio_pull_mode_t>(mode.get_pull_mode())));
|
|
}
|
|
|
|
void GPIOInput::wakeup_enable(GPIOWakeupIntrType interrupt_type)
|
|
{
|
|
GPIO_CHECK_THROW(gpio_wakeup_enable(gpio_to_driver_type(gpio_num),
|
|
static_cast<gpio_int_type_t>(interrupt_type.get_level())));
|
|
}
|
|
|
|
void GPIOInput::wakeup_disable()
|
|
{
|
|
GPIO_CHECK_THROW(gpio_wakeup_disable(gpio_to_driver_type(gpio_num)));
|
|
}
|
|
|
|
GPIO_OpenDrain::GPIO_OpenDrain(GPIONum num) : GPIOInput(num)
|
|
{
|
|
GPIO_CHECK_THROW(gpio_set_direction(gpio_to_driver_type(gpio_num), GPIO_MODE_INPUT_OUTPUT_OD));
|
|
}
|
|
|
|
void GPIO_OpenDrain::set_floating()
|
|
{
|
|
GPIO_CHECK_THROW(gpio_set_level(gpio_to_driver_type(gpio_num), 1));
|
|
}
|
|
|
|
void GPIO_OpenDrain::set_low()
|
|
{
|
|
GPIO_CHECK_THROW(gpio_set_level(gpio_to_driver_type(gpio_num), 0));
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|