/* * Copyright (C) 2018 Alkgrove * * This file is subject to the terms and conditions of the GNU Lesser * General Public License v2.1. See the file LICENSE in the top level * directory for more details. */ /** * @defgroup cpu_sam0_common_sdhc sam0 SD Host Controller * @ingroup cpu_sam0_common * @brief SD card interface functions for sam0 class devices * * @{ * * @file * @brief SD card interface functions for sam0 class devices * * @author alkgrove * @author Benjamin Valentin */ #ifndef SDHC_H #define SDHC_H #ifdef __cplusplus extern "C" { #endif #include #include #include "periph/gpio.h" #include "mutex.h" /** * @brief SD Card driver context */ typedef struct { Sdhc *dev; /**< SDHC instance */ gpio_t cd; /**< Card detect pin */ gpio_t wp; /**< Write Protect pin */ mutex_t lock; /**< Ensure thread-safe access */ mutex_t sync; /**< ISR mutex */ uint32_t sectors; /**< Capacity in bytes */ uint32_t clock; /**< Accepted Cloc Rate in Hz */ uint16_t rca; /**< Relative Card Address */ uint16_t error; /**< Last error state */ uint8_t type; /**< Type of Card */ uint8_t version; /**< Version of Card */ uint8_t bus_width; /**< Acceptable Bus Width (1 or 4) */ bool high_speed; /**< Turbo mode */ bool need_init; /**< Card installed but not initialized if true */ } sdhc_state_t; /** @name Card Types */ /** @{ */ #define CARD_TYPE_UNKNOWN (0) /**< Unknown type card */ #define CARD_TYPE_SD (1 << 0) /**< SD card */ #define CARD_TYPE_MMC (1 << 1) /**< MMC card */ #define CARD_TYPE_SDIO (1 << 2) /**< SDIO card */ #define CARD_TYPE_HC (1 << 3) /**< High capacity card */ /** SD combo card (io + memory) */ #define CARD_TYPE_SD_COMBO (CARD_TYPE_SD | CARD_TYPE_SDIO) /** @} */ /** @name Card Versions */ /** @{ */ #define CARD_VER_UNKNOWN (0) /**< Unknown card version */ #define CARD_VER_SD_1_0 (0x10) /**< SD version 1.0 and 1.01 */ #define CARD_VER_SD_1_10 (0x1A) /**< SD version 1.10 */ #define CARD_VER_SD_2_0 (0X20) /**< SD version 2.00 */ #define CARD_VER_SD_3_0 (0X30) /**< SD version 3.0X */ #define CARD_VER_MMC_1_2 (0x12) /**< MMC version 1.2 */ #define CARD_VER_MMC_1_4 (0x14) /**< MMC version 1.4 */ #define CARD_VER_MMC_2_2 (0x22) /**< MMC version 2.2 */ #define CARD_VER_MMC_3 (0x30) /**< MMC version 3 */ #define CARD_VER_MMC_4 (0x40) /**< MMC version 4 */ /** @} */ /** @name Flags used to define MCI parser for SD/MMC command */ /** @{ */ #define MCI_RESP_PRESENT (1ul << 8) /**< Have response */ #define MCI_RESP_136 (1ul << 11) /**< 136 bit response */ #define MCI_RESP_CRC (1ul << 12) /**< Expect valid crc */ #define MCI_RESP_BUSY (1ul << 13) /**< Card may send busy */ #define MCI_CMD_OPENDRAIN (1ul << 14) /**< Open drain for a broadcast command */ #define MCI_CMD_WRITE (1ul << 15) /**< To signal a data write operation */ #define MCI_CMD_SDIO_BYTE (1ul << 16) /**< To signal a SDIO transfer in multi byte mode */ #define MCI_CMD_SDIO_BLOCK (1ul << 17) /**< To signal a SDIO transfer in block mode */ #define MCI_CMD_STREAM (1ul << 18) /**< To signal a data transfer in stream mode */ #define MCI_CMD_SINGLE_BLOCK (1ul << 19) /**< To signal a data transfer in single block mode */ #define MCI_CMD_MULTI_BLOCK (1ul << 20) /**< To signal a data transfer in multi block mode */ /** @} */ /** This SD stack uses the maximum block size authorized (512 bytes) */ #define SD_MMC_BLOCK_SIZE 512 /**< SD card block size */ #define SDHC_SLOW_CLOCK_HZ 400000 /**< Clock frequency on init */ /** * @brief Initialize the SD host controller * * @param[in] state driver context * * @return int 0 on success, error otherwise */ int sdhc_init(sdhc_state_t *state); /** * @brief Send a command to the SD card * * @param[in] state driver context * @param[in] cmd the command code * @param[in] arg command argument * * @return true command was successful * @return false command returned error */ bool sdhc_send_cmd(sdhc_state_t *state, uint32_t cmd, uint32_t arg); /** * @brief Read blocks from the SD card into memory * * Reads n 512 byte blocks from the SD card * * @param[in] state driver context * @param[in] block block number to read from * @param[out] dst destination address * @param[in] num number of blocks to read * * @return 0 if success, negative error if failed */ int sdhc_read_blocks(sdhc_state_t *state, uint32_t block, void *dst, uint16_t num); /** * @brief Write memory to SD card blocks * * Writes n 512 bytes blocks on the SD card * * @param[in] state driver context * @param[in] block block number to write to * @param[in] src pointer to memory to write * @param[in] num number of blocks to write * * @return 0 if success, negative error if failed */ int sdhc_write_blocks(sdhc_state_t *state, uint32_t block, const void *src, uint16_t num); /** * @brief Erase memory from SD card blocks * * Erases n 512 byte blocks on the SD card * * @param[in] state driver context * @param[in] block first block number to erase * @param[in] num number of blocks to erase * * @return 0 if success, negative error if failed */ int sdhc_erase_blocks(sdhc_state_t *state, uint32_t block, uint16_t num); #ifdef __cplusplus } #endif #endif /* SDHC_H */ /** @} */