aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/clk
diff options
context:
space:
mode:
authorMike Turquette <mturquette@linaro.org>2012-05-02 19:23:43 -0400
committerMike Turquette <mturquette@linaro.org>2012-05-08 17:11:53 -0400
commit63f5c3b2b18dcaca0fc8983b52a3f5d4d70a0590 (patch)
tree966cda40bacfc4be9dfc7c3b13427d7661665c2e /drivers/clk
parent7704addb60e274d4e98c69a02f7ebe3f77c6c3a4 (diff)
clk: prevent spurious parent rate propagation
Patch 'clk: always pass parent_rate into .round_rate' made a subtle change to the semantics of .round_rate. It is now expected for the parent's rate to always be passed in, simplifying the implemenation of various .round_rate callback definitions. However the patch also introduced a bug in clk_calc_new_rates whereby a clock without the CLK_SET_RATE_PARENT flag set could still propagate a rate change up to a parent clock if the the .round_rate callback modified the &best_parent_rate value in any way. This patch fixes the issue at the framework level (in clk_calc_new_rates) by specifically handling the case where the CLK_SET_RATE_PARENT flag is not set. Signed-off-by: Mike Turquette <mturquette@linaro.org> Acked-by: Sascha Hauer <s.hauer@pengutronix.de>
Diffstat (limited to 'drivers/clk')
-rw-r--r--drivers/clk/clk.c7
1 files changed, 6 insertions, 1 deletions
diff --git a/drivers/clk/clk.c b/drivers/clk/clk.c
index 8149764f8438..7ceca0e8645a 100644
--- a/drivers/clk/clk.c
+++ b/drivers/clk/clk.c
@@ -774,12 +774,18 @@ static struct clk *clk_calc_new_rates(struct clk *clk, unsigned long rate)
774 if (IS_ERR_OR_NULL(clk)) 774 if (IS_ERR_OR_NULL(clk))
775 return NULL; 775 return NULL;
776 776
777 /* save parent rate, if it exists */
778 if (clk->parent)
779 best_parent_rate = clk->parent->rate;
780
777 /* never propagate up to the parent */ 781 /* never propagate up to the parent */
778 if (!(clk->flags & CLK_SET_RATE_PARENT)) { 782 if (!(clk->flags & CLK_SET_RATE_PARENT)) {
779 if (!clk->ops->round_rate) { 783 if (!clk->ops->round_rate) {
780 clk->new_rate = clk->rate; 784 clk->new_rate = clk->rate;
781 return NULL; 785 return NULL;
782 } 786 }
787 new_rate = clk->ops->round_rate(clk->hw, rate, &best_parent_rate);
788 goto out;
783 } 789 }
784 790
785 /* need clk->parent from here on out */ 791 /* need clk->parent from here on out */
@@ -795,7 +801,6 @@ static struct clk *clk_calc_new_rates(struct clk *clk, unsigned long rate)
795 goto out; 801 goto out;
796 } 802 }
797 803
798 best_parent_rate = clk->parent->rate;
799 new_rate = clk->ops->round_rate(clk->hw, rate, &best_parent_rate); 804 new_rate = clk->ops->round_rate(clk->hw, rate, &best_parent_rate);
800 805
801 if (best_parent_rate != clk->parent->rate) { 806 if (best_parent_rate != clk->parent->rate) {