driver: sdmmc: add esp32s3 support

* Extend sdmmc_slot_config_t with GPIO pin numbers for all SD bus
  signals. These new fields are available if SOC_SDMMC_USE_GPIO_MATRIX
  is set.
* Add shorter "sd" and "wp" aliases for "gpio_sd" and "gpio_wp" field
  names in sdmmc_slot_config_t.
* Deprecate sdmmc_host_pullup_en, prefer to enable this feature using
  SDMMC_SLOT_FLAG_INTERNAL_PULLUP instead.
This commit is contained in:
Ivan Grokhotkov
2021-01-04 21:05:27 +01:00
parent 17c65dad27
commit 2c8f413059
3 changed files with 192 additions and 55 deletions

View File

@@ -57,6 +57,8 @@ if(${target} STREQUAL "esp32s3")
list(APPEND srcs "adc_common.c" list(APPEND srcs "adc_common.c"
"dedic_gpio.c" "dedic_gpio.c"
"gdma.c" "gdma.c"
"sdmmc_host.c"
"sdmmc_transaction.c"
"spi_slave_hd.c" "spi_slave_hd.c"
"touch_sensor_common.c" "touch_sensor_common.c"
) )

View File

@@ -59,8 +59,26 @@ extern "C" {
* Extra configuration for SDMMC peripheral slot * Extra configuration for SDMMC peripheral slot
*/ */
typedef struct { typedef struct {
gpio_num_t gpio_cd; ///< GPIO number of card detect signal #ifdef SOC_SDMMC_USE_GPIO_MATRIX
gpio_num_t gpio_wp; ///< GPIO number of write protect signal gpio_num_t clk; ///< GPIO number of CLK signal.
gpio_num_t cmd; ///< GPIO number of CMD signal.
gpio_num_t d0; ///< GPIO number of D0 signal.
gpio_num_t d1; ///< GPIO number of D1 signal.
gpio_num_t d2; ///< GPIO number of D2 signal.
gpio_num_t d3; ///< GPIO number of D3 signal.
gpio_num_t d4; ///< GPIO number of D4 signal. Ignored in 1- or 4- line mode.
gpio_num_t d5; ///< GPIO number of D5 signal. Ignored in 1- or 4- line mode.
gpio_num_t d6; ///< GPIO number of D6 signal. Ignored in 1- or 4- line mode.
gpio_num_t d7; ///< GPIO number of D7 signal. Ignored in 1- or 4- line mode.
#endif // SOC_SDMMC_USE_GPIO_MATRIX
union {
gpio_num_t gpio_cd; ///< GPIO number of card detect signal
gpio_num_t cd; ///< GPIO number of card detect signal; shorter name.
};
union {
gpio_num_t gpio_wp; ///< GPIO number of write protect signal
gpio_num_t wp; ///< GPIO number of write protect signal; shorter name.
};
uint8_t width; ///< Bus width used by the slot (might be less than the max width supported) uint8_t width; ///< Bus width used by the slot (might be less than the max width supported)
uint32_t flags; ///< Features used by this slot uint32_t flags; ///< Features used by this slot
#define SDMMC_SLOT_FLAG_INTERNAL_PULLUP BIT(0) #define SDMMC_SLOT_FLAG_INTERNAL_PULLUP BIT(0)
@@ -72,18 +90,44 @@ typedef struct {
#define SDMMC_SLOT_NO_CD GPIO_NUM_NC ///< indicates that card detect line is not used #define SDMMC_SLOT_NO_CD GPIO_NUM_NC ///< indicates that card detect line is not used
#define SDMMC_SLOT_NO_WP GPIO_NUM_NC ///< indicates that write protect line is not used #define SDMMC_SLOT_NO_WP GPIO_NUM_NC ///< indicates that write protect line is not used
#define SDMMC_SLOT_WIDTH_DEFAULT 0 ///< use the default width for the slot (8 for slot 0, 4 for slot 1) #define SDMMC_SLOT_WIDTH_DEFAULT 0 ///< use the maximum possible width for the slot
#ifdef SOC_SDMMC_USE_GPIO_MATRIX
/** /**
* Macro defining default configuration of SDMMC host slot * Macro defining default configuration of SDMMC host slot
*/ */
#define SDMMC_SLOT_CONFIG_DEFAULT() {\ #define SDMMC_SLOT_CONFIG_DEFAULT() {\
.gpio_cd = SDMMC_SLOT_NO_CD, \ .clk = GPIO_NUM_14, \
.gpio_wp = SDMMC_SLOT_NO_WP, \ .cmd = GPIO_NUM_15, \
.d0 = GPIO_NUM_2, \
.d1 = GPIO_NUM_4, \
.d2 = GPIO_NUM_12, \
.d3 = GPIO_NUM_13, \
.d4 = GPIO_NUM_33, \
.d5 = GPIO_NUM_34, \
.d6 = GPIO_NUM_35, \
.d7 = GPIO_NUM_36, \
.cd = SDMMC_SLOT_NO_CD, \
.wp = SDMMC_SLOT_NO_WP, \
.width = SDMMC_SLOT_WIDTH_DEFAULT, \ .width = SDMMC_SLOT_WIDTH_DEFAULT, \
.flags = 0, \ .flags = 0, \
} }
#else // SOC_SDMMC_USE_GPIO_MATRIX
/**
* Macro defining default configuration of SDMMC host slot
*/
#define SDMMC_SLOT_CONFIG_DEFAULT() {\
.cd = SDMMC_SLOT_NO_CD, \
.wp = SDMMC_SLOT_NO_WP, \
.width = SDMMC_SLOT_WIDTH_DEFAULT, \
.flags = 0, \
}
#endif // SOC_SDMMC_USE_GPIO_MATRIX
/** /**
* @brief Initialize SDMMC host peripheral * @brief Initialize SDMMC host peripheral
* *
@@ -226,6 +270,9 @@ esp_err_t sdmmc_host_deinit(void);
/** /**
* @brief Enable the pull-ups of sd pins. * @brief Enable the pull-ups of sd pins.
* *
* This function is deprecated. Please set SDMMC_SLOT_FLAG_INTERNAL_PULLUP flag in
* sdmmc_slot_config_t::flags instead.
*
* @note You should always place actual pullups on the lines instead of using * @note You should always place actual pullups on the lines instead of using
* this function. Internal pullup resistance are high and not sufficient, may * this function. Internal pullup resistance are high and not sufficient, may
* cause instability in products. This is for debug or examples only. * cause instability in products. This is for debug or examples only.
@@ -238,7 +285,7 @@ esp_err_t sdmmc_host_deinit(void);
* - ESP_ERR_INVALID_ARG: if configured width larger than maximum the slot can * - ESP_ERR_INVALID_ARG: if configured width larger than maximum the slot can
* support * support
*/ */
esp_err_t sdmmc_host_pullup_en(int slot, int width); esp_err_t sdmmc_host_pullup_en(int slot, int width) __attribute__((deprecated));
#ifdef __cplusplus #ifdef __cplusplus
} }

