phy: rockchip-emmc: enable ctrl-base before waiting pll
authorShawn Lin <shawn.lin@rock-chips.com>
Mon, 9 May 2016 08:11:25 +0000 (16:11 +0800)
committerHuang, Tao <huangtao@rock-chips.com>
Fri, 13 May 2016 02:39:40 +0000 (10:39 +0800)
Need to control phy's digital block before enabling pll and
waiting for it into locked state.

Change-Id: I04037f5496fd5c1ef4e24853eb32b43ce326ff01
Signed-off-by: Shawn Lin <shawn.lin@rock-chips.com>
drivers/phy/phy-rockchip-emmc.c

index 3a2bbb7032539c9c207f7320f41589fd612da30b..b0763d38dae8efefbbc5713bc0b38999a66af21f 100644 (file)
@@ -15,6 +15,7 @@
  */
 
 #include <linux/delay.h>
+#include <linux/io.h>
 #include <linux/mfd/syscon.h>
 #include <linux/module.h>
 #include <linux/of.h>
 #define GRF_EMMCPHY_CON5               0x14
 #define GRF_EMMCPHY_CON6               0x18
 #define GRF_EMMCPHY_STATUS             0x20
+#define CTRL_OFFSET                    0x2c
 
+#define CTRL_INTER_CLKEN               0x1
+#define CTRL_INTER_CLKRDY              0x1
+#define CTRL_INTER_CLKOUT              0x1
 #define PHYCTRL_PDB_MASK               0x1
 #define PHYCTRL_PDB_SHIFT              0x0
 #define PHYCTRL_PDB_PWR_ON             0x1
@@ -78,6 +83,7 @@
 struct rockchip_emmc_phy {
        unsigned int    reg_offset;
        struct regmap   *reg_base;
+       void __iomem *ctrl_base;
        u32     freq_sel;
        u32     dr_sel;
        u32     opdelay;
@@ -88,6 +94,8 @@ static int rockchip_emmc_phy_power(struct rockchip_emmc_phy *rk_phy,
 {
        unsigned int caldone;
        unsigned int dllrdy;
+       u16 ctrl_val;
+       unsigned long timeout;
 
        /*
         * Keep phyctrl_pdb and phyctrl_endll low to allow
@@ -108,6 +116,22 @@ static int rockchip_emmc_phy_power(struct rockchip_emmc_phy *rk_phy,
        if (on_off == PHYCTRL_PDB_PWR_OFF)
                return 0;
 
+       ctrl_val = readw(rk_phy->ctrl_base + CTRL_OFFSET);
+       ctrl_val |= CTRL_INTER_CLKEN;
+       writew(ctrl_val, rk_phy->ctrl_base + CTRL_OFFSET);
+       /* Wait max 20 ms */
+       while (!((ctrl_val = readw(rk_phy->ctrl_base + CTRL_OFFSET))
+               & CTRL_INTER_CLKRDY)) {
+               if (timeout == 0) {
+                       pr_err("rockchip_emmc_phy_power_on: inter_clk not rdy\n");
+                       return -EINVAL;
+               }
+               timeout--;
+               mdelay(1);
+       }
+       ctrl_val |= CTRL_INTER_CLKOUT;
+       writew(ctrl_val, rk_phy->ctrl_base + CTRL_OFFSET);
+
        /*
         * According to the user manual, calpad calibration
         * cycle takes more than 2us without the minimal recommended
@@ -232,6 +256,7 @@ static int rockchip_emmc_phy_probe(struct platform_device *pdev)
        u32 freq_sel;
        u32 dr_sel;
        u32 opdelay;
+       u32 ctrl_base;
 
        grf = syscon_regmap_lookup_by_phandle(dev->of_node, "rockchip,grf");
        if (IS_ERR(grf)) {
@@ -249,6 +274,18 @@ static int rockchip_emmc_phy_probe(struct platform_device *pdev)
                return -EINVAL;
        }
 
+       if (of_property_read_u32(dev->of_node, "ctrl-base", &ctrl_base)) {
+               dev_err(dev, "missing ctrl-base property in node %s\n",
+                       dev->of_node->name);
+               return -EINVAL;
+       }
+
+       rk_phy->ctrl_base = ioremap(ctrl_base, SZ_1K);
+       if (!rk_phy->ctrl_base) {
+               dev_err(dev, "failed to remap ctrl_base!\n");
+               return -ENOMEM;
+       }
+
        rk_phy->freq_sel = 0x0;
        if (!of_property_read_u32(dev->of_node, "freq-sel", &freq_sel)) {
                switch (freq_sel) {
@@ -308,6 +345,7 @@ static int rockchip_emmc_phy_probe(struct platform_device *pdev)
        generic_phy = devm_phy_create(dev, dev->of_node, &ops);
        if (IS_ERR(generic_phy)) {
                dev_err(dev, "failed to create PHY\n");
+               iounmap(rk_phy->ctrl_base);
                return PTR_ERR(generic_phy);
        }