Return-Path: Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1754035AbdHUTnS convert rfc822-to-8bit (ORCPT ); Mon, 21 Aug 2017 15:43:18 -0400 Received: from mail.free-electrons.com ([62.4.15.54]:33247 "EHLO mail.free-electrons.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1753768AbdHUTnQ (ORCPT ); Mon, 21 Aug 2017 15:43:16 -0400 Date: Mon, 21 Aug 2017 21:43:03 +0200 From: Boris Brezillon To: Elaine Zhang Cc: mturquette@baylibre.com, sboyd@codeaurora.org, heiko@sntech.de, linux-clk@vger.kernel.org, linux-rockchip@lists.infradead.org, linux-kernel@vger.kernel.org, xxx@rock-chips.com, xf@rock-chips.com, huangtao@rock-chips.com, david.wu@rock-chips.com, xjq@rock-chips.com, Doug Anderson Subject: Re: [PATCH v1] clk: Keep clocks in their initial state until clk_disable_unused() is called Message-ID: <20170821214303.25169c7d@bbrezillon> In-Reply-To: <1502264249-30509-1-git-send-email-zhangqing@rock-chips.com> References: <1502264249-30509-1-git-send-email-zhangqing@rock-chips.com> X-Mailer: Claws Mail 3.14.1 (GTK+ 2.24.31; x86_64-pc-linux-gnu) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8BIT Sender: linux-kernel-owner@vger.kernel.org List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Content-Length: 3831 Lines: 110 +Doug Le Wed, 9 Aug 2017 15:37:29 +0800, Elaine Zhang a écrit : > ome drivers are briefly preparing+enabling the clock in their *Some > ->probe() hook and disable+unprepare them before leaving the function. > > This can be problem if a clock is shared between different devices, and > one of these devices is critical to the system. If this clock is > enabled/disabled by a non-critical device before the driver of the > critical one had a chance to enable+prepare it, there might be a short > period of time during which the critical device is not clocked. > > To solve this problem, we save the initial clock state (at registration > time) and prevent the clock from being disabled until kernel init is done > (which is when clk_disable_unused() is called). Well, my patch was just an informal proposal, and Doug pointed one thing that needs to be addressed before considering this approach: we are breaking clk users that expect clk_disable/unprepare() to be synchronous even when they're called before clk_disable_unused(). Mike, Stephen, any idea how to solve this problem properly? > > Signed-off-by: Boris Brezillon > Signed-off-by: Elaine Zhang > --- > drivers/clk/clk.c | 29 +++++++++++++++++++++++++++-- > 1 file changed, 27 insertions(+), 2 deletions(-) > > diff --git a/drivers/clk/clk.c b/drivers/clk/clk.c > index fc58c52a26b4..3f61374a364b 100644 > --- a/drivers/clk/clk.c > +++ b/drivers/clk/clk.c > @@ -58,6 +58,8 @@ struct clk_core { > struct clk_core *new_child; > unsigned long flags; > bool orphan; > + bool keep_enabled; > + bool keep_prepared; > unsigned int enable_count; > unsigned int prepare_count; > unsigned long min_rate; > @@ -486,7 +488,7 @@ static void clk_core_unprepare(struct clk_core *core) > > trace_clk_unprepare(core); > > - if (core->ops->unprepare) > + if (core->ops->unprepare && !core->keep_prepared) > core->ops->unprepare(core->hw); > > trace_clk_unprepare_complete(core); > @@ -602,7 +604,7 @@ static void clk_core_disable(struct clk_core *core) > > trace_clk_disable_rcuidle(core); > > - if (core->ops->disable) > + if (core->ops->disable && !core->keep_enabled) > core->ops->disable(core->hw); > > trace_clk_disable_complete_rcuidle(core); > @@ -739,6 +741,12 @@ static void clk_unprepare_unused_subtree(struct clk_core *core) > hlist_for_each_entry(child, &core->children, child_node) > clk_unprepare_unused_subtree(child); > > + /* > + * Reset the ->keep_prepared flag so that subsequent calls to > + * clk_unprepare() on this clk actually unprepare it. > + */ > + core->keep_prepared = false; > + > if (core->prepare_count) > return; > > @@ -770,6 +778,12 @@ static void clk_disable_unused_subtree(struct clk_core *core) > > flags = clk_enable_lock(); > > + /* > + * Reset the ->keep_enabled flag so that subsequent calls to > + * clk_disable() on this clk actually disable it. > + */ > + core->keep_enabled = false; > + > if (core->enable_count) > goto unlock_out; > > @@ -2446,6 +2460,17 @@ static int __clk_core_init(struct clk_core *core) > core->accuracy = 0; > > /* > + * We keep track of the initial clk status to keep clks in the state > + * they were left in by the bootloader until all drivers had a chance > + * to keep them prepared/enabled if they need to. > + */ > + if (core->ops->is_prepared && !clk_ignore_unused) > + core->keep_prepared = core->ops->is_prepared(core->hw); > + > + if (core->ops->is_enabled && !clk_ignore_unused) > + core->keep_enabled = core->ops->is_enabled(core->hw); > + > + /* > * Set clk's phase. > * Since a phase is by definition relative to its parent, just > * query the current clock phase, or just assume it's in phase.