/* * Copyright (c) 2018 naehrwert * Copyright (c) 2018-2019 CTCaer * * This program is free software; you can redistribute it and/or modify it * under the terms and conditions of the GNU General Public License, * version 2, as published by the Free Software Foundation. * * This program is distributed in the hope it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for * more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #ifndef _SDMMC_DRIVER_H_ #define _SDMMC_DRIVER_H_ #include #include /*! SDMMC controller IDs. */ #define SDMMC_1 0 #define SDMMC_2 1 #define SDMMC_3 2 #define SDMMC_4 3 /*! SDMMC power types. */ #define SDMMC_POWER_OFF 0 #define SDMMC_POWER_1_8 1 #define SDMMC_POWER_3_3 2 /*! SDMMC bus widths. */ #define SDMMC_BUS_WIDTH_1 0 #define SDMMC_BUS_WIDTH_4 1 #define SDMMC_BUS_WIDTH_8 2 /*! SDMMC response types. */ #define SDMMC_RSP_TYPE_0 0 #define SDMMC_RSP_TYPE_1 1 #define SDMMC_RSP_TYPE_2 2 #define SDMMC_RSP_TYPE_3 3 #define SDMMC_RSP_TYPE_4 4 #define SDMMC_RSP_TYPE_5 5 /*! SDMMC mask interrupt status. */ #define SDMMC_MASKINT_MASKED 0 #define SDMMC_MASKINT_NOERROR -1 #define SDMMC_MASKINT_ERROR -2 /*! SDMMC present state. */ #define SDHCI_CMD_INHIBIT 0x1 #define SDHCI_DATA_INHIBIT 0x2 #define SDHCI_DOING_WRITE 0x100 #define SDHCI_DOING_READ 0x200 #define SDHCI_SPACE_AVAILABLE 0x400 #define SDHCI_DATA_AVAILABLE 0x800 #define SDHCI_CARD_PRESENT 0x10000 #define SDHCI_CD_STABLE 0x20000 #define SDHCI_CD_LVL 0x40000 #define SDHCI_WRITE_PROTECT 0x80000 #define SDHCI_DATA_LVL_MASK 0xF00000 #define SDHCI_DATA_0_LVL_MASK 0x100000 #define SDHCI_CMD_LVL 0x1000000 /*! SDMMC transfer mode. */ #define SDHCI_TRNS_DMA 0x01 #define SDHCI_TRNS_BLK_CNT_EN 0x02 #define SDHCI_TRNS_AUTO_CMD12 0x04 #define SDHCI_TRNS_AUTO_CMD23 0x08 #define SDHCI_TRNS_AUTO_SEL 0x0C #define SDHCI_TRNS_WRITE 0x00 #define SDHCI_TRNS_READ 0x10 #define SDHCI_TRNS_MULTI 0x20 /*! SDMMC command. */ #define SDHCI_CMD_RESP_MASK 0x3 #define SDHCI_CMD_RESP_NO_RESP 0x0 #define SDHCI_CMD_RESP_LEN136 0x1 #define SDHCI_CMD_RESP_LEN48 0x2 #define SDHCI_CMD_RESP_LEN48_BUSY 0x3 #define SDHCI_CMD_CRC 0x08 #define SDHCI_CMD_INDEX 0x10 #define SDHCI_CMD_DATA 0x20 #define SDHCI_CMD_ABORTCMD 0xC0 /*! SDMMC host control. */ #define SDHCI_CTRL_LED 0x01 #define SDHCI_CTRL_4BITBUS 0x02 #define SDHCI_CTRL_HISPD 0x04 #define SDHCI_CTRL_DMA_MASK 0x18 #define SDHCI_CTRL_SDMA 0x00 #define SDHCI_CTRL_ADMA1 0x08 #define SDHCI_CTRL_ADMA32 0x10 #define SDHCI_CTRL_ADMA64 0x18 #define SDHCI_CTRL_8BITBUS 0x20 #define SDHCI_CTRL_CDTEST_INS 0x40 #define SDHCI_CTRL_CDTEST_EN 0x80 /*! SDMMC host control 2. */ #define SDHCI_CTRL_UHS_MASK 0xFFF8 #define SDHCI_CTRL_VDD_180 8 #define SDHCI_CTRL_DRV_TYPE_B 0x00 #define SDHCI_CTRL_DRV_TYPE_A 0x10 #define SDHCI_CTRL_DRV_TYPE_C 0x20 #define SDHCI_CTRL_DRV_TYPE_D 0x30 #define SDHCI_CTRL_EXEC_TUNING 0x40 #define SDHCI_CTRL_TUNED_CLK 0x80 #define SDHCI_HOST_VERSION_4_EN 0x1000 #define SDHCI_ADDRESSING_64BIT_EN 0x2000 #define SDHCI_CTRL_PRESET_VAL_EN 0x8000 /*! SDMMC power control. */ #define SDHCI_POWER_ON 0x01 #define SDHCI_POWER_180 0x0A #define SDHCI_POWER_300 0x0C #define SDHCI_POWER_330 0x0E #define SDHCI_POWER_MASK 0xF1 // /*! SDMMC max current. */ // #define SDHCI_MAX_CURRENT_330_MASK 0xFF // #define SDHCI_MAX_CURRENT_180_MASK 0xFF0000 // #define SDHCI_MAX_CURRENT_MULTIPLIER 4 /*! SDMMC clock control. */ #define SDHCI_DIVIDER_SHIFT 8 #define SDHCI_DIVIDER_HI_SHIFT 6 #define SDHCI_DIV_MASK 0xFF00 #define SDHCI_DIV_HI_MASK 0xC0 #define SDHCI_PROG_CLOCK_MODE 0x20 #define SDHCI_CLOCK_CARD_EN 0x4 #define SDHCI_CLOCK_INT_STABLE 0x2 #define SDHCI_CLOCK_INT_EN 0x1 /*! SDMMC software reset. */ #define SDHCI_RESET_ALL 0x01 #define SDHCI_RESET_CMD 0x02 #define SDHCI_RESET_DATA 0x04 /*! SDMMC interrupt status and control. */ #define SDHCI_INT_RESPONSE 0x1 #define SDHCI_INT_DATA_END 0x2 #define SDHCI_INT_BLK_GAP 0x4 #define SDHCI_INT_DMA_END 0x8 #define SDHCI_INT_SPACE_AVAIL 0x10 #define SDHCI_INT_DATA_AVAIL 0x20 #define SDHCI_INT_CARD_INSERT 0x40 #define SDHCI_INT_CARD_REMOVE 0x80 #define SDHCI_INT_CARD_INT 0x100 #define SDHCI_INT_RETUNE 0x1000 #define SDHCI_INT_CQE 0x4000 #define SDHCI_INT_ERROR 0x8000 /*! SDMMC error interrupt status and control. */ #define SDHCI_ERR_INT_TIMEOUT 0x1 #define SDHCI_ERR_INT_CRC 0x2 #define SDHCI_ERR_INT_END_BIT 0x4 #define SDHCI_ERR_INT_INDEX 0x8 #define SDHCI_ERR_INT_DATA_TIMEOUT 0x10 #define SDHCI_ERR_INT_DATA_CRC 0x20 #define SDHCI_ERR_INT_DATA_END_BIT 0x40 #define SDHCI_ERR_INT_BUS_POWER 0x80 #define SDHCI_ERR_INT_AUTO_CMD_ERR 0x100 #define SDHCI_ERR_INT_ADMA_ERROR 0x200 #define SDHCI_ERR_INT_ALL_EXCEPT_ADMA_BUSPWR \ (SDHCI_ERR_INT_AUTO_CMD_ERR | SDHCI_ERR_INT_DATA_END_BIT | \ SDHCI_ERR_INT_DATA_CRC | SDHCI_ERR_INT_DATA_TIMEOUT | \ SDHCI_ERR_INT_INDEX | SDHCI_ERR_INT_END_BIT | \ SDHCI_ERR_INT_CRC | SDHCI_ERR_INT_TIMEOUT) /*! SD bus speeds. */ #define UHS_SDR12_BUS_SPEED 0 #define HIGH_SPEED_BUS_SPEED 1 #define UHS_SDR25_BUS_SPEED 1 #define UHS_SDR50_BUS_SPEED 2 #define UHS_SDR104_BUS_SPEED 3 #define UHS_DDR50_BUS_SPEED 4 #define HS400_BUS_SPEED 5 /*! SDMMC timmings. */ #define SDHCI_TIMING_MMC_ID 0 #define SDHCI_TIMING_MMC_LS26 1 #define SDHCI_TIMING_MMC_HS52 2 #define SDHCI_TIMING_MMC_HS200 3 #define SDHCI_TIMING_MMC_HS400 4 #define SDHCI_TIMING_SD_ID 5 #define SDHCI_TIMING_SD_DS12 6 #define SDHCI_TIMING_SD_HS25 7 #define SDHCI_TIMING_UHS_SDR12 8 #define SDHCI_TIMING_UHS_SDR25 9 #define SDHCI_TIMING_UHS_SDR50 10 #define SDHCI_TIMING_UHS_SDR104 11 #define SDHCI_TIMING_UHS_SDR82 12 // SDR104 with a 163.2MHz -> 81.6MHz clock. #define SDHCI_TIMING_UHS_DDR50 13 #define SDHCI_TIMING_MMC_HS102 14 #define SDHCI_CAN_64BIT 0x10000000 /*! SDMMC Low power features. */ #define SDMMC_POWER_SAVE_DISABLE 0 #define SDMMC_POWER_SAVE_ENABLE 1 /*! Helper for SWITCH command argument. */ #define SDMMC_SWITCH(mode, index, value) (((mode) << 24) | ((index) << 16) | ((value) << 8)) /*! SDMMC controller context. */ typedef struct _sdmmc_t { t210_sdmmc_t *regs; u32 id; u32 divisor; u32 clock_stopped; int powersave_enabled; int manual_cal; int card_clock_enabled; int venclkctl_set; u32 venclkctl_tap; u32 expected_rsp_type; u32 dma_addr_next; u32 rsp[4]; u32 rsp3; int t210b01; } sdmmc_t; /*! SDMMC command. */ typedef struct _sdmmc_cmd_t { u16 cmd; u32 arg; u32 rsp_type; u32 check_busy; } sdmmc_cmd_t; /*! SDMMC request. */ typedef struct _sdmmc_req_t { void *buf; u32 blksize; u32 num_sectors; int is_write; int is_multi_block; int is_auto_stop_trn; } sdmmc_req_t; int sdmmc_get_io_power(sdmmc_t *sdmmc); u32 sdmmc_get_bus_width(sdmmc_t *sdmmc); void sdmmc_set_bus_width(sdmmc_t *sdmmc, u32 bus_width); void sdmmc_save_tap_value(sdmmc_t *sdmmc); int sdmmc_setup_clock(sdmmc_t *sdmmc, u32 type); void sdmmc_card_clock_powersave(sdmmc_t *sdmmc, int powersave_enable); int sdmmc_get_rsp(sdmmc_t *sdmmc, u32 *rsp, u32 size, u32 type); int sdmmc_tuning_execute(sdmmc_t *sdmmc, u32 type, u32 cmd); int sdmmc_stop_transmission(sdmmc_t *sdmmc, u32 *rsp); bool sdmmc_get_sd_inserted(); int sdmmc_init(sdmmc_t *sdmmc, u32 id, u32 power, u32 bus_width, u32 type, int powersave_enable); void sdmmc_end(sdmmc_t *sdmmc); void sdmmc_init_cmd(sdmmc_cmd_t *cmdbuf, u16 cmd, u32 arg, u32 rsp_type, u32 check_busy); int sdmmc_execute_cmd(sdmmc_t *sdmmc, sdmmc_cmd_t *cmd, sdmmc_req_t *req, u32 *blkcnt_out); int sdmmc_enable_low_voltage(sdmmc_t *sdmmc); #endif