From: Bjorn Andersson <bjorn.andersson@linaro.org>
To: Dmitry Baryshkov <dmitry.baryshkov@linaro.org>
Cc: Andy Gross <agross@kernel.org>, Stephen Boyd <sboyd@kernel.org>,
Michael Turquette <mturquette@baylibre.com>,
Taniya Das <tdas@codeaurora.org>,
linux-arm-msm@vger.kernel.org, linux-clk@vger.kernel.org,
Ulf Hansson <ulf.hansson@linaro.org>
Subject: Re: [PATCH 2/2] clk: qcom: move pm_clk functionality into common code
Date: Fri, 16 Jul 2021 18:26:58 -0500 [thread overview]
Message-ID: <YPIVwgAjb/JdTowQ@yoga> (raw)
In-Reply-To: <20210710140130.1176657-3-dmitry.baryshkov@linaro.org>
On Sat 10 Jul 09:01 CDT 2021, Dmitry Baryshkov wrote:
> Several Qualcomm clock controller drivers use pm_clk functionality.
> Instead of having common code in all the drivers, move the pm_clk
> handling to the qcom_cc_map/qcom_cc_probe.
>
> Signed-off-by: Dmitry Baryshkov <dmitry.baryshkov@linaro.org>
> ---
> drivers/clk/qcom/camcc-sc7180.c | 44 ++--------
> drivers/clk/qcom/common.c | 113 +++++++++++++++++++++++---
> drivers/clk/qcom/common.h | 6 ++
> drivers/clk/qcom/lpasscorecc-sc7180.c | 56 +++----------
> drivers/clk/qcom/mss-sc7180.c | 40 ++-------
> drivers/clk/qcom/q6sstop-qcs404.c | 36 ++------
> drivers/clk/qcom/turingcc-qcs404.c | 34 ++------
> 7 files changed, 142 insertions(+), 187 deletions(-)
>
> diff --git a/drivers/clk/qcom/camcc-sc7180.c b/drivers/clk/qcom/camcc-sc7180.c
> index 9bcf2f8ed4de..1c6c1b7fab51 100644
> --- a/drivers/clk/qcom/camcc-sc7180.c
> +++ b/drivers/clk/qcom/camcc-sc7180.c
> @@ -9,7 +9,6 @@
> #include <linux/of.h>
> #include <linux/of_device.h>
> #include <linux/pm_clock.h>
> -#include <linux/pm_runtime.h>
> #include <linux/regmap.h>
>
> #include <dt-bindings/clock/qcom,camcc-sc7180.h>
> @@ -1631,8 +1630,12 @@ static const struct regmap_config cam_cc_sc7180_regmap_config = {
> .fast_io = true,
> };
>
> +static const char * const cam_cc_sc7180_pm_clks[] = { "xo", "iface" };
> +
> static const struct qcom_cc_desc cam_cc_sc7180_desc = {
> .config = &cam_cc_sc7180_regmap_config,
> + .pm_clks = cam_cc_sc7180_pm_clks,
> + .num_pm_clks = ARRAY_SIZE(cam_cc_sc7180_pm_clks),
> .clk_hws = cam_cc_sc7180_hws,
> .num_clk_hws = ARRAY_SIZE(cam_cc_sc7180_hws),
> .clks = cam_cc_sc7180_clocks,
> @@ -1652,33 +1655,9 @@ static int cam_cc_sc7180_probe(struct platform_device *pdev)
> struct regmap *regmap;
> int ret;
>
> - pm_runtime_enable(&pdev->dev);
> - ret = pm_clk_create(&pdev->dev);
> - if (ret < 0)
> - return ret;
> -
> - ret = pm_clk_add(&pdev->dev, "xo");
> - if (ret < 0) {
> - dev_err(&pdev->dev, "Failed to acquire XO clock\n");
> - goto disable_pm_runtime;
> - }
> -
> - ret = pm_clk_add(&pdev->dev, "iface");
> - if (ret < 0) {
> - dev_err(&pdev->dev, "Failed to acquire iface clock\n");
> - goto disable_pm_runtime;
> - }
> -
> - ret = pm_runtime_get(&pdev->dev);
> - if (ret)
> - goto destroy_pm_clk;
> -
> regmap = qcom_cc_map(pdev, &cam_cc_sc7180_desc);
> - if (IS_ERR(regmap)) {
> - ret = PTR_ERR(regmap);
> - pm_runtime_put(&pdev->dev);
> - goto destroy_pm_clk;
> - }
> + if (IS_ERR(regmap))
> + return PTR_ERR(regmap);
>
> clk_fabia_pll_configure(&cam_cc_pll0, regmap, &cam_cc_pll0_config);
> clk_fabia_pll_configure(&cam_cc_pll1, regmap, &cam_cc_pll1_config);
> @@ -1686,21 +1665,12 @@ static int cam_cc_sc7180_probe(struct platform_device *pdev)
> clk_fabia_pll_configure(&cam_cc_pll3, regmap, &cam_cc_pll3_config);
>
> ret = qcom_cc_really_probe(pdev, &cam_cc_sc7180_desc, regmap);
> - pm_runtime_put(&pdev->dev);
> if (ret < 0) {
> dev_err(&pdev->dev, "Failed to register CAM CC clocks\n");
> - goto destroy_pm_clk;
> + return 0;
> }
>
> return 0;
> -
> -destroy_pm_clk:
> - pm_clk_destroy(&pdev->dev);
> -
> -disable_pm_runtime:
> - pm_runtime_disable(&pdev->dev);
> -
> - return ret;
> }
>
> static const struct dev_pm_ops cam_cc_pm_ops = {
> diff --git a/drivers/clk/qcom/common.c b/drivers/clk/qcom/common.c
> index ed7c516a597a..e1d34561cab7 100644
> --- a/drivers/clk/qcom/common.c
> +++ b/drivers/clk/qcom/common.c
> @@ -7,6 +7,8 @@
> #include <linux/module.h>
> #include <linux/regmap.h>
> #include <linux/platform_device.h>
> +#include <linux/pm_clock.h>
> +#include <linux/pm_runtime.h>
> #include <linux/clk-provider.h>
> #include <linux/reset-controller.h>
> #include <linux/of.h>
> @@ -69,12 +71,86 @@ int qcom_find_src_index(struct clk_hw *hw, const struct parent_map *map, u8 src)
> }
> EXPORT_SYMBOL_GPL(qcom_find_src_index);
>
> +static void qcom_cc_pm_runtime_disable(void *data)
> +{
> + pm_runtime_disable(data);
> +}
> +
> +static void qcom_cc_pm_clk_destroy(void *data)
> +{
> + pm_clk_destroy(data);
> +}
> +
> +static int
> +qcom_cc_add_pm_clks(struct platform_device *pdev, const struct qcom_cc_desc *desc)
> +{
> + struct device *dev = &pdev->dev;
> + int ret;
> + int i;
> +
> + if (!desc->num_pm_clks)
> + return 0;
> +
> + ret = pm_clk_create(dev);
> + if (ret < 0)
> + return ret;
> + ret = devm_add_action_or_reset(dev, qcom_cc_pm_clk_destroy, dev);
> + if (ret)
> + return ret;
> +
> + for (i = 0; i < desc->num_pm_clks; i++) {
> + ret = pm_clk_add(dev, desc->pm_clks[i]);
> + if (ret < 0) {
> + dev_err(dev, "Failed to acquire %s pm clk\n", desc->pm_clks[i]);
> + return ret;
> + }
> + }
> +
> + return 0;
> +}
> +
> +static int
> +qcom_cc_manage_pm(struct platform_device *pdev, const struct qcom_cc_desc *desc)
> +{
> + struct device *dev = &pdev->dev;
> + int ret;
> +
> + /* For now enable runtime PM only if we have PM clocks in use */
> + if (desc->num_pm_clks && !pm_runtime_enabled(dev)) {
> + pm_runtime_enable(dev);
> +
> + ret = devm_add_action_or_reset(dev, qcom_cc_pm_runtime_disable, dev);
> + if (ret)
> + return ret;
> + }
> +
> + ret = qcom_cc_add_pm_clks(pdev, desc);
> + if (ret)
> + return ret;
> +
> + /* Other code might have enabled runtime PM, resume device here */
> + if (pm_runtime_enabled(dev)) {
> + ret = pm_runtime_get_sync(dev);
As I said previously, don't do this. Look at how clk_pm_runtime_get()
and clk_pm_runtime_put() are invoked throughout the clock framework.
At best this would be an optimization to ensure that the pm_runtime
state isn't toggled back and forth between every operation, but this is
typically not how we deal with that and this is certainly unrelated to
the rest of what the patch does.
> + if (ret) {
> + pm_runtime_put_noidle(dev);
> + return ret;
> + }
> + }
> +
> + return 0;
> +}
> +
> static struct regmap *
> qcom_cc_map_by_index(struct platform_device *pdev, const struct qcom_cc_desc *desc, int index)
I really don't like the idea of having a function with a name that
indicates that it's mapping hardware blocks to under the hood also play
pm_runtime.
Afaict the reason for this patch is to avoid having to sprinkle
pm_runtime_enable(), pm_clk_create(), pm_clk_destroy() and
pm_runtime_disable() in the various clock drivers that needs this.
But as I said previously, there's a lot of drivers in the kernel that
does exactly and only that, so there's definitely motivation to create
devm_pm_runtime_enable() and devm_pm_clk_create() and then go back and
clean up these and a lot of other drivers.
Perhaps I'm overly optimistic about getting those interfaces landed, if
that's the case I would like to have some explicit
devres-pm_runtime_enable/pm_clk_create/pm_clk_add added in the common
code, rather than piggy backing the existing map function.
But either way, I would much rather see you land the subdomain setup in
gdsc_register(), the pm_runtime_enable/disable we need in the
dispcc-sm8250 and the pm_runtime_get()/put() we need in gdsc_init(),
gdsc_enable() and gdsc_disable() - before refactoring all this.
> {
> void __iomem *base;
> struct resource *res;
> struct device *dev = &pdev->dev;
> + int ret;
> +
> + ret = qcom_cc_manage_pm(pdev, desc);
> + if (ret)
> + return ERR_PTR(ret);
>
> res = platform_get_resource(pdev, IORESOURCE_MEM, index);
> base = devm_ioremap_resource(dev, res);
> @@ -244,8 +320,10 @@ int qcom_cc_really_probe(struct platform_device *pdev,
> struct clk_hw **clk_hws = desc->clk_hws;
>
> cc = devm_kzalloc(dev, sizeof(*cc), GFP_KERNEL);
> - if (!cc)
> - return -ENOMEM;
> + if (!cc) {
> + ret = -ENOMEM;
> + goto err;
> + }
>
> reset = &cc->reset;
> reset->rcdev.of_node = dev->of_node;
> @@ -257,22 +335,25 @@ int qcom_cc_really_probe(struct platform_device *pdev,
>
> ret = devm_reset_controller_register(dev, &reset->rcdev);
> if (ret)
> - return ret;
> + goto err;
>
> if (desc->gdscs && desc->num_gdscs) {
> scd = devm_kzalloc(dev, sizeof(*scd), GFP_KERNEL);
> - if (!scd)
> - return -ENOMEM;
> + if (!scd) {
> + ret = -ENOMEM;
> + goto err;
> + }
> +
> scd->dev = dev;
> scd->scs = desc->gdscs;
> scd->num = desc->num_gdscs;
> ret = gdsc_register(scd, &reset->rcdev, regmap);
> if (ret)
> - return ret;
> + goto err;
> ret = devm_add_action_or_reset(dev, qcom_cc_gdsc_unregister,
> scd);
> if (ret)
> - return ret;
> + goto err;
> }
>
> cc->rclks = rclks;
> @@ -283,7 +364,7 @@ int qcom_cc_really_probe(struct platform_device *pdev,
> for (i = 0; i < num_clk_hws; i++) {
> ret = devm_clk_hw_register(dev, clk_hws[i]);
> if (ret)
> - return ret;
> + goto err;
> }
>
> for (i = 0; i < num_clks; i++) {
> @@ -292,14 +373,26 @@ int qcom_cc_really_probe(struct platform_device *pdev,
>
> ret = devm_clk_register_regmap(dev, rclks[i]);
> if (ret)
> - return ret;
> + goto err;
> }
>
> ret = devm_of_clk_add_hw_provider(dev, qcom_cc_clk_hw_get, cc);
> if (ret)
> - return ret;
> + goto err;
> +
> + if (pm_runtime_enabled(dev)) {
> + /* for the LPASS on sc7180, which uses autosuspend */
> + pm_runtime_mark_last_busy(dev);
> + pm_runtime_put(dev);
> + }
>
> return 0;
> +
> +err:
> + if (pm_runtime_enabled(dev))
> + pm_runtime_put(dev);
> +
> + return ret;
> }
> EXPORT_SYMBOL_GPL(qcom_cc_really_probe);
>
> diff --git a/drivers/clk/qcom/common.h b/drivers/clk/qcom/common.h
> index bb39a7e106d8..5b45e2033a92 100644
> --- a/drivers/clk/qcom/common.h
> +++ b/drivers/clk/qcom/common.h
> @@ -19,8 +19,14 @@ struct clk_hw;
> #define PLL_VOTE_FSM_ENA BIT(20)
> #define PLL_VOTE_FSM_RESET BIT(21)
>
> +/*
> + * Note: if pm_clks are used, pm_clk_suspend/resume should be called manually
> + * from runtime pm callbacks (or just passed to SET_RUNTIME_PM_OPS).
> + */
I don't like the fact that you're hiding most of the pm_runtime boiler
plate code, but each driver still needs to do this.
Perhaps there is merit to have a qcom_cc_pm_enable_and_add_clocks() and
qcom_cc_pm_ops exposed by common.c, but please let's add the pieces we
need first.
Regards,
Bjorn
next prev parent reply other threads:[~2021-07-16 23:27 UTC|newest]
Thread overview: 6+ messages / expand[flat|nested] mbox.gz Atom feed top
2021-07-10 14:01 [PATCH 0/2] clk: qcom: move pm_clk handling to common code Dmitry Baryshkov
2021-07-10 14:01 ` [PATCH 1/2] clk: qcom: use common code for qcom_cc_probe_by_index Dmitry Baryshkov
2021-07-16 23:29 ` Bjorn Andersson
2021-07-10 14:01 ` [PATCH 2/2] clk: qcom: move pm_clk functionality into common code Dmitry Baryshkov
2021-07-16 23:26 ` Bjorn Andersson [this message]
2021-07-17 7:33 ` Dmitry Baryshkov
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=YPIVwgAjb/JdTowQ@yoga \
--to=bjorn.andersson@linaro.org \
--cc=agross@kernel.org \
--cc=dmitry.baryshkov@linaro.org \
--cc=linux-arm-msm@vger.kernel.org \
--cc=linux-clk@vger.kernel.org \
--cc=mturquette@baylibre.com \
--cc=sboyd@kernel.org \
--cc=tdas@codeaurora.org \
--cc=ulf.hansson@linaro.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox