652 lines
27 KiB
C
Raw Normal View History

/*
* Copyright : (C) 2022 Phytium Information Technology, Inc.
* All Rights Reserved.
*
* This program is OPEN SOURCE software: you can redistribute it and/or modify it
* under the terms of the Phytium Public License as published by the Phytium Technology Co.,Ltd,
* either version 1.0 of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,but WITHOUT ANY WARRANTY;
* without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
* See the Phytium Public License for more details.
*
*
* FilePath: fsdio_hw.h
* Date: 2022-05-26 15:32:34
* LastEditTime: 2022-05-26 15:32:35
* Description:  This file is for SDIO register function definition
*
* Modify History:
* Ver   Who        Date         Changes
* ----- ------     --------    --------------------------------------
* 1.0 zhugengyu 2021/12/2 init
* 1.1 zhugengyu 2022/5/26 modify according to tech manual.
*/
#ifndef FSDIO_HW_H
#define FSDIO_HW_H
/***************************** Include Files *********************************/
#include "fparameters.h"
#include "fio.h"
#include "ftypes.h"
#include "fassert.h"
#include "fkernel.h"
#ifdef __aarch64__
#include "faarch64.h"
#else
#include "fcp15.h"
#endif
#ifdef __cplusplus
extern "C"
{
#endif
/************************** Constant Definitions *****************************/
/** @name Register Map
*
* Register offsets from the base address of an SD device.
* @{
*/
#define FSDIO_CNTRL_OFFSET 0x00U /* the controller config reg */
#define FSDIO_PWREN_OFFSET 0x04U /* the power enable reg */
#define FSDIO_CLKDIV_OFFSET 0x08U /* the clock divider reg */
#define FSDIO_CLKENA_OFFSET 0x10U /* the clock enable reg */
#define FSDIO_TMOUT_OFFSET 0x14U /* the timeout reg */
#define FSDIO_CTYPE_OFFSET 0x18U /* the card type reg */
#define FSDIO_BLK_SIZ_OFFSET 0x1CU /* the block size reg */
#define FSDIO_BYT_CNT_OFFSET 0x20U /* the byte count reg */
#define FSDIO_INT_MASK_OFFSET 0x24U /* the interrupt mask reg */
#define FSDIO_CMD_ARG_OFFSET 0x28U /* the command argument reg */
#define FSDIO_CMD_OFFSET 0x2CU /* the command reg */
#define FSDIO_RESP0_OFFSET 0x30U /* the response reg0 */
#define FSDIO_RESP1_OFFSET 0x34U /* the response reg1 */
#define FSDIO_RESP2_OFFSET 0x38U /* the response reg2 */
#define FSDIO_RESP3_OFFSET 0x3CU /* the response reg3 */
#define FSDIO_MASKED_INTS_OFFSET 0x40U /* the masked interrupt status reg */
#define FSDIO_RAW_INTS_OFFSET 0x44U /* the raw interrupt status reg */
#define FSDIO_STATUS_OFFSET 0x48U /* the status reg */
#define FSDIO_FIFOTH_OFFSET 0x4CU /* the FIFO threshold watermark reg */
#define FSDIO_CARD_DETECT_OFFSET 0x50U /* the card detect reg */
#define FSDIO_CARD_WRTPRT_OFFSET 0x54U /* the card write protect reg */
#define FSDIO_GPIO_OFFSET 0x58U /* the ciu ready */
#define FSDIO_TRAN_CARD_CNT_OFFSET 0x5CU /* the transferred CIU card byte count reg */
#define FSDIO_TRAN_FIFO_CNT_OFFSET 0x60U /* the transferred host to FIFO byte count reg */
#define FSDIO_DEBNCE_OFFSET 0x64U /* the debounce count reg */
#define FSDIO_UID_OFFSET 0x68U /* the user ID reg */
#define FSDIO_VID_OFFSET 0x6CU /* the controller version ID reg */
#define FSDIO_HWCONF_OFFSET 0x70U /* the hardware configuration reg */
#define FSDIO_UHS_REG_OFFSET 0x74U /* the UHS-I reg */
#define FSDIO_CARD_RESET_OFFSET 0x78U /* the card reset reg */
#define FSDIO_BUS_MODE_OFFSET 0x80U /* the bus mode reg */
#define FSDIO_DESC_LIST_ADDRL_OFFSET 0x88U /* the descriptor list low base address reg */
#define FSDIO_DESC_LIST_ADDRH_OFFSET 0x8CU /* the descriptor list high base address reg */
#define FSDIO_DMAC_STATUS_OFFSET 0x90U /* the internal DMAC status reg */
#define FSDIO_DMAC_INT_EN_OFFSET 0x94U /* the internal DMAC interrupt enable reg */
#define FSDIO_CUR_DESC_ADDRL_OFFSET 0x98U /* the current host descriptor low address reg */
#define FSDIO_CUR_DESC_ADDRH_OFFSET 0x9CU /* the current host descriptor high address reg */
#define FSDIO_CUR_BUF_ADDRL_OFFSET 0xA0U /* the current buffer low address reg */
#define FSDIO_CUR_BUF_ADDRH_OFFSET 0xA4U /* the current buffer high address reg */
#define FSDIO_CARD_THRCTL_OFFSET 0x100U /* the card threshold control reg */
#define FSDIO_UHS_REG_EXT_OFFSET 0x108U /* the UHS register extension */
#define FSDIO_EMMC_DDR_REG_OFFSET 0x10CU /* the EMMC DDR reg */
#define FSDIO_ENABLE_SHIFT_OFFSET 0x110U /* the enable phase shift reg */
#define FSDIO_DATA_OFFSET 0x200U /* the data FIFO access */
/** @name FSDIO_CNTRL_OFFSET x0 Register
*/
#define FSDIO_CNTRL_CONTROLLER_RESET BIT(0) /* RW 复位控制器,除 DMAFIFO */
#define FSDIO_CNTRL_FIFO_RESET BIT(1) /* RW 复位 FIFO, 1 有效 */
#define FSDIO_CNTRL_DMA_RESET BIT(2) /* RW 复位内部 DMA, 1 有效 */
#define FSDIO_CNTRL_INT_ENABLE BIT(4) /* RW 全局中断使能配置, 1 使能 */
#define FSDIO_CNTRL_DMA_ENABLE BIT(5) /* RW 外部 DMA 模式使能 */
#define FSDIO_CNTRL_READ_WAIT BIT(6) /* RW SDIO 读等待 1 有效 */
#define FSDIO_CNTRL_SEND_IRQ_RESPONSE BIT(7) /* RW MMC 中断自动响应配置 1 有效 */
#define FSDIO_CNTRL_ABORT_READ_DATA BIT(8) /* RW 读暂停异常清除 */
#define FSDIO_CNTRL_SEND_CCSD BIT(9) /* RW 发送CCD (NOT USED) */
#define FSDIO_CNTRL_SEND_AUTO_STOP_CCSD BIT(10) /* RW 发送CCD自动STOP (NOT USED) */
#define FSDIO_CNTRL_ENDIAN BIT(11) /* RW 0小端1大端 */
#define FSDIO_CNTRL_CARD_VOLTAGE_A_MASK GENMASK(19, 16) /* RW A电压选择 */
#define FSDIO_CNTRL_CARD_VOLTAGE_B_MASK GENMASK(23, 20) /* RW B电压选择 */
#define FSDIO_CNTRL_ENABLE_OD_PULLUP BIT(24) /* RW 外部开漏输出 */
#define FSDIO_CNTRL_USE_INTERNAL_DMAC BIT(25) /* RW 使用内部DMA */
/** @name FSDIO_PWREN_OFFSET 0x4 Register
*/
#define FSDIO_PWREN_ENABLE BIT(0) /* RW 卡供电开关, 01开*/
/** @name FSDIO_CLKDIV_OFFSET 0x8 Register
*/
/* CLK_SAMPLE 和 CLK_SAMPLE 必须小于 CLK_DIVIDER */
#define FSDIO_CLK_SAMPLE_SET(x) SET_REG32_BITS((x), 23, 16)
#define FSDIO_CLK_DRV_SET(x) SET_REG32_BITS((x), 15, 8)
#define FSDIO_CLK_DIVIDER_SET(x) SET_REG32_BITS((x), 7, 0) /* 分频系数 = 2 * bit[7:0] */
#define FSDIO_CLK_DIV(samp, drv, div) FSDIO_CLK_SAMPLE_SET(samp) | \
FSDIO_CLK_DRV_SET(drv) | \
FSDIO_CLK_DIVIDER_SET(div)
#define FSDIO_CLK_DIVDER_GET(div_reg) GET_REG32_BITS((div_reg), 7, 0)
/** @name FSDIO_CLKENA_OFFSET Register
*/
#define FSDIO_CLKENA_CCLK_ENABLE BIT(0) /* RW 0Clock disabled1Clock enabled */
#define FSDIO_CLKENA_CCLK_LOW_POWER BIT(16) /* RW 0x0非低功耗0x1低功耗 */
/** @name FSDIO_TMOUT_OFFSET Register
*/
#define FSDIO_MAX_DATA_TIMEOUT 0xffffff /* RW 读卡超时(以卡时钟为单位) */
#define FSDIO_MAX_RESP_TIMEOUT 0xff /* RW 响应超时(以卡时钟为单位) */
#define FSDIO_TIMEOUT_DATA(data_timeout, resp_timeout) \
((GENMASK(31, 8) & ((data_timeout) << 8)) | \
(GENMASK(7, 0) & ((resp_timeout))))
/** @name FSDIO_CTYPE_OFFSET Register
*/
#define FSDIO_CARD0_WIDTH1_8BIT BIT(16) /* 1: 8-bit mode */
#define FSDIO_CARD0_WIDTH2_4BIT BIT(0) /* 1: 4-bit mode */
#define FSDIO_CARD0_WIDTH2_1BIT 0x0U /* 0: 1-bit mode */
/** @name FSDIO_INT_MASK_OFFSET Register
* @name FSDIO_MASKED_INTS_OFFSET Register
* @name FSDIO_RAW_INTS_OFFSET Register
*/
#define FSDIO_INT_CD_BIT BIT(0) /* RW Card detect (CD) */
#define FSDIO_INT_RE_BIT BIT(1) /* RW Response error (RE) */
#define FSDIO_INT_CMD_BIT BIT(2) /* RW Command done (CD) */
#define FSDIO_INT_DTO_BIT BIT(3) /* RW Data transfer over (DTO) */
#define FSDIO_INT_TXDR_BIT BIT(4) /* RW Transmit FIFO data request (TXDR) */
#define FSDIO_INT_RXDR_BIT BIT(5) /* RW Receive FIFO data request (RXDR) */
#define FSDIO_INT_RCRC_BIT BIT(6) /* RW Response CRC error (RCRC) */
#define FSDIO_INT_DCRC_BIT BIT(7) /* RW Data CRC error (DCRC) */
#define FSDIO_INT_RTO_BIT BIT(8) /* RW Response timeout (RTO) */
#define FSDIO_INT_DRTO_BIT BIT(9) /* RW Data read timeout (DRTO) */
#define FSDIO_INT_HTO_BIT BIT(10) /* RW Data starvation-by-host timeout (HTO) */
#define FSDIO_INT_FRUN_BIT BIT(11) /* RW FIFO underrun/overrun error (FRUN) */
#define FSDIO_INT_HLE_BIT BIT(12) /* RW Hardware locked write error (HLE) */
#define FSDIO_INT_SBE_BCI_BIT BIT(13) /* RW Start-bit error (SBE) */
#define FSDIO_INT_ACD_BIT BIT(14) /* RW Auto command done (ACD) */
#define FSDIO_INT_EBE_BIT BIT(15) /* RW End-bit error (read)/Write no CRC (EBE) */
#define FSDIO_INT_SDIO_BIT BIT(16) /* RW SDIO interrupt for card */
#define FSDIO_INT_ALL_BITS GENMASK(16, 0)
#define FSDIO_INTS_CMD_MASK (FSDIO_INT_RE_BIT | FSDIO_INT_CMD_BIT | FSDIO_INT_RCRC_BIT | \
FSDIO_INT_RTO_BIT | FSDIO_INT_HTO_BIT | FSDIO_INT_HLE_BIT)
#define FSDIO_INTS_DATA_MASK (FSDIO_INT_DTO_BIT | FSDIO_INT_DCRC_BIT | FSDIO_INT_DRTO_BIT | \
FSDIO_INT_SBE_BCI_BIT)
/** @name FSDIO_CMD_OFFSET Register
*/
#define FSDIO_CMD_START BIT(31) /* 启动命令 */
#define FSDIO_CMD_USE_HOLD_REG BIT(29) /* 0: 旁路HOLD寄存器1: 使能HOLD寄存器 */
#define FSDIO_CMD_VOLT_SWITCH BIT(28) /* 0: 无电压转换1: 有电压转换 */
#define FSDIO_CMD_BOOT_MODE BIT(27) /* 0: Mandatory boot, 1: Alternate boot */
#define FSDIO_CMD_DISABLE_BOOT BIT(26) /* 中止boot进程 */
#define FSDIO_CMD_EXPECT_BOOT_ACK BIT(25) /* 1: Expect book ack */
#define FSDIO_CMD_ENABLE_BOOT BIT(24) /* 1: 使能 boot for mandatory */
#define FSDIO_CMD_UPD_CLK BIT(21) /* 1不发送指令仅更新时钟寄存器的值到卡时钟域内 */
#define FSDIO_CMD_CARD_NUM_SET(num) SET_REG32_BITS((num), 20, 16)
#define FSDIO_CMD_INIT BIT(15) /* 0在发送指令前不发送初始化序列80 个周期) 1: 发送 */
#define FSDIO_CMD_STOP_ABORT BIT(14) /* 1停止或中止命令用于停止当前的数据传输 */
#define FSDIO_CMD_WAIT_PRVDATA_COMPLETE BIT(13) /* 1等待前面的数据传输完成后再发送指令 0: 立即发送命令 */
#define FSDIO_CMD_SEND_AUTO_STOP BIT(12) /* 1在数据传送结束时发送停止命令 */
#define FSDIO_CMD_TRANSF_MODE_SET(mode) SET_REG32_BITS((mode), 12, 11) /* 1: 流数据传输指令 */
#define FSDIO_CMD_DAT_WRITE BIT(10) /* 0读卡 1写卡 */
#define FSDIO_CMD_DAT_EXP BIT(9) /* 0不等待数据传输, 1等待数据传输 */
#define FSDIO_CMD_RESP_CRC BIT(8) /* 1检查响应 CRC */
#define FSDIO_CMD_RESP_LONG BIT(7) /* 0等待卡的短响应 1等待卡的长响应 */
#define FSDIO_CMD_RESP_EXP BIT(6) /* 1等待卡的响应0命令不需要卡响应 */
#define FSDIO_CMD_INDX_SET(ind) SET_REG32_BITS((ind), 5, 0) /* 命令索引号 */
/** @name FSDIO_STATUS_OFFSET Register
*/
#define FSDIO_STATUS_FIFO_RX BIT(0) /* RO, 达到 FIFO_RX 标记 */
#define FSDIO_STATUS_FIFO_TX BIT(1) /* RO, 达到 FIFO_TX 标记 */
#define FSDIO_STATUS_FIFO_EMPTY BIT(2) /* RO, FIFO empty */
#define FSDIO_STATUS_FIFO_FULL BIT(3) /* RO, FIFO full */
#define FSDIO_STATUS_CMD_FSM_GET(reg_val) GET_REG32_BITS((reg_val), 7, 4)
#define FSDIO_STATUS_DATA3_STATUS BIT(8) /* RO DATA[3] 卡在位检测1在位 */
#define FSDIO_STATUS_DATA_BUSY BIT(9) /* RO 1: 卡 busy */
#define FSDIO_STATUS_DATA_STATE_MC_BUSY BIT(10) /* RO DATA TX|RX FSM busy */
#define FSDIO_STATUS_RESP_INDEX_GET(reg_val) GET_REG32_BITS((reg_val), 16, 11)
#define FSDIO_STATUS_FIFO_CNT_GET(reg_val) GET_REG32_BITS((reg_val), 29, 17) /* RO: FIFO 填充计数器 */
#define FSDIO_STATUS_DMA_ACK BIT(30) /* RO DMA 确认 */
#define FSDIO_STATUS_DMA_REQ BIT(31) /* RO DMA 请求 */
/** @name FSDIO_FIFOTH_OFFSET Register
*/
enum
{
FSDIO_FIFOTH_DMA_TRANS_1 = 0b000,
FSDIO_FIFOTH_DMA_TRANS_4 = 0b001,
FSDIO_FIFOTH_DMA_TRANS_8 = 0b010,
FSDIO_FIFOTH_DMA_TRANS_16 = 0b011,
FSDIO_FIFOTH_DMA_TRANS_32 = 0b100,
FSDIO_FIFOTH_DMA_TRANS_64 = 0b101,
FSDIO_FIFOTH_DMA_TRANS_128 = 0b110,
FSDIO_FIFOTH_DMA_TRANS_256 = 0b111
};
#define FSDIO_FIFOTH_DMA_TRANS_MASK GENMASK(30, 28) /* 多次传输的突发大小 */
#define FSDIO_FIFOTH_RX_WMARK_MASK GENMASK(27, 16) /* 当接收数据给卡时FIFO的阈值 */
#define FSDIO_FIFOTH_TX_WMARK_MASK GENMASK(11, 0) /* 当发送数据给卡时FIFO的阈值 */
#define FSDIO_RX_WMARK 0x7U
#define FSDIO_TX_WMARK 0x100U
/*
trans_size: Burst size of multiple transaction;
rx_wmark: FIFO threshold watermark level when receiving data to card.
tx_wmark: FIFO threshold watermark level when transmitting data to card
*/
#define FSDIO_FIFOTH(trans_size, rx_wmark, tx_wmark) \
(((FSDIO_FIFOTH_DMA_TRANS_MASK) & ((trans_size) << 28)) | \
((FSDIO_FIFOTH_RX_WMARK_MASK) & ((rx_wmark) << 16)) | \
((FSDIO_FIFOTH_TX_WMARK_MASK) & (tx_wmark)))
#define FSDIO_DMA_TRANS_SIZE_SET(x) SET_REG32_BITS((x), 30, 28)
#define FSDIO_RX_MARK_SIZE_SET(x) SET_REG32_BITS((x), 27, 16)
#define FSDIO_TX_MARK_SIZE_SET(x) SET_REG32_BITS((x), 11, 0)
/** @name FSDIO_CARD_DETECT_OFFSET Register
*/
#define FSDIO_CARD_DETECTED BIT(0) /* 1卡不在位0卡在位 */
/** @name FSDIO_CARD_WRTPRT_OFFSET Register
*/
#define FSDIO_CARD_WRITE_PROTECTED BIT(0) /* 1写保护0无写保护 */
/** @name FSDIO_GPIO_OFFSET Register
*/
#define FSDIO_CLK_READY BIT(0) /* CIU 时钟 ready */
/** @name FSDIO_UHS_REG_OFFSET Register
*/
#define FSDIO_UHS_REG_VOLT_180 BIT(0) /* RW 外部调压器接口电压 0: 3.3v, 1: 1.8v */
#define FSDIO_UHS_REG_VOLT_330 0U
#define FSDIO_UHS_REG_DDR BIT(16) /* RW DDR 模式 */
/** @name FSDIO_CARD_RESET_OFFSET Register
*/
#define FSDIO_CARD_RESET_ENABLE BIT(0) /* RW 1运行0复位 */
/** @name FSDIO_BUS_MODE_OFFSET Register
*/
#define FSDIO_BUS_MODE_SWR BIT(0) /* RW 软复位复位idma内部寄存器 */
#define FSDIO_BUS_MODE_FB BIT(1) /* RW 固定burst */
#define FSDIO_BUS_MODE_DE BIT(7) /* RW idma使能 */
#define FSDIO_BUS_MODE_PBL_GET(reg_val) GET_REG32_BITS((reg_val), 10, 8) /* burst LEN */
/** @name FSDIO_DMAC_STATUS_OFFSET Register
*/
#define FSDIO_DMAC_STATUS_TI BIT(0) /* RW 发送中断。表示链表的数据发送完成 */
#define FSDIO_DMAC_STATUS_RI BIT(1) /* RW 接收中断。表示链表的数据接收完成 */
#define FSDIO_DMAC_STATUS_FBE BIT(2) /* RW 致命总线错误中断 */
#define FSDIO_DMAC_STATUS_DU BIT(4) /* RW 链表不可用中断 */
#define FSDIO_DMAC_STATUS_CES BIT(5) /* RW 卡错误汇总 */
#define FSDIO_DMAC_STATUS_NIS BIT(8) /* RW 正常中断汇总 */
#define FSDIO_DMAC_STATUS_AIS BIT(9) /* RW 异常中断汇总 */
#define FSDIO_DMAC_STATUS_EB_GET(reg_val) GET_REG32_BITS((reg_val), 12, 10)
#define FSDIO_DMAC_STATUS_ALL_BITS GENMASK(9, 0)
#define FSDIO_DMAC_STATUS_EB_TX 0b001
#define FSDIO_DMAC_STATUS_EB_RX 0b010
/** @name FSDIO_DMAC_INT_EN_OFFSET Register
*/
#define FSDIO_DMAC_INT_ENA_TI BIT(0) /* RW 发送完成中断使能 */
#define FSDIO_DMAC_INT_ENA_RI BIT(1) /* RW 接收完成中断使能 */
#define FSDIO_DMAC_INT_ENA_FBE BIT(2) /* RW 总线错误中断使能 */
#define FSDIO_DMAC_INT_ENA_DU BIT(4) /* RW 描述符不可读中断使能 */
#define FSDIO_DMAC_INT_ENA_CES BIT(5) /* RW 卡错误中断使能 */
#define FSDIO_DMAC_INT_ENA_NIS BIT(8) /* RW 正常中断汇总使能 */
#define FSDIO_DMAC_INT_ENA_AIS BIT(9) /* RW 异常中断汇总使能 */
#define FSDIO_DMAC_INT_ENA_ALL GENMASK(9, 0)
#define FSDIO_DMAC_INTS_MASK (FSDIO_DMAC_INT_ENA_FBE | FSDIO_DMAC_INT_ENA_DU | \
FSDIO_DMAC_INT_ENA_NIS | FSDIO_DMAC_INT_ENA_AIS)
/** @name FSDIO_CARD_THRCTL_OFFSET Register
*/
#define FSDIO_CARD_THRCTL_CARDRD BIT(0) /* RW 读卡threshold使能 */
#define FSDIO_CARD_THRCTL_BUSY_CLR BIT(1) /* RW busy清中断 */
#define FSDIO_CARD_THRCTL_CARDWR BIT(2) /* RO 写卡threshold使能 */
enum
{
FSDIO_FIFO_DEPTH_8 = 23,
FSDIO_FIFO_DEPTH_16 = 24,
FSDIO_FIFO_DEPTH_32 = 25,
FSDIO_FIFO_DEPTH_64 = 26,
FSDIO_FIFO_DEPTH_128 = 27
};
#define FSDIO_CARD_THRCTL_THRESHOLD(n) BIT(n) /* 读卡 Threshold */
/** @name FSDIO_UHS_REG_EXT_OFFSET Register
*/
#define FSDIO_UHS_EXT_MMC_VOLT BIT(0) /* RW 1.2V供电选择 */
#define FSDIO_UHS_EXT_CLK_ENA BIT(1) /* RW 外部时钟CIU时钟使能 */
#define FSDIO_UHS_CLK_DIV_MASK GENMASK(14, 8) /* RW 分频系数ciu_f = clk_div_ctrl + 1, min=1*/
#define FSDIO_UHS_CLK_DIV(x) (FSDIO_UHS_CLK_DIV_MASK & ((x) << 8))
#define FSDIO_UHS_CLK_SAMP_MASK GENMASK(22, 16) /* RW 采样相位参数相对于ciu时钟相位点 */
#define FSDIO_UHS_CLK_SAMP(x) (FSDIO_UHS_CLK_SAMP_MASK & ((x) << 16))
#define FSDIO_UHS_CLK_DRV_MASK GENMASK(30, 24) /* RW 输出相位参数相对于ciu时钟相位点 */
#define FSDIO_UHS_CLK_DRV(x) (FSDIO_UHS_CLK_DRV_MASK & ((x) << 24))
#define FSDIO_UHS_EXT_CLK_MUX BIT(31)
/* FSDIO_UHS_REG_EXT_OFFSET 和 FSDIO_CLKDIV_OFFSET 两个寄存器配合完成卡时钟和驱动采样相位调整
UHS_REG_EXT CLK_DIV CARD工作时钟频率, DRV SAMP
= bit [14 : 8] + 1
CLKDIV , DIVIDER , DRV SAMP
= bit [7: 0] * 2
*/
#define FSDIO_UHS_REG(drv_phase, samp_phase, clk_div) \
(FSDIO_UHS_CLK_DRV(drv_phase) | \
FSDIO_UHS_CLK_SAMP(samp_phase) | \
FSDIO_UHS_CLK_DIV(clk_div))
#define FSDIO_UHS_CLK_DIV_SET(x) SET_REG32_BITS((x), 14, 8)
#define FSDIO_UHS_CLK_DIV_GET(reg_val) GET_REG32_BITS((reg_val), 14, 8)
#define FSDIO_UHS_CLK_SAMP_SET(x) SET_REG32_BITS((x), 22, 16)
#define FSDIO_UHS_CLK_DRV_SET(x) SET_REG32_BITS((x), 30, 24)
/** @name FSDIO_REG_EMMC_DDR_REG_OFFSET Register
*/
#define FSDIO_EMMC_DDR_CYCLE BIT(0) /* RW 1: start bit小于一个周期0start bit 为一个周期 */
#define FSDIO_TIMEOUT (50000) /* timeout for retries */
#define FSDIO_DELAY_US (5)
#define FSDIO_400_KHZ (400000UL)
#define FSDIO_25_MHZ (25000000UL)
#define FSDIO_MAX_FIFO_CNT (0x800U)
/************************** Variable Definitions *****************************/
/***************** Macros (Inline Functions) Definitions *********************/
#define FSDIO_READ_REG(addr, reg_off) FtIn32((addr) + (u32)(reg_off))
#define FSDIO_WRITE_REG(addr, reg_off, reg_val) FtOut32((addr) + (u32)(reg_off), (u32)(reg_val))
#define FSDIO_CLR_BIT(addr, reg_off, bits) FtClearBit32((addr) + (u32)(reg_off), bits)
#define FSDIO_SET_BIT(addr, reg_off, bits) FtSetBit32((addr) + (u32)(reg_off), bits)
#define FSDIO_DATA_BARRIER() WMB()
/************************** Function Prototypes ******************************/
FError FSdioSendPrivateCmd(uintptr base_addr, u32 cmd, u32 arg);
FError FSdioResetCtrl(uintptr base_addr, u32 reset_bits);
/*****************************************************************************/
/**
* @name: FSdioSetClock
* @msg: Enable/Disable controller clock
* @return {NONE}
* @param {uintptr} base_addr, base address of SDIO controller
* @param {boolean} enable, TRUE: enable clock
*/
static inline void FSdioSetClock(uintptr base_addr, boolean enable)
{
u32 reg_val = FSDIO_READ_REG(base_addr, FSDIO_CLKENA_OFFSET);
if (enable)
{
reg_val |= FSDIO_CLKENA_CCLK_ENABLE;
}
else
{
reg_val &= ~FSDIO_CLKENA_CCLK_ENABLE;
}
FSDIO_WRITE_REG(base_addr, FSDIO_CLKENA_OFFSET, reg_val);
}
static inline void FSdioSetPower(uintptr base_addr, boolean enable)
{
u32 reg_val = FSDIO_READ_REG(base_addr, FSDIO_PWREN_OFFSET);
if (enable)
{
reg_val |= FSDIO_PWREN_ENABLE;
}
else
{
reg_val &= ~FSDIO_PWREN_ENABLE;
}
FSDIO_WRITE_REG(base_addr, FSDIO_PWREN_OFFSET, reg_val);
}
static inline void FSdioSetExtClock(uintptr base_addr, boolean enable)
{
u32 reg_val = FSDIO_READ_REG(base_addr, FSDIO_UHS_REG_EXT_OFFSET);
if (enable)
{
reg_val |= FSDIO_UHS_EXT_CLK_ENA;
}
else
{
reg_val &= ~FSDIO_UHS_EXT_CLK_ENA;
}
FSDIO_WRITE_REG(base_addr, FSDIO_UHS_REG_EXT_OFFSET, reg_val);
}
static inline void FSdioSetVoltage1_8V(uintptr base_addr, boolean v1_8)
{
u32 reg_val = FSDIO_READ_REG(base_addr, FSDIO_UHS_REG_OFFSET);
if (v1_8)
{
reg_val |= FSDIO_UHS_REG_VOLT_180;
}
else
{
reg_val &= ~FSDIO_UHS_REG_VOLT_180; /* 3.3v */
}
FSDIO_WRITE_REG(base_addr, FSDIO_UHS_REG_OFFSET, reg_val);
}
/**
* @name: FSdioGetRawStatus
* @msg: Get raw interrupt status of controller
* @return {u32} raw interrupt status
* @param {uintptr} base_addr, base address of SDIO controller
*/
static inline u32 FSdioGetRawStatus(uintptr base_addr)
{
return FSDIO_READ_REG(base_addr, FSDIO_RAW_INTS_OFFSET);
}
/**
* @name: FSdioClearRawStatus
* @msg: Clear raw interrupt status of controller
* @return {NONE}
* @param {uintptr} base_addr, base address of SDIO controller
*/
static inline void FSdioClearRawStatus(uintptr base_addr)
{
u32 reg_val = FSdioGetRawStatus(base_addr);
FSDIO_WRITE_REG(base_addr, FSDIO_RAW_INTS_OFFSET, reg_val);
}
/**
* @name: FSdioGetStatus
* @msg: Get status of controller
* @return {u32} controller status
* @param {uintptr} base_addr, base address of SDIO controller
*/
static inline u32 FSdioGetStatus(uintptr base_addr)
{
return FSDIO_READ_REG(base_addr, FSDIO_STATUS_OFFSET);
}
/**
* @name: FSdioGetDMAStatus
* @msg: Get interrupt status of DMA
* @return {u32} DMA interrupt status
* @param {uintptr} base_addr, base address of SDIO controller
*/
static inline u32 FSdioGetDMAStatus(uintptr base_addr)
{
return FSDIO_READ_REG(base_addr, FSDIO_DMAC_STATUS_OFFSET);
}
/**
* @name: FSdioClearDMAStatus
* @msg: Clear interrupt status of DMA
* @return {NONE}
* @param {uintptr} base_addr, base address of SDIO controller
*/
static inline void FSdioClearDMAStatus(uintptr base_addr)
{
u32 reg_val = FSdioGetDMAStatus(base_addr);
FSDIO_WRITE_REG(base_addr, FSDIO_DMAC_STATUS_OFFSET, reg_val);
}
/**
* @name: FSdioSetDescriptor
* @msg: Set base address of DMA descriptors
* @return {NONE}
* @param {uintptr} base_addr, base address of SDIO controller
* @param {uintptr} descriptor, base address of DMA descriptors
*/
static inline void FSdioSetDescriptor(uintptr base_addr, uintptr descriptor)
{
#ifdef __aarch64___
FSDIO_WRITE_REG(base_addr, FSDIO_DESC_LIST_ADDRH_OFFSET, UPPER_32_BITS(descriptor));
FSDIO_WRITE_REG(base_addr, FSDIO_DESC_LIST_ADDRL_OFFSET, LOWER_32_BITS(descriptor));
#else
FSDIO_WRITE_REG(base_addr, FSDIO_DESC_LIST_ADDRH_OFFSET, 0x0U);
FSDIO_WRITE_REG(base_addr, FSDIO_DESC_LIST_ADDRL_OFFSET, (u32)(descriptor));
#endif
}
/**
* @name: FSdioSetTransBytes
* @msg: Set number of bytes to transfer
* @return {NONE}
* @param {uintptr} base_addr, base address of SDIO controller
* @param {u32} bytes, number of bytes to transfer
*/
static inline void FSdioSetTransBytes(uintptr base_addr, u32 bytes)
{
FSDIO_WRITE_REG(base_addr, FSDIO_BYT_CNT_OFFSET, bytes);
}
/**
* @name: FSdioSetBlockSize
* @msg: Set size of blocks in card
* @return {NONE}
* @param {uintptr} base_addr, base address of SDIO controller
* @param {u32} block_size, size of blocks in card
*/
static inline void FSdioSetBlockSize(uintptr base_addr, u32 block_size)
{
FSDIO_WRITE_REG(base_addr, FSDIO_BLK_SIZ_OFFSET, block_size);
}
/**
* @name: FSdioSetBusWidth
* @msg: Set bus width of card
* @return {NONE}
* @param {uintptr} base_addr, base address of SDIO controller
* @param {u32} width, bus width of card 1-bit, 4-bit, 8-bit
*/
static inline void FSdioSetBusWidth(uintptr base_addr, u32 width)
{
u32 reg_val;
switch (width)
{
case 1:
reg_val = FSDIO_CARD0_WIDTH2_1BIT;
break;
case 4:
reg_val = FSDIO_CARD0_WIDTH2_4BIT;
break;
case 8:
reg_val = FSDIO_CARD0_WIDTH1_8BIT;
break;
default:
FASSERT_MSG(0, "invalid bus width %d", width);
break;
}
FSDIO_WRITE_REG(base_addr, FSDIO_CTYPE_OFFSET, reg_val);
}
/**
* @name: FSdioGetBusWidth
* @msg: Get bus width for card
* @return {u32} current bus width setting
* @param {uintptr} base_addr, base address of SDIO controller
*/
static inline u32 FSdioGetBusWidth(uintptr base_addr)
{
u32 bus_width = 1;
u32 reg_val = FSDIO_READ_REG(base_addr, FSDIO_CTYPE_OFFSET);
if (FSDIO_CARD0_WIDTH2_4BIT & reg_val)
{
bus_width = 4;
}
else if (FSDIO_CARD0_WIDTH1_8BIT & reg_val)
{
bus_width = 8;
}
return bus_width;
}
/**
* @name: FSdioResetIDMA
* @msg: Reset for internal DMA
* @return {NONE}
* @param {uintptr} base_addr, base address of SDIO controller
*/
static inline void FSdioResetIDMA(uintptr base_addr)
{
u32 reg_val = FSDIO_READ_REG(base_addr, FSDIO_BUS_MODE_OFFSET);
reg_val |= FSDIO_BUS_MODE_SWR; /* 写1软复位idma复位完成后硬件自动清0 */
FSDIO_WRITE_REG(base_addr, FSDIO_BUS_MODE_OFFSET, reg_val);
}
/**
* @name: FSdioSetDDRMode
* @msg: Enable/Disable DDR mode
* @return {*}
* @param {uintptr} base_addr, base address of SDIO controller
* @param {boolean} enable, TRUE: enable DDR mode
*/
static inline void FSdioSetDDRMode(uintptr base_addr, boolean enable)
{
u32 uhs_val = FSDIO_READ_REG(base_addr, FSDIO_UHS_REG_OFFSET);
u32 emmc_val = FSDIO_READ_REG(base_addr, FSDIO_EMMC_DDR_REG_OFFSET);
if (enable)
{
uhs_val |= FSDIO_UHS_REG_DDR;
emmc_val |= FSDIO_EMMC_DDR_CYCLE;
}
else
{
uhs_val &= ~FSDIO_UHS_REG_DDR;
emmc_val &= ~FSDIO_EMMC_DDR_CYCLE;
}
FSDIO_WRITE_REG(base_addr, FSDIO_UHS_REG_OFFSET, uhs_val);
FSDIO_WRITE_REG(base_addr, FSDIO_EMMC_DDR_REG_OFFSET, emmc_val);
return;
}
/**
* @name: FSdioCheckIfCardExists
* @msg: Check if card inserted
* @return {boolean} TRUE: inserted, FALSE: not-found
* @param {uintptr} base_addr, base address of SDIO controller
*/
static inline boolean FSdioCheckIfCardExists(uintptr base_addr)
{
return (FSDIO_READ_REG(base_addr, FSDIO_CARD_DETECT_OFFSET) & FSDIO_CARD_DETECTED) ?
FALSE : TRUE;
}
#ifdef __cplusplus
}
#endif
#endif