Received: by 10.213.65.68 with SMTP id h4csp550138imn; Wed, 28 Mar 2018 08:25:34 -0700 (PDT) X-Google-Smtp-Source: AIpwx4+6ryTdVaOdJuC7prgB7KLbIV42Q1hkVZtN+w+OFDV4ttHzsbxLEWOwPGVgm4p6Q7dRZj7o X-Received: by 2002:a17:902:d695:: with SMTP id v21-v6mr4276266ply.34.1522250734441; Wed, 28 Mar 2018 08:25:34 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1522250734; cv=none; d=google.com; s=arc-20160816; b=ayjS6BP7H734EV+Dlj8sP20X/9zju2tj6fV+vRigX6KLahOEX6zgPDf5LorL0yezHc LkstBxRYRyZ14rv+3HgFvPGDxum5L0hjkB23HTnD025CqXgi0PhyC1ONg2FnKB8Gfb1/ +caKPgVgTpqkQrF1DqRF+q02nn/ei1ESe0x8ctvvSMzRu1aU+jA9mpNwGf53SZWdrehy 2ib8HagOmVbk9cGXw87U4xUewDy5isLxVt1rP0+FE2S44s+alU5w5j6RfdNBaNykNQXS yr7WAwVpROYslFwpHDnuMlSVb7oyxQ1mwn8QlFViKATknZ7P66Hoqij/u8yxa9rhnonY ZhOg== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:user-agent:in-reply-to :content-transfer-encoding:content-disposition:mime-version :references:message-id:subject:cc:to:from:date:dkim-signature :arc-authentication-results; bh=Ij+7uzQD4ox8rpaJh+TAOjZ06jowMP6yXC08kYwQvgQ=; b=rQ5TnqKon5/jNEa6eoYcItTlhl3Wl+GNPR8D7HMcgcvv3s6K+a86V43PA/xxfEIfMI A6I02Hjf/oGKuO3Yhzdi2Y48irkBGCK3/ljC2C6keYxvGPAS+/JS3dS3hc4iYepDO8lK yP4QadlqRMY4Wa+XxQefZKcVFMT/ouKBMUwd5I/iOoBwXv/Hlt/zaauuxRua4KR61noD Er3OMX7hEIQrlumOG0vT0wdg3NS9tyH3HIAiVdrP2CgDRYmtU+UaI945BDL8dPJFvQ6j Ls/N0aOMHIz9GRAZFZXv++5lHveF7E7aMr6/yKXFPifnb8ZoaC+clzokj2VKvY9w6Y20 K4iQ== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@linaro.org header.s=google header.b=PnFh8Pkc; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=linaro.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id r20si2939578pff.361.2018.03.28.08.25.11; Wed, 28 Mar 2018 08:25:34 -0700 (PDT) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; dkim=pass header.i=@linaro.org header.s=google header.b=PnFh8Pkc; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=linaro.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1753988AbeC1PW6 (ORCPT + 99 others); Wed, 28 Mar 2018 11:22:58 -0400 Received: from mail-wm0-f65.google.com ([74.125.82.65]:33488 "EHLO mail-wm0-f65.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1753958AbeC1PW4 (ORCPT ); Wed, 28 Mar 2018 11:22:56 -0400 Received: by mail-wm0-f65.google.com with SMTP id o23so14655620wmf.0 for ; Wed, 28 Mar 2018 08:22:55 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=linaro.org; s=google; h=date:from:to:cc:subject:message-id:references:mime-version :content-disposition:content-transfer-encoding:in-reply-to :user-agent; bh=Ij+7uzQD4ox8rpaJh+TAOjZ06jowMP6yXC08kYwQvgQ=; b=PnFh8PkcTQKQMf8Et8pg1cG27Bbo9p9JZP5In2L+qZ2h8+qOcLe3LsFnZ4+da3KQw+ E7Z1d1Q5F5eZDtJ8Ojz1p/GB+bFoaosw1wfiMrRDs0rsWyyq2Yx1cIH+Nlx39C+4LVX8 RL0oX+NkpjUqt4NQJJoKO6rgDzWHA8Pjwjges= X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:date:from:to:cc:subject:message-id:references :mime-version:content-disposition:content-transfer-encoding :in-reply-to:user-agent; bh=Ij+7uzQD4ox8rpaJh+TAOjZ06jowMP6yXC08kYwQvgQ=; b=UaysrvmxNzkBXbkn35HAjJXNtkfbuHhBcIXJ4kvTDgmtlmL1nBLU2uhzkr0xWwJZ2B 4xQYLiCmO/VttyuyH0824SVfBBeLm8gpYyEyoZW5F035k8r1gUJODBNc30umAuqyIyn6 1Kso+cdwga9855xTrVX+xtMz1GsCrBeWKOO7ozod12Dbxo8hbU2NXfzbSMWQIAKObM6D scZouWyKqLJeirpm9cd1j2NR3mZB6TGv8Jh13uY6fWpdHQ2bkxRtvC5d/2A6KxT18fB/ a8phHrbcICAlUEVxzsy/0F2eiZKh+mIbUIpiZxRu6QjGQ61dRNdBseYKK5HFEuephEe5 /pLw== X-Gm-Message-State: AElRT7HpK1F1PJx7J8BOE62FacP6MUtH86IVpyrBTiMhOfVjxDKNDlir 4ci93SpIB3OtGTcVIC61o6vspQ== X-Received: by 10.80.151.69 with SMTP id d5mr3818305edb.87.1522250575049; Wed, 28 Mar 2018 08:22:55 -0700 (PDT) Received: from dell ([2.31.163.57]) by smtp.gmail.com with ESMTPSA id j92sm2523402edd.81.2018.03.28.08.22.53 (version=TLS1_2 cipher=ECDHE-RSA-CHACHA20-POLY1305 bits=256/256); Wed, 28 Mar 2018 08:22:53 -0700 (PDT) Date: Wed, 28 Mar 2018 16:22:51 +0100 From: Lee Jones To: Fabrice Gasnier Cc: thierry.reding@gmail.com, alexandre.torgue@st.com, benjamin.gaignard@linaro.org, robh+dt@kernel.org, mark.rutland@arm.com, linux@armlinux.org.uk, mcoquelin.stm32@gmail.com, benjamin.gaignard@st.com, devicetree@vger.kernel.org, linux-arm-kernel@lists.infradead.org, linux-kernel@vger.kernel.org, linux-pwm@vger.kernel.org Subject: Re: [RESEND PATCH v2 4/8] mfd: stm32-timers: add support for dmas Message-ID: <20180328152251.xeh72dpeflzohn4k@dell> References: <1518602679-3064-1-git-send-email-fabrice.gasnier@st.com> <1518602679-3064-5-git-send-email-fabrice.gasnier@st.com> MIME-Version: 1.0 Content-Type: text/plain; charset=utf-8 Content-Disposition: inline Content-Transfer-Encoding: 8bit In-Reply-To: <1518602679-3064-5-git-send-email-fabrice.gasnier@st.com> User-Agent: NeoMutt/20170609 (1.8.3) Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On Wed, 14 Feb 2018, Fabrice Gasnier wrote: > STM32 Timers can support up to 7 DMA requests: > - 4 channels, update, compare and trigger. > Optionally request part, or all DMAs from stm32-timers MFD core. > > Also add routine to implement burst reads using DMA from timer registers. > This is exported. So, it can be used by child drivers, PWM capture > for instance (but not limited to). > > Signed-off-by: Fabrice Gasnier > Reviewed-by: Benjamin Gaignard > --- > Changes in v2: > - Abstract DMA handling from child driver: move it to MFD core > - Add comments on optional dma support > --- > drivers/mfd/stm32-timers.c | 215 ++++++++++++++++++++++++++++++++++++++- > include/linux/mfd/stm32-timers.h | 27 +++++ > 2 files changed, 238 insertions(+), 4 deletions(-) > > diff --git a/drivers/mfd/stm32-timers.c b/drivers/mfd/stm32-timers.c > index a6675a4..2cdad2c 100644 > --- a/drivers/mfd/stm32-timers.c > +++ b/drivers/mfd/stm32-timers.c > @@ -6,16 +6,166 @@ > * License terms: GNU General Public License (GPL), version 2 > */ > > +#include > +#include > +#include > #include > #include > #include > #include > > +#define STM32_TIMERS_MAX_REGISTERS 0x3fc > + > +struct stm32_timers_priv { > + struct device *dev; > + struct completion completion; > + phys_addr_t phys_base; /* timers physical addr for dma */ > + struct mutex lock; /* protect dma access */ > + struct dma_chan *dma_chan; /* dma channel in use */ I think kerneldoc would be better than inline comments. > + struct dma_chan *dmas[STM32_TIMERS_MAX_DMAS]; > + struct stm32_timers ddata; This looks odd to me. Why can't you expand the current ddata structure? Wouldn't it be better to create a stm32_timers_dma structure to place all this information in (except *dev, that should live in the ddata struct), then place a reference in the existing stm32_timers struct? > +}; > + > +static struct stm32_timers_priv *to_stm32_timers_priv(struct stm32_timers *d) > +{ > + return container_of(d, struct stm32_timers_priv, ddata); > +} If you take my other suggestions, you can remove this function. > +/* DIER register DMA enable bits */ > +static const u32 stm32_timers_dier_dmaen[STM32_TIMERS_MAX_DMAS] = { > + TIM_DIER_CC1DE, TIM_DIER_CC2DE, TIM_DIER_CC3DE, TIM_DIER_CC4DE, > + TIM_DIER_UIE, TIM_DIER_TDE, TIM_DIER_COMDE > +}; Maybe one per line would be kinder on the eye? > +static void stm32_timers_dma_done(void *p) > +{ > + struct stm32_timers_priv *priv = p; > + struct dma_chan *dma_chan = priv->dma_chan; > + struct dma_tx_state state; > + enum dma_status status; > + > + status = dmaengine_tx_status(dma_chan, dma_chan->cookie, &state); > + if (status == DMA_COMPLETE) > + complete(&priv->completion); > +} > + > +/** > + * stm32_timers_dma_burst_read - Read from timers registers using DMA. > + * > + * Read from STM32 timers registers using DMA on a single event. > + * @ddata: reference to stm32_timers It's odd to pass device data back like this. Better to pass a pointer to 'struct device', then use the normal helpers. > + * @buf: dma'able destination buffer > + * @id: stm32_timers_dmas event identifier (ch[1..4], up, trig or com) > + * @reg: registers start offset for DMA to read from (like CCRx for capture) > + * @num_reg: number of registers to read upon each dma request, starting @reg. > + * @bursts: number of bursts to read (e.g. like two for pwm period capture) > + * @tmo_ms: timeout (milliseconds) > + */ > +int stm32_timers_dma_burst_read(struct stm32_timers *ddata, u32 *buf, > + enum stm32_timers_dmas id, u32 reg, > + unsigned int num_reg, unsigned int bursts, > + unsigned long tmo_ms) > +{ > + struct stm32_timers_priv *priv = to_stm32_timers_priv(ddata); > + unsigned long timeout = msecs_to_jiffies(tmo_ms); > + struct regmap *regmap = priv->ddata.regmap; > + size_t len = num_reg * bursts * sizeof(u32); > + struct dma_async_tx_descriptor *desc; > + struct dma_slave_config config; > + dma_cookie_t cookie; > + dma_addr_t dma_buf; > + u32 dbl, dba; > + long err; > + int ret; > + > + /* sanity check */ > + if (id < STM32_TIMERS_DMA_CH1 || id >= STM32_TIMERS_MAX_DMAS) > + return -EINVAL; > + > + if (!num_reg || !bursts || reg > STM32_TIMERS_MAX_REGISTERS || > + (reg + num_reg * sizeof(u32)) > STM32_TIMERS_MAX_REGISTERS) > + return -EINVAL; > + > + if (!priv->dmas[id]) > + return -ENODEV; > + mutex_lock(&priv->lock); > + priv->dma_chan = priv->dmas[id]; > + > + dma_buf = dma_map_single(priv->dev, buf, len, DMA_FROM_DEVICE); > + ret = dma_mapping_error(priv->dev, dma_buf); > + if (ret) > + goto unlock; > + > + /* Prepare DMA read from timer registers, using DMA burst mode */ > + memset(&config, 0, sizeof(config)); > + config.src_addr = (dma_addr_t)priv->phys_base + TIM_DMAR; > + config.src_addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES; > + ret = dmaengine_slave_config(priv->dma_chan, &config); > + if (ret) > + goto unmap; > + > + desc = dmaengine_prep_slave_single(priv->dma_chan, dma_buf, len, > + DMA_DEV_TO_MEM, DMA_PREP_INTERRUPT); > + if (!desc) { > + ret = -EBUSY; > + goto unmap; > + } > + > + desc->callback = stm32_timers_dma_done; > + desc->callback_param = priv; > + cookie = dmaengine_submit(desc); > + ret = dma_submit_error(cookie); > + if (ret) > + goto dma_term; > + > + reinit_completion(&priv->completion); > + dma_async_issue_pending(priv->dma_chan); > + > + /* Setup and enable timer DMA burst mode */ > + dbl = FIELD_PREP(TIM_DCR_DBL, bursts - 1); > + dba = FIELD_PREP(TIM_DCR_DBA, reg >> 2); > + ret = regmap_write(regmap, TIM_DCR, dbl | dba); > + if (ret) > + goto dma_term; > + > + /* Clear pending flags before enabling DMA request */ > + ret = regmap_write(regmap, TIM_SR, 0); > + if (ret) > + goto dcr_clr; > + > + ret = regmap_update_bits(regmap, TIM_DIER, stm32_timers_dier_dmaen[id], > + stm32_timers_dier_dmaen[id]); > + if (ret) > + goto dcr_clr; > + > + err = wait_for_completion_interruptible_timeout(&priv->completion, > + timeout); > + if (err == 0) > + ret = -ETIMEDOUT; > + else if (err < 0) > + ret = err; > + > + regmap_update_bits(regmap, TIM_DIER, stm32_timers_dier_dmaen[id], 0); > + regmap_write(regmap, TIM_SR, 0); > +dcr_clr: > + regmap_write(regmap, TIM_DCR, 0); > +dma_term: > + dmaengine_terminate_all(priv->dma_chan); > +unmap: > + dma_unmap_single(priv->dev, dma_buf, len, DMA_FROM_DEVICE); > +unlock: > + priv->dma_chan = NULL; > + mutex_unlock(&priv->lock); > + > + return ret; > +} > +EXPORT_SYMBOL_GPL(stm32_timers_dma_burst_read); > + > static const struct regmap_config stm32_timers_regmap_cfg = { > .reg_bits = 32, > .val_bits = 32, > .reg_stride = sizeof(u32), > - .max_register = 0x3fc, > + .max_register = STM32_TIMERS_MAX_REGISTERS, > }; > > static void stm32_timers_get_arr_size(struct stm32_timers *ddata) > @@ -29,21 +179,55 @@ static void stm32_timers_get_arr_size(struct stm32_timers *ddata) > regmap_write(ddata->regmap, TIM_ARR, 0x0); > } > > +static void stm32_timers_dma_probe(struct device *dev, > + struct stm32_timers_priv *priv) > +{ > + int i; > + char name[4]; > + struct dma_chan **dmas = priv->dmas; > + > + /* Optional DMA support: get valid dma channel(s) or NULL */ > + for (i = STM32_TIMERS_DMA_CH1; i <= STM32_TIMERS_DMA_CH4; i++) { > + snprintf(name, ARRAY_SIZE(name), "ch%1d", i + 1); > + dmas[i] = dma_request_slave_channel(dev, name); > + } > + dmas[STM32_TIMERS_DMA_UP] = dma_request_slave_channel(dev, "up"); > + dmas[STM32_TIMERS_DMA_TRIG] = dma_request_slave_channel(dev, "trig"); > + dmas[STM32_TIMERS_DMA_COM] = dma_request_slave_channel(dev, "com"); > +} > + > +static void stm32_timers_dma_remove(struct device *dev, > + struct stm32_timers_priv *priv) > +{ > + int i; > + > + for (i = STM32_TIMERS_DMA_CH1; i < STM32_TIMERS_MAX_DMAS; i++) > + if (priv->dmas[i]) > + dma_release_channel(priv->dmas[i]); > +} > + > static int stm32_timers_probe(struct platform_device *pdev) > { > struct device *dev = &pdev->dev; > + struct stm32_timers_priv *priv; > struct stm32_timers *ddata; > struct resource *res; > void __iomem *mmio; > + int ret; > > - ddata = devm_kzalloc(dev, sizeof(*ddata), GFP_KERNEL); > - if (!ddata) > + priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL); > + if (!priv) > return -ENOMEM; > + ddata = &priv->ddata; > + init_completion(&priv->completion); > + mutex_init(&priv->lock); > + priv->dev = dev; > > res = platform_get_resource(pdev, IORESOURCE_MEM, 0); > mmio = devm_ioremap_resource(dev, res); > if (IS_ERR(mmio)) > return PTR_ERR(mmio); > + priv->phys_base = res->start; > > ddata->regmap = devm_regmap_init_mmio_clk(dev, "int", mmio, > &stm32_timers_regmap_cfg); > @@ -56,9 +240,31 @@ static int stm32_timers_probe(struct platform_device *pdev) > > stm32_timers_get_arr_size(ddata); > > + stm32_timers_dma_probe(dev, priv); > + > platform_set_drvdata(pdev, ddata); > > - return devm_of_platform_populate(&pdev->dev); > + ret = of_platform_populate(pdev->dev.of_node, NULL, NULL, &pdev->dev); > + if (ret) > + goto dma_remove; > + > + return 0; > + > +dma_remove: > + stm32_timers_dma_remove(dev, priv); You can easily remove this label and the goto. > + return ret; > +} > + > +static int stm32_timers_remove(struct platform_device *pdev) > +{ > + struct stm32_timers *ddata = platform_get_drvdata(pdev); > + struct stm32_timers_priv *priv = to_stm32_timers_priv(ddata); > + > + of_platform_depopulate(&pdev->dev); Why can't you continue using devm_*? > + stm32_timers_dma_remove(&pdev->dev, priv); > + > + return 0; > } > > static const struct of_device_id stm32_timers_of_match[] = { > @@ -69,6 +275,7 @@ static int stm32_timers_probe(struct platform_device *pdev) > > static struct platform_driver stm32_timers_driver = { > .probe = stm32_timers_probe, > + .remove = stm32_timers_remove, > .driver = { > .name = "stm32-timers", > .of_match_table = stm32_timers_of_match, > diff --git a/include/linux/mfd/stm32-timers.h b/include/linux/mfd/stm32-timers.h > index ce7346e..5fd2d6b 100644 > --- a/include/linux/mfd/stm32-timers.h > +++ b/include/linux/mfd/stm32-timers.h > @@ -29,6 +29,8 @@ > #define TIM_CCR3 0x3C /* Capt/Comp Register 3 */ > #define TIM_CCR4 0x40 /* Capt/Comp Register 4 */ > #define TIM_BDTR 0x44 /* Break and Dead-Time Reg */ > +#define TIM_DCR 0x48 /* DMA control register */ > +#define TIM_DMAR 0x4C /* DMA register for transfer */ > > #define TIM_CR1_CEN BIT(0) /* Counter Enable */ > #define TIM_CR1_DIR BIT(4) /* Counter Direction */ > @@ -38,6 +40,13 @@ > #define TIM_SMCR_SMS (BIT(0) | BIT(1) | BIT(2)) /* Slave mode selection */ > #define TIM_SMCR_TS (BIT(4) | BIT(5) | BIT(6)) /* Trigger selection */ > #define TIM_DIER_UIE BIT(0) /* Update interrupt */ > +#define TIM_DIER_UDE BIT(8) /* Update DMA request Enable */ > +#define TIM_DIER_CC1DE BIT(9) /* CC1 DMA request Enable */ > +#define TIM_DIER_CC2DE BIT(10) /* CC2 DMA request Enable */ > +#define TIM_DIER_CC3DE BIT(11) /* CC3 DMA request Enable */ > +#define TIM_DIER_CC4DE BIT(12) /* CC4 DMA request Enable */ > +#define TIM_DIER_COMDE BIT(13) /* COM DMA request Enable */ > +#define TIM_DIER_TDE BIT(14) /* Trigger DMA request Enable */ > #define TIM_SR_UIF BIT(0) /* Update interrupt flag */ > #define TIM_EGR_UG BIT(0) /* Update Generation */ > #define TIM_CCMR_PE BIT(3) /* Channel Preload Enable */ > @@ -58,6 +67,8 @@ > #define TIM_BDTR_BK2F (BIT(20) | BIT(21) | BIT(22) | BIT(23)) > #define TIM_BDTR_BK2E BIT(24) /* Break 2 input enable */ > #define TIM_BDTR_BK2P BIT(25) /* Break 2 input polarity */ > +#define TIM_DCR_DBA GENMASK(4, 0) /* DMA base addr */ > +#define TIM_DCR_DBL GENMASK(12, 8) /* DMA burst len */ > > #define MAX_TIM_PSC 0xFFFF > #define TIM_CR2_MMS_SHIFT 4 > @@ -67,9 +78,25 @@ > #define TIM_BDTR_BKF_SHIFT 16 > #define TIM_BDTR_BK2F_SHIFT 20 > > +enum stm32_timers_dmas { > + STM32_TIMERS_DMA_CH1, > + STM32_TIMERS_DMA_CH2, > + STM32_TIMERS_DMA_CH3, > + STM32_TIMERS_DMA_CH4, > + STM32_TIMERS_DMA_UP, > + STM32_TIMERS_DMA_TRIG, > + STM32_TIMERS_DMA_COM, > + STM32_TIMERS_MAX_DMAS, > +}; > + > struct stm32_timers { > struct clk *clk; > struct regmap *regmap; > u32 max_arr; > }; > + > +int stm32_timers_dma_burst_read(struct stm32_timers *ddata, u32 *buf, > + enum stm32_timers_dmas id, u32 reg, > + unsigned int num_reg, unsigned int bursts, > + unsigned long tmo_ms); > #endif -- Lee Jones [李琼斯] Linaro Services Technical Lead Linaro.org │ Open source software for ARM SoCs Follow Linaro: Facebook | Twitter | Blog