Received: by 2002:a25:8b91:0:0:0:0:0 with SMTP id j17csp1787338ybl; Sat, 7 Dec 2019 02:39:08 -0800 (PST) X-Google-Smtp-Source: APXvYqwJWuXVFNCmhmRa1nqIIxwhjnOIWRiU2Kz3PGLp2rVFY76aCihm2e6vfEe/OTdrovJHjcQv X-Received: by 2002:aca:df07:: with SMTP id w7mr81247oig.145.1575715148582; Sat, 07 Dec 2019 02:39:08 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1575715148; cv=none; d=google.com; s=arc-20160816; b=okZTmJO8XWb2lYleVtlLrVfK5unEj/wnxd6O67FqXDrCuVx8ur4QfPO4GPiXF5DXFa BskQLHOoDVLL7iWJU9MB+yOCW347aryyNzs+ZN/EEiExECOnsyEtChSwcnu116CeMQcb 0nON9UcHAM/9kTNXS+G5kF9onK9QEIeZeFhDvsdh7roDNi86MY0tyFlLTgsLEcaEGQ7U zmhlkpm+M7JXxMp7iBZLpTZ4OPf37nqqeQEYCB1m0V1LuFYKdyZdW03uDhH/XB3YdMjN r6xZbiOKgC4Y6JKaC8XNUp8xgRVe27zrfNTcojK+es7yphv9l+0dGlBW09xZzeInrfZU Ab6Q== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-transfer-encoding:mime-version :references:in-reply-to:message-id:subject:cc:to:from:date :dkim-signature; bh=pA4/5homGKQiAFnBg9QXTx1eKQRG5e1YdnhOFZiBeWE=; b=O9fvgxgiTikwGX+2yhQOTwEwyoCOeFROHkiBqDKaK8E2EFn4RwYXCyihr4O1NZgeiW WlTmCYJ+2eN/L5MBIFL4+eW+yyBMxy1AcFQTO69tYlSYybDzCjEfXYvwSrL5m1j8P0IQ 2vQsKjL166dwu3o2tPVa/RCu+Y1zU7RyKP9EIAxnnTT/SrHcC+gh1QW4dapxdbPcP1QW /sM8K1ZGWYCPqSaMJz08XyEcKStvLLUTGjiDYWnqC6k2GpV8GVK9d0gKastPta5qRSjV xr/I6wTYUZLBOsNVM7VnK9KIyV6YmEfeCAmShwZU8pDzqH68CeeRE6P6Kuug7RIcZwUF eHOQ== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@kernel.org header.s=default header.b=mfVMPLDi; 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=kernel.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id z203si2088704oia.124.2019.12.07.02.38.56; Sat, 07 Dec 2019 02:39:08 -0800 (PST) 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=@kernel.org header.s=default header.b=mfVMPLDi; 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=kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1726371AbfLGKhd (ORCPT + 99 others); Sat, 7 Dec 2019 05:37:33 -0500 Received: from mail.kernel.org ([198.145.29.99]:34024 "EHLO mail.kernel.org" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1725992AbfLGKhc (ORCPT ); Sat, 7 Dec 2019 05:37:32 -0500 Received: from archlinux (cpc149474-cmbg20-2-0-cust94.5-4.cable.virginm.net [82.4.196.95]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) by mail.kernel.org (Postfix) with ESMTPSA id 6C34F2053B; Sat, 7 Dec 2019 10:37:29 +0000 (UTC) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=kernel.org; s=default; t=1575715051; bh=03tkZ4Qslko3/IrvTMxw42gSR7I/sOU150wfj2BXJ20=; h=Date:From:To:Cc:Subject:In-Reply-To:References:From; b=mfVMPLDitdPuCa0bB4HhDQEkEzDvFKUcbtWoyMxJiZKc6OMSUlaaby9h+PTgZE3cm c/MjBocSvb8ft6Oo/cwzF63LoobU3XpH/07owt3AUQO3U16P8FZ1uTmw8+zDk7Jgm5 GNV6FIrBlS19xnPyyBADUWdFTysb/NnEMueAbr8E= Date: Sat, 7 Dec 2019 10:37:26 +0000 From: Jonathan Cameron To: Fabrice Gasnier Cc: , , , , , , , , , Subject: Re: [PATCH v2] iio: adc: stm32-adc: Add check on overrun interrupt Message-ID: <20191207103726.6d431237@archlinux> In-Reply-To: <1575277339-30237-1-git-send-email-fabrice.gasnier@st.com> References: <1575277339-30237-1-git-send-email-fabrice.gasnier@st.com> X-Mailer: Claws Mail 3.17.4 (GTK+ 2.24.32; x86_64-pc-linux-gnu) MIME-Version: 1.0 Content-Type: text/plain; charset=US-ASCII Content-Transfer-Encoding: 7bit Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On Mon, 2 Dec 2019 10:02:19 +0100 Fabrice Gasnier wrote: > Enable overrun interrupt on STM32 ADC. In case data register hasn't been > read (by CPU or DMA), overrun condition is detected when there's new > conversion data available. Stop grabbing data and log an error message. > Use a threaded irq to avoid printing the error message from hard irq > context. > > Signed-off-by: Fabrice Gasnier Given you have explicitly mentioned this as a fix I haven't assumed it was particularly urgent. Hence applied to the togreg branch of iio.git and pushed out as testing. If we want to request this being added to stable, we should do that after it is in mainline after the next merge window. If you need it faster, let me know and add a fixes tag. Thanks, Jonathan > --- > Changes in v2: > - Add a comment in the code, to give a hint on how to restart the capture, > as suggested by Jonathan. Also update the error message to mention a > restart is needed. > --- > drivers/iio/adc/stm32-adc-core.c | 14 +++++----- > drivers/iio/adc/stm32-adc-core.h | 9 +++++++ > drivers/iio/adc/stm32-adc.c | 55 ++++++++++++++++++++++++++++++++++++++-- > 3 files changed, 69 insertions(+), 9 deletions(-) > > diff --git a/drivers/iio/adc/stm32-adc-core.c b/drivers/iio/adc/stm32-adc-core.c > index 6537f4f..97655d7 100644 > --- a/drivers/iio/adc/stm32-adc-core.c > +++ b/drivers/iio/adc/stm32-adc-core.c > @@ -280,21 +280,21 @@ static int stm32h7_adc_clk_sel(struct platform_device *pdev, > static const struct stm32_adc_common_regs stm32f4_adc_common_regs = { > .csr = STM32F4_ADC_CSR, > .ccr = STM32F4_ADC_CCR, > - .eoc1_msk = STM32F4_EOC1, > - .eoc2_msk = STM32F4_EOC2, > - .eoc3_msk = STM32F4_EOC3, > + .eoc1_msk = STM32F4_EOC1 | STM32F4_OVR1, > + .eoc2_msk = STM32F4_EOC2 | STM32F4_OVR2, > + .eoc3_msk = STM32F4_EOC3 | STM32F4_OVR3, > .ier = STM32F4_ADC_CR1, > - .eocie_msk = STM32F4_EOCIE, > + .eocie_msk = STM32F4_EOCIE | STM32F4_OVRIE, > }; > > /* STM32H7 common registers definitions */ > static const struct stm32_adc_common_regs stm32h7_adc_common_regs = { > .csr = STM32H7_ADC_CSR, > .ccr = STM32H7_ADC_CCR, > - .eoc1_msk = STM32H7_EOC_MST, > - .eoc2_msk = STM32H7_EOC_SLV, > + .eoc1_msk = STM32H7_EOC_MST | STM32H7_OVR_MST, > + .eoc2_msk = STM32H7_EOC_SLV | STM32H7_OVR_SLV, > .ier = STM32H7_ADC_IER, > - .eocie_msk = STM32H7_EOCIE, > + .eocie_msk = STM32H7_EOCIE | STM32H7_OVRIE, > }; > > static const unsigned int stm32_adc_offset[STM32_ADC_MAX_ADCS] = { > diff --git a/drivers/iio/adc/stm32-adc-core.h b/drivers/iio/adc/stm32-adc-core.h > index 2579d51..2322809 100644 > --- a/drivers/iio/adc/stm32-adc-core.h > +++ b/drivers/iio/adc/stm32-adc-core.h > @@ -51,10 +51,12 @@ > #define STM32F4_ADC_CCR (STM32_ADCX_COMN_OFFSET + 0x04) > > /* STM32F4_ADC_SR - bit fields */ > +#define STM32F4_OVR BIT(5) > #define STM32F4_STRT BIT(4) > #define STM32F4_EOC BIT(1) > > /* STM32F4_ADC_CR1 - bit fields */ > +#define STM32F4_OVRIE BIT(26) > #define STM32F4_RES_SHIFT 24 > #define STM32F4_RES_MASK GENMASK(25, 24) > #define STM32F4_SCAN BIT(8) > @@ -72,8 +74,11 @@ > #define STM32F4_ADON BIT(0) > > /* STM32F4_ADC_CSR - bit fields */ > +#define STM32F4_OVR3 BIT(21) > #define STM32F4_EOC3 BIT(17) > +#define STM32F4_OVR2 BIT(13) > #define STM32F4_EOC2 BIT(9) > +#define STM32F4_OVR1 BIT(5) > #define STM32F4_EOC1 BIT(1) > > /* STM32F4_ADC_CCR - bit fields */ > @@ -103,10 +108,12 @@ > > /* STM32H7_ADC_ISR - bit fields */ > #define STM32MP1_VREGREADY BIT(12) > +#define STM32H7_OVR BIT(4) > #define STM32H7_EOC BIT(2) > #define STM32H7_ADRDY BIT(0) > > /* STM32H7_ADC_IER - bit fields */ > +#define STM32H7_OVRIE STM32H7_OVR > #define STM32H7_EOCIE STM32H7_EOC > > /* STM32H7_ADC_CR - bit fields */ > @@ -155,7 +162,9 @@ enum stm32h7_adc_dmngt { > #define STM32H7_LINCALFACT_MASK GENMASK(29, 0) > > /* STM32H7_ADC_CSR - bit fields */ > +#define STM32H7_OVR_SLV BIT(20) > #define STM32H7_EOC_SLV BIT(18) > +#define STM32H7_OVR_MST BIT(4) > #define STM32H7_EOC_MST BIT(2) > > /* STM32H7_ADC_CCR - bit fields */ > diff --git a/drivers/iio/adc/stm32-adc.c b/drivers/iio/adc/stm32-adc.c > index 3b291d7..5f05bf9 100644 > --- a/drivers/iio/adc/stm32-adc.c > +++ b/drivers/iio/adc/stm32-adc.c > @@ -117,7 +117,9 @@ struct stm32_adc_regs { > * struct stm32_adc_regspec - stm32 registers definition > * @dr: data register offset > * @ier_eoc: interrupt enable register & eocie bitfield > + * @ier_ovr: interrupt enable register & overrun bitfield > * @isr_eoc: interrupt status register & eoc bitfield > + * @isr_ovr: interrupt status register & overrun bitfield > * @sqr: reference to sequence registers array > * @exten: trigger control register & bitfield > * @extsel: trigger selection register & bitfield > @@ -128,7 +130,9 @@ struct stm32_adc_regs { > struct stm32_adc_regspec { > const u32 dr; > const struct stm32_adc_regs ier_eoc; > + const struct stm32_adc_regs ier_ovr; > const struct stm32_adc_regs isr_eoc; > + const struct stm32_adc_regs isr_ovr; > const struct stm32_adc_regs *sqr; > const struct stm32_adc_regs exten; > const struct stm32_adc_regs extsel; > @@ -337,7 +341,9 @@ static const unsigned int stm32f4_adc_smp_cycles[STM32_ADC_MAX_SMP + 1] = { > static const struct stm32_adc_regspec stm32f4_adc_regspec = { > .dr = STM32F4_ADC_DR, > .ier_eoc = { STM32F4_ADC_CR1, STM32F4_EOCIE }, > + .ier_ovr = { STM32F4_ADC_CR1, STM32F4_OVRIE }, > .isr_eoc = { STM32F4_ADC_SR, STM32F4_EOC }, > + .isr_ovr = { STM32F4_ADC_SR, STM32F4_OVR }, > .sqr = stm32f4_sq, > .exten = { STM32F4_ADC_CR2, STM32F4_EXTEN_MASK, STM32F4_EXTEN_SHIFT }, > .extsel = { STM32F4_ADC_CR2, STM32F4_EXTSEL_MASK, > @@ -429,7 +435,9 @@ static const unsigned int stm32h7_adc_smp_cycles[STM32_ADC_MAX_SMP + 1] = { > static const struct stm32_adc_regspec stm32h7_adc_regspec = { > .dr = STM32H7_ADC_DR, > .ier_eoc = { STM32H7_ADC_IER, STM32H7_EOCIE }, > + .ier_ovr = { STM32H7_ADC_IER, STM32H7_OVRIE }, > .isr_eoc = { STM32H7_ADC_ISR, STM32H7_EOC }, > + .isr_ovr = { STM32H7_ADC_ISR, STM32H7_OVR }, > .sqr = stm32h7_sq, > .exten = { STM32H7_ADC_CFGR, STM32H7_EXTEN_MASK, STM32H7_EXTEN_SHIFT }, > .extsel = { STM32H7_ADC_CFGR, STM32H7_EXTSEL_MASK, > @@ -506,6 +514,18 @@ static void stm32_adc_conv_irq_disable(struct stm32_adc *adc) > adc->cfg->regs->ier_eoc.mask); > } > > +static void stm32_adc_ovr_irq_enable(struct stm32_adc *adc) > +{ > + stm32_adc_set_bits(adc, adc->cfg->regs->ier_ovr.reg, > + adc->cfg->regs->ier_ovr.mask); > +} > + > +static void stm32_adc_ovr_irq_disable(struct stm32_adc *adc) > +{ > + stm32_adc_clr_bits(adc, adc->cfg->regs->ier_ovr.reg, > + adc->cfg->regs->ier_ovr.mask); > +} > + > static void stm32_adc_set_res(struct stm32_adc *adc) > { > const struct stm32_adc_regs *res = &adc->cfg->regs->res; > @@ -1205,6 +1225,19 @@ static int stm32_adc_read_raw(struct iio_dev *indio_dev, > } > } > > +static irqreturn_t stm32_adc_threaded_isr(int irq, void *data) > +{ > + struct stm32_adc *adc = data; > + struct iio_dev *indio_dev = iio_priv_to_dev(adc); > + const struct stm32_adc_regspec *regs = adc->cfg->regs; > + u32 status = stm32_adc_readl(adc, regs->isr_eoc.reg); > + > + if (status & regs->isr_ovr.mask) > + dev_err(&indio_dev->dev, "Overrun, stopping: restart needed\n"); > + > + return IRQ_HANDLED; > +} > + > static irqreturn_t stm32_adc_isr(int irq, void *data) > { > struct stm32_adc *adc = data; > @@ -1212,6 +1245,19 @@ static irqreturn_t stm32_adc_isr(int irq, void *data) > const struct stm32_adc_regspec *regs = adc->cfg->regs; > u32 status = stm32_adc_readl(adc, regs->isr_eoc.reg); > > + if (status & regs->isr_ovr.mask) { > + /* > + * Overrun occurred on regular conversions: data for wrong > + * channel may be read. Unconditionally disable interrupts > + * to stop processing data and print error message. > + * Restarting the capture can be done by disabling, then > + * re-enabling it (e.g. write 0, then 1 to buffer/enable). > + */ > + stm32_adc_ovr_irq_disable(adc); > + stm32_adc_conv_irq_disable(adc); > + return IRQ_WAKE_THREAD; > + } > + > if (status & regs->isr_eoc.mask) { > /* Reading DR also clears EOC status flag */ > adc->buffer[adc->bufi] = stm32_adc_readw(adc, regs->dr); > @@ -1441,6 +1487,8 @@ static int __stm32_adc_buffer_postenable(struct iio_dev *indio_dev) > /* Reset adc buffer index */ > adc->bufi = 0; > > + stm32_adc_ovr_irq_enable(adc); > + > if (!adc->dma_chan) > stm32_adc_conv_irq_enable(adc); > > @@ -1481,6 +1529,8 @@ static void __stm32_adc_buffer_predisable(struct iio_dev *indio_dev) > if (!adc->dma_chan) > stm32_adc_conv_irq_disable(adc); > > + stm32_adc_ovr_irq_disable(adc); > + > if (adc->dma_chan) > dmaengine_terminate_sync(adc->dma_chan); > > @@ -1818,8 +1868,9 @@ static int stm32_adc_probe(struct platform_device *pdev) > if (adc->irq < 0) > return adc->irq; > > - ret = devm_request_irq(&pdev->dev, adc->irq, stm32_adc_isr, > - 0, pdev->name, adc); > + ret = devm_request_threaded_irq(&pdev->dev, adc->irq, stm32_adc_isr, > + stm32_adc_threaded_isr, > + 0, pdev->name, adc); > if (ret) { > dev_err(&pdev->dev, "failed to request IRQ\n"); > return ret;