mirror of
https://github.com/CTCaer/hekate
synced 2024-11-05 11:26:35 +00:00
c13eabcde8
The driver was working before this, but adding the changes provides a proper and better sdmmc controller inner state.
264 lines
7.9 KiB
C
264 lines
7.9 KiB
C
/*
|
|
* 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 <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#ifndef _SDMMC_DRIVER_H_
|
|
#define _SDMMC_DRIVER_H_
|
|
|
|
#include <utils/types.h>
|
|
#include <storage/sdmmc_t210.h>
|
|
|
|
/*! 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_cmd12;
|
|
} 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
|