[<prev] [next>] [<thread-prev] [day] [month] [year] [list]
Message-ID: <21b52acb-9710-4363-803e-280773da0351@baylibre.com>
Date: Tue, 23 Sep 2025 11:03:54 -0500
From: David Lechner <dlechner@...libre.com>
To: Marcelo Schmitt <marcelo.schmitt1@...il.com>
Cc: Marcelo Schmitt <marcelo.schmitt@...log.com>, linux-iio@...r.kernel.org,
devicetree@...r.kernel.org, linux-doc@...r.kernel.org,
linux-spi@...r.kernel.org, linux-kernel@...r.kernel.org, jic23@...nel.org,
michael.hennerich@...log.com, nuno.sa@...log.com, eblanc@...libre.com,
andy@...nel.org, robh@...nel.org, krzk+dt@...nel.org, conor+dt@...nel.org,
corbet@....net, Sergiu Cuciurean <sergiu.cuciurean@...log.com>,
Trevor Gamblin <tgamblin@...libre.com>, Axel Haslam <ahaslam@...libre.com>
Subject: Re: [PATCH v2 6/8] iio: adc: ad4030: Add SPI offload support
On 9/23/25 10:27 AM, Marcelo Schmitt wrote:
> Hi David, thanks for the insightful review.
>
> On 09/22, David Lechner wrote:
>> On 9/18/25 12:39 PM, Marcelo Schmitt wrote:
...
>>> + cnv_wf.period_length_ns = DIV_ROUND_CLOSEST(NSEC_PER_SEC, freq);
>>> + /*
>>> + * The datasheet lists a minimum time of 9.8 ns, but no maximum. If the
>>> + * rounded PWM's value is less than 10, increase the target value by 10
>>> + * and attempt to round the waveform again, until the value is at least
>>> + * 10 ns. Use a separate variable to represent the target in case the
>>> + * rounding is severe enough to keep putting the first few results under
>>> + * the minimum 10ns condition checked by the while loop.
>>> + */
>>> + do {
>>> + cnv_wf.duty_length_ns = target;
>>> + ret = pwm_round_waveform_might_sleep(st->cnv_trigger, &cnv_wf);
>>> + if (ret)
>>> + return ret;
>>> + target += AD4030_TCNVH_NS;
>>> + } while (cnv_wf.duty_length_ns < AD4030_TCNVH_NS);
>>> +
>>> + if (!in_range(cnv_wf.period_length_ns, AD4030_TCYC_NS, INT_MAX))
>>> + return -EINVAL;
>>
>> I hit this error during testing with the default max_sample_rate_hz assigned
>> in probe. We could have a loop for this too to try to get the closest valid
>> period rather than erroring if the exact value isn't available.
>>
> Yes, this makes sense. Though, looping to try to get a suitable period wouldn't
> potentially also change the duty_length we settled above?
I didn't think too hard about it or debug too deep. So it might be fine the
way it is. We'll just want to make sure that when testing with a 2 MSPS part
that we can get the max sample rate without error. The ZedBoard has some funny
rounding due to clocks being divided by 3, so it could just be a case of
having to to put in 1.998 MHz to actually get 2 MHz or something like
that because of the lack of accuracy due to rounding.
>
>>> +
>>> + offload_period_ns = cnv_wf.period_length_ns;
>>> + if (st->mode == AD4030_OUT_DATA_MD_30_AVERAGED_DIFF)
>>
> ...
>>> +static int ad4030_set_sampling_freq(struct iio_dev *indio_dev, int freq)
>>> +{
>>> + struct ad4030_state *st = iio_priv(indio_dev);
>>> +
>>> + /*
>>> + * We have no control over the sampling frequency without SPI offload
>>> + * triggering.
>>> + */
>>> + if (!st->offload_trigger)
>>> + return -ENODEV;
>>> +
>>> + if (!in_range(freq, 1, st->chip->max_sample_rate_hz))
>>> + return -EINVAL;
>>> +
>>> + guard(mutex)(&st->lock);
>>
>> Why not iio_device_claim_direct() instead of a new lock? We wouldn't
>> want to change the sampling frequency during a buffered read anyway.
>> This driver already uses iio_device_claim_direct() to protect other
>> register access.
>
> The new lock is to protect concurrent updates of the oversampling and sampling
> frequency. Since, oversampling and the sampling frequency properties are
> mutually dependent one from another, a simultaneous write to those attributes
> could lead to an invalid oversamp + samp freq configuration.
I understand the need for the protection. And using iio_device_claim_direct()
seems like it could do the job without the need for an additional lock.
Powered by blists - more mailing lists