View File

@@ -37,13 +37,44 @@
static void sdmmc_isr(void* arg); static void sdmmc_isr(void* arg);
static void sdmmc_host_dma_init(void); static void sdmmc_host_dma_init(void);
static const char* TAG = "sdmmc_periph"; static const char* TAG = "sdmmc_periph";
static intr_handle_t s_intr_handle; static intr_handle_t s_intr_handle;
static QueueHandle_t s_event_queue; static QueueHandle_t s_event_queue;
static SemaphoreHandle_t s_io_intr_event; static SemaphoreHandle_t s_io_intr_event;
size_t s_slot_width[2] = {1,1}; static size_t s_slot_width[2] = {1, 1};
/* The following definitions are used to simplify GPIO configuration in the driver,
* whether IOMUX or GPIO Matrix is used by the chip.
* Two simple "APIs" are provided to the driver code:
* - configure_pin(name, slot, mode): Configures signal "name" for the given slot and mode.
* - GPIO_NUM(slot, name): Returns the GPIO number of signal "name" for the given slot.
*
* To make this work, configure_pin is defined as a macro that picks the parameters required
* for configuring GPIO matrix or IOMUX from relevant arrays, and passes them to either of
* configure_pin_gpio_matrix, configure_pin_iomux functions.
* Likewise, GPIO_NUM is a macro that picks the pin number from one of the two structures.
*
* Macros are used rather than inline functions to look up members of different structures
* with same names. E.g. the number of pin d3 is obtained either from .d3 member of
* sdmmc_slot_gpio_num array (for IOMUX) or from .d3 member of s_sdmmc_slot_gpio_num array
* (for GPIO matrix).
*/
#ifdef SOC_SDMMC_USE_GPIO_MATRIX
static void configure_pin_gpio_matrix(uint8_t gpio_num, uint8_t gpio_matrix_sig, gpio_mode_t mode, const char* name);
#define configure_pin(name, slot, mode) \
configure_pin_gpio_matrix(s_sdmmc_slot_gpio_num[slot].name, sdmmc_slot_gpio_sig[slot].name, mode, #name)
static sdmmc_slot_io_info_t s_sdmmc_slot_gpio_num[SOC_SDMMC_NUM_SLOTS];
#define GPIO_NUM(slot, name) s_sdmmc_slot_gpio_num[slot].name
#elif SOC_SDMMC_USE_IOMUX
static void configure_pin_iomux(uint8_t gpio_num);
#define configure_pin(name, slot, mode) configure_pin_iomux(sdmmc_slot_gpio_num[slot].name)
#define GPIO_NUM(slot, name) sdmmc_slot_gpio_num[slot].name
#endif // SOC_SDMMC_USE_GPIO_MATRIX
static esp_err_t sdmmc_host_pullup_en_internal(int slot, int width);
void sdmmc_host_reset(void) void sdmmc_host_reset(void)
{ {
@@ -94,9 +125,21 @@ static void sdmmc_host_set_clk_div(int div)
SDMMC.clock.div_factor_p = p; SDMMC.clock.div_factor_p = p;
SDMMC.clock.div_factor_h = h; SDMMC.clock.div_factor_h = h;
SDMMC.clock.div_factor_m = p; SDMMC.clock.div_factor_m = p;
// Make sure 160 MHz source clock is used
#if SOC_SDMMC_SUPPORT_XTAL_CLOCK
SDMMC.clock.clk_sel = 1;
#endif
#if SOC_SDMMC_USE_GPIO_MATRIX
// 90 degree phase on input and output clocks
const int inout_clock_phase = 1;
#else
// 180 degree phase on input and output clocks
const int inout_clock_phase = 4;
#endif
// Set phases for in/out clocks // Set phases for in/out clocks
SDMMC.clock.phase_dout = 4; // 180 degree phase on the output clock SDMMC.clock.phase_dout = inout_clock_phase;
SDMMC.clock.phase_din = 4; // 180 degree phase on the input clock SDMMC.clock.phase_din = inout_clock_phase;
SDMMC.clock.phase_core = 0; SDMMC.clock.phase_core = 0;
// Wait for the clock to propagate // Wait for the clock to propagate
esp_rom_delay_us(10); esp_rom_delay_us(10);
@@ -294,20 +337,41 @@ esp_err_t sdmmc_host_init(void)
return ESP_OK; return ESP_OK;
} }
static void configure_pin(int pin) #ifdef SOC_SDMMC_USE_IOMUX
static void configure_pin_iomux(uint8_t gpio_num)
{ {
const int sdmmc_func = 3; const int sdmmc_func = 3;
const int drive_strength = 3; const int drive_strength = 3;
assert(pin!=GPIO_NUM_NC); assert(gpio_num != (uint8_t) GPIO_NUM_NC);
gpio_pulldown_dis(pin); gpio_pulldown_dis(gpio_num);
uint32_t reg = GPIO_PIN_MUX_REG[pin]; uint32_t reg = GPIO_PIN_MUX_REG[gpio_num];
assert(reg != UINT32_MAX); assert(reg != UINT32_MAX);
PIN_INPUT_ENABLE(reg); PIN_INPUT_ENABLE(reg);
gpio_hal_iomux_func_sel(reg, sdmmc_func); gpio_hal_iomux_func_sel(reg, sdmmc_func);
PIN_SET_DRV(reg, drive_strength); PIN_SET_DRV(reg, drive_strength);
} }
#elif SOC_SDMMC_USE_GPIO_MATRIX
static void configure_pin_gpio_matrix(uint8_t gpio_num, uint8_t gpio_matrix_sig, gpio_mode_t mode, const char* name)
{
assert (gpio_num != (uint8_t) GPIO_NUM_NC);
ESP_LOGD(TAG, "using GPIO%d as %s pin", gpio_num, name);
gpio_reset_pin(gpio_num);
gpio_set_direction(gpio_num, mode);
gpio_pulldown_dis(gpio_num);
if (mode == GPIO_MODE_INPUT || mode == GPIO_MODE_INPUT_OUTPUT) {
esp_rom_gpio_connect_in_signal(gpio_num, gpio_matrix_sig, false);
}
if (mode == GPIO_MODE_OUTPUT || mode == GPIO_MODE_INPUT_OUTPUT) {
esp_rom_gpio_connect_out_signal(gpio_num, gpio_matrix_sig, false, false);
}
}
#endif // SOC_SDMMC_USE_{IOMUX,GPIO_MATRIX}
esp_err_t sdmmc_host_init_slot(int slot, const sdmmc_slot_config_t* slot_config) esp_err_t sdmmc_host_init_slot(int slot, const sdmmc_slot_config_t* slot_config)
{ {
if (!s_intr_handle) { if (!s_intr_handle) {
@@ -319,54 +383,75 @@ esp_err_t sdmmc_host_init_slot(int slot, const sdmmc_slot_config_t* slot_config)
if (slot_config == NULL) { if (slot_config == NULL) {
return ESP_ERR_INVALID_ARG; return ESP_ERR_INVALID_ARG;
} }
bool pullup = slot_config->flags & SDMMC_SLOT_FLAG_INTERNAL_PULLUP; int gpio_cd = slot_config->cd;
if (pullup) { int gpio_wp = slot_config->wp;
sdmmc_host_pullup_en(slot, slot_config->width);
}
int gpio_cd = slot_config->gpio_cd;
int gpio_wp = slot_config->gpio_wp;
uint8_t slot_width = slot_config->width; uint8_t slot_width = slot_config->width;
// Configure pins // Configure pins
const sdmmc_slot_info_t* pslot = &sdmmc_slot_info[slot]; const sdmmc_slot_info_t* slot_info = &sdmmc_slot_info[slot];
if (slot_width == SDMMC_SLOT_WIDTH_DEFAULT) { if (slot_width == SDMMC_SLOT_WIDTH_DEFAULT) {
slot_width = pslot->width; slot_width = slot_info->width;
} }
else if (slot_width > pslot->width) { else if (slot_width > slot_info->width) {
return ESP_ERR_INVALID_ARG; return ESP_ERR_INVALID_ARG;
} }
s_slot_width[slot] = slot_width; s_slot_width[slot] = slot_width;
configure_pin(pslot->clk_gpio); #if SOC_SDMMC_USE_GPIO_MATRIX
configure_pin(pslot->cmd_gpio); /* Save pin configuration for this slot */
configure_pin(pslot->d0_gpio); s_sdmmc_slot_gpio_num[slot].clk = slot_config->clk;
s_sdmmc_slot_gpio_num[slot].cmd = slot_config->cmd;
s_sdmmc_slot_gpio_num[slot].d0 = slot_config->d0;
/* Save d1 even in 1-line mode, it might be needed for SDIO INT line */
s_sdmmc_slot_gpio_num[slot].d1 = slot_config->d1;
if (slot_width >= 4) {
s_sdmmc_slot_gpio_num[slot].d2 = slot_config->d2;
}
/* Save d3 even for 1-line mode, as it needs to be set high */
s_sdmmc_slot_gpio_num[slot].d3 = slot_config->d3;
if (slot_width >= 8) {
s_sdmmc_slot_gpio_num[slot].d4 = slot_config->d4;
s_sdmmc_slot_gpio_num[slot].d5 = slot_config->d5;
s_sdmmc_slot_gpio_num[slot].d6 = slot_config->d6;
s_sdmmc_slot_gpio_num[slot].d7 = slot_config->d7;
}
#endif
bool pullup = slot_config->flags & SDMMC_SLOT_FLAG_INTERNAL_PULLUP;
if (pullup) {
sdmmc_host_pullup_en_internal(slot, slot_config->width);
}
configure_pin(clk, slot, GPIO_MODE_OUTPUT);
configure_pin(cmd, slot, GPIO_MODE_INPUT_OUTPUT);
configure_pin(d0, slot, GPIO_MODE_INPUT_OUTPUT);
if (slot_width >= 4) { if (slot_width >= 4) {
configure_pin(pslot->d1_gpio); configure_pin(d1, slot, GPIO_MODE_INPUT_OUTPUT);
configure_pin(pslot->d2_gpio); configure_pin(d2, slot, GPIO_MODE_INPUT_OUTPUT);
// Force D3 high to make slave enter SD mode. // Force D3 high to make slave enter SD mode.
// Connect to peripheral after width configuration. // Connect to peripheral after width configuration.
gpio_config_t gpio_conf = { gpio_config_t gpio_conf = {
.pin_bit_mask = BIT64(pslot->d3_gpio), .pin_bit_mask = BIT64(GPIO_NUM(slot, d3)),
.mode = GPIO_MODE_OUTPUT , .mode = GPIO_MODE_OUTPUT,
.pull_up_en = 0, .pull_up_en = 0,
.pull_down_en = 0, .pull_down_en = 0,
.intr_type = GPIO_INTR_DISABLE, .intr_type = GPIO_INTR_DISABLE,
}; };
gpio_config(&gpio_conf); gpio_config(&gpio_conf);
gpio_set_level(pslot->d3_gpio, 1); gpio_set_level(GPIO_NUM(slot, d3), 1);
if (slot_width == 8) { }
configure_pin(pslot->d4_gpio); if (slot_width == 8) {
configure_pin(pslot->d5_gpio); configure_pin(d4, slot, GPIO_MODE_INPUT_OUTPUT);
configure_pin(pslot->d6_gpio); configure_pin(d5, slot, GPIO_MODE_INPUT_OUTPUT);
configure_pin(pslot->d7_gpio); configure_pin(d6, slot, GPIO_MODE_INPUT_OUTPUT);
} configure_pin(d7, slot, GPIO_MODE_INPUT_OUTPUT);
} }
// SDIO slave interrupt is edge sensitive to ~(int_n | card_int | card_detect) // SDIO slave interrupt is edge sensitive to ~(int_n | card_int | card_detect)
// set this and card_detect to high to enable sdio interrupt // set this and card_detect to high to enable sdio interrupt
esp_rom_gpio_connect_in_signal(GPIO_MATRIX_CONST_ONE_INPUT, pslot->card_int, false); esp_rom_gpio_connect_in_signal(GPIO_MATRIX_CONST_ONE_INPUT, slot_info->card_int, false);
// Set up Card Detect input // Set up Card Detect input
int matrix_in_cd; int matrix_in_cd;
@@ -379,7 +464,7 @@ esp_err_t sdmmc_host_init_slot(int slot, const sdmmc_slot_config_t* slot_config)
// if not set, default to CD low (card present) // if not set, default to CD low (card present)
matrix_in_cd = GPIO_MATRIX_CONST_ZERO_INPUT; matrix_in_cd = GPIO_MATRIX_CONST_ZERO_INPUT;
} }
esp_rom_gpio_connect_in_signal(matrix_in_cd, pslot->card_detect, false); esp_rom_gpio_connect_in_signal(matrix_in_cd, slot_info->card_detect, false);
// Set up Write Protect input // Set up Write Protect input
int matrix_in_wp; int matrix_in_wp;
@@ -394,7 +479,7 @@ esp_err_t sdmmc_host_init_slot(int slot, const sdmmc_slot_config_t* slot_config)
} }
// WP signal is normally active low, but hardware expects // WP signal is normally active low, but hardware expects
// an active-high signal, so invert it in GPIO matrix // an active-high signal, so invert it in GPIO matrix
esp_rom_gpio_connect_in_signal(matrix_in_wp, pslot->write_protect, true); esp_rom_gpio_connect_in_signal(matrix_in_wp, slot_info->write_protect, true);
// By default, set probing frequency (400kHz) and 1-bit bus // By default, set probing frequency (400kHz) and 1-bit bus
esp_err_t ret = sdmmc_host_set_card_clk(slot, 400); esp_err_t ret = sdmmc_host_set_card_clk(slot, 400);
@@ -456,11 +541,11 @@ esp_err_t sdmmc_host_set_bus_width(int slot, size_t width)
SDMMC.ctype.card_width_8 &= ~mask; SDMMC.ctype.card_width_8 &= ~mask;
SDMMC.ctype.card_width |= mask; SDMMC.ctype.card_width |= mask;
// D3 was set to GPIO high to force slave into SD mode, until 4-bit mode is set // D3 was set to GPIO high to force slave into SD mode, until 4-bit mode is set
configure_pin(sdmmc_slot_info[slot].d3_gpio); configure_pin(d3, slot, GPIO_MODE_INPUT_OUTPUT);
} else if (width == 8) { } else if (width == 8) {
SDMMC.ctype.card_width_8 |= mask; SDMMC.ctype.card_width_8 |= mask;
// D3 was set to GPIO high to force slave into SD mode, until 4-bit mode is set // D3 was set to GPIO high to force slave into SD mode, until 4-bit mode is set
configure_pin(sdmmc_slot_info[slot].d3_gpio); configure_pin(d3, slot, GPIO_MODE_INPUT_OUTPUT);
} else { } else {
return ESP_ERR_INVALID_ARG; return ESP_ERR_INVALID_ARG;
} }
@@ -542,7 +627,7 @@ bool sdmmc_host_card_busy(void)
esp_err_t sdmmc_host_io_int_enable(int slot) esp_err_t sdmmc_host_io_int_enable(int slot)
{ {
configure_pin(sdmmc_slot_info[slot].d1_gpio); configure_pin(d1, slot, GPIO_MODE_INPUT_OUTPUT);
return ESP_OK; return ESP_OK;
} }
@@ -558,7 +643,7 @@ esp_err_t sdmmc_host_io_int_wait(int slot, TickType_t timeout_ticks)
SDMMC.intmask.sdio &= ~BIT(slot); /* Disable SDIO interrupt */ SDMMC.intmask.sdio &= ~BIT(slot); /* Disable SDIO interrupt */
SDMMC.rintsts.sdio = BIT(slot); SDMMC.rintsts.sdio = BIT(slot);
if (gpio_get_level(sdmmc_slot_info[slot].d1_gpio) == 0) { if (gpio_get_level(GPIO_NUM(slot, d1)) == 0) {
return ESP_OK; return ESP_OK;
} }
/* Otherwise, need to wait for an interrupt. Since D1 was high, /* Otherwise, need to wait for an interrupt. Since D1 was high,
@@ -619,25 +704,28 @@ static void sdmmc_isr(void* arg) {
} }
} }
esp_err_t sdmmc_host_pullup_en(int slot, int width) static esp_err_t sdmmc_host_pullup_en_internal(int slot, int width)
{ {
if (width > sdmmc_slot_info[slot].width) { if (width > sdmmc_slot_info[slot].width) {
//in esp32 we only support 8 bit in slot 0, note this is occupied by the flash by default //in esp32 we only support 8 bit in slot 0, note this is occupied by the flash by default
return ESP_ERR_INVALID_ARG; return ESP_ERR_INVALID_ARG;
} }
//according to the spec, the host control the clk, we don't to pull it up here // according to the spec, the host controls the clk, we don't to pull it up here
gpio_pullup_en(sdmmc_slot_info[slot].cmd_gpio); gpio_pullup_en(GPIO_NUM(slot, cmd));
gpio_pullup_en(sdmmc_slot_info[slot].d0_gpio); gpio_pullup_en(GPIO_NUM(slot, d0));
if (width >= 4) { if (width >= 4) {
gpio_pullup_en(sdmmc_slot_info[slot].d1_gpio); gpio_pullup_en(GPIO_NUM(slot, d1));
gpio_pullup_en(sdmmc_slot_info[slot].d2_gpio); gpio_pullup_en(GPIO_NUM(slot, d2));
gpio_pullup_en(sdmmc_slot_info[slot].d3_gpio); gpio_pullup_en(GPIO_NUM(slot, d3));
} }
if (width == 8) { if (width == 8) {
gpio_pullup_en(sdmmc_slot_info[slot].d4_gpio); gpio_pullup_en(GPIO_NUM(slot, d4));
gpio_pullup_en(sdmmc_slot_info[slot].d5_gpio); gpio_pullup_en(GPIO_NUM(slot, d5));
gpio_pullup_en(sdmmc_slot_info[slot].d6_gpio); gpio_pullup_en(GPIO_NUM(slot, d6));
gpio_pullup_en(sdmmc_slot_info[slot].d7_gpio); gpio_pullup_en(GPIO_NUM(slot, d7));
} }
return ESP_OK; return ESP_OK;
} }
/* Deprecared public function */
esp_err_t sdmmc_host_pullup_en(int slot, int width) __attribute__((alias("sdmmc_host_pullup_en_internal")));