Merge branch 'for-4.2/ti-clk-move' of https://github.com/t-kristo/linux-pm into clk...
authorStephen Boyd <sboyd@codeaurora.org>
Tue, 28 Jul 2015 18:58:26 +0000 (11:58 -0700)
committerStephen Boyd <sboyd@codeaurora.org>
Tue, 28 Jul 2015 18:58:26 +0000 (11:58 -0700)
From Tero Kristo:
  "This pull request contains the TI clock driver set to move the
   clock implementations under clock driver. Some small portions of
   the clock driver code still remain under mach-omap2 after this,
   it should be decided whether this code is now obsolete and should
   be deleted or should someone try to fix it."

Slight merge conflicts with determine_rate prototype changes.

1  2 
arch/arm/mach-omap2/Makefile
drivers/clk/ti/clk-7xx.c
drivers/clk/ti/clk.c
drivers/clk/ti/clock.h
drivers/clk/ti/clockdomain.c
drivers/clk/ti/dpll3xxx.c
drivers/clk/ti/dpll44xx.c

Simple merge
Simple merge
Simple merge
index 404158d2d7f8800757cb95255525aa4fe87ee501,d4d232fd89bc12aaf29bf37f4dba4f738a60e655..d8aafd3330583789b8be93988dfb7757213a6e89
@@@ -169,4 -198,88 +198,80 @@@ void ti_clk_patch_legacy_clks(struct ti
  struct clk *ti_clk_register_clk(struct ti_clk *setup);
  int ti_clk_register_legacy_clks(struct ti_clk_alias *clks);
  
 -long omap3_noncore_dpll_determine_rate(struct clk_hw *hw,
 -                                     unsigned long rate,
 -                                     unsigned long min_rate,
 -                                     unsigned long max_rate,
 -                                     unsigned long *best_parent_rate,
 -                                     struct clk_hw **best_parent_clk);
+ void __iomem *ti_clk_get_reg_addr(struct device_node *node, int index);
+ void ti_dt_clocks_register(struct ti_dt_clk *oclks);
+ int ti_clk_retry_init(struct device_node *node, struct clk_hw *hw,
+                     ti_of_clk_init_cb_t func);
+ int ti_clk_add_component(struct device_node *node, struct clk_hw *hw, int type);
+ void omap2_init_clk_hw_omap_clocks(struct clk *clk);
+ int of_ti_clk_autoidle_setup(struct device_node *node);
+ void omap2_clk_enable_init_clocks(const char **clk_names, u8 num_clocks);
+ extern const struct clk_hw_omap_ops clkhwops_omap3_dpll;
+ extern const struct clk_hw_omap_ops clkhwops_omap4_dpllmx;
+ extern const struct clk_hw_omap_ops clkhwops_wait;
+ extern const struct clk_hw_omap_ops clkhwops_iclk;
+ extern const struct clk_hw_omap_ops clkhwops_iclk_wait;
+ extern const struct clk_hw_omap_ops clkhwops_omap2430_i2chs_wait;
+ extern const struct clk_hw_omap_ops clkhwops_omap3430es2_dss_usbhost_wait;
+ extern const struct clk_hw_omap_ops clkhwops_omap3430es2_iclk_hsotgusb_wait;
+ extern const struct clk_hw_omap_ops clkhwops_omap3430es2_iclk_dss_usbhost_wait;
+ extern const struct clk_hw_omap_ops clkhwops_omap3430es2_iclk_ssi_wait;
+ extern const struct clk_hw_omap_ops clkhwops_am35xx_ipss_module_wait;
+ extern const struct clk_hw_omap_ops clkhwops_am35xx_ipss_wait;
+ extern const struct clk_ops ti_clk_divider_ops;
+ extern const struct clk_ops ti_clk_mux_ops;
+ int omap2_clkops_enable_clkdm(struct clk_hw *hw);
+ void omap2_clkops_disable_clkdm(struct clk_hw *hw);
+ int omap2_dflt_clk_enable(struct clk_hw *hw);
+ void omap2_dflt_clk_disable(struct clk_hw *hw);
+ int omap2_dflt_clk_is_enabled(struct clk_hw *hw);
+ void omap2_clk_dflt_find_companion(struct clk_hw_omap *clk,
+                                  void __iomem **other_reg,
+                                  u8 *other_bit);
+ void omap2_clk_dflt_find_idlest(struct clk_hw_omap *clk,
+                               void __iomem **idlest_reg,
+                               u8 *idlest_bit, u8 *idlest_val);
+ void omap2_clkt_iclk_allow_idle(struct clk_hw_omap *clk);
+ void omap2_clkt_iclk_deny_idle(struct clk_hw_omap *clk);
+ u8 omap2_init_dpll_parent(struct clk_hw *hw);
+ int omap3_noncore_dpll_enable(struct clk_hw *hw);
+ void omap3_noncore_dpll_disable(struct clk_hw *hw);
+ int omap3_noncore_dpll_set_parent(struct clk_hw *hw, u8 index);
+ int omap3_noncore_dpll_set_rate(struct clk_hw *hw, unsigned long rate,
+                               unsigned long parent_rate);
+ int omap3_noncore_dpll_set_rate_and_parent(struct clk_hw *hw,
+                                          unsigned long rate,
+                                          unsigned long parent_rate,
+                                          u8 index);
 -long omap4_dpll_regm4xen_determine_rate(struct clk_hw *hw,
 -                                      unsigned long rate,
 -                                      unsigned long min_rate,
 -                                      unsigned long max_rate,
 -                                      unsigned long *best_parent_rate,
 -                                      struct clk_hw **best_parent_clk);
++int omap3_noncore_dpll_determine_rate(struct clk_hw *hw,
++                                    struct clk_rate_request *req);
+ long omap2_dpll_round_rate(struct clk_hw *hw, unsigned long target_rate,
+                          unsigned long *parent_rate);
+ unsigned long omap3_clkoutx2_recalc(struct clk_hw *hw,
+                                   unsigned long parent_rate);
+ unsigned long omap3_dpll_recalc(struct clk_hw *hw, unsigned long parent_rate);
+ int omap3_dpll4_set_rate(struct clk_hw *clk, unsigned long rate,
+                        unsigned long parent_rate);
+ int omap3_dpll4_set_rate_and_parent(struct clk_hw *hw, unsigned long rate,
+                                   unsigned long parent_rate, u8 index);
+ void omap3_clk_lock_dpll5(void);
+ unsigned long omap4_dpll_regm4xen_recalc(struct clk_hw *hw,
+                                        unsigned long parent_rate);
+ long omap4_dpll_regm4xen_round_rate(struct clk_hw *hw,
+                                   unsigned long target_rate,
+                                   unsigned long *parent_rate);
++int omap4_dpll_regm4xen_determine_rate(struct clk_hw *hw,
++                                     struct clk_rate_request *req);
+ extern struct ti_clk_ll_ops *ti_clk_ll_ops;
  #endif
Simple merge
index 0000000000000000000000000000000000000000,22d77a3312879e3b62e29fdcdd95fa22f3843200..b0aa87b35f243095eb2ebc4ce0b65e2153b38f5e
mode 000000,100644..100644
--- /dev/null
@@@ -1,0 -1,825 +1,820 @@@
 - * @rate: target rate for the DPLL
 - * @best_parent_rate: pointer for returning best parent rate
 - * @best_parent_clk: pointer for returning best parent clock
+ /*
+  * OMAP3/4 - specific DPLL control functions
+  *
+  * Copyright (C) 2009-2010 Texas Instruments, Inc.
+  * Copyright (C) 2009-2010 Nokia Corporation
+  *
+  * Written by Paul Walmsley
+  * Testing and integration fixes by Jouni Högander
+  *
+  * 36xx support added by Vishwanath BS, Richard Woodruff, and Nishanth
+  * Menon
+  *
+  * Parts of this code are based on code written by
+  * Richard Woodruff, Tony Lindgren, Tuukka Tikkanen, Karthik Dasu
+  *
+  * This program is free software; you can redistribute it and/or modify
+  * it under the terms of the GNU General Public License version 2 as
+  * published by the Free Software Foundation.
+  */
+ #include <linux/kernel.h>
+ #include <linux/device.h>
+ #include <linux/list.h>
+ #include <linux/errno.h>
+ #include <linux/delay.h>
+ #include <linux/clk.h>
+ #include <linux/io.h>
+ #include <linux/bitops.h>
+ #include <linux/clkdev.h>
+ #include <linux/clk/ti.h>
+ #include "clock.h"
+ /* CM_AUTOIDLE_PLL*.AUTO_* bit values */
+ #define DPLL_AUTOIDLE_DISABLE                 0x0
+ #define DPLL_AUTOIDLE_LOW_POWER_STOP          0x1
+ #define MAX_DPLL_WAIT_TRIES           1000000
+ #define OMAP3XXX_EN_DPLL_LOCKED               0x7
+ /* Forward declarations */
+ static u32 omap3_dpll_autoidle_read(struct clk_hw_omap *clk);
+ static void omap3_dpll_deny_idle(struct clk_hw_omap *clk);
+ static void omap3_dpll_allow_idle(struct clk_hw_omap *clk);
+ /* Private functions */
+ /* _omap3_dpll_write_clken - write clken_bits arg to a DPLL's enable bits */
+ static void _omap3_dpll_write_clken(struct clk_hw_omap *clk, u8 clken_bits)
+ {
+       const struct dpll_data *dd;
+       u32 v;
+       dd = clk->dpll_data;
+       v = ti_clk_ll_ops->clk_readl(dd->control_reg);
+       v &= ~dd->enable_mask;
+       v |= clken_bits << __ffs(dd->enable_mask);
+       ti_clk_ll_ops->clk_writel(v, dd->control_reg);
+ }
+ /* _omap3_wait_dpll_status: wait for a DPLL to enter a specific state */
+ static int _omap3_wait_dpll_status(struct clk_hw_omap *clk, u8 state)
+ {
+       const struct dpll_data *dd;
+       int i = 0;
+       int ret = -EINVAL;
+       const char *clk_name;
+       dd = clk->dpll_data;
+       clk_name = __clk_get_name(clk->hw.clk);
+       state <<= __ffs(dd->idlest_mask);
+       while (((ti_clk_ll_ops->clk_readl(dd->idlest_reg) & dd->idlest_mask)
+               != state) && i < MAX_DPLL_WAIT_TRIES) {
+               i++;
+               udelay(1);
+       }
+       if (i == MAX_DPLL_WAIT_TRIES) {
+               pr_err("clock: %s failed transition to '%s'\n",
+                      clk_name, (state) ? "locked" : "bypassed");
+       } else {
+               pr_debug("clock: %s transition to '%s' in %d loops\n",
+                        clk_name, (state) ? "locked" : "bypassed", i);
+               ret = 0;
+       }
+       return ret;
+ }
+ /* From 3430 TRM ES2 4.7.6.2 */
+ static u16 _omap3_dpll_compute_freqsel(struct clk_hw_omap *clk, u8 n)
+ {
+       unsigned long fint;
+       u16 f = 0;
+       fint = __clk_get_rate(clk->dpll_data->clk_ref) / n;
+       pr_debug("clock: fint is %lu\n", fint);
+       if (fint >= 750000 && fint <= 1000000)
+               f = 0x3;
+       else if (fint > 1000000 && fint <= 1250000)
+               f = 0x4;
+       else if (fint > 1250000 && fint <= 1500000)
+               f = 0x5;
+       else if (fint > 1500000 && fint <= 1750000)
+               f = 0x6;
+       else if (fint > 1750000 && fint <= 2100000)
+               f = 0x7;
+       else if (fint > 7500000 && fint <= 10000000)
+               f = 0xB;
+       else if (fint > 10000000 && fint <= 12500000)
+               f = 0xC;
+       else if (fint > 12500000 && fint <= 15000000)
+               f = 0xD;
+       else if (fint > 15000000 && fint <= 17500000)
+               f = 0xE;
+       else if (fint > 17500000 && fint <= 21000000)
+               f = 0xF;
+       else
+               pr_debug("clock: unknown freqsel setting for %d\n", n);
+       return f;
+ }
+ /*
+  * _omap3_noncore_dpll_lock - instruct a DPLL to lock and wait for readiness
+  * @clk: pointer to a DPLL struct clk
+  *
+  * Instructs a non-CORE DPLL to lock.  Waits for the DPLL to report
+  * readiness before returning.  Will save and restore the DPLL's
+  * autoidle state across the enable, per the CDP code.  If the DPLL
+  * locked successfully, return 0; if the DPLL did not lock in the time
+  * allotted, or DPLL3 was passed in, return -EINVAL.
+  */
+ static int _omap3_noncore_dpll_lock(struct clk_hw_omap *clk)
+ {
+       const struct dpll_data *dd;
+       u8 ai;
+       u8 state = 1;
+       int r = 0;
+       pr_debug("clock: locking DPLL %s\n", __clk_get_name(clk->hw.clk));
+       dd = clk->dpll_data;
+       state <<= __ffs(dd->idlest_mask);
+       /* Check if already locked */
+       if ((ti_clk_ll_ops->clk_readl(dd->idlest_reg) & dd->idlest_mask) ==
+           state)
+               goto done;
+       ai = omap3_dpll_autoidle_read(clk);
+       if (ai)
+               omap3_dpll_deny_idle(clk);
+       _omap3_dpll_write_clken(clk, DPLL_LOCKED);
+       r = _omap3_wait_dpll_status(clk, 1);
+       if (ai)
+               omap3_dpll_allow_idle(clk);
+ done:
+       return r;
+ }
+ /*
+  * _omap3_noncore_dpll_bypass - instruct a DPLL to bypass and wait for readiness
+  * @clk: pointer to a DPLL struct clk
+  *
+  * Instructs a non-CORE DPLL to enter low-power bypass mode.  In
+  * bypass mode, the DPLL's rate is set equal to its parent clock's
+  * rate.  Waits for the DPLL to report readiness before returning.
+  * Will save and restore the DPLL's autoidle state across the enable,
+  * per the CDP code.  If the DPLL entered bypass mode successfully,
+  * return 0; if the DPLL did not enter bypass in the time allotted, or
+  * DPLL3 was passed in, or the DPLL does not support low-power bypass,
+  * return -EINVAL.
+  */
+ static int _omap3_noncore_dpll_bypass(struct clk_hw_omap *clk)
+ {
+       int r;
+       u8 ai;
+       if (!(clk->dpll_data->modes & (1 << DPLL_LOW_POWER_BYPASS)))
+               return -EINVAL;
+       pr_debug("clock: configuring DPLL %s for low-power bypass\n",
+                __clk_get_name(clk->hw.clk));
+       ai = omap3_dpll_autoidle_read(clk);
+       _omap3_dpll_write_clken(clk, DPLL_LOW_POWER_BYPASS);
+       r = _omap3_wait_dpll_status(clk, 0);
+       if (ai)
+               omap3_dpll_allow_idle(clk);
+       return r;
+ }
+ /*
+  * _omap3_noncore_dpll_stop - instruct a DPLL to stop
+  * @clk: pointer to a DPLL struct clk
+  *
+  * Instructs a non-CORE DPLL to enter low-power stop. Will save and
+  * restore the DPLL's autoidle state across the stop, per the CDP
+  * code.  If DPLL3 was passed in, or the DPLL does not support
+  * low-power stop, return -EINVAL; otherwise, return 0.
+  */
+ static int _omap3_noncore_dpll_stop(struct clk_hw_omap *clk)
+ {
+       u8 ai;
+       if (!(clk->dpll_data->modes & (1 << DPLL_LOW_POWER_STOP)))
+               return -EINVAL;
+       pr_debug("clock: stopping DPLL %s\n", __clk_get_name(clk->hw.clk));
+       ai = omap3_dpll_autoidle_read(clk);
+       _omap3_dpll_write_clken(clk, DPLL_LOW_POWER_STOP);
+       if (ai)
+               omap3_dpll_allow_idle(clk);
+       return 0;
+ }
+ /**
+  * _lookup_dco - Lookup DCO used by j-type DPLL
+  * @clk: pointer to a DPLL struct clk
+  * @dco: digital control oscillator selector
+  * @m: DPLL multiplier to set
+  * @n: DPLL divider to set
+  *
+  * See 36xx TRM section 3.5.3.3.3.2 "Type B DPLL (Low-Jitter)"
+  *
+  * XXX This code is not needed for 3430/AM35xx; can it be optimized
+  * out in non-multi-OMAP builds for those chips?
+  */
+ static void _lookup_dco(struct clk_hw_omap *clk, u8 *dco, u16 m, u8 n)
+ {
+       unsigned long fint, clkinp; /* watch out for overflow */
+       clkinp = __clk_get_rate(__clk_get_parent(clk->hw.clk));
+       fint = (clkinp / n) * m;
+       if (fint < 1000000000)
+               *dco = 2;
+       else
+               *dco = 4;
+ }
+ /**
+  * _lookup_sddiv - Calculate sigma delta divider for j-type DPLL
+  * @clk: pointer to a DPLL struct clk
+  * @sd_div: target sigma-delta divider
+  * @m: DPLL multiplier to set
+  * @n: DPLL divider to set
+  *
+  * See 36xx TRM section 3.5.3.3.3.2 "Type B DPLL (Low-Jitter)"
+  *
+  * XXX This code is not needed for 3430/AM35xx; can it be optimized
+  * out in non-multi-OMAP builds for those chips?
+  */
+ static void _lookup_sddiv(struct clk_hw_omap *clk, u8 *sd_div, u16 m, u8 n)
+ {
+       unsigned long clkinp, sd; /* watch out for overflow */
+       int mod1, mod2;
+       clkinp = __clk_get_rate(__clk_get_parent(clk->hw.clk));
+       /*
+        * target sigma-delta to near 250MHz
+        * sd = ceil[(m/(n+1)) * (clkinp_MHz / 250)]
+        */
+       clkinp /= 100000; /* shift from MHz to 10*Hz for 38.4 and 19.2 */
+       mod1 = (clkinp * m) % (250 * n);
+       sd = (clkinp * m) / (250 * n);
+       mod2 = sd % 10;
+       sd /= 10;
+       if (mod1 || mod2)
+               sd++;
+       *sd_div = sd;
+ }
+ /*
+  * _omap3_noncore_dpll_program - set non-core DPLL M,N values directly
+  * @clk:      struct clk * of DPLL to set
+  * @freqsel:  FREQSEL value to set
+  *
+  * Program the DPLL with the last M, N values calculated, and wait for
+  * the DPLL to lock. Returns -EINVAL upon error, or 0 upon success.
+  */
+ static int omap3_noncore_dpll_program(struct clk_hw_omap *clk, u16 freqsel)
+ {
+       struct dpll_data *dd = clk->dpll_data;
+       u8 dco, sd_div;
+       u32 v;
+       /* 3430 ES2 TRM: 4.7.6.9 DPLL Programming Sequence */
+       _omap3_noncore_dpll_bypass(clk);
+       /*
+        * Set jitter correction. Jitter correction applicable for OMAP343X
+        * only since freqsel field is no longer present on other devices.
+        */
+       if (ti_clk_get_features()->flags & TI_CLK_DPLL_HAS_FREQSEL) {
+               v = ti_clk_ll_ops->clk_readl(dd->control_reg);
+               v &= ~dd->freqsel_mask;
+               v |= freqsel << __ffs(dd->freqsel_mask);
+               ti_clk_ll_ops->clk_writel(v, dd->control_reg);
+       }
+       /* Set DPLL multiplier, divider */
+       v = ti_clk_ll_ops->clk_readl(dd->mult_div1_reg);
+       /* Handle Duty Cycle Correction */
+       if (dd->dcc_mask) {
+               if (dd->last_rounded_rate >= dd->dcc_rate)
+                       v |= dd->dcc_mask; /* Enable DCC */
+               else
+                       v &= ~dd->dcc_mask; /* Disable DCC */
+       }
+       v &= ~(dd->mult_mask | dd->div1_mask);
+       v |= dd->last_rounded_m << __ffs(dd->mult_mask);
+       v |= (dd->last_rounded_n - 1) << __ffs(dd->div1_mask);
+       /* Configure dco and sd_div for dplls that have these fields */
+       if (dd->dco_mask) {
+               _lookup_dco(clk, &dco, dd->last_rounded_m, dd->last_rounded_n);
+               v &= ~(dd->dco_mask);
+               v |= dco << __ffs(dd->dco_mask);
+       }
+       if (dd->sddiv_mask) {
+               _lookup_sddiv(clk, &sd_div, dd->last_rounded_m,
+                             dd->last_rounded_n);
+               v &= ~(dd->sddiv_mask);
+               v |= sd_div << __ffs(dd->sddiv_mask);
+       }
+       ti_clk_ll_ops->clk_writel(v, dd->mult_div1_reg);
+       /* Set 4X multiplier and low-power mode */
+       if (dd->m4xen_mask || dd->lpmode_mask) {
+               v = ti_clk_ll_ops->clk_readl(dd->control_reg);
+               if (dd->m4xen_mask) {
+                       if (dd->last_rounded_m4xen)
+                               v |= dd->m4xen_mask;
+                       else
+                               v &= ~dd->m4xen_mask;
+               }
+               if (dd->lpmode_mask) {
+                       if (dd->last_rounded_lpmode)
+                               v |= dd->lpmode_mask;
+                       else
+                               v &= ~dd->lpmode_mask;
+               }
+               ti_clk_ll_ops->clk_writel(v, dd->control_reg);
+       }
+       /* We let the clock framework set the other output dividers later */
+       /* REVISIT: Set ramp-up delay? */
+       _omap3_noncore_dpll_lock(clk);
+       return 0;
+ }
+ /* Public functions */
+ /**
+  * omap3_dpll_recalc - recalculate DPLL rate
+  * @clk: DPLL struct clk
+  *
+  * Recalculate and propagate the DPLL rate.
+  */
+ unsigned long omap3_dpll_recalc(struct clk_hw *hw, unsigned long parent_rate)
+ {
+       struct clk_hw_omap *clk = to_clk_hw_omap(hw);
+       return omap2_get_dpll_rate(clk);
+ }
+ /* Non-CORE DPLL (e.g., DPLLs that do not control SDRC) clock functions */
+ /**
+  * omap3_noncore_dpll_enable - instruct a DPLL to enter bypass or lock mode
+  * @clk: pointer to a DPLL struct clk
+  *
+  * Instructs a non-CORE DPLL to enable, e.g., to enter bypass or lock.
+  * The choice of modes depends on the DPLL's programmed rate: if it is
+  * the same as the DPLL's parent clock, it will enter bypass;
+  * otherwise, it will enter lock.  This code will wait for the DPLL to
+  * indicate readiness before returning, unless the DPLL takes too long
+  * to enter the target state.  Intended to be used as the struct clk's
+  * enable function.  If DPLL3 was passed in, or the DPLL does not
+  * support low-power stop, or if the DPLL took too long to enter
+  * bypass or lock, return -EINVAL; otherwise, return 0.
+  */
+ int omap3_noncore_dpll_enable(struct clk_hw *hw)
+ {
+       struct clk_hw_omap *clk = to_clk_hw_omap(hw);
+       int r;
+       struct dpll_data *dd;
+       struct clk_hw *parent;
+       dd = clk->dpll_data;
+       if (!dd)
+               return -EINVAL;
+       if (clk->clkdm) {
+               r = ti_clk_ll_ops->clkdm_clk_enable(clk->clkdm, hw->clk);
+               if (r) {
+                       WARN(1,
+                            "%s: could not enable %s's clockdomain %s: %d\n",
+                            __func__, __clk_get_name(hw->clk),
+                            clk->clkdm_name, r);
+                       return r;
+               }
+       }
+       parent = __clk_get_hw(__clk_get_parent(hw->clk));
+       if (__clk_get_rate(hw->clk) == __clk_get_rate(dd->clk_bypass)) {
+               WARN_ON(parent != __clk_get_hw(dd->clk_bypass));
+               r = _omap3_noncore_dpll_bypass(clk);
+       } else {
+               WARN_ON(parent != __clk_get_hw(dd->clk_ref));
+               r = _omap3_noncore_dpll_lock(clk);
+       }
+       return r;
+ }
+ /**
+  * omap3_noncore_dpll_disable - instruct a DPLL to enter low-power stop
+  * @clk: pointer to a DPLL struct clk
+  *
+  * Instructs a non-CORE DPLL to enter low-power stop.  This function is
+  * intended for use in struct clkops.  No return value.
+  */
+ void omap3_noncore_dpll_disable(struct clk_hw *hw)
+ {
+       struct clk_hw_omap *clk = to_clk_hw_omap(hw);
+       _omap3_noncore_dpll_stop(clk);
+       if (clk->clkdm)
+               ti_clk_ll_ops->clkdm_clk_disable(clk->clkdm, hw->clk);
+ }
+ /* Non-CORE DPLL rate set code */
+ /**
+  * omap3_noncore_dpll_determine_rate - determine rate for a DPLL
+  * @hw: pointer to the clock to determine rate for
 - * Returns a positive clock rate with success, negative error value
 - * in failure.
++ * @req: target rate request
+  *
+  * Determines which DPLL mode to use for reaching a desired target rate.
+  * Checks whether the DPLL shall be in bypass or locked mode, and if
+  * locked, calculates the M,N values for the DPLL via round-rate.
 -long omap3_noncore_dpll_determine_rate(struct clk_hw *hw, unsigned long rate,
 -                                     unsigned long min_rate,
 -                                     unsigned long max_rate,
 -                                     unsigned long *best_parent_rate,
 -                                     struct clk_hw **best_parent_clk)
++ * Returns a 0 on success, negative error value in failure.
+  */
 -      if (!hw || !rate)
++int omap3_noncore_dpll_determine_rate(struct clk_hw *hw,
++                                    struct clk_rate_request *req)
+ {
+       struct clk_hw_omap *clk = to_clk_hw_omap(hw);
+       struct dpll_data *dd;
 -      if (__clk_get_rate(dd->clk_bypass) == rate &&
++      if (!req->rate)
+               return -EINVAL;
+       dd = clk->dpll_data;
+       if (!dd)
+               return -EINVAL;
 -              *best_parent_clk = __clk_get_hw(dd->clk_bypass);
++      if (__clk_get_rate(dd->clk_bypass) == req->rate &&
+           (dd->modes & (1 << DPLL_LOW_POWER_BYPASS))) {
 -              rate = omap2_dpll_round_rate(hw, rate, best_parent_rate);
 -              *best_parent_clk = __clk_get_hw(dd->clk_ref);
++              req->best_parent_hw = __clk_get_hw(dd->clk_bypass);
+       } else {
 -      *best_parent_rate = rate;
++              req->rate = omap2_dpll_round_rate(hw, req->rate,
++                                        &req->best_parent_rate);
++              req->best_parent_hw = __clk_get_hw(dd->clk_ref);
+       }
 -      return rate;
++      req->best_parent_rate = req->rate;
++      return 0;
+ }
+ /**
+  * omap3_noncore_dpll_set_parent - set parent for a DPLL clock
+  * @hw: pointer to the clock to set parent for
+  * @index: parent index to select
+  *
+  * Sets parent for a DPLL clock. This sets the DPLL into bypass or
+  * locked mode. Returns 0 with success, negative error value otherwise.
+  */
+ int omap3_noncore_dpll_set_parent(struct clk_hw *hw, u8 index)
+ {
+       struct clk_hw_omap *clk = to_clk_hw_omap(hw);
+       int ret;
+       if (!hw)
+               return -EINVAL;
+       if (index)
+               ret = _omap3_noncore_dpll_bypass(clk);
+       else
+               ret = _omap3_noncore_dpll_lock(clk);
+       return ret;
+ }
+ /**
+  * omap3_noncore_dpll_set_rate - set rate for a DPLL clock
+  * @hw: pointer to the clock to set parent for
+  * @rate: target rate for the clock
+  * @parent_rate: rate of the parent clock
+  *
+  * Sets rate for a DPLL clock. First checks if the clock parent is
+  * reference clock (in bypass mode, the rate of the clock can't be
+  * changed) and proceeds with the rate change operation. Returns 0
+  * with success, negative error value otherwise.
+  */
+ int omap3_noncore_dpll_set_rate(struct clk_hw *hw, unsigned long rate,
+                               unsigned long parent_rate)
+ {
+       struct clk_hw_omap *clk = to_clk_hw_omap(hw);
+       struct dpll_data *dd;
+       u16 freqsel = 0;
+       int ret;
+       if (!hw || !rate)
+               return -EINVAL;
+       dd = clk->dpll_data;
+       if (!dd)
+               return -EINVAL;
+       if (__clk_get_hw(__clk_get_parent(hw->clk)) !=
+           __clk_get_hw(dd->clk_ref))
+               return -EINVAL;
+       if (dd->last_rounded_rate == 0)
+               return -EINVAL;
+       /* Freqsel is available only on OMAP343X devices */
+       if (ti_clk_get_features()->flags & TI_CLK_DPLL_HAS_FREQSEL) {
+               freqsel = _omap3_dpll_compute_freqsel(clk, dd->last_rounded_n);
+               WARN_ON(!freqsel);
+       }
+       pr_debug("%s: %s: set rate: locking rate to %lu.\n", __func__,
+                __clk_get_name(hw->clk), rate);
+       ret = omap3_noncore_dpll_program(clk, freqsel);
+       return ret;
+ }
+ /**
+  * omap3_noncore_dpll_set_rate_and_parent - set rate and parent for a DPLL clock
+  * @hw: pointer to the clock to set rate and parent for
+  * @rate: target rate for the DPLL
+  * @parent_rate: clock rate of the DPLL parent
+  * @index: new parent index for the DPLL, 0 - reference, 1 - bypass
+  *
+  * Sets rate and parent for a DPLL clock. If new parent is the bypass
+  * clock, only selects the parent. Otherwise proceeds with a rate
+  * change, as this will effectively also change the parent as the
+  * DPLL is put into locked mode. Returns 0 with success, negative error
+  * value otherwise.
+  */
+ int omap3_noncore_dpll_set_rate_and_parent(struct clk_hw *hw,
+                                          unsigned long rate,
+                                          unsigned long parent_rate,
+                                          u8 index)
+ {
+       int ret;
+       if (!hw || !rate)
+               return -EINVAL;
+       /*
+        * clk-ref at index[0], in which case we only need to set rate,
+        * the parent will be changed automatically with the lock sequence.
+        * With clk-bypass case we only need to change parent.
+        */
+       if (index)
+               ret = omap3_noncore_dpll_set_parent(hw, index);
+       else
+               ret = omap3_noncore_dpll_set_rate(hw, rate, parent_rate);
+       return ret;
+ }
+ /* DPLL autoidle read/set code */
+ /**
+  * omap3_dpll_autoidle_read - read a DPLL's autoidle bits
+  * @clk: struct clk * of the DPLL to read
+  *
+  * Return the DPLL's autoidle bits, shifted down to bit 0.  Returns
+  * -EINVAL if passed a null pointer or if the struct clk does not
+  * appear to refer to a DPLL.
+  */
+ static u32 omap3_dpll_autoidle_read(struct clk_hw_omap *clk)
+ {
+       const struct dpll_data *dd;
+       u32 v;
+       if (!clk || !clk->dpll_data)
+               return -EINVAL;
+       dd = clk->dpll_data;
+       if (!dd->autoidle_reg)
+               return -EINVAL;
+       v = ti_clk_ll_ops->clk_readl(dd->autoidle_reg);
+       v &= dd->autoidle_mask;
+       v >>= __ffs(dd->autoidle_mask);
+       return v;
+ }
+ /**
+  * omap3_dpll_allow_idle - enable DPLL autoidle bits
+  * @clk: struct clk * of the DPLL to operate on
+  *
+  * Enable DPLL automatic idle control.  This automatic idle mode
+  * switching takes effect only when the DPLL is locked, at least on
+  * OMAP3430.  The DPLL will enter low-power stop when its downstream
+  * clocks are gated.  No return value.
+  */
+ static void omap3_dpll_allow_idle(struct clk_hw_omap *clk)
+ {
+       const struct dpll_data *dd;
+       u32 v;
+       if (!clk || !clk->dpll_data)
+               return;
+       dd = clk->dpll_data;
+       if (!dd->autoidle_reg)
+               return;
+       /*
+        * REVISIT: CORE DPLL can optionally enter low-power bypass
+        * by writing 0x5 instead of 0x1.  Add some mechanism to
+        * optionally enter this mode.
+        */
+       v = ti_clk_ll_ops->clk_readl(dd->autoidle_reg);
+       v &= ~dd->autoidle_mask;
+       v |= DPLL_AUTOIDLE_LOW_POWER_STOP << __ffs(dd->autoidle_mask);
+       ti_clk_ll_ops->clk_writel(v, dd->autoidle_reg);
+ }
+ /**
+  * omap3_dpll_deny_idle - prevent DPLL from automatically idling
+  * @clk: struct clk * of the DPLL to operate on
+  *
+  * Disable DPLL automatic idle control.  No return value.
+  */
+ static void omap3_dpll_deny_idle(struct clk_hw_omap *clk)
+ {
+       const struct dpll_data *dd;
+       u32 v;
+       if (!clk || !clk->dpll_data)
+               return;
+       dd = clk->dpll_data;
+       if (!dd->autoidle_reg)
+               return;
+       v = ti_clk_ll_ops->clk_readl(dd->autoidle_reg);
+       v &= ~dd->autoidle_mask;
+       v |= DPLL_AUTOIDLE_DISABLE << __ffs(dd->autoidle_mask);
+       ti_clk_ll_ops->clk_writel(v, dd->autoidle_reg);
+ }
+ /* Clock control for DPLL outputs */
+ /* Find the parent DPLL for the given clkoutx2 clock */
+ static struct clk_hw_omap *omap3_find_clkoutx2_dpll(struct clk_hw *hw)
+ {
+       struct clk_hw_omap *pclk = NULL;
+       struct clk *parent;
+       /* Walk up the parents of clk, looking for a DPLL */
+       do {
+               do {
+                       parent = __clk_get_parent(hw->clk);
+                       hw = __clk_get_hw(parent);
+               } while (hw && (__clk_get_flags(hw->clk) & CLK_IS_BASIC));
+               if (!hw)
+                       break;
+               pclk = to_clk_hw_omap(hw);
+       } while (pclk && !pclk->dpll_data);
+       /* clk does not have a DPLL as a parent?  error in the clock data */
+       if (!pclk) {
+               WARN_ON(1);
+               return NULL;
+       }
+       return pclk;
+ }
+ /**
+  * omap3_clkoutx2_recalc - recalculate DPLL X2 output virtual clock rate
+  * @clk: DPLL output struct clk
+  *
+  * Using parent clock DPLL data, look up DPLL state.  If locked, set our
+  * rate to the dpll_clk * 2; otherwise, just use dpll_clk.
+  */
+ unsigned long omap3_clkoutx2_recalc(struct clk_hw *hw,
+                                   unsigned long parent_rate)
+ {
+       const struct dpll_data *dd;
+       unsigned long rate;
+       u32 v;
+       struct clk_hw_omap *pclk = NULL;
+       if (!parent_rate)
+               return 0;
+       pclk = omap3_find_clkoutx2_dpll(hw);
+       if (!pclk)
+               return 0;
+       dd = pclk->dpll_data;
+       WARN_ON(!dd->enable_mask);
+       v = ti_clk_ll_ops->clk_readl(dd->control_reg) & dd->enable_mask;
+       v >>= __ffs(dd->enable_mask);
+       if ((v != OMAP3XXX_EN_DPLL_LOCKED) || (dd->flags & DPLL_J_TYPE))
+               rate = parent_rate;
+       else
+               rate = parent_rate * 2;
+       return rate;
+ }
+ /* OMAP3/4 non-CORE DPLL clkops */
+ const struct clk_hw_omap_ops clkhwops_omap3_dpll = {
+       .allow_idle     = omap3_dpll_allow_idle,
+       .deny_idle      = omap3_dpll_deny_idle,
+ };
+ /**
+  * omap3_dpll4_set_rate - set rate for omap3 per-dpll
+  * @hw: clock to change
+  * @rate: target rate for clock
+  * @parent_rate: rate of the parent clock
+  *
+  * Check if the current SoC supports the per-dpll reprogram operation
+  * or not, and then do the rate change if supported. Returns -EINVAL
+  * if not supported, 0 for success, and potential error codes from the
+  * clock rate change.
+  */
+ int omap3_dpll4_set_rate(struct clk_hw *hw, unsigned long rate,
+                        unsigned long parent_rate)
+ {
+       /*
+        * According to the 12-5 CDP code from TI, "Limitation 2.5"
+        * on 3430ES1 prevents us from changing DPLL multipliers or dividers
+        * on DPLL4.
+        */
+       if (ti_clk_get_features()->flags & TI_CLK_DPLL4_DENY_REPROGRAM) {
+               pr_err("clock: DPLL4 cannot change rate due to silicon 'Limitation 2.5' on 3430ES1.\n");
+               return -EINVAL;
+       }
+       return omap3_noncore_dpll_set_rate(hw, rate, parent_rate);
+ }
+ /**
+  * omap3_dpll4_set_rate_and_parent - set rate and parent for omap3 per-dpll
+  * @hw: clock to change
+  * @rate: target rate for clock
+  * @parent_rate: rate of the parent clock
+  * @index: parent index, 0 - reference clock, 1 - bypass clock
+  *
+  * Check if the current SoC support the per-dpll reprogram operation
+  * or not, and then do the rate + parent change if supported. Returns
+  * -EINVAL if not supported, 0 for success, and potential error codes
+  * from the clock rate change.
+  */
+ int omap3_dpll4_set_rate_and_parent(struct clk_hw *hw, unsigned long rate,
+                                   unsigned long parent_rate, u8 index)
+ {
+       if (ti_clk_get_features()->flags & TI_CLK_DPLL4_DENY_REPROGRAM) {
+               pr_err("clock: DPLL4 cannot change rate due to silicon 'Limitation 2.5' on 3430ES1.\n");
+               return -EINVAL;
+       }
+       return omap3_noncore_dpll_set_rate_and_parent(hw, rate, parent_rate,
+                                                     index);
+ }
index 0000000000000000000000000000000000000000,ef1a5b43d01ffd06c2c4da336f32f7ef8e3a7f0d..73af77a90586e86db2955104470812f993c883cf
mode 000000,100644..100644
--- /dev/null
@@@ -1,0 -1,233 +1,227 @@@
 - * @rate: target rate for the DPLL
 - * @best_parent_rate: pointer for returning best parent rate
 - * @best_parent_clk: pointer for returning best parent clock
+ /*
+  * OMAP4-specific DPLL control functions
+  *
+  * Copyright (C) 2011 Texas Instruments, Inc.
+  * Rajendra Nayak
+  *
+  * This program is free software; you can redistribute it and/or modify
+  * it under the terms of the GNU General Public License version 2 as
+  * published by the Free Software Foundation.
+  */
+ #include <linux/kernel.h>
+ #include <linux/errno.h>
+ #include <linux/clk.h>
+ #include <linux/io.h>
+ #include <linux/bitops.h>
+ #include <linux/clk/ti.h>
+ #include "clock.h"
+ /*
+  * Maximum DPLL input frequency (FINT) and output frequency (FOUT) that
+  * can supported when using the DPLL low-power mode. Frequencies are
+  * defined in OMAP4430/60 Public TRM section 3.6.3.3.2 "Enable Control,
+  * Status, and Low-Power Operation Mode".
+  */
+ #define OMAP4_DPLL_LP_FINT_MAX        1000000
+ #define OMAP4_DPLL_LP_FOUT_MAX        100000000
+ /*
+  * Bitfield declarations
+  */
+ #define OMAP4430_DPLL_CLKOUT_GATE_CTRL_MASK           BIT(8)
+ #define OMAP4430_DPLL_CLKOUTX2_GATE_CTRL_MASK         BIT(10)
+ #define OMAP4430_DPLL_REGM4XEN_MASK                   BIT(11)
+ /* Static rate multiplier for OMAP4 REGM4XEN clocks */
+ #define OMAP4430_REGM4XEN_MULT                                4
+ static void omap4_dpllmx_allow_gatectrl(struct clk_hw_omap *clk)
+ {
+       u32 v;
+       u32 mask;
+       if (!clk || !clk->clksel_reg)
+               return;
+       mask = clk->flags & CLOCK_CLKOUTX2 ?
+                       OMAP4430_DPLL_CLKOUTX2_GATE_CTRL_MASK :
+                       OMAP4430_DPLL_CLKOUT_GATE_CTRL_MASK;
+       v = ti_clk_ll_ops->clk_readl(clk->clksel_reg);
+       /* Clear the bit to allow gatectrl */
+       v &= ~mask;
+       ti_clk_ll_ops->clk_writel(v, clk->clksel_reg);
+ }
+ static void omap4_dpllmx_deny_gatectrl(struct clk_hw_omap *clk)
+ {
+       u32 v;
+       u32 mask;
+       if (!clk || !clk->clksel_reg)
+               return;
+       mask = clk->flags & CLOCK_CLKOUTX2 ?
+                       OMAP4430_DPLL_CLKOUTX2_GATE_CTRL_MASK :
+                       OMAP4430_DPLL_CLKOUT_GATE_CTRL_MASK;
+       v = ti_clk_ll_ops->clk_readl(clk->clksel_reg);
+       /* Set the bit to deny gatectrl */
+       v |= mask;
+       ti_clk_ll_ops->clk_writel(v, clk->clksel_reg);
+ }
+ const struct clk_hw_omap_ops clkhwops_omap4_dpllmx = {
+       .allow_idle     = omap4_dpllmx_allow_gatectrl,
+       .deny_idle      = omap4_dpllmx_deny_gatectrl,
+ };
+ /**
+  * omap4_dpll_lpmode_recalc - compute DPLL low-power setting
+  * @dd: pointer to the dpll data structure
+  *
+  * Calculates if low-power mode can be enabled based upon the last
+  * multiplier and divider values calculated. If low-power mode can be
+  * enabled, then the bit to enable low-power mode is stored in the
+  * last_rounded_lpmode variable. This implementation is based upon the
+  * criteria for enabling low-power mode as described in the OMAP4430/60
+  * Public TRM section 3.6.3.3.2 "Enable Control, Status, and Low-Power
+  * Operation Mode".
+  */
+ static void omap4_dpll_lpmode_recalc(struct dpll_data *dd)
+ {
+       long fint, fout;
+       fint = __clk_get_rate(dd->clk_ref) / (dd->last_rounded_n + 1);
+       fout = fint * dd->last_rounded_m;
+       if ((fint < OMAP4_DPLL_LP_FINT_MAX) && (fout < OMAP4_DPLL_LP_FOUT_MAX))
+               dd->last_rounded_lpmode = 1;
+       else
+               dd->last_rounded_lpmode = 0;
+ }
+ /**
+  * omap4_dpll_regm4xen_recalc - compute DPLL rate, considering REGM4XEN bit
+  * @clk: struct clk * of the DPLL to compute the rate for
+  *
+  * Compute the output rate for the OMAP4 DPLL represented by @clk.
+  * Takes the REGM4XEN bit into consideration, which is needed for the
+  * OMAP4 ABE DPLL.  Returns the DPLL's output rate (before M-dividers)
+  * upon success, or 0 upon error.
+  */
+ unsigned long omap4_dpll_regm4xen_recalc(struct clk_hw *hw,
+                                        unsigned long parent_rate)
+ {
+       struct clk_hw_omap *clk = to_clk_hw_omap(hw);
+       u32 v;
+       unsigned long rate;
+       struct dpll_data *dd;
+       if (!clk || !clk->dpll_data)
+               return 0;
+       dd = clk->dpll_data;
+       rate = omap2_get_dpll_rate(clk);
+       /* regm4xen adds a multiplier of 4 to DPLL calculations */
+       v = ti_clk_ll_ops->clk_readl(dd->control_reg);
+       if (v & OMAP4430_DPLL_REGM4XEN_MASK)
+               rate *= OMAP4430_REGM4XEN_MULT;
+       return rate;
+ }
+ /**
+  * omap4_dpll_regm4xen_round_rate - round DPLL rate, considering REGM4XEN bit
+  * @clk: struct clk * of the DPLL to round a rate for
+  * @target_rate: the desired rate of the DPLL
+  *
+  * Compute the rate that would be programmed into the DPLL hardware
+  * for @clk if set_rate() were to be provided with the rate
+  * @target_rate.  Takes the REGM4XEN bit into consideration, which is
+  * needed for the OMAP4 ABE DPLL.  Returns the rounded rate (before
+  * M-dividers) upon success, -EINVAL if @clk is null or not a DPLL, or
+  * ~0 if an error occurred in omap2_dpll_round_rate().
+  */
+ long omap4_dpll_regm4xen_round_rate(struct clk_hw *hw,
+                                   unsigned long target_rate,
+                                   unsigned long *parent_rate)
+ {
+       struct clk_hw_omap *clk = to_clk_hw_omap(hw);
+       struct dpll_data *dd;
+       long r;
+       if (!clk || !clk->dpll_data)
+               return -EINVAL;
+       dd = clk->dpll_data;
+       dd->last_rounded_m4xen = 0;
+       /*
+        * First try to compute the DPLL configuration for
+        * target rate without using the 4X multiplier.
+        */
+       r = omap2_dpll_round_rate(hw, target_rate, NULL);
+       if (r != ~0)
+               goto out;
+       /*
+        * If we did not find a valid DPLL configuration, try again, but
+        * this time see if using the 4X multiplier can help. Enabling the
+        * 4X multiplier is equivalent to dividing the target rate by 4.
+        */
+       r = omap2_dpll_round_rate(hw, target_rate / OMAP4430_REGM4XEN_MULT,
+                                 NULL);
+       if (r == ~0)
+               return r;
+       dd->last_rounded_rate *= OMAP4430_REGM4XEN_MULT;
+       dd->last_rounded_m4xen = 1;
+ out:
+       omap4_dpll_lpmode_recalc(dd);
+       return dd->last_rounded_rate;
+ }
+ /**
+  * omap4_dpll_regm4xen_determine_rate - determine rate for a DPLL
+  * @hw: pointer to the clock to determine rate for
 - * Returns a positive clock rate with success, negative error value
 - * in failure.
++ * @req: target rate request
+  *
+  * Determines which DPLL mode to use for reaching a desired rate.
+  * Checks whether the DPLL shall be in bypass or locked mode, and if
+  * locked, calculates the M,N values for the DPLL via round-rate.
 -long omap4_dpll_regm4xen_determine_rate(struct clk_hw *hw, unsigned long rate,
 -                                      unsigned long min_rate,
 -                                      unsigned long max_rate,
 -                                      unsigned long *best_parent_rate,
 -                                      struct clk_hw **best_parent_clk)
++ * Returns 0 on success and a negative error value otherwise.
+  */
 -      if (!hw || !rate)
++int omap4_dpll_regm4xen_determine_rate(struct clk_hw *hw,
++                                     struct clk_rate_request *req)
+ {
+       struct clk_hw_omap *clk = to_clk_hw_omap(hw);
+       struct dpll_data *dd;
 -      if (__clk_get_rate(dd->clk_bypass) == rate &&
++      if (!req->rate)
+               return -EINVAL;
+       dd = clk->dpll_data;
+       if (!dd)
+               return -EINVAL;
 -              *best_parent_clk = __clk_get_hw(dd->clk_bypass);
++      if (__clk_get_rate(dd->clk_bypass) == req->rate &&
+           (dd->modes & (1 << DPLL_LOW_POWER_BYPASS))) {
 -              rate = omap4_dpll_regm4xen_round_rate(hw, rate,
 -                                                    best_parent_rate);
 -              *best_parent_clk = __clk_get_hw(dd->clk_ref);
++              req->best_parent_hw = __clk_get_hw(dd->clk_bypass);
+       } else {
 -      *best_parent_rate = rate;
++              req->rate = omap4_dpll_regm4xen_round_rate(hw, req->rate,
++                                              &req->best_parent_rate);
++              req->best_parent_hw = __clk_get_hw(dd->clk_ref);
+       }
 -      return rate;
++      req->best_parent_rate = req->rate;
++      return 0;
+ }