/***************************************************************************//** * \file cyhal_sdhc.h * * \brief * Provides a high level interface for interacting with the Cypress SDHC. * This interface abstracts out the chip specific details. If any chip specific * functionality is necessary, or performance is critical the low level functions * can be used directly. * ******************************************************************************** * \copyright * Copyright 2018-2021 Cypress Semiconductor Corporation * SPDX-License-Identifier: Apache-2.0 * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. *******************************************************************************/ /** * \addtogroup group_hal_sdhc SDHC (SD Host Controller) * \ingroup group_hal * \{ * High level interface to the Secure Digital Host Controller (SDHC). * * The SDHC driver allows data to be read from and written to an SD Card using the SDHC block. * The data is transferred in blocks with a block size of 512 bytes. * * \section subsection_sdhc_features Features * * Supports the 4-bit interface * * Supports Ultra High Speed (UHS-I) mode * * Supports Default Speed (DS), High Speed (HS), SDR12, SDR25 and SDR50 speed modes * * \section subsection_sdhc_quickstart Quick Start * Initialize SDHC by using \ref cyhal_sdhc_init by selecting the pins according to the target device used. * Specify the SDHC configuration using the configuration structure (const \ref cyhal_sdhc_config_t * config). <br> * See \ref subsection_sdhc_snippet_1 * * \section subsection_sdhc_code_snippets Code Snippets * * \subsection subsection_sdhc_snippet_1 Snippet 1: SDHC Initialization and configuration * The following snippet is used to initialize the SDHC block. SDHC object - \ref cyhal_sdhc_t, * SDHC card configuration structure (const \ref cyhal_sdhc_config_t * config). The pins connected to the SDHC block * needs to be provided to the \ref cyhal_sdhc_init function. * \snippet hal_sdhc.c snippet_cyhal_sdhc_init * * \subsection subsection_sdhc_snippet_2 Snippet 2: Reading a block of data from an SD Card * The following snippet reads a block of data from the SD Card. * \snippet hal_sdhc.c snippet_cyhal_sdhc_read * \subsection subsection_sdhc_snippet_3 Snippet 3: Writing a block of data to an SD Card * The following snippet writes a block of data to the SD Card. * \snippet hal_sdhc.c snippet_cyhal_sdhc_write */ #pragma once #include <stdint.h> #include <stdbool.h> #include "cy_result.h" #include "cyhal_hw_types.h" #if defined(__cplusplus) extern "C" { #endif /******************************************************************************* * Defines *******************************************************************************/ /** \addtogroup group_hal_results_sdhc SDHC HAL Results * SDHC specific return codes * \ingroup group_hal_results * \{ *//** */ /** Pin related Error. */ #define CYHAL_SDHC_RSLT_ERR_PIN \ (CYHAL_RSLT_CREATE(CY_RSLT_TYPE_ERROR, CYHAL_RSLT_MODULE_SDHC, 0)) /** Requested feature is not supported on this hardware. */ #define CYHAL_SDHC_RSLT_ERR_UNSUPPORTED \ (CYHAL_RSLT_CREATE(CY_RSLT_TYPE_ERROR, CYHAL_RSLT_MODULE_SDHC, 1)) /** Timeout during waiting for erase complete. */ #define CYHAL_SDHC_RSLT_ERR_ERASE_CMPLT_TIMEOUT \ (CYHAL_RSLT_CREATE(CY_RSLT_TYPE_ERROR, CYHAL_RSLT_MODULE_SDHC, 2)) /** Block count cannot be retrieved. */ #define CYHAL_SDHC_RSLT_ERR_BLOCK_COUNT_GET_FAILURE \ (CYHAL_RSLT_CREATE(CY_RSLT_TYPE_ERROR, CYHAL_RSLT_MODULE_SDHC, 3)) /** Cannot set desired SD bus frequency. */ #define CYHAL_SDHC_RSLT_ERR_SET_FREQ \ (CYHAL_RSLT_CREATE(CY_RSLT_TYPE_ERROR, CYHAL_RSLT_MODULE_SDHC, 4)) /** Incorrect function parameter. */ #define CYHAL_SDHC_RSLT_ERR_WRONG_PARAM \ (CYHAL_RSLT_CREATE(CY_RSLT_TYPE_ERROR, CYHAL_RSLT_MODULE_SDHC, 5)) /** Error occured during I/O voltage switch sequence. */ #define CYHAL_SDHC_RSLT_ERR_IO_VOLT_SWITCH_SEQ \ (CYHAL_RSLT_CREATE(CY_RSLT_TYPE_ERROR, CYHAL_RSLT_MODULE_SDHC, 6)) /** Cannot configure data timeout. */ #define CYHAL_SDHC_RSLT_ERR_TOUT_CFG \ (CYHAL_RSLT_CREATE(CY_RSLT_TYPE_ERROR, CYHAL_RSLT_MODULE_SDHC, 7)) /** * \} */ /******************************************************************************* * Enumerations *******************************************************************************/ /** Card types */ typedef enum { CYHAL_SDHC_SD, //!< Secure Digital card CYHAL_SDHC_SDIO, //!< SD Input Output card CYHAL_SDHC_EMMC, //!< Embedded Multimedia card CYHAL_SDHC_COMBO, //!< Combo Card (SD + SDIO) CYHAL_SDHC_UNUSABLE, //!< Unusable card or unsupported type CYHAL_SDHC_NOT_EMMC, //!< Not an eMMC card } cyhal_sdhc_card_type_t; /** SDHC interrupt triggers */ typedef enum { CYHAL_SDHC_CMD_COMPLETE = 0x00001, //!< Command Complete CYHAL_SDHC_XFER_COMPLETE = 0x00002, //!< Host read/write transfer is complete CYHAL_SDHC_BGAP_EVENT = 0x00004, //!< This bit is set when both read/write transaction is stopped at the block gap CYHAL_SDHC_DMA_INTERRUPT = 0x00008, //!< Host controller detects an SDMA Buffer Boundary during transfer CYHAL_SDHC_BUF_WR_READY = 0x00010, //!< This bit is set if the Buffer Write Enable changes from 0 to 1 CYHAL_SDHC_BUF_RD_READY = 0x00020, //!< This bit is set if the Buffer Read Enable changes from 0 to 1 CYHAL_SDHC_CARD_INSERTION = 0x00040, //!< This bit is set if the Card Inserted in the Present State register changes from 0 to 1. CYHAL_SDHC_CARD_REMOVAL = 0x00080, //!< This bit is set if the Card Inserted in the Present State register changes from 1 to 0. CYHAL_SDHC_INT_A = 0x00200, CYHAL_SDHC_INT_B = 0x00400, CYHAL_SDHC_INT_C = 0x00800, CYHAL_SDHC_RE_TUNE_EVENT = 0x01000, //!< This bit is set if the Re-Tuning Request changes from 0 to 1 CYHAL_SDHC_FX_EVENT = 0x02000, //!< This status is set when R[14] of response register is set to 1 CYHAL_SDHC_CQE_EVENT = 0x04000, //!< This status is set if Command Queuing/Crypto event has occurred CYHAL_SDHC_ERR_INTERRUPT = 0x08000, //!< If any of the bits in the Error Interrupt Status register are set CYHAL_SDHC_ALL_INTERRUPTS = 0x0FEFF, //!< Is used to enable/disable all interrupts } cyhal_sdhc_event_t; /** I/O voltage levels */ typedef enum { CYHAL_SDHC_IO_VOLTAGE_3_3V, //!< I/O voltage is 3.3V. CYHAL_SDHC_IO_VOLTAGE_1_8V //!< I/O voltage is 1.8V. } cyhal_sdhc_io_voltage_t; /** SDHC I/O voltage select principle */ typedef enum { CYHAL_SDHC_IO_VOLT_ACTION_NEGOTIATE = 0U, //!< (Recommended) HAL driver performs all steps, needed for switching I/O bus voltage to certain level: sends needed commands to prepare card, changes level of io_volt_sel pin and performs switching sequence according to SD specification. CYHAL_SDHC_IO_VOLT_ACTION_SWITCH_SEQ_ONLY = 1U, //!< HAL driver performs switching sequence (if voltage is being switched to 1.8V) and changes io_volt_sel pin level accordingly. No commands are being send to the card in this mode. CYHAL_SDHC_IO_VOLT_ACTION_NONE = 2U, //!< I/O voltage is changed by changing io_volt_sel pin's level. No commands are being send to the card in this mode. } cyhal_sdhc_io_volt_action_type_t; /** SDHC response types */ typedef enum { CYHAL_SDHC_RESPONSE_NONE = 0U, //!< No Response. CYHAL_SDHC_RESPONSE_LEN_136 = 1U, //!< Response Length 136. CYHAL_SDHC_RESPONSE_LEN_48 = 2U, //!< Response Length 48. CYHAL_SDHC_RESPONSE_LEN_48B = 3U //!< Response Length 48. Check Busy after response. } cyhal_sdhc_cmd_response_type_t; /** SDHC auto command enable selection. */ typedef enum { CYHAL_SDHC_AUTO_CMD_NONE = 0U, //!< Auto command disable. CYHAL_SDHC_AUTO_CMD_12 = 1U, //!< Auto command 12 enable. CYHAL_SDHC_AUTO_CMD_23 = 2U, //!< Auto command 23 enable. CYHAL_SDHC_AUTO_CMD_AUTO = 3U //!< Auto command Auto enable. } cyhal_sdhc_auto_cmd_t; /** SDHC command types */ typedef enum { CYHAL_SDHC_CMD_NORMAL = 0U, //!< Other commands. CYHAL_SDHC_CMD_SUSPEND = 1U, //!< CMD52 for writing "Bus Suspend" in CCCR. CYHAL_SDHC_CMD_RESUME = 2U, //!< CMD52 for writing "Function Select" in CCCR. CYHAL_SDHC_CMD_ABORT = 3U //!< CMD12, CMD52 for writing "I/O Abort" in CCCR. } cyhal_sdhc_cmd_type_t; /** SDHC command error states */ typedef enum { /** Last operation did not cause any error status. */ CYHAL_SDHC_NO_ERR = 0x0000U, /** Command timeout error. In SD/eMMC Mode, this event is set only if no response is returned * within 64 SD clock cycles from the end bit of the command. If the Host Controller detects a CMD line conflict, * along with Command CRC Error bit, this event is set to 1, without waiting for 64 SD/eMMC card clock cycles. */ CYHAL_SDHC_CMD_TOUT_ERR = 0x0001U, /** Command CRC error. A Command CRC Error is generated in SD/eMMC mode when: * 1. A response is returned and the Command Timeout Error is set to 0 (indicating no timeout), * this bit is set to 1 when detecting a CRC error in the command response. * 2. The Host Controller detects a CMD line conflict by monitoring the CMD line when a command is issued. If * the Host Controller drives the CMD line to level 1, but detects level 0 on the CMD line at the next SD clock * edge, then the Host Controller aborts the command (stop driving CMD line) and sets this bit to 1. The Command * Timeout Error is also set to 1 to distinguish a CMD line conflict. */ CYHAL_SDHC_CMD_CRC_ERR = 0x0002U, /** Command End Bit error. This bit is set after detecting that the end bit of a command response is 0 in SD/eMMC mode. */ CYHAL_SDHC_CMD_END_BIT_ERR = 0x0004U, /** Command Index error. This bit is set if a Command Index error occurs in the command response in SD/eMMC mode. */ CYHAL_SDHC_CMD_IDX_ERR = 0x0008U, /** Data Timeout error. This bit is set in SD/eMMC mode when detecting one of the following timeout conditions: * * Busy timeout for R1b, R5b type * * Busy timeout after Write CRC status * * Write CRC Status timeout * * Read Data timeout. */ CYHAL_SDHC_DATA_TOUT_ERR = 0x0010U, /** Data CRC error. This error occurs in SD/eMMC mode after detecting a CRC error while transferring read data * that uses the DAT line, detecting the Write CRC status having a value other than 010 or when writing a CRC status timeout. */ CYHAL_SDHC_DATA_CRC_ERR = 0x0020U, /** Data End Bit error. This error occurs in SD/eMMC mode either when detecting 0 at the end bit position of read * data that uses the DAT line or at the end bit position of the CRC status. */ CYHAL_SDHC_DATA_END_BIT_ERR = 0x0040U, /** Current Limit error. */ CYHAL_SDHC_CUR_LMT_ERR = 0x0080U, /** Auto CMD error. This error status is used by Auto CMD12 and Auto CMD23 in SD/eMMC mode. This bit is set after * detecting that any of the bits D00 to D05 in the Auto CMD Error Status register has changed from 0 to 1. D07 is * effective in case for Auto CMD12. The Auto CMD Error Status register is valid while this bit is set to 1 and may * be cleared by clearing this bit. */ CYHAL_SDHC_AUTO_CMD_ERR = 0x0100U, /** ADMA error. This bit is set when the Host Controller detects an error during an ADMA-based data transfer. * The possible reasons for an error: * * An error response is received from the System bus; * * ADMA3, ADMA2 Descriptors are invalid; * * CQE Task or Transfer descriptors are invalid. * When an error occurs, the state of the ADMA is saved in the ADMA Error Status register. */ CYHAL_SDHC_ADMA_ERR = 0x0200U, /** Tuning error. */ CYHAL_SDHC_TUNNING_ERR = 0x0400U, /** Response error. Host Controller Version 4.00 supports the response error check function to avoid overhead of * the response error check by Host Driver during DMA execution. If the Response Error Check Enable is set to 1 in * the Transfer Mode register, the Host Controller checks R1 or R5 response. If an error is detected in a response, * this bit is set to 1. This is applicable in SD/eMMC mode. */ CYHAL_SDHC_RESP_ERR = 0x0800U, /** Boot Acknowledgement error. This bit is set when there is a timeout for boot acknowledgement or after detecting * the boot ACK status having a value other than 010. This is applicable only when boot acknowledgement is expected in eMMC mode. */ CYHAL_SDHC_BOOT_ACK_ERR = 0x1000U } cyhal_sdhc_error_type_t; /******************************************************************************* * Typedefs *******************************************************************************/ /** Handler for SDHC interrupts */ typedef void (*cyhal_sdhc_event_callback_t)(void *callback_arg, cyhal_sdhc_event_t event); /******************************************************************************* * Data Structures *******************************************************************************/ /** Defines configuration options for the SDHC block */ typedef struct { bool enableLedControl; //!< Drive one IO to indicate when the card is being accessed bool lowVoltageSignaling; //!< Whether 1.8V signaling is supported bool isEmmc; //!< true if eMMC card, otherwise false uint8_t busWidth; //!< The desired bus width, 1-bit, 4-bit, 8-bit } cyhal_sdhc_config_t; /** Defines data configuration */ typedef struct { uint32_t* data_ptr; //!< The pointer to data for send/receive. Data will be transfered using DMA, which will be configured automaticaly by SDHC HAL driver. uint32_t block_size; //!< The size of the data block. uint32_t number_of_blocks; //!< The number of blocks with size block_size to send. cyhal_sdhc_auto_cmd_t auto_command; //!< Selects which auto commands are used if any. bool is_read; //!< true = Read from the card, false = Write to the card. } cyhal_sdhc_data_config_t; /** Defines command configuration for \ref cyhal_sdhc_send_cmd function */ typedef struct { uint32_t command_index; //!< The index of the command. uint32_t command_argument; //!< The argument for the command. bool enable_crc_check; //!< Enables the CRC check on the response. cyhal_sdhc_cmd_response_type_t response_type; //!< The response type. bool enable_idx_check; //!< Checks the index of the response. cyhal_sdhc_cmd_type_t command_type; //!< The command type. cyhal_sdhc_data_config_t *data_config; //!< Data transfer configuration, defined in \ref cyhal_sdhc_data_config_t. Should be NULL if data transfer is not expected for provided command. } cyhal_sdhc_cmd_config_t; /******************************************************************************* * Functions *******************************************************************************/ /** Initialize SDHC driver and corresponding hardware. * * @param[out] obj Pointer to an SDHC object. The caller must allocate the memory for this object but * the init function will initialize its contents. * @param[in] config The card configuration object * @param[out] clk The pin connected to the clk signal * @param[in] cmd The pin connected to the command signal * @param[in] data0 The pin connected to the data0 signal * @param[in] data1 The pin connected to the data1 signal. This pin can be NC if bus width is 1-bit. * @param[in] data2 The pin connected to the data2 signal. This pin can be NC if bus width is 1-bit. * @param[in] data3 The pin connected to the data3 signal. This pin can be NC if bus width is 1-bit. * @param[in] data4 The pin connected to the data4 signal. This pin can be NC if bus width is less than 8-bit. * @param[in] data5 The pin connected to the data5 signal. This pin can be NC if bus width is less than 8-bit. * @param[in] data6 The pin connected to the data6 signal. This pin can be NC if bus width is less than 8-bit. * @param[in] data7 The pin connected to the data7 signal. This pin can be NC if bus width is less than 8-bit. * @param[in] card_detect The pin connected to the card_detect signal * Card is considered as inserted if card_detect pin in low state. * This pin can be NC. * @param[out] io_volt_sel The pin connected to the io_volt_sel signal. This pin changes the logic level on the * sd_io_volt_sel line. It assumes that this line is used to control a regulator connected to the VDDIO of the MCU. * This regulator allows for switching between the 3.3V and 1.8V signaling. High level on the pin stands for * 1.8V signaling, while low - for 3.3V. * This pin can be NC. * @param[out] card_pwr_en The pin connected to the card_pwr_en signal This pin can be used to enable a voltage * regulator used to power the card. High level on the pin - card is powered. Low - card is not powered. * This pin can be NC. * @param[in] card_mech_write_prot The pin connected to the card_mech_write_prot signal This pin is used for card * mechanical write protect checking. * If pin in high state - card is mechanically write protected, if in low state - card is not mechanically write * protected. * \ref cyhal_sdhc_is_card_mech_write_protected function can be used to check card write protection state. * This pin can be NC. * @param[in] led_ctrl The pin connected to the led_ctrl signal. This pin can be NC. * @param[in] emmc_reset The pin connected to the emmc_reset signal. This pin can be NC. * @return The status of the init request * */ cy_rslt_t cyhal_sdhc_init_hw(cyhal_sdhc_t *obj, const cyhal_sdhc_config_t *config, cyhal_gpio_t cmd, cyhal_gpio_t clk, cyhal_gpio_t data0, cyhal_gpio_t data1, cyhal_gpio_t data2, cyhal_gpio_t data3, cyhal_gpio_t data4, cyhal_gpio_t data5, cyhal_gpio_t data6, cyhal_gpio_t data7, cyhal_gpio_t card_detect, cyhal_gpio_t io_volt_sel, cyhal_gpio_t card_pwr_en, cyhal_gpio_t card_mech_write_prot, cyhal_gpio_t led_ctrl, cyhal_gpio_t emmc_reset); /** Initialize the connected card. * \note This function should be called after \ref cyhal_sdhc_init_hw * * @param[in,out] obj The SDHC object * @return The status of the card init request */ cy_rslt_t cyhal_sdhc_init_card(cyhal_sdhc_t *obj); /** Initialize the SDHC block and connected card. * This function is a combination of \ref cyhal_sdhc_init_hw and \ref cyhal_sdhc_init_card calls. * \note refer to \ref cyhal_sdhc_init_hw function description for detailed parameter guidance. * * @param[out] obj Pointer to an SDHC object. The caller must allocate the memory for this object but * the init function will initialize its contents. * @param[in] config The card configuration object * @param[out] clk The pin connected to the clk signal * @param[in] cmd The pin connected to the command signal * @param[in] data0 The pin connected to the data0 signal * @param[in] data1 The pin connected to the data1 signal. This pin can be NC if bus width is 1-bit. * @param[in] data2 The pin connected to the data2 signal. This pin can be NC if bus width is 1-bit. * @param[in] data3 The pin connected to the data3 signal. This pin can be NC if bus width is 1-bit. * @param[in] data4 The pin connected to the data4 signal. This pin can be NC if bus width is less than 8-bit. * @param[in] data5 The pin connected to the data5 signal. This pin can be NC if bus width is less than 8-bit. * @param[in] data6 The pin connected to the data6 signal. This pin can be NC if bus width is less than 8-bit. * @param[in] data7 The pin connected to the data7 signal. This pin can be NC if bus width is less than 8-bit. * @param[in] card_detect The pin connected to the card_detect signal * @param[out] io_volt_sel The pin connected to the io_volt_sel signal * @param[out] card_pwr_en The pin connected to the card_pwr_en signal * @param[in] card_mech_write_prot The pin connected to the card_mech_write_prot signal * @param[in] led_ctrl The pin connected to the led_ctrl signal * @param[in] emmc_reset The pin connected to the emmc_reset signal * @return The status of the init request * */ cy_rslt_t cyhal_sdhc_init(cyhal_sdhc_t *obj, const cyhal_sdhc_config_t *config, cyhal_gpio_t cmd, cyhal_gpio_t clk, cyhal_gpio_t data0, cyhal_gpio_t data1, cyhal_gpio_t data2, cyhal_gpio_t data3, cyhal_gpio_t data4, cyhal_gpio_t data5, cyhal_gpio_t data6, cyhal_gpio_t data7, cyhal_gpio_t card_detect, cyhal_gpio_t io_volt_sel, cyhal_gpio_t card_pwr_en, cyhal_gpio_t card_mech_write_prot, cyhal_gpio_t led_ctrl, cyhal_gpio_t emmc_reset); /** Release the SDHC peripheral, not currently invoked. It requires further * resource management. * * @param[in,out] obj The SDHC object */ void cyhal_sdhc_free(cyhal_sdhc_t *obj); /** Attempts to read data synchronously over the SDHC peripheral. * * This will read as many blocks as possible, up to `length` blocks, into the buffer * pointed to by `data`, then return. The value pointed to by `length` will be * updated to reflect the number of words that were actually read. * * See \ref subsection_sdhc_snippet_2 * * @param[in] obj The SDHC object * @param[in] address The address to read data from * @param[out] data Pointer to the byte-array where data read from the device should be stored * @param[in,out] length Number of 512 byte blocks to read, updated with the number actually read * @return The status of the read request */ cy_rslt_t cyhal_sdhc_read(const cyhal_sdhc_t *obj, uint32_t address, uint8_t *data, size_t *length); /** Attempts to write data synchronously over SDHC peripheral * * This will write as many blocks as possible, up to `length` blocks, then return. * The value pointed to by `length` will be updated to reflect the number of words * that were actually read. * * See \ref subsection_sdhc_snippet_3 * * @param[in] obj The SDHC object * @param[in] address The address to write data to * @param[in] data Pointer to the byte-array of data to write to the device * @param[in,out] length Number of 512 byte blocks to write, updated with the number actually written * @return The status of the write request * */ cy_rslt_t cyhal_sdhc_write(const cyhal_sdhc_t *obj, uint32_t address, const uint8_t *data, size_t *length); /** Erases a block of data over the SDHC peripheral * * @param[in] obj The SDHC object * @param[in] start_addr Is the address of the first byte to erase * @param[in] length Number of 512 byte blocks (starting at start_addr) to erase * @return The status of the erase request * */ cy_rslt_t cyhal_sdhc_erase(const cyhal_sdhc_t *obj, uint32_t start_addr, size_t length); /** Start SDHC asynchronous read * * This will transfer `length` 512 byte blocks into the buffer pointed to by `data` in the background. * When the requested quantity of data has been read, the \ref CYHAL_SDHC_XFER_COMPLETE event will * be raised. See \ref cyhal_sdhc_register_callback and \ref cyhal_sdhc_enable_event. * * @param[in] obj The SDHC object that holds the transfer information * @param[in] address The address to read data from * @param[out] data The receive buffer * @param[in,out] length Number of 512 byte blocks to read, updated with the number actually read * @return The status of the read_async request */ cy_rslt_t cyhal_sdhc_read_async(const cyhal_sdhc_t *obj, uint32_t address, uint8_t *data, size_t *length); /** Start asynchronous SDHC write * * This will transfer `length` 512 byte blocks from the buffer pointed to by `data` in the background. * When the requested quantity of data has been written, the \ref CYHAL_SDHC_XFER_COMPLETE event * will be raised. See \ref cyhal_sdhc_register_callback and \ref cyhal_sdhc_enable_event. * * @param[in] obj The SDHC object that holds the transfer information * @param[in] address The address to write data to * @param[in] data The transmit buffer * @param[in,out] length The number of 512 byte blocks to write, updated with the number actually written * @return The status of the write_async request */ cy_rslt_t cyhal_sdhc_write_async(const cyhal_sdhc_t *obj, uint32_t address, const uint8_t *data, size_t *length); /** Checks if the specified SDHC peripheral is in use * * @param[in] obj The SDHC peripheral to check * @return Indication of whether the SDHC is still transmitting */ bool cyhal_sdhc_is_busy(const cyhal_sdhc_t *obj); /** Abort an SDHC transfer * * @param[in] obj The SDHC peripheral to stop * @return The status of the abort_async request */ cy_rslt_t cyhal_sdhc_abort_async(const cyhal_sdhc_t *obj); /** Register an SDHC callback handler * * This function will be called when one of the events enabled by \ref cyhal_sdhc_enable_event occurs. * * @param[in] obj The SDHC object * @param[in] callback The callback handler which will be invoked when the event fires * @param[in] callback_arg Generic argument that will be provided to the callback when called */ void cyhal_sdhc_register_callback(cyhal_sdhc_t *obj, cyhal_sdhc_event_callback_t callback, void *callback_arg); /** Configure SDHC event enablement. * * When an enabled event occurs, the function specified by \ref cyhal_sdhc_register_callback will be called. * * @param[in] obj The SDHC object * @param[in] event The SDHC event type * @param[in] intr_priority The priority for NVIC interrupt events * @param[in] enable True to turn on interrupts, False to turn off */ void cyhal_sdhc_enable_event(cyhal_sdhc_t *obj, cyhal_sdhc_event_t event, uint8_t intr_priority, bool enable); /** Checks if SD card is inserted * * @param[in] obj The SDHC peripheral to check * @return Indication of whether the card is inserted. */ bool cyhal_sdhc_is_card_inserted(const cyhal_sdhc_t *obj); /** Checks if the inserted SD card is mechanically write protected * * @param[in] obj The SDHC peripheral to check * @return Indication of whether the inserted SD card is mechanically write protected */ bool cyhal_sdhc_is_card_mech_write_protected(const cyhal_sdhc_t *obj); /** Get block count of inserted SD card / eMMC * \note SDHC driver should be initialized with \ref cyhal_sdhc_init prior to using this function. * * @param[in] obj The SDHC object * @param[in] block_count Pointer to variable where block count will be stored * @return The status of the operation */ cy_rslt_t cyhal_sdhc_get_block_count(cyhal_sdhc_t *obj, uint32_t *block_count); /** Sets the SD bus frequency (frequency on which SD card / eMMC is accessed) * \note Actual frequency may differ from the desired frequency due to available dividers and the frequency of source clock. * Function will set the closest possible (but not greater than) frequency to what was requested. * Use \ref cyhal_sdhc_get_frequency function to get actual frequency value that was achieved and set. * \note If data timeout was configured by \ref cyhal_sdhc_set_data_read_timeout, it can be automaticaly recalculated * according to new SD bus frequency. For details, please refer to \ref cyhal_sdhc_set_data_read_timeout function description. * * @param[in] obj The SDHC object * @param[in] hz Desired SD bus frequency in Hz * @param[in] negotiate Whether new frequency value needs to be negotiated with card or not. true is * recommended and it means that new frequency will be negotiated. * @return The status of the operation */ cy_rslt_t cyhal_sdhc_set_frequency(cyhal_sdhc_t *obj, uint32_t hz, bool negotiate); /** Get the actual frequency that SD bus is configured for * * @param[in] obj The SDHC object * @return Frequency in Hz */ uint32_t cyhal_sdhc_get_frequency(cyhal_sdhc_t *obj); /** Sets the maximum time to wait for data from the card. The time is specified in number of card clock cycles. * With SD bus frequency changed by \ref cyhal_sdhc_set_frequency, timeout can automaticaly be recalculated according * to new clock frequency. This can be activated by 'auto_reconfigure' parameter. * * @param[in] obj The SDHC object * @param[in] timeout Time to wait for data from the card. * @param[in] auto_reconfigure Timeout value will be automaticaly reconfigured upon clock change * * @return The status of the operation */ cy_rslt_t cyhal_sdhc_set_data_read_timeout(cyhal_sdhc_t *obj, uint32_t timeout, bool auto_reconfigure); /** Initializes the SD block and DMA for a data transfer. It does not start a transfer. * \ref cyhal_sdhc_send_cmd needs to be called after this function in order to start data transfer. * * @param[in] obj The SDHC object * @param[in] data_config Data transfer configuration * @return The status of the operation */ cy_rslt_t cyhal_sdhc_config_data_transfer(cyhal_sdhc_t *obj, cyhal_sdhc_data_config_t *data_config); /** Sends a command to the card and wait until it is sent. If the command assumes data transfer via data lines, * \ref cyhal_sdhc_config_data_transfer function needs to be called prior to this one. The response of issued command * can be retrieved by using \ref cyhal_sdhc_get_response function. * \note Function does not wait until data (configured with \ref cyhal_sdhc_config_data_transfer) transfer complete. * In order to do so, user can: * - register \ref CYHAL_SDHC_XFER_COMPLETE and wait for it * - wait until \ref cyhal_sdhc_is_busy return false * - use \ref cyhal_sdhc_wait_transfer_complete function * * @param[in] obj The SDHC object * @param[in] cmd_config Command configuration * @return The status of the operation */ cy_rslt_t cyhal_sdhc_send_cmd(cyhal_sdhc_t *obj, cyhal_sdhc_cmd_config_t *cmd_config); /** Returns a response of last issued by \ref cyhal_sdhc_send_cmd function command. * * @param[in] obj The SDHC object * @param[in] response Pointer to array where response will be stored * @param[in] large_response If true, the expected response is 136 bits, false - 48 bits, which corresponds to * 120 and 32 bits of useful for application data respectively. So for large_response 4 uint32_t element array can be used * while for not large_response 1 uint32_t value will be enough. * @return The status of the operation */ cy_rslt_t cyhal_sdhc_get_response(cyhal_sdhc_t *obj, uint32_t *response, bool large_response); /** Wait for asynchronous data transfer to complete. Such data transfer can be triggered by \ref cyhal_sdhc_write_async, * \ref cyhal_sdhc_read_async or by \ref cyhal_sdhc_config_data_transfer + \ref cyhal_sdhc_send_cmd functions. * * @param[in] obj The SDHC object * @return The status of the operation */ cy_rslt_t cyhal_sdhc_wait_transfer_complete(cyhal_sdhc_t *obj); /** Sets the voltage level of the I/O lines. * \note This function requires io_volt_sel and (for some cases) card_pwr_en pins to be assigned. Please refer to * \ref cyhal_sdhc_init_hw for pin details. * \note Switching from 1.8V to 3.3V can be done only via power cycle sequence (power down card, wait, power up card), * which is supported by HAL driver and performed only if CYHAL_SDHC_IO_VOLT_ACTION_NEGOTIATE selected. card_pwr_en pin * has to be assigned. Please refer to \ref cyhal_sdhc_init_hw for pin details. * * @param[in] obj The SDHC object * @param[in] io_voltage I/O voltage to be set on lines * @param[in] io_switch_type Defines how I/O voltage will be switched * @return The status of the operation */ cy_rslt_t cyhal_sdhc_set_io_voltage(cyhal_sdhc_t *obj, cyhal_sdhc_io_voltage_t io_voltage, cyhal_sdhc_io_volt_action_type_t io_switch_type); /** Returns the current voltage level of the I/O lines * * @param[in] obj The SDHC object * @return Current I/O voltage setting value */ cyhal_sdhc_io_voltage_t cyhal_sdhc_get_io_voltage(cyhal_sdhc_t *obj); /** Configures data bus width on host side and (optionally) informs the card about new width configuration. * * @param[in] obj The SDHC object * @param[in] bus_width The desired bus width, 1-bit, 4-bit, 8-bit * @param[in] configure_card Whether card needs to be configured with new bus width. true is recommended. * @return The status of the operation */ cy_rslt_t cyhal_sdhc_set_bus_width(cyhal_sdhc_t *obj, uint8_t bus_width, bool configure_card); /** Returns currently configured data bus width * * @param[in] obj The SDHC object * @return Currently configured bus width, 1-bit, 4-bit, 8-bit */ uint8_t cyhal_sdhc_get_bus_width(cyhal_sdhc_t *obj); /** Returns last issued SD operation error states. This function can be used for error checking after any of cmd / data * transfer-related operations. For list of possible errors, that are being tracked, please refer to \ref cyhal_sdhc_error_type_t. * * @param[in] obj The SDHC object * @return Errors occurred during last command */ cyhal_sdhc_error_type_t cyhal_sdhc_get_last_command_errors(cyhal_sdhc_t *obj); /** Clears SDHC hardware error states. Error statuses are indicated by \ref cyhal_sdhc_get_last_command_errors function. * * @param[in] obj The SDHC object */ void cyhal_sdhc_clear_errors(cyhal_sdhc_t *obj); /** Resets CMD and Data lines and corresponding circuits of SD Host. * * @param[in] obj The SDHC object */ void cyhal_sdhc_software_reset(cyhal_sdhc_t *obj); /** Powers up / down the card based on provided parameter. This function uses card_pwr_en pin to change card power * state. Please refer to \ref cyhal_sdhc_init_hw for pin description. * * @param[in] obj The SDHC peripheral to configure * @param[in] enable Card is powered if true, not powered if false. * @return The status of the operation */ cy_rslt_t cyhal_sdhc_enable_card_power(cyhal_sdhc_t *obj, bool enable); #if defined(__cplusplus) } #endif #ifdef CYHAL_SDHC_IMPL_HEADER #include CYHAL_SDHC_IMPL_HEADER #endif /* CYHAL_SDHC_IMPL_HEADER */ /** \} group_hal_sdhc */