Merge branches 'clk-range', 'clk-uniphier', 'clk-apple' and 'clk-qcom' into clk-next
[linux-2.6-microblaze.git] / drivers / clk / imx / clk-pll14xx.c
index b464e11..1d0f79e 100644 (file)
@@ -3,6 +3,8 @@
  * Copyright 2017-2018 NXP.
  */
 
+#define pr_fmt(fmt) "pll14xx: " fmt
+
 #include <linux/bitfield.h>
 #include <linux/bits.h>
 #include <linux/clk-provider.h>
@@ -27,6 +29,8 @@
 #define PDIV_MASK      GENMASK(9, 4)
 #define SDIV_MASK      GENMASK(2, 0)
 #define KDIV_MASK      GENMASK(15, 0)
+#define KDIV_MIN       SHRT_MIN
+#define KDIV_MAX       SHRT_MAX
 
 #define LOCK_TIMEOUT_US                10000
 
@@ -111,20 +115,130 @@ static long pll14xx_calc_rate(struct clk_pll14xx *pll, int mdiv, int pdiv,
        return fvco;
 }
 
-static long clk_pll14xx_round_rate(struct clk_hw *hw, unsigned long rate,
+static long pll1443x_calc_kdiv(int mdiv, int pdiv, int sdiv,
+               unsigned long rate, unsigned long prate)
+{
+       long kdiv;
+
+       /* calc kdiv = round(rate * pdiv * 65536 * 2^sdiv / prate) - (mdiv * 65536) */
+       kdiv = ((rate * ((pdiv * 65536) << sdiv) + prate / 2) / prate) - (mdiv * 65536);
+
+       return clamp_t(short, kdiv, KDIV_MIN, KDIV_MAX);
+}
+
+static void imx_pll14xx_calc_settings(struct clk_pll14xx *pll, unsigned long rate,
+                                     unsigned long prate, struct imx_pll14xx_rate_table *t)
+{
+       u32 pll_div_ctl0, pll_div_ctl1;
+       int mdiv, pdiv, sdiv, kdiv;
+       long fvco, rate_min, rate_max, dist, best = LONG_MAX;
+       const struct imx_pll14xx_rate_table *tt;
+
+       /*
+        * Fractional PLL constrains:
+        *
+        * a) 6MHz <= prate <= 25MHz
+        * b) 1 <= p <= 63 (1 <= p <= 4 prate = 24MHz)
+        * c) 64 <= m <= 1023
+        * d) 0 <= s <= 6
+        * e) -32768 <= k <= 32767
+        *
+        * fvco = (m * 65536 + k) * prate / (p * 65536)
+        */
+
+       /* First try if we can get the desired rate from one of the static entries */
+       tt = imx_get_pll_settings(pll, rate);
+       if (tt) {
+               pr_debug("%s: in=%ld, want=%ld, Using PLL setting from table\n",
+                        clk_hw_get_name(&pll->hw), prate, rate);
+               t->rate = tt->rate;
+               t->mdiv = tt->mdiv;
+               t->pdiv = tt->pdiv;
+               t->sdiv = tt->sdiv;
+               t->kdiv = tt->kdiv;
+               return;
+       }
+
+       pll_div_ctl0 = readl_relaxed(pll->base + DIV_CTL0);
+       mdiv = FIELD_GET(MDIV_MASK, pll_div_ctl0);
+       pdiv = FIELD_GET(PDIV_MASK, pll_div_ctl0);
+       sdiv = FIELD_GET(SDIV_MASK, pll_div_ctl0);
+       pll_div_ctl1 = readl_relaxed(pll->base + DIV_CTL1);
+
+       /* Then see if we can get the desired rate by only adjusting kdiv (glitch free) */
+       rate_min = pll14xx_calc_rate(pll, mdiv, pdiv, sdiv, KDIV_MIN, prate);
+       rate_max = pll14xx_calc_rate(pll, mdiv, pdiv, sdiv, KDIV_MAX, prate);
+
+       if (rate >= rate_min && rate <= rate_max) {
+               kdiv = pll1443x_calc_kdiv(mdiv, pdiv, sdiv, rate, prate);
+               pr_debug("%s: in=%ld, want=%ld Only adjust kdiv %ld -> %d\n",
+                        clk_hw_get_name(&pll->hw), prate, rate,
+                        FIELD_GET(KDIV_MASK, pll_div_ctl1), kdiv);
+               fvco = pll14xx_calc_rate(pll, mdiv, pdiv, sdiv, kdiv, prate);
+               t->rate = (unsigned int)fvco;
+               t->mdiv = mdiv;
+               t->pdiv = pdiv;
+               t->sdiv = sdiv;
+               t->kdiv = kdiv;
+               return;
+       }
+
+       /* Finally calculate best values */
+       for (pdiv = 1; pdiv <= 7; pdiv++) {
+               for (sdiv = 0; sdiv <= 6; sdiv++) {
+                       /* calc mdiv = round(rate * pdiv * 2^sdiv) / prate) */
+                       mdiv = DIV_ROUND_CLOSEST(rate * (pdiv << sdiv), prate);
+                       mdiv = clamp(mdiv, 64, 1023);
+
+                       kdiv = pll1443x_calc_kdiv(mdiv, pdiv, sdiv, rate, prate);
+                       fvco = pll14xx_calc_rate(pll, mdiv, pdiv, sdiv, kdiv, prate);
+
+                       /* best match */
+                       dist = abs((long)rate - (long)fvco);
+                       if (dist < best) {
+                               best = dist;
+                               t->rate = (unsigned int)fvco;
+                               t->mdiv = mdiv;
+                               t->pdiv = pdiv;
+                               t->sdiv = sdiv;
+                               t->kdiv = kdiv;
+
+                               if (!dist)
+                                       goto found;
+                       }
+               }
+       }
+found:
+       pr_debug("%s: in=%ld, want=%ld got=%d (pdiv=%d sdiv=%d mdiv=%d kdiv=%d)\n",
+                clk_hw_get_name(&pll->hw), prate, rate, t->rate, t->pdiv, t->sdiv,
+                t->mdiv, t->kdiv);
+}
+
+static long clk_pll1416x_round_rate(struct clk_hw *hw, unsigned long rate,
                        unsigned long *prate)
 {
        struct clk_pll14xx *pll = to_clk_pll14xx(hw);
        const struct imx_pll14xx_rate_table *rate_table = pll->rate_table;
        int i;
 
-       /* Assumming rate_table is in descending order */
+       /* Assuming rate_table is in descending order */
        for (i = 0; i < pll->rate_count; i++)
                if (rate >= rate_table[i].rate)
                        return rate_table[i].rate;
 
        /* return minimum supported value */
-       return rate_table[i - 1].rate;
+       return rate_table[pll->rate_count - 1].rate;
+}
+
+static long clk_pll1443x_round_rate(struct clk_hw *hw, unsigned long rate,
+                       unsigned long *prate)
+{
+       struct clk_pll14xx *pll = to_clk_pll14xx(hw);
+       struct imx_pll14xx_rate_table t;
+
+       imx_pll14xx_calc_settings(pll, rate, *prate, &t);
+
+       return t.rate;
 }
 
 static unsigned long clk_pll14xx_recalc_rate(struct clk_hw *hw,
@@ -177,8 +291,8 @@ static int clk_pll1416x_set_rate(struct clk_hw *hw, unsigned long drate,
 
        rate = imx_get_pll_settings(pll, drate);
        if (!rate) {
-               pr_err("%s: Invalid rate : %lu for pll clk %s\n", __func__,
-                      drate, clk_hw_get_name(hw));
+               pr_err("Invalid rate %lu for pll clk %s\n", drate,
+                      clk_hw_get_name(hw));
                return -EINVAL;
        }
 
@@ -237,25 +351,21 @@ static int clk_pll1443x_set_rate(struct clk_hw *hw, unsigned long drate,
                                 unsigned long prate)
 {
        struct clk_pll14xx *pll = to_clk_pll14xx(hw);
-       const struct imx_pll14xx_rate_table *rate;
+       struct imx_pll14xx_rate_table rate;
        u32 gnrl_ctl, div_ctl0;
        int ret;
 
-       rate = imx_get_pll_settings(pll, drate);
-       if (!rate) {
-               pr_err("%s: Invalid rate : %lu for pll clk %s\n", __func__,
-                       drate, clk_hw_get_name(hw));
-               return -EINVAL;
-       }
+       imx_pll14xx_calc_settings(pll, drate, prate, &rate);
 
        div_ctl0 = readl_relaxed(pll->base + DIV_CTL0);
 
-       if (!clk_pll14xx_mp_change(rate, div_ctl0)) {
+       if (!clk_pll14xx_mp_change(&rate, div_ctl0)) {
+               /* only sdiv and/or kdiv changed - no need to RESET PLL */
                div_ctl0 &= ~SDIV_MASK;
-               div_ctl0 |= FIELD_PREP(SDIV_MASK, rate->sdiv);
+               div_ctl0 |= FIELD_PREP(SDIV_MASK, rate.sdiv);
                writel_relaxed(div_ctl0, pll->base + DIV_CTL0);
 
-               writel_relaxed(FIELD_PREP(KDIV_MASK, rate->kdiv),
+               writel_relaxed(FIELD_PREP(KDIV_MASK, rate.kdiv),
                               pll->base + DIV_CTL1);
 
                return 0;
@@ -270,11 +380,12 @@ static int clk_pll1443x_set_rate(struct clk_hw *hw, unsigned long drate,
        gnrl_ctl |= BYPASS_MASK;
        writel_relaxed(gnrl_ctl, pll->base + GNRL_CTL);
 
-       div_ctl0 = FIELD_PREP(MDIV_MASK, rate->mdiv) |
-                  FIELD_PREP(PDIV_MASK, rate->pdiv) |
-                  FIELD_PREP(SDIV_MASK, rate->sdiv);
+       div_ctl0 = FIELD_PREP(MDIV_MASK, rate.mdiv) |
+                  FIELD_PREP(PDIV_MASK, rate.pdiv) |
+                  FIELD_PREP(SDIV_MASK, rate.sdiv);
        writel_relaxed(div_ctl0, pll->base + DIV_CTL0);
-       writel_relaxed(FIELD_PREP(KDIV_MASK, rate->kdiv), pll->base + DIV_CTL1);
+
+       writel_relaxed(FIELD_PREP(KDIV_MASK, rate.kdiv), pll->base + DIV_CTL1);
 
        /*
         * According to SPEC, t3 - t2 need to be greater than
@@ -357,7 +468,7 @@ static const struct clk_ops clk_pll1416x_ops = {
        .unprepare      = clk_pll14xx_unprepare,
        .is_prepared    = clk_pll14xx_is_prepared,
        .recalc_rate    = clk_pll14xx_recalc_rate,
-       .round_rate     = clk_pll14xx_round_rate,
+       .round_rate     = clk_pll1416x_round_rate,
        .set_rate       = clk_pll1416x_set_rate,
 };
 
@@ -370,7 +481,7 @@ static const struct clk_ops clk_pll1443x_ops = {
        .unprepare      = clk_pll14xx_unprepare,
        .is_prepared    = clk_pll14xx_is_prepared,
        .recalc_rate    = clk_pll14xx_recalc_rate,
-       .round_rate     = clk_pll14xx_round_rate,
+       .round_rate     = clk_pll1443x_round_rate,
        .set_rate       = clk_pll1443x_set_rate,
 };
 
@@ -404,8 +515,7 @@ struct clk_hw *imx_dev_clk_hw_pll14xx(struct device *dev, const char *name,
                init.ops = &clk_pll1443x_ops;
                break;
        default:
-               pr_err("%s: Unknown pll type for pll clk %s\n",
-                      __func__, name);
+               pr_err("Unknown pll type for pll clk %s\n", name);
                kfree(pll);
                return ERR_PTR(-EINVAL);
        }
@@ -424,8 +534,7 @@ struct clk_hw *imx_dev_clk_hw_pll14xx(struct device *dev, const char *name,
 
        ret = clk_hw_register(dev, hw);
        if (ret) {
-               pr_err("%s: failed to register pll %s %d\n",
-                       __func__, name, ret);
+               pr_err("failed to register pll %s %d\n", name, ret);
                kfree(pll);
                return ERR_PTR(ret);
        }