Return-Path: Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S934792AbbKTHZx (ORCPT ); Fri, 20 Nov 2015 02:25:53 -0500 Received: from mx18-09.smtp.antispamcloud.com ([207.244.64.178]:40356 "EHLO mx18-09.smtp.antispamcloud.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1758765AbbKTHZu convert rfc822-to-8bit (ORCPT ); Fri, 20 Nov 2015 02:25:50 -0500 Subject: Re: [PATCH v2] fpga: zynq-fpga: Enable pm_runtime (suspend, resume) To: Moritz Fischer , References: <1447970832-774-1-git-send-email-moritz.fischer@ettus.com> CC: , , , , From: Mike Looijmans Organization: TOPIC Message-ID: <564ECAE7.9060900@topic.nl> Date: Fri, 20 Nov 2015 08:25:27 +0100 User-Agent: Mozilla/5.0 (X11; Linux x86_64; rv:38.0) Gecko/20100101 Thunderbird/38.3.0 MIME-Version: 1.0 In-Reply-To: <1447970832-774-1-git-send-email-moritz.fischer@ettus.com> Content-Type: text/plain; charset="utf-8"; format=flowed Content-Transfer-Encoding: 8BIT X-Originating-IP: [192.168.80.121] X-EXCLAIMER-MD-CONFIG: 9833cda7-5b21-4d34-9a38-8d025ddc3664 X-EXCLAIMER-MD-BIFURCATION-INSTANCE: 0 X-Filter-ID: s0sct1PQhAABKnZB5plbIbbvfIHzQjPVmPLZeVYSu3xU9luQrU+8/8qthi+0Jd/W6KAUC/fjyuDn NXFr4uarw4L+0Gk8Er9EDAmooWNG6NUXhYU8Jrr8yzFMFofcPkGGaOJv1yhqRSmeI+CmjdBgg6Ta MzkscmNkGq0QLaxj17WRxCCVjKGNHHobxmphk6KCqQvGnEF5AYOOOFRrW/jtHsQmyO56CnIWdFG7 HGeBoVyYxLloU+fO6iZzJO1autPxdjxmrRG+Yr2w44GUVRd4Lx8scZPEo7N6Pf5PJywZ6bTN3Et1 R9JkebCP5t3901uwuIiWfbXmGP/7JVzpf9dZ9AlNiPSjtaF8E/p/HJFsoC5gJPC5qlgckLhODYCw jbzKgENJ/9r3xpUCmyHH/LzsqK96FsAwDJ+L/f6U7xJA+uzQNhM1gygbvpDTU2f9+POp853EH1Co uyuN8TQvPLbdcXLYM3A6BXfvel8OEFDbU52d4QBMwRns26XOCF4YJs758eSmAkvysNuGCNDEi81V gtsfwlvSHWaU11skDPCC6TbVCZ1my7qTGj+pYSPpnV3tgSAqeihFev3wsZ/mTPLzZZsQrgz9ii6P 9NMYh5vidkhifFDKEuqo+y7SM5LKmlqk3XSq2E2jSZlKz92hhfD4QZ9WIA+d24zNKCR+8vOjc8c1 dOt47E4hX88q7EuJ0BoQmTHQX02yj7sQTBJF1UYnWcQxJoo2IkxG24CDQ98uI/0hEWyJzIkwSFAW 0Pw8uiKeFDhyrhWiOFT8srmpEcmA7F6L0DH2dm7YexLmt0wleAOuFwkDDG/lQ7mxnF16qsWYcfHb pla9upwqbpLVZbiC0cSyJEFHLnf7plZdvFHh9EU5SJeoVHj5h7lL06fEAVxIh17Nk1nbKoTTiTSZ IPzkLiIDBmPzR5caWyZhO+mhjAemU241EnRsE6VIrGoONZJ9aWpjWhBvczBesa4xOtx44w== X-Report-Abuse-To: spam@mx99.antispamcloud.com X-Filter-Fingerprint: IFrWXGses7OKB5S5G8/dJUb3OPwsHaH0Fvg5oXltHd/JUWjZ8+qhjyB23tbDuyLOYL8Ff78gYsez 4Rl08xudmXi4esCQ0R1MchVjt7wblGlvhFgW0MjUMRkF5sMCDfftTXNFDzN17hnrWeZYOJvLq0Ic WjZ+XcEjj/7Pkld0zkmvziDInX9WdMov2kn2yXjdwv61T+KDYyYtREgszdyFwv8IxCB3p/oCKvxr eyISh3JGb7OS5oVgiO+kDxZrVPLz3MmEGC2PrUKqLq5WmHK+Nw== X-Originating-IP: 88.159.208.100 X-Spampanel-Domain: topic.nl X-Spampanel-Username: 88.159.208.100 Authentication-Results: antispamcloud.com; auth=pass smtp.auth=88.159.208.100@topic.nl X-Spampanel-Outgoing-Class: ham X-Spampanel-Outgoing-Evidence: Combined (0.00) X-Recommended-Action: accept Sender: linux-kernel-owner@vger.kernel.org List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Content-Length: 6773 Lines: 237 On 19-11-15 23:07, Moritz Fischer wrote: > Replaced constant clock_{enable,disable} calls with pm_runtime > hooks for suspend and resume to avoid constant clk_enable / > clk_disable. > > Acked-by: Alan Tull > Signed-off-by: Moritz Fischer > --- > Changes: > > v1: > - Removed superfluous #ifdef CONFIG_PM as suggested by Michal > - Changed commit message to include suspend / resume > - Added Alan's Acked-by > > drivers/fpga/zynq-fpga.c | 76 +++++++++++++++++++++++++++++++++++++----------- > 1 file changed, 59 insertions(+), 17 deletions(-) > > diff --git a/drivers/fpga/zynq-fpga.c b/drivers/fpga/zynq-fpga.c > index c2fb412..3f5469d 100644 > --- a/drivers/fpga/zynq-fpga.c > +++ b/drivers/fpga/zynq-fpga.c > @@ -28,6 +28,7 @@ > #include > #include > #include > +#include > #include > #include > > @@ -184,8 +185,8 @@ static int zynq_fpga_ops_write_init(struct fpga_manager *mgr, u32 flags, > > priv = mgr->priv; > > - err = clk_enable(priv->clk); > - if (err) > + err = pm_runtime_get_sync(priv->dev); > + if (err < 0) > return err; > > /* don't globally reset PL if we're doing partial reconfig */ > @@ -271,12 +272,12 @@ static int zynq_fpga_ops_write_init(struct fpga_manager *mgr, u32 flags, > ctrl = zynq_fpga_read(priv, MCTRL_OFFSET); > zynq_fpga_write(priv, MCTRL_OFFSET, (~MCTRL_PCAP_LPBK_MASK & ctrl)); > > - clk_disable(priv->clk); > + pm_runtime_put(priv->dev); > > return 0; > > out_err: > - clk_disable(priv->clk); > + pm_runtime_put(priv->dev); > > return err; > } > @@ -301,9 +302,8 @@ static int zynq_fpga_ops_write(struct fpga_manager *mgr, > > memcpy(kbuf, buf, count); > > - /* enable clock */ > - err = clk_enable(priv->clk); > - if (err) > + err = pm_runtime_get_sync(priv->dev); > + if (err < 0) > goto out_free; > > zynq_fpga_write(priv, INT_STS_OFFSET, IXR_ALL_MASK); > @@ -335,7 +335,7 @@ static int zynq_fpga_ops_write(struct fpga_manager *mgr, > err = -EFAULT; > } > > - clk_disable(priv->clk); > + pm_runtime_put(priv->dev); > > out_free: > dma_free_coherent(priv->dev, in_count, kbuf, dma_addr); > @@ -349,8 +349,8 @@ static int zynq_fpga_ops_write_complete(struct fpga_manager *mgr, u32 flags) > int err; > u32 intr_status; > > - err = clk_enable(priv->clk); > - if (err) > + err = pm_runtime_get_sync(priv->dev); > + if (err < 0) > return err; > > err = zynq_fpga_poll_timeout(priv, INT_STS_OFFSET, intr_status, > @@ -358,7 +358,7 @@ static int zynq_fpga_ops_write_complete(struct fpga_manager *mgr, u32 flags) > INIT_POLL_DELAY, > INIT_POLL_TIMEOUT); > > - clk_disable(priv->clk); > + pm_runtime_put(priv->dev); > > if (err) > return err; > @@ -385,12 +385,12 @@ static enum fpga_mgr_states zynq_fpga_ops_state(struct fpga_manager *mgr) > > priv = mgr->priv; > > - err = clk_enable(priv->clk); > - if (err) > + err = pm_runtime_get_sync(priv->dev); > + if (err < 0) > return FPGA_MGR_STATE_UNKNOWN; > > intr_status = zynq_fpga_read(priv, INT_STS_OFFSET); > - clk_disable(priv->clk); > + pm_runtime_put(priv->dev); > > if (intr_status & IXR_PCFG_DONE_MASK) > return FPGA_MGR_STATE_OPERATING; > @@ -457,19 +457,26 @@ static int zynq_fpga_probe(struct platform_device *pdev) > return err; > } > > + pm_runtime_get_noresume(&pdev->dev); > + pm_runtime_set_active(&pdev->dev); > + pm_runtime_enable(&pdev->dev); > + > /* unlock the device */ > zynq_fpga_write(priv, UNLOCK_OFFSET, UNLOCK_MASK); > > - clk_disable(priv->clk); > > err = fpga_mgr_register(dev, "Xilinx Zynq FPGA Manager", > &zynq_fpga_ops, priv); > if (err) { > dev_err(dev, "unable to register FPGA manager"); > - clk_unprepare(priv->clk); > + clk_disable_unprepare(priv->clk); > + pm_runtime_put_noidle(&pdev->dev); > + pm_runtime_disable(&pdev->dev); > return err; > } > > + pm_runtime_put(&pdev->dev); > + > return 0; > } > > @@ -483,11 +490,45 @@ static int zynq_fpga_remove(struct platform_device *pdev) > > fpga_mgr_unregister(&pdev->dev); > > - clk_unprepare(priv->clk); > + pm_runtime_get_sync(&pdev->dev); > + clk_disable_unprepare(priv->clk); > + pm_runtime_put_noidle(&pdev->dev); > + pm_runtime_disable(&pdev->dev); > > return 0; > } > > +static int __maybe_unused zynq_fpga_runtime_suspend(struct device *dev) > +{ > + struct zynq_fpga_priv *priv; > + struct fpga_manager *mgr; > + > + mgr = dev_get_drvdata(dev); > + priv = mgr->priv; > + > + clk_disable(priv->clk); From what I understand, this call is done in a sleepable context, so you can use clk_disable_unprepare here (and its counterpart in resume). And remove the prepare at probe time and unprepare at removal. Not all clocks can implement atomic enable/disable, for example I2C and SPI controlled clocks only implement the prepare/unprepare routines. I guess the "clk" here will always be a SOC provided one, so it won't make any difference for the Zynq, but someone is likely to some day copy/paste this driver and use it for some externally connected FPGA instead. > + > + return 0; > +} > + > +static int __maybe_unused zynq_fpga_runtime_resume(struct device *dev) > +{ > + struct zynq_fpga_priv *priv; > + struct fpga_manager *mgr; > + > + mgr = dev_get_drvdata(dev); > + priv = mgr->priv; > + > + clk_enable(priv->clk); > + > + return 0; > +} > + > +static const struct dev_pm_ops zynq_fpga_pm_ops = { > + SET_RUNTIME_PM_OPS(zynq_fpga_runtime_suspend, > + zynq_fpga_runtime_resume, NULL) > +}; > + > #ifdef CONFIG_OF > static const struct of_device_id zynq_fpga_of_match[] = { > { .compatible = "xlnx,zynq-devcfg-1.0", }, > @@ -503,6 +544,7 @@ static struct platform_driver zynq_fpga_driver = { > .driver = { > .name = "zynq_fpga_manager", > .of_match_table = of_match_ptr(zynq_fpga_of_match), > + .pm = &zynq_fpga_pm_ops, > }, > }; > > Kind regards, Mike Looijmans System Expert TOPIC Embedded Products Eindhovenseweg 32-C, NL-5683 KH Best Postbus 440, NL-5680 AK Best Telefoon: +31 (0) 499 33 69 79 Telefax: +31 (0) 499 33 69 70 E-mail: mike.looijmans@topicproducts.com Website: www.topicproducts.com Please consider the environment before printing this e-mail Visit us at : Aerospace Electrical Systems Expo Europe which will be held from 17.11.2015 till 19.11.2015, Findorffstrasse 101 Bremen, Germany, Hall 5, stand number C65 http://www.aesexpo.eu -- To unsubscribe from this list: send the line "unsubscribe linux-kernel" in the body of a message to majordomo@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html Please read the FAQ at http://www.tux.org/lkml/