mirror of
https://github.com/CTCaer/switch-l4t-atf.git
synced 2025-01-21 01:34:27 +00:00
drivers: add designware ufs driver
Initialized the designware UFS PHY. Signed-off-by: Haojian Zhuang <haojian.zhuang@linaro.org>
This commit is contained in:
parent
eb5073f49e
commit
7e08084213
190
drivers/synopsys/ufs/dw_ufs.c
Normal file
190
drivers/synopsys/ufs/dw_ufs.c
Normal file
@ -0,0 +1,190 @@
|
||||
/*
|
||||
* Copyright (c) 2017, ARM Limited and Contributors. All rights reserved.
|
||||
*
|
||||
* SPDX-License-Identifier: BSD-3-Clause
|
||||
*/
|
||||
|
||||
#include <assert.h>
|
||||
#include <debug.h>
|
||||
#include <dw_ufs.h>
|
||||
#include <mmio.h>
|
||||
#include <stdint.h>
|
||||
#include <string.h>
|
||||
#include <ufs.h>
|
||||
|
||||
static int dwufs_phy_init(ufs_params_t *params)
|
||||
{
|
||||
uintptr_t base;
|
||||
unsigned int fsm0, fsm1;
|
||||
unsigned int data;
|
||||
int result;
|
||||
|
||||
assert((params != NULL) && (params->reg_base != 0));
|
||||
|
||||
base = params->reg_base;
|
||||
|
||||
/* Unipro VS_MPHY disable */
|
||||
ufshc_dme_set(VS_MPHY_DISABLE_OFFSET, 0, VS_MPHY_DISABLE_MPHYDIS);
|
||||
ufshc_dme_set(PA_HS_SERIES_OFFSET, 0, 2);
|
||||
/* MPHY CBRATESEL */
|
||||
ufshc_dme_set(0x8114, 0, 1);
|
||||
/* MPHY CBOVRCTRL2 */
|
||||
ufshc_dme_set(0x8121, 0, 0x2d);
|
||||
/* MPHY CBOVRCTRL3 */
|
||||
ufshc_dme_set(0x8122, 0, 0x1);
|
||||
ufshc_dme_set(VS_MPHY_CFG_UPDT_OFFSET, 0, 1);
|
||||
|
||||
/* MPHY RXOVRCTRL4 rx0 */
|
||||
ufshc_dme_set(0x800d, 4, 0x58);
|
||||
/* MPHY RXOVRCTRL4 rx1 */
|
||||
ufshc_dme_set(0x800d, 5, 0x58);
|
||||
/* MPHY RXOVRCTRL5 rx0 */
|
||||
ufshc_dme_set(0x800e, 4, 0xb);
|
||||
/* MPHY RXOVRCTRL5 rx1 */
|
||||
ufshc_dme_set(0x800e, 5, 0xb);
|
||||
/* MPHY RXSQCONTROL rx0 */
|
||||
ufshc_dme_set(0x8009, 4, 0x1);
|
||||
/* MPHY RXSQCONTROL rx1 */
|
||||
ufshc_dme_set(0x8009, 5, 0x1);
|
||||
ufshc_dme_set(VS_MPHY_CFG_UPDT_OFFSET, 0, 1);
|
||||
|
||||
ufshc_dme_set(0x8113, 0, 0x1);
|
||||
ufshc_dme_set(VS_MPHY_CFG_UPDT_OFFSET, 0, 1);
|
||||
|
||||
ufshc_dme_set(RX_HS_G3_SYNC_LENGTH_CAP_OFFSET, 4, 0x4a);
|
||||
ufshc_dme_set(RX_HS_G3_SYNC_LENGTH_CAP_OFFSET, 5, 0x4a);
|
||||
ufshc_dme_set(RX_HS_G2_SYNC_LENGTH_CAP_OFFSET, 4, 0x4a);
|
||||
ufshc_dme_set(RX_HS_G2_SYNC_LENGTH_CAP_OFFSET, 5, 0x4a);
|
||||
ufshc_dme_set(RX_MIN_ACTIVATETIME_CAP_OFFSET, 4, 0x7);
|
||||
ufshc_dme_set(RX_MIN_ACTIVATETIME_CAP_OFFSET, 5, 0x7);
|
||||
ufshc_dme_set(TX_HIBERN8TIME_CAP_OFFSET, 0, 0x5);
|
||||
ufshc_dme_set(TX_HIBERN8TIME_CAP_OFFSET, 1, 0x5);
|
||||
ufshc_dme_set(VS_MPHY_CFG_UPDT_OFFSET, 0, 1);
|
||||
|
||||
result = ufshc_dme_get(VS_MPHY_DISABLE_OFFSET, 0, &data);
|
||||
assert((result == 0) && (data == VS_MPHY_DISABLE_MPHYDIS));
|
||||
/* enable Unipro VS MPHY */
|
||||
ufshc_dme_set(VS_MPHY_DISABLE_OFFSET, 0, 0);
|
||||
|
||||
while (1) {
|
||||
result = ufshc_dme_get(TX_FSM_STATE_OFFSET, 0, &fsm0);
|
||||
assert(result == 0);
|
||||
result = ufshc_dme_get(TX_FSM_STATE_OFFSET, 1, &fsm1);
|
||||
assert(result == 0);
|
||||
if ((fsm0 == TX_FSM_STATE_HIBERN8) &&
|
||||
(fsm1 == TX_FSM_STATE_HIBERN8))
|
||||
break;
|
||||
}
|
||||
|
||||
mmio_write_32(base + HCLKDIV, 0xE4);
|
||||
mmio_clrbits_32(base + AHIT, 0x3FF);
|
||||
|
||||
ufshc_dme_set(PA_LOCAL_TX_LCC_ENABLE_OFFSET, 0, 0);
|
||||
ufshc_dme_set(VS_MK2_EXTN_SUPPORT_OFFSET, 0, 0);
|
||||
|
||||
result = ufshc_dme_get(VS_MK2_EXTN_SUPPORT_OFFSET, 0, &data);
|
||||
assert((result == 0) && (data == 0));
|
||||
|
||||
ufshc_dme_set(DL_AFC0_CREDIT_THRESHOLD_OFFSET, 0, 0);
|
||||
ufshc_dme_set(DL_TC0_OUT_ACK_THRESHOLD_OFFSET, 0, 0);
|
||||
ufshc_dme_set(DL_TC0_TX_FC_THRESHOLD_OFFSET, 0, 9);
|
||||
(void)result;
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int dwufs_phy_set_pwr_mode(ufs_params_t *params)
|
||||
{
|
||||
int result;
|
||||
unsigned int data, tx_lanes, rx_lanes;
|
||||
uintptr_t base;
|
||||
|
||||
assert((params != NULL) && (params->reg_base != 0));
|
||||
|
||||
base = params->reg_base;
|
||||
|
||||
result = ufshc_dme_get(PA_TACTIVATE_OFFSET, 0, &data);
|
||||
assert(result == 0);
|
||||
if (data < 7) {
|
||||
result = ufshc_dme_set(PA_TACTIVATE_OFFSET, 0, 7);
|
||||
assert(result == 0);
|
||||
}
|
||||
result = ufshc_dme_get(PA_CONNECTED_TX_DATA_LANES_OFFSET, 0, &tx_lanes);
|
||||
assert(result == 0);
|
||||
result = ufshc_dme_get(PA_CONNECTED_RX_DATA_LANES_OFFSET, 0, &rx_lanes);
|
||||
assert(result == 0);
|
||||
|
||||
result = ufshc_dme_set(PA_TX_SKIP_OFFSET, 0, 0);
|
||||
assert(result == 0);
|
||||
result = ufshc_dme_set(PA_TX_GEAR_OFFSET, 0, 3);
|
||||
assert(result == 0);
|
||||
result = ufshc_dme_set(PA_RX_GEAR_OFFSET, 0, 3);
|
||||
assert(result == 0);
|
||||
result = ufshc_dme_set(PA_HS_SERIES_OFFSET, 0, 2);
|
||||
assert(result == 0);
|
||||
result = ufshc_dme_set(PA_TX_TERMINATION_OFFSET, 0, 1);
|
||||
assert(result == 0);
|
||||
result = ufshc_dme_set(PA_RX_TERMINATION_OFFSET, 0, 1);
|
||||
assert(result == 0);
|
||||
result = ufshc_dme_set(PA_SCRAMBLING_OFFSET, 0, 0);
|
||||
assert(result == 0);
|
||||
result = ufshc_dme_set(PA_ACTIVE_TX_DATA_LANES_OFFSET, 0, tx_lanes);
|
||||
assert(result == 0);
|
||||
result = ufshc_dme_set(PA_ACTIVE_RX_DATA_LANES_OFFSET, 0, rx_lanes);
|
||||
assert(result == 0);
|
||||
result = ufshc_dme_set(PA_PWR_MODE_USER_DATA0_OFFSET, 0, 8191);
|
||||
assert(result == 0);
|
||||
result = ufshc_dme_set(PA_PWR_MODE_USER_DATA1_OFFSET, 0, 65535);
|
||||
assert(result == 0);
|
||||
result = ufshc_dme_set(PA_PWR_MODE_USER_DATA2_OFFSET, 0, 32767);
|
||||
assert(result == 0);
|
||||
result = ufshc_dme_set(DME_FC0_PROTECTION_TIMEOUT_OFFSET, 0, 8191);
|
||||
assert(result == 0);
|
||||
result = ufshc_dme_set(DME_TC0_REPLAY_TIMEOUT_OFFSET, 0, 65535);
|
||||
assert(result == 0);
|
||||
result = ufshc_dme_set(DME_AFC0_REQ_TIMEOUT_OFFSET, 0, 32767);
|
||||
assert(result == 0);
|
||||
result = ufshc_dme_set(PA_PWR_MODE_USER_DATA3_OFFSET, 0, 8191);
|
||||
assert(result == 0);
|
||||
result = ufshc_dme_set(PA_PWR_MODE_USER_DATA4_OFFSET, 0, 65535);
|
||||
assert(result == 0);
|
||||
result = ufshc_dme_set(PA_PWR_MODE_USER_DATA5_OFFSET, 0, 32767);
|
||||
assert(result == 0);
|
||||
result = ufshc_dme_set(DME_FC1_PROTECTION_TIMEOUT_OFFSET, 0, 8191);
|
||||
assert(result == 0);
|
||||
result = ufshc_dme_set(DME_TC1_REPLAY_TIMEOUT_OFFSET, 0, 65535);
|
||||
assert(result == 0);
|
||||
result = ufshc_dme_set(DME_AFC1_REQ_TIMEOUT_OFFSET, 0, 32767);
|
||||
assert(result == 0);
|
||||
|
||||
result = ufshc_dme_set(PA_PWR_MODE_OFFSET, 0, 0x11);
|
||||
assert(result == 0);
|
||||
do {
|
||||
data = mmio_read_32(base + IS);
|
||||
} while ((data & UFS_INT_UPMS) == 0);
|
||||
mmio_write_32(base + IS, UFS_INT_UPMS);
|
||||
data = mmio_read_32(base + HCS);
|
||||
if ((data & HCS_UPMCRS_MASK) == HCS_PWR_LOCAL)
|
||||
INFO("ufs: change power mode success\n");
|
||||
else
|
||||
WARN("ufs: HCS.UPMCRS error, HCS:0x%x\n", data);
|
||||
(void)result;
|
||||
return 0;
|
||||
}
|
||||
|
||||
const ufs_ops_t dw_ufs_ops = {
|
||||
.phy_init = dwufs_phy_init,
|
||||
.phy_set_pwr_mode = dwufs_phy_set_pwr_mode,
|
||||
};
|
||||
|
||||
int dw_ufs_init(dw_ufs_params_t *params)
|
||||
{
|
||||
ufs_params_t ufs_params;
|
||||
|
||||
memset(&ufs_params, 0, sizeof(ufs_params));
|
||||
ufs_params.reg_base = params->reg_base;
|
||||
ufs_params.desc_base = params->desc_base;
|
||||
ufs_params.desc_size = params->desc_size;
|
||||
ufs_params.flags = params->flags;
|
||||
ufs_init(&dw_ufs_ops, &ufs_params);
|
||||
return 0;
|
||||
}
|
110
include/drivers/dw_ufs.h
Normal file
110
include/drivers/dw_ufs.h
Normal file
@ -0,0 +1,110 @@
|
||||
/*
|
||||
* Copyright (c) 2017, ARM Limited and Contributors. All rights reserved.
|
||||
*
|
||||
* SPDX-License-Identifier: BSD-3-Clause
|
||||
*/
|
||||
|
||||
#ifndef __DW_UFS_H__
|
||||
#define __DW_UFS_H__
|
||||
|
||||
#include <sys/types.h>
|
||||
|
||||
/* Bus Throtting */
|
||||
#define BUSTHRTL 0xC0
|
||||
/* Outstanding OCP Requests */
|
||||
#define OOCPR 0xC4
|
||||
/* Fatal Error Interrupt Enable */
|
||||
#define FEIE 0xC8
|
||||
/* C-Port Direct Access Configuration register */
|
||||
#define CDACFG 0xD0
|
||||
/* C-Port Direct Access Transmit 1 register */
|
||||
#define CDATX1 0xD4
|
||||
/* C-Port Direct Access Transmit 2 register */
|
||||
#define CDATX2 0xD8
|
||||
/* C-Port Direct Access Receive 1 register */
|
||||
#define CDARX1 0xDC
|
||||
/* C-Port Direct Access Receive 2 register */
|
||||
#define CDARX2 0xE0
|
||||
/* C-Port Direct Access Status register */
|
||||
#define CDASTA 0xE4
|
||||
/* UPIU Loopback Configuration register */
|
||||
#define LBMCFG 0xF0
|
||||
/* UPIU Loopback Status */
|
||||
#define LBMSTA 0xF4
|
||||
/* Debug register */
|
||||
#define DBG 0xF8
|
||||
/* HClk Divider register */
|
||||
#define HCLKDIV 0xFC
|
||||
|
||||
#define TX_HIBERN8TIME_CAP_OFFSET 0x000F
|
||||
#define TX_FSM_STATE_OFFSET 0x0041
|
||||
#define TX_FSM_STATE_LINE_RESET 7
|
||||
#define TX_FSM_STATE_LINE_CFG 6
|
||||
#define TX_FSM_STATE_HS_BURST 5
|
||||
#define TX_FSM_STATE_LS_BURST 4
|
||||
#define TX_FSM_STATE_STALL 3
|
||||
#define TX_FSM_STATE_SLEEP 2
|
||||
#define TX_FSM_STATE_HIBERN8 1
|
||||
#define TX_FSM_STATE_DISABLE 0
|
||||
|
||||
#define RX_MIN_ACTIVATETIME_CAP_OFFSET 0x008F
|
||||
#define RX_HS_G2_SYNC_LENGTH_CAP_OFFSET 0x0094
|
||||
#define RX_HS_G3_SYNC_LENGTH_CAP_OFFSET 0x0095
|
||||
|
||||
#define PA_AVAIL_TX_DATA_LANES_OFFSET 0x1520
|
||||
#define PA_TX_SKIP_OFFSET 0x155C
|
||||
#define PA_TX_SKIP_PERIOD_OFFSET 0x155D
|
||||
#define PA_LOCAL_TX_LCC_ENABLE_OFFSET 0x155E
|
||||
#define PA_ACTIVE_TX_DATA_LANES_OFFSET 0x1560
|
||||
#define PA_CONNECTED_TX_DATA_LANES_OFFSET 0x1561
|
||||
#define PA_TX_TRAILING_CLOCKS_OFFSET 0x1564
|
||||
#define PA_TX_GEAR_OFFSET 0x1568
|
||||
#define PA_TX_TERMINATION_OFFSET 0x1569
|
||||
#define PA_HS_SERIES_OFFSET 0x156A
|
||||
#define PA_PWR_MODE_OFFSET 0x1571
|
||||
#define PA_ACTIVE_RX_DATA_LANES_OFFSET 0x1580
|
||||
#define PA_CONNECTED_RX_DATA_LANES_OFFSET 0x1581
|
||||
#define PA_RX_PWR_STATUS_OFFSET 0x1582
|
||||
#define PA_RX_GEAR_OFFSET 0x1583
|
||||
#define PA_RX_TERMINATION_OFFSET 0x1584
|
||||
#define PA_SCRAMBLING_OFFSET 0x1585
|
||||
#define PA_MAX_RX_PWM_GEAR_OFFSET 0x1586
|
||||
#define PA_MAX_RX_HS_GEAR_OFFSET 0x1587
|
||||
#define PA_PACP_REQ_TIMEOUT_OFFSET 0x1590
|
||||
#define PA_PACP_REQ_EOB_TIMEOUT_OFFSET 0x1591
|
||||
#define PA_REMOTE_VER_INFO_OFFSET 0x15A0
|
||||
#define PA_LOGICAL_LANE_MAP_OFFSET 0x15A1
|
||||
#define PA_TACTIVATE_OFFSET 0x15A8
|
||||
#define PA_PWR_MODE_USER_DATA0_OFFSET 0x15B0
|
||||
#define PA_PWR_MODE_USER_DATA1_OFFSET 0x15B1
|
||||
#define PA_PWR_MODE_USER_DATA2_OFFSET 0x15B2
|
||||
#define PA_PWR_MODE_USER_DATA3_OFFSET 0x15B3
|
||||
#define PA_PWR_MODE_USER_DATA4_OFFSET 0x15B4
|
||||
#define PA_PWR_MODE_USER_DATA5_OFFSET 0x15B5
|
||||
|
||||
#define DL_TC0_TX_FC_THRESHOLD_OFFSET 0x2040
|
||||
#define DL_AFC0_CREDIT_THRESHOLD_OFFSET 0x2044
|
||||
#define DL_TC0_OUT_ACK_THRESHOLD_OFFSET 0x2045
|
||||
|
||||
#define DME_FC0_PROTECTION_TIMEOUT_OFFSET 0xD041
|
||||
#define DME_TC0_REPLAY_TIMEOUT_OFFSET 0xD042
|
||||
#define DME_AFC0_REQ_TIMEOUT_OFFSET 0xD043
|
||||
#define DME_FC1_PROTECTION_TIMEOUT_OFFSET 0xD044
|
||||
#define DME_TC1_REPLAY_TIMEOUT_OFFSET 0xD045
|
||||
#define DME_AFC1_REQ_TIMEOUT_OFFSET 0xD046
|
||||
|
||||
#define VS_MPHY_CFG_UPDT_OFFSET 0xD085
|
||||
#define VS_MK2_EXTN_SUPPORT_OFFSET 0xD0AB
|
||||
#define VS_MPHY_DISABLE_OFFSET 0xD0C1
|
||||
#define VS_MPHY_DISABLE_MPHYDIS (1 << 0)
|
||||
|
||||
typedef struct dw_ufs_params {
|
||||
uintptr_t reg_base;
|
||||
uintptr_t desc_base;
|
||||
size_t desc_size;
|
||||
unsigned long flags;
|
||||
} dw_ufs_params_t;
|
||||
|
||||
int dw_ufs_init(dw_ufs_params_t *params);
|
||||
|
||||
#endif /* __DW_UFS_H__ */
|
Loading…
x
Reference in New Issue
Block a user