phy fixes for 7.1
Couple of driver fixes
- Big pile of Qualcomm DP/eDP config fixes and kaanapali PHY PLL
lock failure fix
- Apple typec switch/mux leak fix
- Marvell incoorect register fix for mvebu utmi phy
- Tegra per-pad calibration fix
-----BEGIN PGP SIGNATURE-----
iQIzBAABCgAdFiEE+vs47OPLdNbVcHzyfBQHDyUjg0cFAmoQjSAACgkQfBQHDyUj
g0e5NA//dOb7/jrQf2jFlzlK3MprA4DWxpB8o3cFzWWxTix20+jSyBqyVtC/g0EM
zf0hGPbh3dhmA6hN6wAaYLj+ly9k5TEibqMbziEaZNNYlvO/ugkeOGrFHBLwNkS1
gpT4n4Zojdarr3OtqkEsFCuaaWtjRl6f5Qz0mT8Q1D1qgb1wHuiQDeHKG/o4yWyZ
ROyawyCmKebV/T35XUxn+MtGrhVC8CUGhFebn2Cn4JEog915f9uuacFZtRV+SO3C
wc8s2kEwnCkCiOQjE/f/8Jgx4NtP0FpHu09dJJ81IciOlhPOWIBGmVE1wWwFNdGs
orJMYwIBcdixfWoFUaxcJaGp/VtDx5cyVqL8ZUyVCj9cN5Az/C917ELkpflPitqv
jFsclq/fZoYWjvaGGILe+pW6qRG4XPFL3PzqClGKPi4bJpip4mvi6EEKUFpZzg4B
SlpCal8N6UN1l6Vl8YSTW0WNyeafUpeANRddrVKoignVGbvVS/C7Hs2b4WRE3Ahh
R8tt8w6Toe69g/syu1X46qAUe9M9/S5zIsKSgELp5zI4TzKtZJ70lLRCm49oNUgO
AHDCxeIAE9jeokroiEitzO5tvfGhCXLy+1RIx2uLMphlzhSnP4r8RXN+r81/lBtq
KDwqw2NcZ7V/TQmhOz9rr7Lmy1ltpti76b0isb3MOHVPytWpHpo=
=SxJy
-----END PGP SIGNATURE-----
Merge tag 'phy-fixes-7.1' of git://git.kernel.org/pub/scm/linux/kernel/git/phy/linux-phy
Pull phy fixes from Vinod Koul:
- Big pile of Qualcomm DP/eDP config fixes and kaanapali PHY PLL
lock failure fix
- Apple typec switch/mux leak fix
- Marvell incoorect register fix for mvebu utmi phy
- Tegra per-pad calibration fix
* tag 'phy-fixes-7.1' of git://git.kernel.org/pub/scm/linux/kernel/git/phy/linux-phy:
phy: qcom: qmp-usbc: Fix out-of-bounds array access in dp swing config
phy: apple: atc: Fix typec switch/mux leak on unbind
phy: spacemit: Remove incorrect clk_disable() in spacemit_usb2phy_init()
phy: eswin: Fix incorrect error check in probe()
phy: qcom-qmp-ufs: Fix kaanapali PHY PLL lock failure after SM8650 G4 fix
phy: exynos5-usbdrd: fix USB 2.0 HS PHY tuning values for Exynos7870
phy: tegra: xusb: Fix per-pad high-speed termination calibration
phy: marvell: mvebu-a3700-utmi: fix incorrect USB2_PHY_CTRL register access
phy: qcom: edp: Add PHY-specific LDO config for eDP low vdiff
phy: qcom: edp: Fix AUX_CFG8 programming for DP mode
phy: qcom: edp: Add SC7280/SC8180X swing/pre-emphasis tables
phy: qcom: edp: Add eDP/DP mode switch support
phy: qcom: edp: Unify generic DP/eDP swing and pre-emphasis tables
master
commit
cca95436be
|
|
@ -628,9 +628,6 @@ struct apple_atcphy {
|
|||
|
||||
struct reset_controller_dev rcdev;
|
||||
|
||||
struct typec_switch *sw;
|
||||
struct typec_mux *mux;
|
||||
|
||||
struct mutex lock;
|
||||
};
|
||||
|
||||
|
|
@ -2066,15 +2063,25 @@ static int atcphy_sw_set(struct typec_switch_dev *sw, enum typec_orientation ori
|
|||
return 0;
|
||||
}
|
||||
|
||||
static void atcphy_typec_switch_unregister(void *data)
|
||||
{
|
||||
typec_switch_unregister(data);
|
||||
}
|
||||
|
||||
static int atcphy_probe_switch(struct apple_atcphy *atcphy)
|
||||
{
|
||||
struct typec_switch_dev *sw;
|
||||
struct typec_switch_desc sw_desc = {
|
||||
.drvdata = atcphy,
|
||||
.fwnode = atcphy->dev->fwnode,
|
||||
.set = atcphy_sw_set,
|
||||
};
|
||||
|
||||
return PTR_ERR_OR_ZERO(typec_switch_register(atcphy->dev, &sw_desc));
|
||||
sw = typec_switch_register(atcphy->dev, &sw_desc);
|
||||
if (IS_ERR(sw))
|
||||
return PTR_ERR(sw);
|
||||
|
||||
return devm_add_action_or_reset(atcphy->dev, atcphy_typec_switch_unregister, sw);
|
||||
}
|
||||
|
||||
static int atcphy_mux_set(struct typec_mux_dev *mux, struct typec_mux_state *state)
|
||||
|
|
@ -2146,15 +2153,25 @@ static int atcphy_mux_set(struct typec_mux_dev *mux, struct typec_mux_state *sta
|
|||
return atcphy_configure(atcphy, target_mode);
|
||||
}
|
||||
|
||||
static void atcphy_typec_mux_unregister(void *data)
|
||||
{
|
||||
typec_mux_unregister(data);
|
||||
}
|
||||
|
||||
static int atcphy_probe_mux(struct apple_atcphy *atcphy)
|
||||
{
|
||||
struct typec_mux_dev *mux;
|
||||
struct typec_mux_desc mux_desc = {
|
||||
.drvdata = atcphy,
|
||||
.fwnode = atcphy->dev->fwnode,
|
||||
.set = atcphy_mux_set,
|
||||
};
|
||||
|
||||
return PTR_ERR_OR_ZERO(typec_mux_register(atcphy->dev, &mux_desc));
|
||||
mux = typec_mux_register(atcphy->dev, &mux_desc);
|
||||
if (IS_ERR(mux))
|
||||
return PTR_ERR(mux);
|
||||
|
||||
return devm_add_action_or_reset(atcphy->dev, atcphy_typec_mux_unregister, mux);
|
||||
}
|
||||
|
||||
static int atcphy_load_tunables(struct apple_atcphy *atcphy)
|
||||
|
|
|
|||
|
|
@ -216,8 +216,8 @@ static int eic7700_sata_phy_probe(struct platform_device *pdev)
|
|||
return -ENOENT;
|
||||
|
||||
regs = devm_ioremap(dev, res->start, resource_size(res));
|
||||
if (IS_ERR(regs))
|
||||
return PTR_ERR(regs);
|
||||
if (!regs)
|
||||
return -ENOMEM;
|
||||
|
||||
sata_phy->regmap = devm_regmap_init_mmio
|
||||
(dev, regs, &eic7700_sata_phy_regmap_config);
|
||||
|
|
|
|||
|
|
@ -168,9 +168,8 @@ static int mvebu_a3700_utmi_phy_power_off(struct phy *phy)
|
|||
u32 reg;
|
||||
|
||||
/* Disable PHY pull-up and enable USB2 suspend */
|
||||
reg = readl(utmi->regs + USB2_PHY_CTRL(usb32));
|
||||
reg &= ~(RB_USB2PHY_PU | RB_USB2PHY_SUSPM(usb32));
|
||||
writel(reg, utmi->regs + USB2_PHY_CTRL(usb32));
|
||||
regmap_update_bits(utmi->usb_misc, USB2_PHY_CTRL(usb32),
|
||||
RB_USB2PHY_PU | RB_USB2PHY_SUSPM(usb32), 0);
|
||||
|
||||
/* Power down OTG module */
|
||||
if (usb32) {
|
||||
|
|
|
|||
|
|
@ -81,13 +81,15 @@ struct phy_ver_ops {
|
|||
int (*com_clk_fwd_cfg)(const struct qcom_edp *edp);
|
||||
int (*com_configure_pll)(const struct qcom_edp *edp);
|
||||
int (*com_configure_ssc)(const struct qcom_edp *edp);
|
||||
int (*com_ldo_config)(const struct qcom_edp *edp);
|
||||
};
|
||||
|
||||
struct qcom_edp_phy_cfg {
|
||||
bool is_edp;
|
||||
const u8 *aux_cfg;
|
||||
const u8 *vco_div_cfg;
|
||||
const struct qcom_edp_swing_pre_emph_cfg *swing_pre_emph_cfg;
|
||||
const struct qcom_edp_swing_pre_emph_cfg *dp_swing_pre_emph_cfg;
|
||||
const struct qcom_edp_swing_pre_emph_cfg *edp_swing_pre_emph_cfg;
|
||||
const struct phy_ver_ops *ver_ops;
|
||||
};
|
||||
|
||||
|
|
@ -116,17 +118,17 @@ struct qcom_edp {
|
|||
};
|
||||
|
||||
static const u8 dp_swing_hbr_rbr[4][4] = {
|
||||
{ 0x08, 0x0f, 0x16, 0x1f },
|
||||
{ 0x07, 0x0f, 0x16, 0x1f },
|
||||
{ 0x11, 0x1e, 0x1f, 0xff },
|
||||
{ 0x16, 0x1f, 0xff, 0xff },
|
||||
{ 0x1f, 0xff, 0xff, 0xff }
|
||||
};
|
||||
|
||||
static const u8 dp_pre_emp_hbr_rbr[4][4] = {
|
||||
{ 0x00, 0x0d, 0x14, 0x1a },
|
||||
{ 0x00, 0x0e, 0x15, 0x1a },
|
||||
{ 0x00, 0x0e, 0x15, 0xff },
|
||||
{ 0x00, 0x0e, 0xff, 0xff },
|
||||
{ 0x03, 0xff, 0xff, 0xff }
|
||||
{ 0x04, 0xff, 0xff, 0xff }
|
||||
};
|
||||
|
||||
static const u8 dp_swing_hbr2_hbr3[4][4] = {
|
||||
|
|
@ -150,6 +152,47 @@ static const struct qcom_edp_swing_pre_emph_cfg dp_phy_swing_pre_emph_cfg = {
|
|||
.pre_emphasis_hbr3_hbr2 = &dp_pre_emp_hbr2_hbr3,
|
||||
};
|
||||
|
||||
static const u8 dp_pre_emp_hbr_rbr_v8[4][4] = {
|
||||
{ 0x00, 0x0e, 0x15, 0x1a },
|
||||
{ 0x00, 0x0e, 0x15, 0xff },
|
||||
{ 0x00, 0x0e, 0xff, 0xff },
|
||||
{ 0x00, 0xff, 0xff, 0xff }
|
||||
};
|
||||
|
||||
static const struct qcom_edp_swing_pre_emph_cfg dp_phy_swing_pre_emph_cfg_v8 = {
|
||||
.swing_hbr_rbr = &dp_swing_hbr_rbr,
|
||||
.swing_hbr3_hbr2 = &dp_swing_hbr2_hbr3,
|
||||
.pre_emphasis_hbr_rbr = &dp_pre_emp_hbr_rbr_v8,
|
||||
.pre_emphasis_hbr3_hbr2 = &dp_pre_emp_hbr2_hbr3,
|
||||
};
|
||||
|
||||
static const u8 dp_swing_hbr2_hbr3_v2[4][4] = {
|
||||
{ 0x27, 0x2f, 0x36, 0xff },
|
||||
{ 0x31, 0x3e, 0x3f, 0xff },
|
||||
{ 0x3a, 0x3f, 0xff, 0xff },
|
||||
{ 0xff, 0xff, 0xff, 0xff }
|
||||
};
|
||||
|
||||
static const u8 dp_pre_emp_hbr2_hbr3_v2[4][4] = {
|
||||
{ 0x20, 0x2e, 0x35, 0xff },
|
||||
{ 0x20, 0x2e, 0x35, 0xff },
|
||||
{ 0x20, 0x2e, 0xff, 0xff },
|
||||
{ 0xff, 0xff, 0xff, 0xff }
|
||||
};
|
||||
|
||||
static const struct qcom_edp_swing_pre_emph_cfg dp_phy_swing_pre_emph_cfg_v2 = {
|
||||
/*
|
||||
* NOTE: The HPG does not specify a separate swing_hbr_rbr table.
|
||||
* Reuse the HBR2/HBR3 table for now.
|
||||
*
|
||||
* TODO: Update this once the HPG explicitly defines RBR/HBR swing values.
|
||||
*/
|
||||
.swing_hbr_rbr = &dp_swing_hbr2_hbr3_v2,
|
||||
.swing_hbr3_hbr2 = &dp_swing_hbr2_hbr3_v2,
|
||||
.pre_emphasis_hbr_rbr = &dp_pre_emp_hbr2_hbr3_v2,
|
||||
.pre_emphasis_hbr3_hbr2 = &dp_pre_emp_hbr2_hbr3_v2,
|
||||
};
|
||||
|
||||
static const u8 edp_swing_hbr_rbr[4][4] = {
|
||||
{ 0x07, 0x0f, 0x16, 0x1f },
|
||||
{ 0x0d, 0x16, 0x1e, 0xff },
|
||||
|
|
@ -158,7 +201,7 @@ static const u8 edp_swing_hbr_rbr[4][4] = {
|
|||
};
|
||||
|
||||
static const u8 edp_pre_emp_hbr_rbr[4][4] = {
|
||||
{ 0x05, 0x12, 0x17, 0x1d },
|
||||
{ 0x05, 0x11, 0x17, 0x1d },
|
||||
{ 0x05, 0x11, 0x18, 0xff },
|
||||
{ 0x06, 0x11, 0xff, 0xff },
|
||||
{ 0x00, 0xff, 0xff, 0xff }
|
||||
|
|
@ -172,10 +215,10 @@ static const u8 edp_swing_hbr2_hbr3[4][4] = {
|
|||
};
|
||||
|
||||
static const u8 edp_pre_emp_hbr2_hbr3[4][4] = {
|
||||
{ 0x08, 0x11, 0x17, 0x1b },
|
||||
{ 0x00, 0x0c, 0x13, 0xff },
|
||||
{ 0x05, 0x10, 0xff, 0xff },
|
||||
{ 0x00, 0xff, 0xff, 0xff }
|
||||
{ 0x0c, 0x15, 0x19, 0x1e },
|
||||
{ 0x0b, 0x15, 0x19, 0xff },
|
||||
{ 0x0e, 0x14, 0xff, 0xff },
|
||||
{ 0x0d, 0xff, 0xff, 0xff }
|
||||
};
|
||||
|
||||
static const struct qcom_edp_swing_pre_emph_cfg edp_phy_swing_pre_emph_cfg = {
|
||||
|
|
@ -193,25 +236,46 @@ static const u8 edp_phy_vco_div_cfg_v4[4] = {
|
|||
0x01, 0x01, 0x02, 0x00,
|
||||
};
|
||||
|
||||
static const u8 edp_pre_emp_hbr_rbr_v5[4][4] = {
|
||||
{ 0x05, 0x11, 0x17, 0x1d },
|
||||
static const u8 edp_pre_emp_hbr_rbr_v2[4][4] = {
|
||||
{ 0x05, 0x12, 0x17, 0x1d },
|
||||
{ 0x05, 0x11, 0x18, 0xff },
|
||||
{ 0x06, 0x11, 0xff, 0xff },
|
||||
{ 0x00, 0xff, 0xff, 0xff }
|
||||
};
|
||||
|
||||
static const u8 edp_pre_emp_hbr2_hbr3_v5[4][4] = {
|
||||
static const u8 edp_pre_emp_hbr2_hbr3_v2[4][4] = {
|
||||
{ 0x0c, 0x15, 0x19, 0x1e },
|
||||
{ 0x0b, 0x15, 0x19, 0xff },
|
||||
{ 0x08, 0x15, 0x19, 0xff },
|
||||
{ 0x0e, 0x14, 0xff, 0xff },
|
||||
{ 0x0d, 0xff, 0xff, 0xff }
|
||||
};
|
||||
|
||||
static const struct qcom_edp_swing_pre_emph_cfg edp_phy_swing_pre_emph_cfg_v5 = {
|
||||
static const struct qcom_edp_swing_pre_emph_cfg edp_phy_swing_pre_emph_cfg_v2 = {
|
||||
.swing_hbr_rbr = &edp_swing_hbr_rbr,
|
||||
.swing_hbr3_hbr2 = &edp_swing_hbr2_hbr3,
|
||||
.pre_emphasis_hbr_rbr = &edp_pre_emp_hbr_rbr_v5,
|
||||
.pre_emphasis_hbr3_hbr2 = &edp_pre_emp_hbr2_hbr3_v5,
|
||||
.pre_emphasis_hbr_rbr = &edp_pre_emp_hbr_rbr_v2,
|
||||
.pre_emphasis_hbr3_hbr2 = &edp_pre_emp_hbr2_hbr3_v2,
|
||||
};
|
||||
|
||||
static const u8 edp_swing_hbr2_hbr3_v3[4][4] = {
|
||||
{ 0x06, 0x11, 0x16, 0x1b },
|
||||
{ 0x0b, 0x19, 0x1f, 0xff },
|
||||
{ 0x18, 0x1f, 0xff, 0xff },
|
||||
{ 0x1f, 0xff, 0xff, 0xff }
|
||||
};
|
||||
|
||||
static const u8 edp_pre_emp_hbr2_hbr3_v3[4][4] = {
|
||||
{ 0x0c, 0x15, 0x19, 0x1e },
|
||||
{ 0x09, 0x14, 0x19, 0xff },
|
||||
{ 0x0f, 0x14, 0xff, 0xff },
|
||||
{ 0x0d, 0xff, 0xff, 0xff }
|
||||
};
|
||||
|
||||
static const struct qcom_edp_swing_pre_emph_cfg edp_phy_swing_pre_emph_cfg_v3 = {
|
||||
.swing_hbr_rbr = &edp_swing_hbr_rbr,
|
||||
.swing_hbr3_hbr2 = &edp_swing_hbr2_hbr3_v3,
|
||||
.pre_emphasis_hbr_rbr = &edp_pre_emp_hbr_rbr,
|
||||
.pre_emphasis_hbr3_hbr2 = &edp_pre_emp_hbr2_hbr3_v3,
|
||||
};
|
||||
|
||||
static const u8 edp_phy_aux_cfg_v5[DP_AUX_CFG_SIZE] = {
|
||||
|
|
@ -262,12 +326,7 @@ static int qcom_edp_phy_init(struct phy *phy)
|
|||
DP_PHY_PD_CTL_PLL_PWRDN | DP_PHY_PD_CTL_DP_CLAMP_EN,
|
||||
edp->edp + DP_PHY_PD_CTL);
|
||||
|
||||
/*
|
||||
* TODO: Re-work the conditions around setting the cfg8 value
|
||||
* when more information becomes available about why this is
|
||||
* even needed.
|
||||
*/
|
||||
if (edp->cfg->swing_pre_emph_cfg && !edp->is_edp)
|
||||
if (!edp->is_edp)
|
||||
aux_cfg[8] = 0xb7;
|
||||
|
||||
writel(0xfc, edp->edp + DP_PHY_MODE);
|
||||
|
|
@ -291,19 +350,18 @@ out_disable_supplies:
|
|||
|
||||
static int qcom_edp_set_voltages(struct qcom_edp *edp, const struct phy_configure_opts_dp *dp_opts)
|
||||
{
|
||||
const struct qcom_edp_swing_pre_emph_cfg *cfg = edp->cfg->swing_pre_emph_cfg;
|
||||
const struct qcom_edp_swing_pre_emph_cfg *cfg;
|
||||
unsigned int v_level = 0;
|
||||
unsigned int p_level = 0;
|
||||
u8 ldo_config;
|
||||
int ret;
|
||||
u8 swing;
|
||||
u8 emph;
|
||||
int i;
|
||||
|
||||
if (!cfg)
|
||||
return 0;
|
||||
|
||||
if (edp->is_edp)
|
||||
cfg = &edp_phy_swing_pre_emph_cfg;
|
||||
cfg = edp->cfg->edp_swing_pre_emph_cfg;
|
||||
else
|
||||
cfg = edp->cfg->dp_swing_pre_emph_cfg;
|
||||
|
||||
for (i = 0; i < dp_opts->lanes; i++) {
|
||||
v_level = max(v_level, dp_opts->voltage[i]);
|
||||
|
|
@ -321,13 +379,13 @@ static int qcom_edp_set_voltages(struct qcom_edp *edp, const struct phy_configur
|
|||
if (swing == 0xff || emph == 0xff)
|
||||
return -EINVAL;
|
||||
|
||||
ldo_config = edp->is_edp ? 0x0 : 0x1;
|
||||
ret = edp->cfg->ver_ops->com_ldo_config(edp);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
writel(ldo_config, edp->tx0 + TXn_LDO_CONFIG);
|
||||
writel(swing, edp->tx0 + TXn_TX_DRV_LVL);
|
||||
writel(emph, edp->tx0 + TXn_TX_EMP_POST1_LVL);
|
||||
|
||||
writel(ldo_config, edp->tx1 + TXn_LDO_CONFIG);
|
||||
writel(swing, edp->tx1 + TXn_TX_DRV_LVL);
|
||||
writel(emph, edp->tx1 + TXn_TX_EMP_POST1_LVL);
|
||||
|
||||
|
|
@ -551,6 +609,52 @@ static int qcom_edp_com_configure_pll_v4(const struct qcom_edp *edp)
|
|||
return 0;
|
||||
}
|
||||
|
||||
static int qcom_edp_ldo_config_v3(const struct qcom_edp *edp)
|
||||
{
|
||||
const struct phy_configure_opts_dp *dp_opts = &edp->dp_opts;
|
||||
u32 ldo_config;
|
||||
|
||||
if (!edp->is_edp)
|
||||
ldo_config = 0x0;
|
||||
else if (dp_opts->link_rate <= 2700)
|
||||
ldo_config = 0x81;
|
||||
else
|
||||
ldo_config = 0x41;
|
||||
|
||||
writel(ldo_config, edp->tx0 + TXn_LDO_CONFIG);
|
||||
writel(dp_opts->lanes > 2 ? ldo_config : 0x00, edp->tx1 + TXn_LDO_CONFIG);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int qcom_edp_ldo_config_v4(const struct qcom_edp *edp)
|
||||
{
|
||||
const struct phy_configure_opts_dp *dp_opts = &edp->dp_opts;
|
||||
u32 ldo_config;
|
||||
|
||||
if (!edp->is_edp)
|
||||
ldo_config = 0x0;
|
||||
else if (dp_opts->link_rate <= 2700)
|
||||
ldo_config = 0xc1;
|
||||
else
|
||||
ldo_config = 0x81;
|
||||
|
||||
writel(ldo_config, edp->tx0 + TXn_LDO_CONFIG);
|
||||
writel(dp_opts->lanes > 2 ? ldo_config : 0x00, edp->tx1 + TXn_LDO_CONFIG);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const struct phy_ver_ops qcom_edp_phy_ops_v3 = {
|
||||
.com_power_on = qcom_edp_phy_power_on_v4,
|
||||
.com_resetsm_cntrl = qcom_edp_phy_com_resetsm_cntrl_v4,
|
||||
.com_bias_en_clkbuflr = qcom_edp_com_bias_en_clkbuflr_v4,
|
||||
.com_clk_fwd_cfg = qcom_edp_com_clk_fwd_cfg_v4,
|
||||
.com_configure_pll = qcom_edp_com_configure_pll_v4,
|
||||
.com_configure_ssc = qcom_edp_com_configure_ssc_v4,
|
||||
.com_ldo_config = qcom_edp_ldo_config_v3,
|
||||
};
|
||||
|
||||
static const struct phy_ver_ops qcom_edp_phy_ops_v4 = {
|
||||
.com_power_on = qcom_edp_phy_power_on_v4,
|
||||
.com_resetsm_cntrl = qcom_edp_phy_com_resetsm_cntrl_v4,
|
||||
|
|
@ -558,26 +662,39 @@ static const struct phy_ver_ops qcom_edp_phy_ops_v4 = {
|
|||
.com_clk_fwd_cfg = qcom_edp_com_clk_fwd_cfg_v4,
|
||||
.com_configure_pll = qcom_edp_com_configure_pll_v4,
|
||||
.com_configure_ssc = qcom_edp_com_configure_ssc_v4,
|
||||
.com_ldo_config = qcom_edp_ldo_config_v4,
|
||||
};
|
||||
|
||||
static const struct qcom_edp_phy_cfg sa8775p_dp_phy_cfg = {
|
||||
.is_edp = false,
|
||||
.aux_cfg = edp_phy_aux_cfg_v5,
|
||||
.vco_div_cfg = edp_phy_vco_div_cfg_v4,
|
||||
.swing_pre_emph_cfg = &edp_phy_swing_pre_emph_cfg_v5,
|
||||
.dp_swing_pre_emph_cfg = &dp_phy_swing_pre_emph_cfg,
|
||||
.edp_swing_pre_emph_cfg = &edp_phy_swing_pre_emph_cfg,
|
||||
.ver_ops = &qcom_edp_phy_ops_v4,
|
||||
};
|
||||
|
||||
static const struct qcom_edp_phy_cfg sc7280_dp_phy_cfg = {
|
||||
.aux_cfg = edp_phy_aux_cfg_v4,
|
||||
.vco_div_cfg = edp_phy_vco_div_cfg_v4,
|
||||
.ver_ops = &qcom_edp_phy_ops_v4,
|
||||
.dp_swing_pre_emph_cfg = &dp_phy_swing_pre_emph_cfg,
|
||||
.edp_swing_pre_emph_cfg = &edp_phy_swing_pre_emph_cfg_v3,
|
||||
.ver_ops = &qcom_edp_phy_ops_v3,
|
||||
};
|
||||
|
||||
static const struct qcom_edp_phy_cfg sc8180x_dp_phy_cfg = {
|
||||
.aux_cfg = edp_phy_aux_cfg_v4,
|
||||
.vco_div_cfg = edp_phy_vco_div_cfg_v4,
|
||||
.dp_swing_pre_emph_cfg = &dp_phy_swing_pre_emph_cfg_v2,
|
||||
.edp_swing_pre_emph_cfg = &edp_phy_swing_pre_emph_cfg_v2,
|
||||
.ver_ops = &qcom_edp_phy_ops_v3,
|
||||
};
|
||||
|
||||
static const struct qcom_edp_phy_cfg sc8280xp_dp_phy_cfg = {
|
||||
.aux_cfg = edp_phy_aux_cfg_v4,
|
||||
.vco_div_cfg = edp_phy_vco_div_cfg_v4,
|
||||
.swing_pre_emph_cfg = &dp_phy_swing_pre_emph_cfg,
|
||||
.dp_swing_pre_emph_cfg = &dp_phy_swing_pre_emph_cfg,
|
||||
.edp_swing_pre_emph_cfg = &edp_phy_swing_pre_emph_cfg,
|
||||
.ver_ops = &qcom_edp_phy_ops_v4,
|
||||
};
|
||||
|
||||
|
|
@ -585,7 +702,8 @@ static const struct qcom_edp_phy_cfg sc8280xp_edp_phy_cfg = {
|
|||
.is_edp = true,
|
||||
.aux_cfg = edp_phy_aux_cfg_v4,
|
||||
.vco_div_cfg = edp_phy_vco_div_cfg_v4,
|
||||
.swing_pre_emph_cfg = &edp_phy_swing_pre_emph_cfg,
|
||||
.dp_swing_pre_emph_cfg = &dp_phy_swing_pre_emph_cfg,
|
||||
.edp_swing_pre_emph_cfg = &edp_phy_swing_pre_emph_cfg,
|
||||
.ver_ops = &qcom_edp_phy_ops_v4,
|
||||
};
|
||||
|
||||
|
|
@ -754,6 +872,24 @@ static int qcom_edp_com_configure_pll_v6(const struct qcom_edp *edp)
|
|||
return 0;
|
||||
}
|
||||
|
||||
static int qcom_edp_ldo_config_v6(const struct qcom_edp *edp)
|
||||
{
|
||||
const struct phy_configure_opts_dp *dp_opts = &edp->dp_opts;
|
||||
u32 ldo_config;
|
||||
|
||||
if (!edp->is_edp)
|
||||
ldo_config = 0x0;
|
||||
else if (dp_opts->link_rate <= 2700)
|
||||
ldo_config = 0x51;
|
||||
else
|
||||
ldo_config = 0x91;
|
||||
|
||||
writel(ldo_config, edp->tx0 + TXn_LDO_CONFIG);
|
||||
writel(dp_opts->lanes > 2 ? ldo_config : 0x00, edp->tx1 + TXn_LDO_CONFIG);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const struct phy_ver_ops qcom_edp_phy_ops_v6 = {
|
||||
.com_power_on = qcom_edp_phy_power_on_v6,
|
||||
.com_resetsm_cntrl = qcom_edp_phy_com_resetsm_cntrl_v6,
|
||||
|
|
@ -761,12 +897,14 @@ static const struct phy_ver_ops qcom_edp_phy_ops_v6 = {
|
|||
.com_clk_fwd_cfg = qcom_edp_com_clk_fwd_cfg_v4,
|
||||
.com_configure_pll = qcom_edp_com_configure_pll_v6,
|
||||
.com_configure_ssc = qcom_edp_com_configure_ssc_v6,
|
||||
.com_ldo_config = qcom_edp_ldo_config_v6,
|
||||
};
|
||||
|
||||
static struct qcom_edp_phy_cfg x1e80100_phy_cfg = {
|
||||
.aux_cfg = edp_phy_aux_cfg_v4,
|
||||
.vco_div_cfg = edp_phy_vco_div_cfg_v4,
|
||||
.swing_pre_emph_cfg = &dp_phy_swing_pre_emph_cfg,
|
||||
.dp_swing_pre_emph_cfg = &dp_phy_swing_pre_emph_cfg,
|
||||
.edp_swing_pre_emph_cfg = &edp_phy_swing_pre_emph_cfg,
|
||||
.ver_ops = &qcom_edp_phy_ops_v6,
|
||||
};
|
||||
|
||||
|
|
@ -940,12 +1078,14 @@ static const struct phy_ver_ops qcom_edp_phy_ops_v8 = {
|
|||
.com_clk_fwd_cfg = qcom_edp_com_clk_fwd_cfg_v8,
|
||||
.com_configure_pll = qcom_edp_com_configure_pll_v8,
|
||||
.com_configure_ssc = qcom_edp_com_configure_ssc_v8,
|
||||
.com_ldo_config = qcom_edp_ldo_config_v6,
|
||||
};
|
||||
|
||||
static struct qcom_edp_phy_cfg glymur_phy_cfg = {
|
||||
.aux_cfg = edp_phy_aux_cfg_v8,
|
||||
.vco_div_cfg = edp_phy_vco_div_cfg_v8,
|
||||
.swing_pre_emph_cfg = &edp_phy_swing_pre_emph_cfg_v5,
|
||||
.dp_swing_pre_emph_cfg = &dp_phy_swing_pre_emph_cfg_v8,
|
||||
.edp_swing_pre_emph_cfg = &edp_phy_swing_pre_emph_cfg,
|
||||
.ver_ops = &qcom_edp_phy_ops_v8,
|
||||
};
|
||||
|
||||
|
|
@ -954,7 +1094,6 @@ static int qcom_edp_phy_power_on(struct phy *phy)
|
|||
const struct qcom_edp *edp = phy_get_drvdata(phy);
|
||||
u32 bias0_en, drvr0_en, bias1_en, drvr1_en;
|
||||
unsigned long pixel_freq;
|
||||
u8 ldo_config = 0x0;
|
||||
int ret;
|
||||
u32 val;
|
||||
u8 cfg1;
|
||||
|
|
@ -963,11 +1102,10 @@ static int qcom_edp_phy_power_on(struct phy *phy)
|
|||
if (ret)
|
||||
return ret;
|
||||
|
||||
if (edp->cfg->swing_pre_emph_cfg && !edp->is_edp)
|
||||
ldo_config = 0x1;
|
||||
ret = edp->cfg->ver_ops->com_ldo_config(edp);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
writel(ldo_config, edp->tx0 + TXn_LDO_CONFIG);
|
||||
writel(ldo_config, edp->tx1 + TXn_LDO_CONFIG);
|
||||
writel(0x00, edp->tx0 + TXn_LANE_MODE_1);
|
||||
writel(0x00, edp->tx1 + TXn_LANE_MODE_1);
|
||||
|
||||
|
|
@ -1347,7 +1485,7 @@ static const struct of_device_id qcom_edp_phy_match_table[] = {
|
|||
{ .compatible = "qcom,glymur-dp-phy", .data = &glymur_phy_cfg, },
|
||||
{ .compatible = "qcom,sa8775p-edp-phy", .data = &sa8775p_dp_phy_cfg, },
|
||||
{ .compatible = "qcom,sc7280-edp-phy", .data = &sc7280_dp_phy_cfg, },
|
||||
{ .compatible = "qcom,sc8180x-edp-phy", .data = &sc7280_dp_phy_cfg, },
|
||||
{ .compatible = "qcom,sc8180x-edp-phy", .data = &sc8180x_dp_phy_cfg, },
|
||||
{ .compatible = "qcom,sc8280xp-dp-phy", .data = &sc8280xp_dp_phy_cfg, },
|
||||
{ .compatible = "qcom,sc8280xp-edp-phy", .data = &sc8280xp_edp_phy_cfg, },
|
||||
{ .compatible = "qcom,x1e80100-dp-phy", .data = &x1e80100_phy_cfg, },
|
||||
|
|
|
|||
|
|
@ -1112,6 +1112,7 @@ static const struct qmp_phy_init_tbl sm8750_ufsphy_pcs[] = {
|
|||
QMP_PHY_INIT_CFG(QPHY_V6_PCS_UFS_MULTI_LANE_CTRL1, 0x02),
|
||||
QMP_PHY_INIT_CFG(QPHY_V6_PCS_UFS_TX_MID_TERM_CTRL1, 0x43),
|
||||
QMP_PHY_INIT_CFG(QPHY_V6_PCS_UFS_PCS_CTRL1, 0x40),
|
||||
QMP_PHY_INIT_CFG(QPHY_V6_PCS_UFS_PLL_CNTL, 0x33),
|
||||
QMP_PHY_INIT_CFG(QPHY_V6_PCS_UFS_TX_LARGE_AMP_DRV_LVL, 0x0f),
|
||||
QMP_PHY_INIT_CFG(QPHY_V6_PCS_UFS_RX_SIGDET_CTRL2, 0x68),
|
||||
QMP_PHY_INIT_CFG(QPHY_V6_PCS_UFS_TX_POST_EMP_LVL_S4, 0x0e),
|
||||
|
|
|
|||
|
|
@ -794,7 +794,7 @@ static int qmp_v2_configure_dp_swing(struct qmp_usbc *qmp)
|
|||
p_level = max(p_level, dp_opts->pre[i]);
|
||||
}
|
||||
|
||||
if (v_level > 4 || p_level > 4) {
|
||||
if (v_level >= 4 || p_level >= 4) {
|
||||
dev_err(qmp->dev, "Invalid v(%d) | p(%d) level)\n",
|
||||
v_level, p_level);
|
||||
return -EINVAL;
|
||||
|
|
|
|||
|
|
@ -1958,13 +1958,14 @@ const struct exynos5_usbdrd_phy_tuning exynos7870_tunes_utmi_postinit[] = {
|
|||
PHYPARAM0_TXPREEMPAMPTUNE | PHYPARAM0_TXHSXVTUNE |
|
||||
PHYPARAM0_TXFSLSTUNE | PHYPARAM0_SQRXTUNE |
|
||||
PHYPARAM0_OTGTUNE | PHYPARAM0_COMPDISTUNE),
|
||||
(FIELD_PREP_CONST(PHYPARAM0_TXVREFTUNE, 14) |
|
||||
(FIELD_PREP_CONST(PHYPARAM0_TXVREFTUNE, 3) |
|
||||
FIELD_PREP_CONST(PHYPARAM0_TXRISETUNE, 1) |
|
||||
FIELD_PREP_CONST(PHYPARAM0_TXRESTUNE, 3) |
|
||||
FIELD_PREP_CONST(PHYPARAM0_TXRESTUNE, 2) |
|
||||
FIELD_PREP_CONST(PHYPARAM0_TXPREEMPPULSETUNE, 0) |
|
||||
FIELD_PREP_CONST(PHYPARAM0_TXPREEMPAMPTUNE, 0) |
|
||||
FIELD_PREP_CONST(PHYPARAM0_TXHSXVTUNE, 0) |
|
||||
FIELD_PREP_CONST(PHYPARAM0_TXFSLSTUNE, 3) |
|
||||
FIELD_PREP_CONST(PHYPARAM0_SQRXTUNE, 6) |
|
||||
FIELD_PREP_CONST(PHYPARAM0_SQRXTUNE, 5) |
|
||||
FIELD_PREP_CONST(PHYPARAM0_OTGTUNE, 2) |
|
||||
FIELD_PREP_CONST(PHYPARAM0_COMPDISTUNE, 3))),
|
||||
PHY_TUNING_ENTRY_LAST
|
||||
|
|
|
|||
|
|
@ -97,7 +97,6 @@ static int spacemit_usb2phy_init(struct phy *phy)
|
|||
ret = clk_enable(sphy->clk);
|
||||
if (ret) {
|
||||
dev_err(&phy->dev, "failed to enable clock\n");
|
||||
clk_disable(sphy->clk);
|
||||
return ret;
|
||||
}
|
||||
|
||||
|
|
|
|||
|
|
@ -20,8 +20,8 @@
|
|||
/* FUSE USB_CALIB registers */
|
||||
#define HS_CURR_LEVEL_PADX_SHIFT(x) ((x) ? (11 + (x - 1) * 6) : 0)
|
||||
#define HS_CURR_LEVEL_PAD_MASK 0x3f
|
||||
#define HS_TERM_RANGE_ADJ_SHIFT 7
|
||||
#define HS_TERM_RANGE_ADJ_MASK 0xf
|
||||
#define HS_TERM_RANGE_ADJ_PADX_SHIFT(x) ((x) ? (5 + (x - 1) * 4) : 7)
|
||||
#define HS_TERM_RANGE_ADJ_PAD_MASK 0xf
|
||||
#define HS_SQUELCH_SHIFT 29
|
||||
#define HS_SQUELCH_MASK 0x7
|
||||
|
||||
|
|
@ -253,7 +253,7 @@
|
|||
struct tegra_xusb_fuse_calibration {
|
||||
u32 *hs_curr_level;
|
||||
u32 hs_squelch;
|
||||
u32 hs_term_range_adj;
|
||||
u32 *hs_term_range_adj;
|
||||
u32 rpd_ctrl;
|
||||
};
|
||||
|
||||
|
|
@ -930,7 +930,7 @@ static int tegra186_utmi_phy_power_on(struct phy *phy)
|
|||
|
||||
value = padctl_readl(padctl, XUSB_PADCTL_USB2_OTG_PADX_CTL1(index));
|
||||
value &= ~TERM_RANGE_ADJ(~0);
|
||||
value |= TERM_RANGE_ADJ(priv->calib.hs_term_range_adj);
|
||||
value |= TERM_RANGE_ADJ(priv->calib.hs_term_range_adj[index]);
|
||||
value &= ~RPD_CTRL(~0);
|
||||
value |= RPD_CTRL(priv->calib.rpd_ctrl);
|
||||
padctl_writel(padctl, value, XUSB_PADCTL_USB2_OTG_PADX_CTL1(index));
|
||||
|
|
@ -1464,17 +1464,23 @@ static const char * const tegra186_usb3_functions[] = {
|
|||
static int
|
||||
tegra186_xusb_read_fuse_calibration(struct tegra186_xusb_padctl *padctl)
|
||||
{
|
||||
const struct tegra_xusb_padctl_soc *soc = padctl->base.soc;
|
||||
struct device *dev = padctl->base.dev;
|
||||
unsigned int i, count;
|
||||
u32 value, *level;
|
||||
u32 *hs_term_range_adj;
|
||||
int err;
|
||||
|
||||
count = padctl->base.soc->ports.usb2.count;
|
||||
count = soc->ports.usb2.count;
|
||||
|
||||
level = devm_kcalloc(dev, count, sizeof(u32), GFP_KERNEL);
|
||||
if (!level)
|
||||
return -ENOMEM;
|
||||
|
||||
hs_term_range_adj = devm_kcalloc(dev, count, sizeof(u32), GFP_KERNEL);
|
||||
if (!hs_term_range_adj)
|
||||
return -ENOMEM;
|
||||
|
||||
err = tegra_fuse_readl(TEGRA_FUSE_SKU_CALIB_0, &value);
|
||||
if (err)
|
||||
return dev_err_probe(dev, err,
|
||||
|
|
@ -1490,8 +1496,8 @@ tegra186_xusb_read_fuse_calibration(struct tegra186_xusb_padctl *padctl)
|
|||
|
||||
padctl->calib.hs_squelch = (value >> HS_SQUELCH_SHIFT) &
|
||||
HS_SQUELCH_MASK;
|
||||
padctl->calib.hs_term_range_adj = (value >> HS_TERM_RANGE_ADJ_SHIFT) &
|
||||
HS_TERM_RANGE_ADJ_MASK;
|
||||
hs_term_range_adj[0] = (value >> HS_TERM_RANGE_ADJ_PADX_SHIFT(0)) &
|
||||
HS_TERM_RANGE_ADJ_PAD_MASK;
|
||||
|
||||
err = tegra_fuse_readl(TEGRA_FUSE_USB_CALIB_EXT_0, &value);
|
||||
if (err) {
|
||||
|
|
@ -1503,6 +1509,17 @@ tegra186_xusb_read_fuse_calibration(struct tegra186_xusb_padctl *padctl)
|
|||
|
||||
padctl->calib.rpd_ctrl = (value >> RPD_CTRL_SHIFT) & RPD_CTRL_MASK;
|
||||
|
||||
for (i = 1; i < count; i++) {
|
||||
if (soc->has_per_pad_term)
|
||||
hs_term_range_adj[i] =
|
||||
(value >> HS_TERM_RANGE_ADJ_PADX_SHIFT(i)) &
|
||||
HS_TERM_RANGE_ADJ_PAD_MASK;
|
||||
else
|
||||
hs_term_range_adj[i] = hs_term_range_adj[0];
|
||||
}
|
||||
|
||||
padctl->calib.hs_term_range_adj = hs_term_range_adj;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
|
@ -1708,6 +1725,7 @@ const struct tegra_xusb_padctl_soc tegra194_xusb_padctl_soc = {
|
|||
.num_supplies = ARRAY_SIZE(tegra194_xusb_padctl_supply_names),
|
||||
.supports_gen2 = true,
|
||||
.poll_trk_completed = true,
|
||||
.has_per_pad_term = true,
|
||||
};
|
||||
EXPORT_SYMBOL_GPL(tegra194_xusb_padctl_soc);
|
||||
|
||||
|
|
@ -1732,6 +1750,7 @@ const struct tegra_xusb_padctl_soc tegra234_xusb_padctl_soc = {
|
|||
.trk_hw_mode = false,
|
||||
.trk_update_on_idle = true,
|
||||
.supports_lp_cfg_en = true,
|
||||
.has_per_pad_term = true,
|
||||
};
|
||||
EXPORT_SYMBOL_GPL(tegra234_xusb_padctl_soc);
|
||||
#endif
|
||||
|
|
|
|||
|
|
@ -435,6 +435,7 @@ struct tegra_xusb_padctl_soc {
|
|||
bool trk_hw_mode;
|
||||
bool trk_update_on_idle;
|
||||
bool supports_lp_cfg_en;
|
||||
bool has_per_pad_term;
|
||||
};
|
||||
|
||||
struct tegra_xusb_padctl {
|
||||
|
|
|
|||
Loading…
Reference in New Issue