All of lore.kernel.org
 help / color / mirror / Atom feed
From: "Popa, Stefan Serban" <StefanSerban.Popa@analog.com>
To: "jic23@kernel.org" <jic23@kernel.org>
Cc: "linux-kernel@vger.kernel.org" <linux-kernel@vger.kernel.org>,
	"robh+dt@kernel.org" <robh+dt@kernel.org>,
	"knaack.h@gmx.de" <knaack.h@gmx.de>,
	"lars@metafoo.de" <lars@metafoo.de>,
	"Hennerich, Michael" <Michael.Hennerich@analog.com>,
	"linux-iio@vger.kernel.org" <linux-iio@vger.kernel.org>,
	"devel@driverdev.osuosl.org" <devel@driverdev.osuosl.org>,
	"mark.rutland@arm.com" <mark.rutland@arm.com>,
	"pmeerw@pmeerw.net" <pmeerw@pmeerw.net>,
	"stefan.popa@analog.co" <stefan.popa@analog.co>,
	"gregkh@linuxfoundation.org" <gregkh@linuxfoundation.org>
Subject: Re: [PATCH 05/11] staging: iio: adc: ad7606: Add support for threaded irq
Date: Mon, 17 Dec 2018 10:28:20 +0000	[thread overview]
Message-ID: <1545042497.12260.9.camel@analog.com> (raw)
In-Reply-To: <20181216134913.1751b640@archlinux>

On Du, 2018-12-16 at 13:49 +0000, Jonathan Cameron wrote:
> On Thu, 13 Dec 2018 14:46:17 +0200
> Stefan Popa <stefan.popa@analog.com> wrote:
> 
> > 
> > This patch replaces the use of a polling ring buffer with a threaded
> > interrupt.
> > 
> > Enabling the buffer sets the CONVST signal to high. When the rising
> > edge
> > of the CONVST is applied, BUSY signal goes logic high and transitions
> > low
> > at the end of the entire conversion process. The falling edge of the
> > BUSY
> > signal triggers the interrupt.
> > 
> > ad7606_trigger_handler() is used as bottom half of the poll function.
> > It reads data from the device and stores it in the internal buffer.
> > 
> > Signed-off-by: Stefan Popa <stefan.popa@analog.com>
> I'd like a little more info as comments in this one. See below.
> Which is another way of saying I have no idea what is going on :)
> 
> Thanks,
> 
> Jonathan.
> 

Hi Jonathan,
Thank you for the review. It turns out that there is no reason to trigger a
conversion before disabling the buffer. I will remove it in v2.

Thank you!
-Stefan

