[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-ID: <44f47927-52aa-5248-6ae4-21028076fd51@axentia.se>
Date: Thu, 2 May 2024 15:49:03 +0200
From: Peter Rosin <peda@...ntia.se>
To: Jonathan Cameron <Jonathan.Cameron@...wei.com>,
João Paulo Gonçalves <jpaulo.silvagoncalves@...il.com>
Cc: linux-iio@...r.kernel.org, linux-kernel@...r.kernel.org,
joao.goncalves@...adex.com
Subject: Re: Supporting a Device with Switchable Current/Voltage Measurement
Hi!
2024-05-02 at 14:36, Jonathan Cameron wrote:
> On Wed, 1 May 2024 20:38:53 -0300
> João Paulo Gonçalves <jpaulo.silvagoncalves@...il.com> wrote:
>
>> Hello all,
>>
>> We need to support a hardware that can measure current and voltage on
>> the same differential analog input, similar to a multimeter. The mode
>> of measurement is controlled by a GPIO switch and goes to different
>> ADC inputs depending on the mode. If the switch is enabled, a current
>> loop with a shunt is enabled for current measurement; otherwise, voltage
>> is measured. From the software point of view, we are considering using
>> the iio-rescale driver as a consumer of an ADC IIO parent device. One
>> of the problems is that we need to change the mode of measurement at
>> runtime, but we are trying to avoid using some userspace "hack". The
>> other is that for a minimal solution to enable the mode from boot, we
>> can use a gpio-hog and control it with overlays. However,
>> still would be better that this was done by the kernel. Do you know
>> or have some guidance on how to properly support this in the kernel?
>>
>> For the in kernel gpio solution, this is a draft of DT we are thinking:
>>
>> current-sense {
>> compatible = "current-sense-shunt";
>> io-channels = <&adc 0>;
>> gpio = <&main_gpio0 29 GPIO_ACTIVE_HIGH>;
>> shunt-resistor-micro-ohms = <3300000>;
>> };
>>
>> voltage-sense {
>> compatible = "voltage-divider";
>> io-channels = <&adc 1>;
>> gpio = <&main_gpio0 29 GPIO_ACTIVE_LOW>;
>> output-ohms = <22>;
>> full-ohms = <222>;
>> };
>>
>> Regards,
>> João Paulo Gonçalves
>>
> +CC Peter Rosin who wrote all the relevant parts you need I think.
>>
>
> Superficially sounds like you want a mixture of appropriate analog front ends
> and a Mux. I haven't tried the combination but it should be possible to do
> something like this with
>
> An IIO mux via this binding
> https://elixir.bootlin.com/linux/v6.9-rc6/source/Documentation/devicetree/bindings/iio/multiplexer/io-channel-mux.yaml
> (that includes a gpio-mux example).
>
> Consumed in turn by a pair of AFE devices.
>
> Then you should be able to just read from which ever of the AFE device you want.
> A sysfs read from
> /sys/bus/iio/devices/iio\:deviceA/in_voltage_raw
> will switch the mux to appropriate place then request the
> voltage from the iio-mux, which in turn requests it from the ADC IIO driver.
>
> /sys/bus/iio/devices/iio\:deviceB/in_current_raw
> switches the mux the other way and otherwise the flow as above.
Since you appear to need to change both the gpio pin and the io-channel, the
mux isn't a perfect fit. The closest you can get with the current code is to
create a gpio mux, I think. You would then use that mux twice to fan out both
io-channels, but only expose the "left leg" on the first fan-out and only the
"right leg" on the other. Something like this (untested, probably riddled with
errors, use salt etc etc):
rcs: raw-current-sense {
compatible = "current-sense-shunt";
io-channels = <&adc 0>;
io-channel-name = "raw-current";
#io-channel-cells = <1>;
shunt-resistor-micro-ohms = <3300000>;
};
rvs: raw-voltage-sense {
compatible = "voltage-divider";
io-channels = <&adc 1>;
io-channel-name = "raw-voltage";
#io-channel-cells = <1>;
output-ohms = <22>;
full-ohms = <222>;
};
mux: gpio-mux {
compatible = "gpio-mux";
#mux-control-cells = <0>;
gpios-mux = <&main_gpio0 29 GPIO_ACTIVE_HIGH>;
};
current-sense {
compatible = "io-channel-mux";
io-channels = <&rcs 0>;
io-channel-names = "parent";
mux-controls = <&mux>;
channels = "current", "";
};
voltage-sense {
compatible = "io-channel-mux";
io-channels = <&rvs 0>;
io-channel-names = "parent";
mux-controls = <&mux>;
channels = "", "voltage";
};
What the mux solves is exclusion, so that the gpio pin is locked while
measurement is made on either current-sense or voltage-sense.
However, the channels from the raw-{current,voltage}-sense nodes are exposed
to user space, and it will be possible to make "raw" measurements without
regard to how the gpio pin is set. That will of course not yield the desired
results, but is also a user error and might not be a big problem?
Cheers,
Peter
Powered by blists - more mailing lists