提交 5dcbc711 编写于 作者: J Jonathan Marek 提交者: Vinod Koul

phy: qcom-qmp: Allow different values for second lane

The primary USB PHY on sm8250 sets some values differently for the second
lane. This makes it possible to represent that.
Signed-off-by: NJonathan Marek <jonathan@marek.ca>
Tested-by: NDmitry Baryshkov <dmitry.baryshkov@linaro.org>
Reviewed-by: NBjorn Andersson <bjorn.andersson@linaro.org>
Link: https://lore.kernel.org/r/20200524021416.17049-2-jonathan@marek.caSigned-off-by: NVinod Koul <vkoul@kernel.org>
上级 1d99d491
...@@ -82,20 +82,34 @@ struct qmp_phy_init_tbl { ...@@ -82,20 +82,34 @@ struct qmp_phy_init_tbl {
* register part of layout ? * register part of layout ?
* if yes, then offset gives index in the reg-layout * if yes, then offset gives index in the reg-layout
*/ */
int in_layout; bool in_layout;
/*
* mask of lanes for which this register is written
* for cases when second lane needs different values
*/
u8 lane_mask;
}; };
#define QMP_PHY_INIT_CFG(o, v) \ #define QMP_PHY_INIT_CFG(o, v) \
{ \ { \
.offset = o, \ .offset = o, \
.val = v, \ .val = v, \
.lane_mask = 0xff, \
} }
#define QMP_PHY_INIT_CFG_L(o, v) \ #define QMP_PHY_INIT_CFG_L(o, v) \
{ \ { \
.offset = o, \ .offset = o, \
.val = v, \ .val = v, \
.in_layout = 1, \ .in_layout = true, \
.lane_mask = 0xff, \
}
#define QMP_PHY_INIT_CFG_LANE(o, v, l) \
{ \
.offset = o, \
.val = v, \
.lane_mask = l, \
} }
/* set of registers with offsets different per-PHY */ /* set of registers with offsets different per-PHY */
...@@ -2085,10 +2099,11 @@ static const struct qmp_phy_cfg sm8150_usb3phy_cfg = { ...@@ -2085,10 +2099,11 @@ static const struct qmp_phy_cfg sm8150_usb3phy_cfg = {
.is_dual_lane_phy = true, .is_dual_lane_phy = true,
}; };
static void qcom_qmp_phy_configure(void __iomem *base, static void qcom_qmp_phy_configure_lane(void __iomem *base,
const unsigned int *regs, const unsigned int *regs,
const struct qmp_phy_init_tbl tbl[], const struct qmp_phy_init_tbl tbl[],
int num) int num,
u8 lane_mask)
{ {
int i; int i;
const struct qmp_phy_init_tbl *t = tbl; const struct qmp_phy_init_tbl *t = tbl;
...@@ -2097,6 +2112,9 @@ static void qcom_qmp_phy_configure(void __iomem *base, ...@@ -2097,6 +2112,9 @@ static void qcom_qmp_phy_configure(void __iomem *base,
return; return;
for (i = 0; i < num; i++, t++) { for (i = 0; i < num; i++, t++) {
if (!(t->lane_mask & lane_mask))
continue;
if (t->in_layout) if (t->in_layout)
writel(t->val, base + regs[t->offset]); writel(t->val, base + regs[t->offset]);
else else
...@@ -2104,6 +2122,14 @@ static void qcom_qmp_phy_configure(void __iomem *base, ...@@ -2104,6 +2122,14 @@ static void qcom_qmp_phy_configure(void __iomem *base,
} }
} }
static void qcom_qmp_phy_configure(void __iomem *base,
const unsigned int *regs,
const struct qmp_phy_init_tbl tbl[],
int num)
{
qcom_qmp_phy_configure_lane(base, regs, tbl, num, 0xff);
}
static int qcom_qmp_phy_com_init(struct qmp_phy *qphy) static int qcom_qmp_phy_com_init(struct qmp_phy *qphy)
{ {
struct qcom_qmp *qmp = qphy->qmp; struct qcom_qmp *qmp = qphy->qmp;
...@@ -2318,16 +2344,18 @@ static int qcom_qmp_phy_enable(struct phy *phy) ...@@ -2318,16 +2344,18 @@ static int qcom_qmp_phy_enable(struct phy *phy)
} }
/* Tx, Rx, and PCS configurations */ /* Tx, Rx, and PCS configurations */
qcom_qmp_phy_configure(tx, cfg->regs, cfg->tx_tbl, cfg->tx_tbl_num); qcom_qmp_phy_configure_lane(tx, cfg->regs,
cfg->tx_tbl, cfg->tx_tbl_num, 1);
/* Configuration for other LANE for USB-DP combo PHY */ /* Configuration for other LANE for USB-DP combo PHY */
if (cfg->is_dual_lane_phy) if (cfg->is_dual_lane_phy)
qcom_qmp_phy_configure(qphy->tx2, cfg->regs, qcom_qmp_phy_configure_lane(qphy->tx2, cfg->regs,
cfg->tx_tbl, cfg->tx_tbl_num); cfg->tx_tbl, cfg->tx_tbl_num, 2);
qcom_qmp_phy_configure(rx, cfg->regs, cfg->rx_tbl, cfg->rx_tbl_num); qcom_qmp_phy_configure_lane(rx, cfg->regs,
cfg->rx_tbl, cfg->rx_tbl_num, 1);
if (cfg->is_dual_lane_phy) if (cfg->is_dual_lane_phy)
qcom_qmp_phy_configure(qphy->rx2, cfg->regs, qcom_qmp_phy_configure_lane(qphy->rx2, cfg->regs,
cfg->rx_tbl, cfg->rx_tbl_num); cfg->rx_tbl, cfg->rx_tbl_num, 2);
qcom_qmp_phy_configure(pcs, cfg->regs, cfg->pcs_tbl, cfg->pcs_tbl_num); qcom_qmp_phy_configure(pcs, cfg->regs, cfg->pcs_tbl, cfg->pcs_tbl_num);
ret = reset_control_deassert(qmp->ufs_reset); ret = reset_control_deassert(qmp->ufs_reset);
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册