> > 
> > ---
> >  drivers/staging/iio/adc/ad7606.c | 116 +++++++++++++++++++++++++++++
> > ----------
> >  drivers/staging/iio/adc/ad7606.h |   6 +-
> >  2 files changed, 89 insertions(+), 33 deletions(-)
> > 
> > diff --git a/drivers/staging/iio/adc/ad7606.c
> > b/drivers/staging/iio/adc/ad7606.c
> > index 7191d51..13aeeec 100644
> > --- a/drivers/staging/iio/adc/ad7606.c
> > +++ b/drivers/staging/iio/adc/ad7606.c
> > @@ -21,6 +21,7 @@
> >  #include <linux/iio/sysfs.h>
> >  #include <linux/iio/buffer.h>
> >  #include <linux/iio/trigger_consumer.h>
> > +#include <linux/iio/trigger.h>
> >  #include <linux/iio/triggered_buffer.h>
> >  
> >  #include "ad7606.h"
> > @@ -81,36 +82,24 @@ static int ad7606_read_samples(struct ad7606_state
> > *st)
> >  static irqreturn_t ad7606_trigger_handler(int irq, void *p)
> >  {
> >  	struct iio_poll_func *pf = p;
> > -	struct ad7606_state *st = iio_priv(pf->indio_dev);
> > -
> > -	gpiod_set_value(st->gpio_convst, 1);
> > -
> > -	return IRQ_HANDLED;
> > -}
> > -
> > -/**
> > - * ad7606_poll_bh_to_ring() bh of trigger launched polling to ring
> > buffer
> > - * @work_s:	the work struct through which this was scheduled
> > - *
> > - * Currently there is no option in this driver to disable the saving
> > of
> > - * timestamps within the ring.
> > - * I think the one copy of this at a time was to avoid problems if the
> > - * trigger was set far too high and the reads then locked up the
> > computer.
> > - **/
> > -static void ad7606_poll_bh_to_ring(struct work_struct *work_s)
> > -{
> > -	struct ad7606_state *st = container_of(work_s, struct
> > ad7606_state,
> > -						poll_work);
> > -	struct iio_dev *indio_dev = iio_priv_to_dev(st);
> > +	struct iio_dev *indio_dev = pf->indio_dev;
> > +	struct ad7606_state *st = iio_priv(indio_dev);
> >  	int ret;
> >  
> > +	mutex_lock(&st->lock);
> > +
> >  	ret = ad7606_read_samples(st);
> >  	if (ret == 0)
> >  		iio_push_to_buffers_with_timestamp(indio_dev, st-
> > >data,
> >  						   iio_get_time_ns(ind
> > io_dev));
> >  
> > -	gpiod_set_value(st->gpio_convst, 0);
> >  	iio_trigger_notify_done(indio_dev->trig);
> > +	/* The rising edge of the CONVST signal starts a new
> > conversion. */
> > +	gpiod_set_value(st->gpio_convst, 1);
> > +
> > +	mutex_unlock(&st->lock);
> > +
> > +	return IRQ_HANDLED;
> >  }
> >  
> >  static int ad7606_scan_direct(struct iio_dev *indio_dev, unsigned int
> > ch)
> > @@ -378,8 +367,11 @@ static int ad7606_request_gpios(struct
> > ad7606_state *st)
> >  	return PTR_ERR_OR_ZERO(st->gpio_os);
> >  }
> >  
> > -/**
> > - *  Interrupt handler
> > +/*
> > + * The BUSY signal indicates when conversions are in progress, so when
> > a rising
> > + * edge of CONVST is applied, BUSY goes logic high and transitions low
> > at the
> > + * end of the entire conversion process. The falling edge of the BUSY
> > signal
> > + * triggers this interrupt.
> >   */
> >  static irqreturn_t ad7606_interrupt(int irq, void *dev_id)
> >  {
> > @@ -387,7 +379,8 @@ static irqreturn_t ad7606_interrupt(int irq, void
> > *dev_id)
> >  	struct ad7606_state *st = iio_priv(indio_dev);
> >  
> >  	if (iio_buffer_enabled(indio_dev)) {
> > -		schedule_work(&st->poll_work);
> > +		gpiod_set_value(st->gpio_convst, 0);
> > +		iio_trigger_poll_chained(st->trig);
> >  	} else {
> >  		complete(&st->completion);
> >  	}
> > @@ -395,26 +388,74 @@ static irqreturn_t ad7606_interrupt(int irq, void
> > *dev_id)
> >  	return IRQ_HANDLED;
> >  };
> >  
> > +static int ad7606_validate_trigger(struct iio_dev *indio_dev,
> > +				   struct iio_trigger *trig)
> > +{
> > +	struct ad7606_state *st = iio_priv(indio_dev);
> > +
> > +	if (st->trig != trig)
> > +		return -EINVAL;
> > +
> > +	return 0;
> > +}
> > +
> > +static int ad7606_buffer_postenable(struct iio_dev *indio_dev)
> > +{
> > +	struct ad7606_state *st = iio_priv(indio_dev);
> > +
> > +	iio_triggered_buffer_postenable(indio_dev);
> > +	gpiod_set_value(st->gpio_convst, 1);
> > +
> > +	return 0;
> > +}
> > +
> > +static int ad7606_buffer_predisable(struct iio_dev *indio_dev)
> > +{
> > +	struct ad7606_state *st = iio_priv(indio_dev);
> > +	int ret;
> > +
> > +	reinit_completion(&st->completion);
> > +	gpiod_set_value(st->gpio_convst, 1);
> > +	ret = wait_for_completion_timeout(&st->completion,
> > +					  msecs_to_jiffies(1000));
> Along with Dan's comment. I'd like to see a comment here on what
> is actually going on.  Not immediately obvious a conversion should
> be triggered to disable the buffer...
> 
> I'll guess there is a race against the normal handler that we
> are closing with this dance, but that race needs explaining.
> 
> > 
> > +	gpiod_set_value(st->gpio_convst, 0);
> > +
> > +	return iio_triggered_buffer_predisable(indio_dev);
> > +}
> > +
> > +static const struct iio_buffer_setup_ops ad7606_buffer_ops = {
> > +	.postenable = &ad7606_buffer_postenable,
> > +	.predisable = &ad7606_buffer_predisable,
> > +};
> > +
> >  static const struct iio_info ad7606_info_no_os_or_range = {
> >  	.read_raw = &ad7606_read_raw,
> > +	.validate_trigger = &ad7606_validate_trigger,
> >  };
> >  
> >  static const struct iio_info ad7606_info_os_and_range = {
> >  	.read_raw = &ad7606_read_raw,
> >  	.write_raw = &ad7606_write_raw,
> >  	.attrs = &ad7606_attribute_group_os_and_range,
> > +	.validate_trigger = &ad7606_validate_trigger,
> >  };
> >  
> >  static const struct iio_info ad7606_info_os = {
> >  	.read_raw = &ad7606_read_raw,
> >  	.write_raw = &ad7606_write_raw,
> >  	.attrs = &ad7606_attribute_group_os,
> > +	.validate_trigger = &ad7606_validate_trigger,
> >  };
> >  
> >  static const struct iio_info ad7606_info_range = {
> >  	.read_raw = &ad7606_read_raw,
> >  	.write_raw = &ad7606_write_raw,
> >  	.attrs = &ad7606_attribute_group_range,
> > +	.validate_trigger = &ad7606_validate_trigger,
> > +};
> > +
> > +static const struct iio_trigger_ops ad7606_trigger_ops = {
> > +	.validate_device = iio_trigger_validate_own_device,
> >  };
> >  
> >  static void ad7606_regulator_disable(void *data)
> > @@ -446,7 +487,6 @@ int ad7606_probe(struct device *dev, int irq, void
> > __iomem *base_address,
> >  	/* tied to logic low, analog input range is +/- 5V */
> >  	st->range = 0;
> >  	st->oversampling = 1;
> > -	INIT_WORK(&st->poll_work, &ad7606_poll_bh_to_ring);
> >  
> >  	st->reg = devm_regulator_get(dev, "avcc");
> >  	if (IS_ERR(st->reg))
> > @@ -491,14 +531,32 @@ int ad7606_probe(struct device *dev, int irq,
> > void __iomem *base_address,
> >  	if (ret)
> >  		dev_warn(st->dev, "failed to RESET: no RESET GPIO
> > specified\n");
> >  
> > -	ret = devm_request_irq(dev, irq, ad7606_interrupt,
> > IRQF_TRIGGER_FALLING,
> > -			       name, indio_dev);
> > +	st->trig = devm_iio_trigger_alloc(dev, "%s-dev%d",
> > +					  indio_dev->name, indio_dev-
> > >id);
> > +	if (!st->trig)
> > +		return -ENOMEM;
> > +
> > +	st->trig->ops = &ad7606_trigger_ops;
> > +	st->trig->dev.parent = dev;
> > +	iio_trigger_set_drvdata(st->trig, indio_dev);
> > +	ret = devm_iio_trigger_register(dev, st->trig);
> > +	if (ret)
> > +		return ret;
> > +
> > +	indio_dev->trig = iio_trigger_get(st->trig);
> > +
> > +	ret = devm_request_threaded_irq(dev, irq,
> > +					NULL,
> > +					&ad7606_interrupt,
> > +					IRQF_TRIGGER_FALLING |
> > IRQF_ONESHOT,
> > +					name, indio_dev);
> >  	if (ret)
> >  		return ret;
> >  
> >  	ret = devm_iio_triggered_buffer_setup(dev, indio_dev,
> > +					      &iio_pollfunc_store_time
> > ,
> >  					      &ad7606_trigger_handler,
> > -					      NULL, NULL);
> > +					      &ad7606_buffer_ops);
> >  	if (ret)
> >  		return ret;
> >  
> > diff --git a/drivers/staging/iio/adc/ad7606.h
> > b/drivers/staging/iio/adc/ad7606.h
> > index 70486ef..b238e9b 100644
> > --- a/drivers/staging/iio/adc/ad7606.h
> > +++ b/drivers/staging/iio/adc/ad7606.h
> > @@ -26,8 +26,6 @@ struct ad7606_chip_info {
> >   * @dev		pointer to kernel device
> >   * @chip_info		entry in the table of chips that
> > describes this device
> >   * @reg		regulator info for the the power supply of the
> > device
> > - * @poll_work		work struct for continuously reading data
> > from the device
> > - *			into an IIO triggered buffer
> >   * @bops		bus operations (SPI or parallel)
> >   * @range		voltage range selection, selects which scale
> > to apply
> >   * @oversampling	oversampling selection
> > @@ -42,14 +40,13 @@ struct ad7606_chip_info {
> >   *			is being read on the first channel
> >   * @gpio_os		GPIO descriptors to control oversampling on
> > the device
> >   * @complete		completion to indicate end of conversion
> > + * @trig		The IIO trigger associated with the device.
> >   * @data		buffer for reading data from the device
> >   */
> > -
> >  struct ad7606_state {
> >  	struct device			*dev;
> >  	const struct ad7606_chip_info	*chip_info;
> >  	struct regulator		*reg;
> > -	struct work_struct		poll_work;
> >  	const struct ad7606_bus_ops	*bops;
> >  	unsigned int			range;
> >  	unsigned int			oversampling;
> > @@ -62,6 +59,7 @@ struct ad7606_state {
> >  	struct gpio_desc		*gpio_standby;
> >  	struct gpio_desc		*gpio_frstdata;
> >  	struct gpio_descs		*gpio_os;
> > +	struct iio_trigger		*trig;
> >  	struct completion		completion;
> >  
> >  	/*

  reply	other threads:[~2018-12-17 10:28 UTC|newest]

Thread overview: 26+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2018-12-13 12:46 [PATCH 00/11] staging: iio: ad7606: Move out of staging Stefan Popa
2018-12-13 12:46 ` [PATCH 01/11] staging: iio: adc: ad7606: Simplify the Kconfing menu Stefan Popa
2018-12-13 15:37   ` Rob Herring
2018-12-13 12:46 ` [PATCH 02/11] staging: iio: adc: ad7606: Use SPDX identifier Stefan Popa
2018-12-13 13:21   ` Dan Carpenter
2018-12-13 12:46 ` [PATCH 03/11] staging: iio: adc: ad7606: Use wait-for-completion handler Stefan Popa
2018-12-16 13:41   ` Jonathan Cameron
2018-12-13 12:46 ` [PATCH 04/11] staging: iio: adc: ad7606: Use devm functions in probe Stefan Popa
2018-12-16 13:43   ` Jonathan Cameron
2018-12-13 12:46 ` [PATCH 05/11] staging: iio: adc: ad7606: Add support for threaded irq Stefan Popa
2018-12-13 13:28   ` Dan Carpenter
2018-12-16 13:49   ` Jonathan Cameron
2018-12-17 10:28     ` Popa, Stefan Serban [this message]
2018-12-13 12:46 ` [PATCH 06/11] staging: iio: adc: ad7606: Use find_closest() macro Stefan Popa
2018-12-16 13:51   ` Jonathan Cameron
2018-12-13 12:46 ` [PATCH 07/11] staging: iio: adc: ad7606: Use vendor prefix for DT properties Stefan Popa
2018-12-16 13:53   ` Jonathan Cameron
2018-12-13 12:46 ` [PATCH 08/11] staging: iio: adc: ad7606: Add OF device ID table Stefan Popa
2018-12-16 13:54   ` Jonathan Cameron
2018-12-13 12:46 ` [PATCH 09/11] staging: iio: adc: ad7606: Misc style fixes (no functional change) Stefan Popa
2018-12-16 13:56   ` Jonathan Cameron
2018-12-13 12:46 ` [PATCH 10/11] staging: iio: adc: ad7606: Move out of staging Stefan Popa
2018-12-16 14:14   ` Jonathan Cameron
2018-12-20 19:45   ` kbuild test robot
2018-12-13 12:46 ` [PATCH 11/11] dt-bindings: iio: adc: Add docs for AD7606 ADC Stefan Popa
2018-12-16 13:58   ` Jonathan Cameron

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=1545042497.12260.9.camel@analog.com \
    --to=stefanserban.popa@analog.com \
    --cc=Michael.Hennerich@analog.com \
    --cc=devel@driverdev.osuosl.org \
    --cc=gregkh@linuxfoundation.org \
    --cc=jic23@kernel.org \
    --cc=knaack.h@gmx.de \
    --cc=lars@metafoo.de \
    --cc=linux-iio@vger.kernel.org \
    --cc=linux-kernel@vger.kernel.org \
    --cc=mark.rutland@arm.com \
    --cc=pmeerw@pmeerw.net \
    --cc=robh+dt@kernel.org \
    --cc=stefan.popa@analog.co \
    /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 an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.