Newer
Older
mbed-os / targets / TARGET_Cypress / TARGET_PSOC6 / mtb-hal-cat1 / include / cyhal_sdhc.h
@Dustin Crossman Dustin Crossman on 4 Jun 2021 36 KB Fix file modes.
/***************************************************************************//**
* \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 */