SD/SDIO/MMC Driver¶
Overview¶
SD/SDIO/MMC driver currently supports SD memory and IO cards. Support for MCC/eMMC cards will be added in the future. This protocol level driver builds on top of SDMMC and SD SPI host drivers.
SDMMC and SD SPI host drivers (driver/sdmmc_host.h
) provide APIs to send commands to the slave device(s), send and receive data, and handle error conditions on the bus.
- See SDMMC Host API for functions used to initialize and configure SDMMC host.
- See SD SPI Host API for functions used to initialize and configure SD SPI host.
SDMMC protocol layer (sdmmc_cmd.h
), described in this document, handles specifics of SD protocol such as card initialization and data transfer commands.
Protocol layer works with the host via sdmmc_host_t
structure. This structure contains pointers to various functions of the host.
Application Example¶
An example which combines SDMMC driver with FATFS library is provided in examples/storage/sd_card
directory. This example initializes the card, writes and reads data from it using POSIX and C library APIs. See README.md file in the example directory for more information.
Protocol layer APIs¶
Protocol layer is given sdmmc_host_t
structure which describes the SD/MMC host driver, lists its capabilites, and provides pointers to functions of the driver. Protocol layer stores card-specific information in sdmmc_card_t
structure. When sending commands to the SD/MMC host driver, protocol layer uses sdmmc_command_t
structure to describe the command, argument, expected return value, and data to transfer, if any.
Usage with SD memory cards¶
- Call the host driver functions to initialize the host (e.g.
sdmmc_host_init()
,sdmmc_host_init_slot()
). - Call
sdmmc_card_init()
to initialize the card, passing it host driver information (host
) and a pointer tosdmmc_card_t
structure which will be filled in (card
). - To read and write sectors of the card, use
sdmmc_read_sectors()
andsdmmc_write_sectors()
, passing the pointer to card information structure (card
). - When card is not used anymore, call the host driver function to disable the host peripheral and free resources allocated by the driver (e.g.
sdmmc_host_deinit()
).
Usage with SDIO cards¶
Initialization an probing process is the same as with SD memory cards. Only data transfer commands differ in SDIO mode.
During probing and card initialization (done by sdmmc_card_init()
), the driver only configures the following registers of the IO card:
- The IO portion of the card is reset by setting RES bit in “I/O Abort” (0x06) register.
- If 4-line mode is enalbed in host and slot configuration, driver attempts to set “Bus width” field in “Bus Interface Control” (0x07) register. If that succeeds (which means that slave supports 4-line mode), host is also switched to 4-line mode.
- If high-speed mode is enabled in host configuration, SHS bit is set in “High Speed” (0x13) register.
In particular, the driver does not set any of the bits in I/O Enable, Int Enable registers, IO block sizes, etc. Applications can set these by calling sdmmc_io_write_byte()
.
For card configuration and data transfer, use one of the following functions:
sdmmc_io_read_byte()
,sdmmc_io_write_byte()
— read and write single byte using IO_RW_DIRECT (CMD52).sdmmc_io_read_bytes()
,sdmmc_io_write_bytes()
— read and write multiple bytes using IO_RW_EXTENDED (CMD53), in byte mode.sdmmc_io_read_blocks()
,sdmmc_io_write_blocks()
— read and write blocks of data using IO_RW_EXTENDED (CMD53), in block mode.
SDIO interrupts can be enabled by the application using sdmmc_io_enable_int()
function. When using SDIO in 1-line mode, D1 line also needs to be connected to use SDIO interrupts.
The application can wait for SDIO interrupt to occur using sdmmc_io_wait_int()
.
Combo (memory + IO) cards¶
The driver does not support SD combo cards. Combo cards will be treated as IO cards.
Thread safety¶
Most applications need to use the protocol layer only in one task; therefore the protocol layer doesn’t implement any kind of locking on the sdmmc_card_t
structure, or when accessing SDMMC or SD SPI host drivers. Such locking is usually implemented in the higher layer (e.g. in the filesystem driver).
API Reference¶
Header File¶
Functions¶
-
esp_err_t
sdmmc_card_init
(const sdmmc_host_t *host, sdmmc_card_t *out_card)¶ Probe and initialize SD/MMC card using given host
- Note
- Only SD cards (SDSC and SDHC/SDXC) are supported now. Support for MMC/eMMC cards will be added later.
- Return
- ESP_OK on success
- One of the error codes from SDMMC host controller
- Parameters
host
: pointer to structure defining host controllerout_card
: pointer to structure which will receive information about the card when the function completes
-
void
sdmmc_card_print_info
(FILE *stream, const sdmmc_card_t *card)¶ Print information about the card to a stream.
- Parameters
stream
: stream obtained using fopen or fdopencard
: card information structure initialized using sdmmc_card_init
-
esp_err_t
sdmmc_write_sectors
(sdmmc_card_t *card, const void *src, size_t start_sector, size_t sector_count)¶ Write given number of sectors to SD/MMC card
- Return
- ESP_OK on success
- One of the error codes from SDMMC host controller
- Parameters
card
: pointer to card information structure previously initialized using sdmmc_card_initsrc
: pointer to data buffer to read data from; data size must be equal to sector_count * card->csd.sector_sizestart_sector
: sector where to start writingsector_count
: number of sectors to write
-
esp_err_t
sdmmc_read_sectors
(sdmmc_card_t *card, void *dst, size_t start_sector, size_t sector_count)¶ Write given number of sectors to SD/MMC card
- Return
- ESP_OK on success
- One of the error codes from SDMMC host controller
- Parameters
card
: pointer to card information structure previously initialized using sdmmc_card_initdst
: pointer to data buffer to write into; buffer size must be at least sector_count * card->csd.sector_sizestart_sector
: sector where to start readingsector_count
: number of sectors to read
-
esp_err_t
sdmmc_io_read_byte
(sdmmc_card_t *card, uint32_t function, uint32_t reg, uint8_t *out_byte)¶ Read one byte from an SDIO card using IO_RW_DIRECT (CMD52)
- Return
- ESP_OK on success
- One of the error codes from SDMMC host controller
- Parameters
card
: pointer to card information structure previously initialized using sdmmc_card_initfunction
: IO function numberreg
: byte address within IO functionout_byte
: output, receives the value read from the card
-
esp_err_t
sdmmc_io_write_byte
(sdmmc_card_t *card, uint32_t function, uint32_t reg, uint8_t in_byte, uint8_t *out_byte)¶ Write one byte to an SDIO card using IO_RW_DIRECT (CMD52)
- Return
- ESP_OK on success
- One of the error codes from SDMMC host controller
- Parameters
card
: pointer to card information structure previously initialized using sdmmc_card_initfunction
: IO function numberreg
: byte address within IO functionin_byte
: value to be writtenout_byte
: if not NULL, receives new byte value read from the card (read-after-write).
-
esp_err_t
sdmmc_io_read_bytes
(sdmmc_card_t *card, uint32_t function, uint32_t addr, void *dst, size_t size)¶ Read multiple bytes from an SDIO card using IO_RW_EXTENDED (CMD53)
This function performs read operation using CMD53 in byte mode. For block mode, see sdmmc_io_read_blocks.
- Return
- ESP_OK on success
- ESP_ERR_INVALID_SIZE if size exceeds 512 bytes
- One of the error codes from SDMMC host controller
- Parameters
card
: pointer to card information structure previously initialized using sdmmc_card_initfunction
: IO function numberaddr
: byte address within IO function where reading startsdst
: buffer which receives the data read from cardsize
: number of bytes to read
-
esp_err_t
sdmmc_io_write_bytes
(sdmmc_card_t *card, uint32_t function, uint32_t addr, const void *src, size_t size)¶ Write multiple bytes to an SDIO card using IO_RW_EXTENDED (CMD53)
This function performs write operation using CMD53 in byte mode. For block mode, see sdmmc_io_write_blocks.
- Return
- ESP_OK on success
- ESP_ERR_INVALID_SIZE if size exceeds 512 bytes
- One of the error codes from SDMMC host controller
- Parameters
card
: pointer to card information structure previously initialized using sdmmc_card_initfunction
: IO function numberaddr
: byte address within IO function where writing startssrc
: data to be writtensize
: number of bytes to write
-
esp_err_t
sdmmc_io_read_blocks
(sdmmc_card_t *card, uint32_t function, uint32_t addr, void *dst, size_t size)¶ Read blocks of data from an SDIO card using IO_RW_EXTENDED (CMD53)
This function performs read operation using CMD53 in block mode. For byte mode, see sdmmc_io_read_bytes.
- Return
- ESP_OK on success
- ESP_ERR_INVALID_SIZE if size is not divisible by 512 bytes
- One of the error codes from SDMMC host controller
- Parameters
card
: pointer to card information structure previously initialized using sdmmc_card_initfunction
: IO function numberaddr
: byte address within IO function where writing startsdst
: buffer which receives the data read from cardsize
: number of bytes to read, must be divisible by the card block size.
-
esp_err_t
sdmmc_io_write_blocks
(sdmmc_card_t *card, uint32_t function, uint32_t addr, const void *src, size_t size)¶ Write blocks of data to an SDIO card using IO_RW_EXTENDED (CMD53)
This function performs write operation using CMD53 in block mode. For byte mode, see sdmmc_io_write_bytes.
- Return
- ESP_OK on success
- ESP_ERR_INVALID_SIZE if size is not divisible by 512 bytes
- One of the error codes from SDMMC host controller
- Parameters
card
: pointer to card information structure previously initialized using sdmmc_card_initfunction
: IO function numberaddr
: byte address within IO function where writing startssrc
: data to be writtensize
: number of bytes to read, must be divisible by the card block size.
-
esp_err_t
sdmmc_io_enable_int
(sdmmc_card_t *card)¶ Enable SDIO interrupt in the SDMMC host
- Return
- ESP_OK on success
- ESP_ERR_NOT_SUPPORTED if the host controller does not support IO interrupts
- Parameters
card
: pointer to card information structure previously initialized using sdmmc_card_init
-
esp_err_t
sdmmc_io_wait_int
(sdmmc_card_t *card, TickType_t timeout_ticks)¶ Block until an SDIO interrupt is received
Slave uses D1 line to signal interrupt condition to the host. This function can be used to wait for the interrupt.
- Return
- ESP_OK if the interrupt is received
- ESP_ERR_NOT_SUPPORTED if the host controller does not support IO interrupts
- ESP_ERR_TIMEOUT if the interrupt does not happen in timeout_ticks
- Parameters
card
: pointer to card information structure previously initialized using sdmmc_card_inittimeout_ticks
: time to wait for the interrupt, in RTOS ticks
Header File¶
Structures¶
-
struct
sdmmc_csd_t
¶ Decoded values from SD card Card Specific Data register
-
struct
sdmmc_cid_t
¶ Decoded values from SD card Card IDentification register
-
struct
sdmmc_scr_t
¶ Decoded values from SD Configuration Register
-
struct
sdmmc_switch_func_rsp_t
¶ SD SWITCH_FUNC response buffer
Public Members
-
uint32_t
data
[512 / 8 / sizeof(uint32_t)]¶ response data
-
uint32_t
-
struct
sdmmc_command_t
¶ SD/MMC command information
Public Members
-
uint32_t
opcode
¶ SD or MMC command index
-
uint32_t
arg
¶ SD/MMC command argument
-
sdmmc_response_t
response
¶ response buffer
-
void *
data
¶ buffer to send or read into
-
size_t
datalen
¶ length of data buffer
-
size_t
blklen
¶ block length
-
int
flags
¶ see below
-
int
timeout_ms
¶ response timeout, in milliseconds
-
uint32_t
-
struct
sdmmc_host_t
¶ SD/MMC Host description
This structure defines properties of SD/MMC host and functions of SD/MMC host which can be used by upper layers.
Public Members
-
uint32_t
flags
¶ flags defining host properties
-
int
slot
¶ slot number, to be passed to host functions
-
int
max_freq_khz
¶ max frequency supported by the host
-
float
io_voltage
¶ I/O voltage used by the controller (voltage switching is not supported)
-
size_t (*
get_bus_width
)(int slot)¶ host function to get bus width
-
esp_err_t (*
do_transaction
)(int slot, sdmmc_command_t *cmdinfo)¶ host function to do a transaction
-
esp_err_t (*
io_int_wait
)(int slot, TickType_t timeout_ticks)¶ Host function to wait for SDIO interrupt line to be active
-
int
command_timeout_ms
¶ timeout, in milliseconds, of a single command. Set to 0 to use the default value.
-
uint32_t
-
struct
sdmmc_card_t
¶ SD/MMC card information structure
Public Members
-
sdmmc_host_t
host
¶ Host with which the card is associated
-
uint32_t
ocr
¶ OCR (Operation Conditions Register) value
-
sdmmc_cid_t
cid
¶ decoded CID (Card IDentification) register value
-
sdmmc_csd_t
csd
¶ decoded CSD (Card-Specific Data) register value
-
sdmmc_scr_t
scr
¶ decoded SCR (SD card Configuration Register) value
-
uint16_t
rca
¶ RCA (Relative Card Address)
-
uint32_t
is_mem
¶ Bit indicates if the card is a memory card
-
uint32_t
is_sdio
¶ Bit indicates if the card is an IO card
-
uint32_t
num_io_functions
¶ If is_sdio is 1, contains the number of IO functions on the card
-
uint32_t
reserved
¶ Reserved for future expansion
-
sdmmc_host_t
Macros¶
-
SDMMC_HOST_FLAG_1BIT
¶ host supports 1-line SD and MMC protocol
-
SDMMC_HOST_FLAG_4BIT
¶ host supports 4-line SD and MMC protocol
-
SDMMC_HOST_FLAG_8BIT
¶ host supports 8-line MMC protocol
-
SDMMC_HOST_FLAG_SPI
¶ host supports SPI protocol
-
SDMMC_FREQ_DEFAULT
¶ SD/MMC Default speed (limited by clock divider)
-
SDMMC_FREQ_HIGHSPEED
¶ SD High speed (limited by clock divider)
-
SDMMC_FREQ_PROBING
¶ SD/MMC probing speed