Re: [PATCH 2/2] iio: light: Add APDS9160 ALS & Proximity sensor driver
From: Matti Vaittinen
Date: Mon Dec 02 2024 - 03:22:34 EST
Hi Jonathan & Mikael,
On 01/12/2024 15:20, Jonathan Cameron wrote:
+};
+MODULE_DEVICE_TABLE(of, apds9160_of_match);
+
+static struct i2c_driver apds9160_driver = {
+ .driver = {
+ .name = APDS9160_DRIVER_NAME,
+ .owner = THIS_MODULE,
+ .of_match_table = apds9160_of_match,
+ },
+ .probe = apds9160_probe,
+ .remove = apds9160_remove,
+ .id_table = apds9160_id,
+};
First, regarding the integration time/gain/scale parameters. I took a look at the datasheet again as there is a table
provided to get lux/count (scale?) for the ALS sensor depending on gain and integration time.
It looks like the correlation in the table is almost linear but it's not as there is a loss of precision.
For example, at 1x gain with integration time 100ms the lux/count is 0.819 but at 3x the table is stating 0.269 instead of exepected 0.273.
Is it still possible to use the gts helpers in that case?
Ah. Probably not if it goes non linear. Matti? (+CC)
Disclaimer - I didn't go through the patch and I just respond from the
top of my head :) So, please take my words with a pinch of salt.
AFAIR, it is not required that the impact of integration time is
_linear_ through the range. The "multiplication factor" can be set for
each integration time separately. So, it is perfectly Ok to say:
time 1 => multiply by 1
time 2 => multiply by 2
time 10 => multiply by 9 <= not linear, as linear would be 10.
time 15 => multiply by 15
...
The notable limitation of _current_ implementation is that the
"multiplication factor" needs to be integer. So, this may result loss of
accuracy.
Second, regarding the use of the IIO_CHAN_INFO_HARDWAREGAIN channel info.
I took a look at a couple of recent drivers, some use the IIO_CHAN_INFO_SCALE to ajust gain type registers.
In my use case, it feels like the scale is read-only as it is affected by the gain and integration time and both can be set independently
with their respective available values. How should I handle this?
The general preference is for the scale to be the primary control.
For a light sensor assuming the device doesn't support very long integration times, the
trade off is normally set the integration time as high as possible (as that gives lowest
noise) then tune the gain as necessary.
Another model is to let the integration time be controllable and then try and adjust
the gain to keep as close as possible to a requested scale. Matti has spent more
brain power on this than anyone so I'll over to him for more precise suggestions!
This is the very reason why I wrote the gts helpers. The recently
removed rohm-bu27008.c light-sensor driver and the rohm-bu27034.c allow
setting integration time. When time is set the driver will also set the
gain, and does this so that the scale is maintained. (Eg, if gain caused
by time is doubled by the user request, the hw-gain is halved by the
driver).
For scale setting I used logic where the scale setting tries to maintain
the integration time when possible, and only set the hardware gain.
Ideology is that when the driver allows user to set the time, the user
is expected to know whether to prefer longer time (and typically better
accuracy), or faster measurement with more hw-gain.
There are still some corner cases in these drivers where the scale can't
be maintained when time is set, or time can't be maintained when scale
is set. AFAIR, These drivers chose to in any case set the requested
time/scale - which means the userland needs to be prepared to "pick up
the pieces" and deal with the other entity changing. Other option would
be to deny such changes, but it would limit the use of the hardware -
and I hate drivers trying to outsmart the users.
Where this gets really hairy is when there are multiple channels and
some of the controls can be set per channel, and some are common for all
channels. As an example, I've faced sensors where gain can be set
separately for the channels but the time is common for all. Here we may
end up in situations where part of the channels can compensate the
changes while others can't. Furthermore, there were situations where
some of the scales could only be achieved with a specific integration
times. This can get very confusing for users.
In order to make it somehow tolerable the gts helpers support also
advertising only scales which can be supported with currently selected
integration time via the "available_scales". Also, I strongly recommend
having at least a read-only hardwaregain to help understanding which
part of the scale is contributed by time, and which by hardwaregain. I
think this is very helpful when debugging ;)
I know Jonathan (and maybe rest of the world :) ) prefers keeping the
scale as the single main way of changing the gain. Still, my personal
opinion on this is that there are cases, where having both the
hardwaregain and integration time directly changeable would be the
simplest and clearest solution.
Putting my opinions aside (as this is not a battle I feel like fighting
- nor do I feel I am the best expert on this!), with the current 'one
scale to rule them all' approach, my recommendation is to consider if
always using largest integration time and smallest hardwaregain to
achieve the requested scale fits your users. (Also, I think your case
where the impact of integration time is not strictly linear will cause
some scales to be 'almost same' but not quite. I am not sure if it makes
sense to support all of these, or just always round to the scale where
the integration time is longest). If yes, then maybe do it.
If you think the longest times may cause unacceptable wait times for
some users, or that the small scale differences really matter, then you
may want to go the route bu27008 / bu27034 drivers took - but it may get
hairy.
Finally, maybe taking a look if the integration time multiplier in gts
helpers could support fixed point wouldn't be completely futile(?) No
promises on that route though, I haven't considered this when writing
those so it may lead to dead end - but one never knows before taking a
look, right? :)
Ha. Looking all the letters I typed, I feel I helped really little while
being very verbose ... Sorry bout that but I really don't have a great
recommendation :(
Yours,
-- Matti