Hi Brian, On 2016/5/13 6:43, Brian Norris wrote: > From: Shawn Lin <shawn.lin at rock-chips.com> > > Signal integrity analysis has suggested we set these values. Do this in > power_on(), so that they get reconfigured after suspend/resume. > > Signed-off-by: Shawn Lin <shawn.lin at rock-chips.com> > Signed-off-by: Brian Norris <briannorris at chromium.org> > --- > drivers/phy/phy-rockchip-emmc.c | 27 +++++++++++++++++++++++++++ > 1 file changed, 27 insertions(+) > > diff --git a/drivers/phy/phy-rockchip-emmc.c b/drivers/phy/phy-rockchip-emmc.c > index 48cbe691a889..5641dede32f6 100644 > --- a/drivers/phy/phy-rockchip-emmc.c > +++ b/drivers/phy/phy-rockchip-emmc.c > @@ -56,6 +56,19 @@ > #define PHYCTRL_DLLRDY_SHIFT 0x5 > #define PHYCTRL_DLLRDY_DONE 0x1 > #define PHYCTRL_DLLRDY_GOING 0x0 > +#define PHYCTRL_FREQSEL_200M 0x0 > +#define PHYCTRL_FREQSEL_50M 0x1 > +#define PHYCTRL_FREQSEL_100M 0x2 > +#define PHYCTRL_FREQSEL_150M 0x3 > +#define PHYCTRL_FREQSEL_MASK 0x3 > +#define PHYCTRL_FREQSEL_SHIFT 0xc > +#define PHYCTRL_DR_MASK 0x7 > +#define PHYCTRL_DR_SHIFT 0x4 > +#define PHYCTRL_DR_50OHM 0x0 > +#define PHYCTRL_DR_33OHM 0x1 > +#define PHYCTRL_DR_66OHM 0x2 > +#define PHYCTRL_DR_100OHM 0x3 > +#define PHYCTRL_DR_40OHM 0x4 > > struct rockchip_emmc_phy { > unsigned int reg_offset; > @@ -154,6 +167,20 @@ static int rockchip_emmc_phy_power_on(struct phy *phy) > struct rockchip_emmc_phy *rk_phy = phy_get_drvdata(phy); > int ret = 0; > > + /* DLL operation: 170 to 200 MHz */ What is 170 here? Should we expose them to dt instead of hardcoding them? Per the commit msg, signal may vary from board to board, so I guess 50ohm may not always be the best selection? Another thing I need to elaborate more here is that emmc phy only supports 50/100/150/200Mhz. Presumably people love to use the highest speed mode with its upper limiting frequency, but in case of some special requirement or bad board design, they want to add max-frequency in dts for emmc controller. In this case PHYCTRL_FREQSEL_XXXM should meet the actual max-frequency, take 100M for example, otherwise emmc_phy's tuning block will use 200M to do some calculation but it certainly should be 100M. This leads emmc_phy to choose the wrong phase. Finally maybe you will see triggering re-tune easily or some worse case of even tuning failure when probing card. > + regmap_write(rk_phy->reg_base, > + rk_phy->reg_offset + GRF_EMMCPHY_CON0, > + HIWORD_UPDATE(PHYCTRL_FREQSEL_200M, > + PHYCTRL_FREQSEL_MASK, > + PHYCTRL_FREQSEL_SHIFT)); > + > + /* Drive impedance: 50 Ohm */ > + regmap_write(rk_phy->reg_base, > + rk_phy->reg_offset + GRF_EMMCPHY_CON6, > + HIWORD_UPDATE(PHYCTRL_DR_50OHM, > + PHYCTRL_DR_MASK, > + PHYCTRL_DR_SHIFT)); > + > /* Power up emmc phy analog blocks */ > ret = rockchip_emmc_phy_power(rk_phy, PHYCTRL_PDB_PWR_ON); > if (ret) > -- Best Regards Shawn Lin