Message ID | 1479384616-12479-3-git-send-email-tom.levens@cern.ch (mailing list archive) |
---|---|
State | Changes Requested |
Headers | show |
On 17 Nov 2016, at 22:54, Guenter Roeck <linux@roeck-us.net> wrote: > On Thu, Nov 17, 2016 at 08:52:12PM +0100, Mike Looijmans wrote: >> On 17-11-2016 19:56, Guenter Roeck wrote: >>> On Thu, Nov 17, 2016 at 06:40:17PM +0100, Mike Looijmans wrote: >>>> On 17-11-16 17:56, Guenter Roeck wrote: >>>>> On 11/17/2016 04:10 AM, Tom Levens wrote: >>>>>> Updated version of the ltc2990 driver which supports all measurement >>>>>> modes available in the chip. The mode can be set through a devicetree >>>>>> attribute. >>>>> >>> [ ... ] >>> >>>>>> >>>>>> static int ltc2990_i2c_probe(struct i2c_client *i2c, >>>>>> const struct i2c_device_id *id) >>>>>> { >>>>>> int ret; >>>>>> struct device *hwmon_dev; >>>>>> + struct ltc2990_data *data; >>>>>> + struct device_node *of_node = i2c->dev.of_node; >>>>>> >>>>>> if (!i2c_check_functionality(i2c->adapter, >>>>>> I2C_FUNC_SMBUS_BYTE_DATA | >>>>>> I2C_FUNC_SMBUS_WORD_DATA)) >>>>>> return -ENODEV; >>>>>> >>>>>> - /* Setup continuous mode, current monitor */ >>>>>> + data = devm_kzalloc(&i2c->dev, sizeof(struct ltc2990_data), >>>>>> GFP_KERNEL); >>>>>> + if (unlikely(!data)) >>>>>> + return -ENOMEM; >>>>>> + data->i2c = i2c; >>>>>> + >>>>>> + if (!of_node || of_property_read_u32(of_node, "lltc,mode", >>>>>> &data->mode)) >>>>>> + data->mode = LTC2990_CONTROL_MODE_DEFAULT; >>>>> >>>>> Iam arguing with myself if we should still do this or if we should read >>>>> the mode >>>>> from the chip instead if it isn't provided (after all, it may have been >>>>> initialized >>>>> by the BIOS/ROMMON). >>>> >>>> I think the mode should be explicitly set, without default. There's no way >>>> to tell whether the BIOS or bootloader has really set it up or whether the >>>> chip is just reporting whatever it happened to default to. And given the >>>> chip's function, it's unlikely a bootloader would want to initialize it. >>>> >>> Unlikely but possible. Even if we all agree that the chip should be configured >>> by the driver, I don't like imposing that view on everyone else. >>> >>>> My advice would be to make it a required property. If not set, display an >>>> error and bail out. >>>> >>> It is not that easy, unfortunately. It also has to work on a non-devicetree >>> system. I would not object to making the property mandatory, but we would >>> still need to provide non-DT support. >>> >>> My "use case" for taking the current mode from the chip if not specified >>> is that it would enable me to run a module test with all modes. I consider >>> this extremely valuable. >> >> Good point. >> >> The chip defaults to measuring internal temperature only, and the mode >> defaults to "0". >> >> Choosing a mode that doesn't match the actual circuitry could be bad for the >> chip or the board (though unlikely, it'll probably just be useless) since it >> will actively drive some of the inputs in the temperature modes (which is >> default for V3/V4 pins). >> >> Instead of failing, one could choose to set the default mode to "7", which >> just measures the 4 voltages, which would be a harmless mode in all cases. >> >> As a way to let a bootloader set things up, I think it would be a good check >> to see if CONTROL register bits 4:3 are set. If "00", the chip is not >> acquiring data at all, and probably needs configuration still. In that case, >> the mode must be provided by the devicetree (or the default "7"). >> If bits 4:3 are "11", it has already been set up to measure its inputs, and >> it's okay to continue doing just that and use the current value of 2:0 >> register as default mode (if the devicetree didn't specify any mode at all). >> > > At first glance, agreed, though by default b[3:4] are 00, and only the > internal temperature is measured. Actually, the 5 mode bits are all > relevant to determine what is measured. Maybe it would be better to take > all 5 bits into account instead of blindly setting b[34]:=11 and a specific > setting of b[0:2]. Sure, that would make the mode table a bit larger, > but then ltc2990_attrs_ena[] could be made an u16 array, and a table size > of 64 bytes would not be that bad. I would tend to agree that it should be possible to configure all 5 mode bits through the devicetree. What I would propose is as follows. If a devicetree node exists, the mode parameter(s?) are required and the chip is initialised. If a devicetree node doesn't exist, it is assumed that the chip has already been configured (by the BIOS, etc). The mode is read from the chip to set the visibility of the sysfs attributes. In the worst case, where the chip has not been configured by another source, it would only be possible to measure the internal temperature -- but I think this is an acceptable limitation. The only case that this does not cover is if the device tree node exists but the chip is expected to be configured by some other source. Maybe I am wrong, but I would not expect this to be a terribly common situation. What do you think? Cheers, Tom
On Thu, Nov 17, 2016 at 11:25:30PM +0000, Tom Levens wrote: > On 17 Nov 2016, at 22:54, Guenter Roeck <linux@roeck-us.net> wrote: > > On Thu, Nov 17, 2016 at 08:52:12PM +0100, Mike Looijmans wrote: > >> On 17-11-2016 19:56, Guenter Roeck wrote: > >>> On Thu, Nov 17, 2016 at 06:40:17PM +0100, Mike Looijmans wrote: > >>>> On 17-11-16 17:56, Guenter Roeck wrote: > >>>>> On 11/17/2016 04:10 AM, Tom Levens wrote: > >>>>>> Updated version of the ltc2990 driver which supports all measurement > >>>>>> modes available in the chip. The mode can be set through a devicetree > >>>>>> attribute. > >>>>> > >>> [ ... ] > >>> > >>>>>> > >>>>>> static int ltc2990_i2c_probe(struct i2c_client *i2c, > >>>>>> const struct i2c_device_id *id) > >>>>>> { > >>>>>> int ret; > >>>>>> struct device *hwmon_dev; > >>>>>> + struct ltc2990_data *data; > >>>>>> + struct device_node *of_node = i2c->dev.of_node; > >>>>>> > >>>>>> if (!i2c_check_functionality(i2c->adapter, > >>>>>> I2C_FUNC_SMBUS_BYTE_DATA | > >>>>>> I2C_FUNC_SMBUS_WORD_DATA)) > >>>>>> return -ENODEV; > >>>>>> > >>>>>> - /* Setup continuous mode, current monitor */ > >>>>>> + data = devm_kzalloc(&i2c->dev, sizeof(struct ltc2990_data), > >>>>>> GFP_KERNEL); > >>>>>> + if (unlikely(!data)) > >>>>>> + return -ENOMEM; > >>>>>> + data->i2c = i2c; > >>>>>> + > >>>>>> + if (!of_node || of_property_read_u32(of_node, "lltc,mode", > >>>>>> &data->mode)) > >>>>>> + data->mode = LTC2990_CONTROL_MODE_DEFAULT; > >>>>> > >>>>> Iam arguing with myself if we should still do this or if we should read > >>>>> the mode > >>>>> from the chip instead if it isn't provided (after all, it may have been > >>>>> initialized > >>>>> by the BIOS/ROMMON). > >>>> > >>>> I think the mode should be explicitly set, without default. There's no way > >>>> to tell whether the BIOS or bootloader has really set it up or whether the > >>>> chip is just reporting whatever it happened to default to. And given the > >>>> chip's function, it's unlikely a bootloader would want to initialize it. > >>>> > >>> Unlikely but possible. Even if we all agree that the chip should be configured > >>> by the driver, I don't like imposing that view on everyone else. > >>> > >>>> My advice would be to make it a required property. If not set, display an > >>>> error and bail out. > >>>> > >>> It is not that easy, unfortunately. It also has to work on a non-devicetree > >>> system. I would not object to making the property mandatory, but we would > >>> still need to provide non-DT support. > >>> > >>> My "use case" for taking the current mode from the chip if not specified > >>> is that it would enable me to run a module test with all modes. I consider > >>> this extremely valuable. > >> > >> Good point. > >> > >> The chip defaults to measuring internal temperature only, and the mode > >> defaults to "0". > >> > >> Choosing a mode that doesn't match the actual circuitry could be bad for the > >> chip or the board (though unlikely, it'll probably just be useless) since it > >> will actively drive some of the inputs in the temperature modes (which is > >> default for V3/V4 pins). > >> > >> Instead of failing, one could choose to set the default mode to "7", which > >> just measures the 4 voltages, which would be a harmless mode in all cases. > >> > >> As a way to let a bootloader set things up, I think it would be a good check > >> to see if CONTROL register bits 4:3 are set. If "00", the chip is not > >> acquiring data at all, and probably needs configuration still. In that case, > >> the mode must be provided by the devicetree (or the default "7"). > >> If bits 4:3 are "11", it has already been set up to measure its inputs, and > >> it's okay to continue doing just that and use the current value of 2:0 > >> register as default mode (if the devicetree didn't specify any mode at all). > >> > > > > At first glance, agreed, though by default b[3:4] are 00, and only the > > internal temperature is measured. Actually, the 5 mode bits are all > > relevant to determine what is measured. Maybe it would be better to take > > all 5 bits into account instead of blindly setting b[34]:=11 and a specific > > setting of b[0:2]. Sure, that would make the mode table a bit larger, > > but then ltc2990_attrs_ena[] could be made an u16 array, and a table size > > of 64 bytes would not be that bad. > > I would tend to agree that it should be possible to configure all 5 mode > bits through the devicetree. What I would propose is as follows. > > If a devicetree node exists, the mode parameter(s?) are required and the > chip is initialised. > > If a devicetree node doesn't exist, it is assumed that the chip has > already been configured (by the BIOS, etc). The mode is read from the > chip to set the visibility of the sysfs attributes. In the worst case, where the > chip has not been configured by another source, it would only be possible > to measure the internal temperature -- but I think this is an acceptable > limitation. > SGTM. > The only case that this does not cover is if the device tree node > exists but the chip is expected to be configured by some other source. > Maybe I am wrong, but I would not expect this to be a terribly common > situation. > > What do you think? > I would not bother about this case. Just make the mode property mandatory. Thanks, Guenter -- To unsubscribe from this list: send the line "unsubscribe linux-hwmon" in the body of a message to majordomo@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html
On Fri, 18 Nov 2016, Guenter Roeck wrote: > On Thu, Nov 17, 2016 at 11:25:30PM +0000, Tom Levens wrote: >> On 17 Nov 2016, at 22:54, Guenter Roeck <linux@roeck-us.net> wrote: >>> On Thu, Nov 17, 2016 at 08:52:12PM +0100, Mike Looijmans wrote: >>>> On 17-11-2016 19:56, Guenter Roeck wrote: >>>>> On Thu, Nov 17, 2016 at 06:40:17PM +0100, Mike Looijmans wrote: >>>>>> On 17-11-16 17:56, Guenter Roeck wrote: >>>>>>> On 11/17/2016 04:10 AM, Tom Levens wrote: >>>>>>>> Updated version of the ltc2990 driver which supports all measurement >>>>>>>> modes available in the chip. The mode can be set through a devicetree >>>>>>>> attribute. >>>>>>> >>>>> [ ... ] >>>>> >>>>>>>> >>>>>>>> static int ltc2990_i2c_probe(struct i2c_client *i2c, >>>>>>>> const struct i2c_device_id *id) >>>>>>>> { >>>>>>>> int ret; >>>>>>>> struct device *hwmon_dev; >>>>>>>> + struct ltc2990_data *data; >>>>>>>> + struct device_node *of_node = i2c->dev.of_node; >>>>>>>> >>>>>>>> if (!i2c_check_functionality(i2c->adapter, >>>>>>>> I2C_FUNC_SMBUS_BYTE_DATA | >>>>>>>> I2C_FUNC_SMBUS_WORD_DATA)) >>>>>>>> return -ENODEV; >>>>>>>> >>>>>>>> - /* Setup continuous mode, current monitor */ >>>>>>>> + data = devm_kzalloc(&i2c->dev, sizeof(struct ltc2990_data), >>>>>>>> GFP_KERNEL); >>>>>>>> + if (unlikely(!data)) >>>>>>>> + return -ENOMEM; >>>>>>>> + data->i2c = i2c; >>>>>>>> + >>>>>>>> + if (!of_node || of_property_read_u32(of_node, "lltc,mode", >>>>>>>> &data->mode)) >>>>>>>> + data->mode = LTC2990_CONTROL_MODE_DEFAULT; >>>>>>> >>>>>>> Iam arguing with myself if we should still do this or if we should read >>>>>>> the mode >>>>>>> from the chip instead if it isn't provided (after all, it may have been >>>>>>> initialized >>>>>>> by the BIOS/ROMMON). >>>>>> >>>>>> I think the mode should be explicitly set, without default. There's no way >>>>>> to tell whether the BIOS or bootloader has really set it up or whether the >>>>>> chip is just reporting whatever it happened to default to. And given the >>>>>> chip's function, it's unlikely a bootloader would want to initialize it. >>>>>> >>>>> Unlikely but possible. Even if we all agree that the chip should be configured >>>>> by the driver, I don't like imposing that view on everyone else. >>>>> >>>>>> My advice would be to make it a required property. If not set, display an >>>>>> error and bail out. >>>>>> >>>>> It is not that easy, unfortunately. It also has to work on a non-devicetree >>>>> system. I would not object to making the property mandatory, but we would >>>>> still need to provide non-DT support. >>>>> >>>>> My "use case" for taking the current mode from the chip if not specified >>>>> is that it would enable me to run a module test with all modes. I consider >>>>> this extremely valuable. >>>> >>>> Good point. >>>> >>>> The chip defaults to measuring internal temperature only, and the mode >>>> defaults to "0". >>>> >>>> Choosing a mode that doesn't match the actual circuitry could be bad for the >>>> chip or the board (though unlikely, it'll probably just be useless) since it >>>> will actively drive some of the inputs in the temperature modes (which is >>>> default for V3/V4 pins). >>>> >>>> Instead of failing, one could choose to set the default mode to "7", which >>>> just measures the 4 voltages, which would be a harmless mode in all cases. >>>> >>>> As a way to let a bootloader set things up, I think it would be a good check >>>> to see if CONTROL register bits 4:3 are set. If "00", the chip is not >>>> acquiring data at all, and probably needs configuration still. In that case, >>>> the mode must be provided by the devicetree (or the default "7"). >>>> If bits 4:3 are "11", it has already been set up to measure its inputs, and >>>> it's okay to continue doing just that and use the current value of 2:0 >>>> register as default mode (if the devicetree didn't specify any mode at all). >>>> >>> >>> At first glance, agreed, though by default b[3:4] are 00, and only the >>> internal temperature is measured. Actually, the 5 mode bits are all >>> relevant to determine what is measured. Maybe it would be better to take >>> all 5 bits into account instead of blindly setting b[34]:=11 and a specific >>> setting of b[0:2]. Sure, that would make the mode table a bit larger, >>> but then ltc2990_attrs_ena[] could be made an u16 array, and a table size >>> of 64 bytes would not be that bad. >> >> I would tend to agree that it should be possible to configure all 5 mode >> bits through the devicetree. What I would propose is as follows. >> >> If a devicetree node exists, the mode parameter(s?) are required and the >> chip is initialised. >> >> If a devicetree node doesn't exist, it is assumed that the chip has >> already been configured (by the BIOS, etc). The mode is read from the >> chip to set the visibility of the sysfs attributes. In the worst case, where the >> chip has not been configured by another source, it would only be possible >> to measure the internal temperature -- but I think this is an acceptable >> limitation. >> > SGTM. > >> The only case that this does not cover is if the device tree node >> exists but the chip is expected to be configured by some other source. >> Maybe I am wrong, but I would not expect this to be a terribly common >> situation. >> >> What do you think? >> > I would not bother about this case. Just make the mode property mandatory. What do you think about making the devicetree property a list of two integers? Something like lltc,mode = <7 3>; which would set mode[2..0]=7 and mode[4..3]=3. To me, this is easier to setup from the datasheet than a single integer value. The other option would be to split it into two properties, but I am struggling to come up with suitable names for them -- the datasheet helpfully calls both fields "mode". Cheers,
On 11/18/2016 04:23 AM, Tom Levens wrote: > > > On Fri, 18 Nov 2016, Guenter Roeck wrote: > >> On Thu, Nov 17, 2016 at 11:25:30PM +0000, Tom Levens wrote: >>> On 17 Nov 2016, at 22:54, Guenter Roeck <linux@roeck-us.net> wrote: >>>> On Thu, Nov 17, 2016 at 08:52:12PM +0100, Mike Looijmans wrote: >>>>> On 17-11-2016 19:56, Guenter Roeck wrote: >>>>>> On Thu, Nov 17, 2016 at 06:40:17PM +0100, Mike Looijmans wrote: >>>>>>> On 17-11-16 17:56, Guenter Roeck wrote: >>>>>>>> On 11/17/2016 04:10 AM, Tom Levens wrote: >>>>>>>>> Updated version of the ltc2990 driver which supports all measurement >>>>>>>>> modes available in the chip. The mode can be set through a devicetree >>>>>>>>> attribute. >>>>>>>> >>>>>> [ ... ] >>>>>> >>>>>>>>> >>>>>>>>> static int ltc2990_i2c_probe(struct i2c_client *i2c, >>>>>>>>> const struct i2c_device_id *id) >>>>>>>>> { >>>>>>>>> int ret; >>>>>>>>> struct device *hwmon_dev; >>>>>>>>> + struct ltc2990_data *data; >>>>>>>>> + struct device_node *of_node = i2c->dev.of_node; >>>>>>>>> >>>>>>>>> if (!i2c_check_functionality(i2c->adapter, >>>>>>>>> I2C_FUNC_SMBUS_BYTE_DATA | >>>>>>>>> I2C_FUNC_SMBUS_WORD_DATA)) >>>>>>>>> return -ENODEV; >>>>>>>>> >>>>>>>>> - /* Setup continuous mode, current monitor */ >>>>>>>>> + data = devm_kzalloc(&i2c->dev, sizeof(struct ltc2990_data), >>>>>>>>> GFP_KERNEL); >>>>>>>>> + if (unlikely(!data)) >>>>>>>>> + return -ENOMEM; >>>>>>>>> + data->i2c = i2c; >>>>>>>>> + >>>>>>>>> + if (!of_node || of_property_read_u32(of_node, "lltc,mode", >>>>>>>>> &data->mode)) >>>>>>>>> + data->mode = LTC2990_CONTROL_MODE_DEFAULT; >>>>>>>> >>>>>>>> Iam arguing with myself if we should still do this or if we should read >>>>>>>> the mode >>>>>>>> from the chip instead if it isn't provided (after all, it may have been >>>>>>>> initialized >>>>>>>> by the BIOS/ROMMON). >>>>>>> >>>>>>> I think the mode should be explicitly set, without default. There's no way >>>>>>> to tell whether the BIOS or bootloader has really set it up or whether the >>>>>>> chip is just reporting whatever it happened to default to. And given the >>>>>>> chip's function, it's unlikely a bootloader would want to initialize it. >>>>>>> >>>>>> Unlikely but possible. Even if we all agree that the chip should be configured >>>>>> by the driver, I don't like imposing that view on everyone else. >>>>>> >>>>>>> My advice would be to make it a required property. If not set, display an >>>>>>> error and bail out. >>>>>>> >>>>>> It is not that easy, unfortunately. It also has to work on a non-devicetree >>>>>> system. I would not object to making the property mandatory, but we would >>>>>> still need to provide non-DT support. >>>>>> >>>>>> My "use case" for taking the current mode from the chip if not specified >>>>>> is that it would enable me to run a module test with all modes. I consider >>>>>> this extremely valuable. >>>>> >>>>> Good point. >>>>> >>>>> The chip defaults to measuring internal temperature only, and the mode >>>>> defaults to "0". >>>>> >>>>> Choosing a mode that doesn't match the actual circuitry could be bad for the >>>>> chip or the board (though unlikely, it'll probably just be useless) since it >>>>> will actively drive some of the inputs in the temperature modes (which is >>>>> default for V3/V4 pins). >>>>> >>>>> Instead of failing, one could choose to set the default mode to "7", which >>>>> just measures the 4 voltages, which would be a harmless mode in all cases. >>>>> >>>>> As a way to let a bootloader set things up, I think it would be a good check >>>>> to see if CONTROL register bits 4:3 are set. If "00", the chip is not >>>>> acquiring data at all, and probably needs configuration still. In that case, >>>>> the mode must be provided by the devicetree (or the default "7"). >>>>> If bits 4:3 are "11", it has already been set up to measure its inputs, and >>>>> it's okay to continue doing just that and use the current value of 2:0 >>>>> register as default mode (if the devicetree didn't specify any mode at all). >>>>> >>>> >>>> At first glance, agreed, though by default b[3:4] are 00, and only the >>>> internal temperature is measured. Actually, the 5 mode bits are all >>>> relevant to determine what is measured. Maybe it would be better to take >>>> all 5 bits into account instead of blindly setting b[34]:=11 and a specific >>>> setting of b[0:2]. Sure, that would make the mode table a bit larger, >>>> but then ltc2990_attrs_ena[] could be made an u16 array, and a table size >>>> of 64 bytes would not be that bad. >>> >>> I would tend to agree that it should be possible to configure all 5 mode >>> bits through the devicetree. What I would propose is as follows. >>> >>> If a devicetree node exists, the mode parameter(s?) are required and the >>> chip is initialised. >>> >>> If a devicetree node doesn't exist, it is assumed that the chip has >>> already been configured (by the BIOS, etc). The mode is read from the >>> chip to set the visibility of the sysfs attributes. In the worst case, where the >>> chip has not been configured by another source, it would only be possible >>> to measure the internal temperature -- but I think this is an acceptable >>> limitation. >>> >> SGTM. >> >>> The only case that this does not cover is if the device tree node >>> exists but the chip is expected to be configured by some other source. >>> Maybe I am wrong, but I would not expect this to be a terribly common >>> situation. >>> >>> What do you think? >>> >> I would not bother about this case. Just make the mode property mandatory. > > What do you think about making the devicetree property a list of two integers? Something like > > lltc,mode = <7 3>; > > which would set mode[2..0]=7 and mode[4..3]=3. > I would personally just use a single value for b[4..0]. But that is really up for bikeshedding (eg should it be <7 3> or <3 7>. I'll leave that up to Rob to decide - he knows better than me of what makes more sense from a DT perspective. Guenter -- To unsubscribe from this list: send the line "unsubscribe linux-hwmon" in the body of a message to majordomo@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html
On 17-11-16 17:56, Guenter Roeck wrote: > On 11/17/2016 04:10 AM, Tom Levens wrote: >> Updated version of the ltc2990 driver which supports all measurement >> modes available in the chip. The mode can be set through a devicetree >> attribute. > > property > >> >> Signed-off-by: Tom Levens <tom.levens@cern.ch> >> --- >> >> Changes since v1: >> * Refactored value conversion (patch 1/3) >> * Split the devicetree binding into separate patch (patch 2/3) >> * Specifying an invalid mode now returns -EINVAL, previously this >> only issued a warning and used the default value >> * Removed the "mode" sysfs attribute, as the mode of the chip is >> hardware specific and should not be user configurable. This allows much >> simpler code as a result. >> >> Documentation/hwmon/ltc2990 | 24 ++++--- >> drivers/hwmon/Kconfig | 7 +-- >> drivers/hwmon/ltc2990.c | 167 ++++++++++++++++++++++++++++++++++++------- >> 3 files changed, 159 insertions(+), 39 deletions(-) >> >> diff --git a/Documentation/hwmon/ltc2990 b/Documentation/hwmon/ltc2990 >> index c25211e..3ed68f6 100644 >> --- a/Documentation/hwmon/ltc2990 >> +++ b/Documentation/hwmon/ltc2990 >> @@ -8,6 +8,7 @@ Supported chips: >> Datasheet: http://www.linear.com/product/ltc2990 >> >> Author: Mike Looijmans <mike.looijmans@topic.nl> >> + Tom Levens <tom.levens@cern.ch> >> >> >> Description >> @@ -16,10 +17,8 @@ Description >> LTC2990 is a Quad I2C Voltage, Current and Temperature Monitor. >> The chip's inputs can measure 4 voltages, or two inputs together (1+2 and 3+4) >> can be combined to measure a differential voltage, which is typically used to >> -measure current through a series resistor, or a temperature. >> - >> -This driver currently uses the 2x differential mode only. In order to support >> -other modes, the driver will need to be expanded. >> +measure current through a series resistor, or a temperature with an external >> +diode. >> >> >> Usage Notes >> @@ -32,12 +31,19 @@ devices explicitly. >> Sysfs attributes >> ---------------- >> >> +in0_input Voltage at Vcc pin in millivolt (range 2.5V to 5V) >> +temp1_input Internal chip temperature in millidegrees Celcius >> + >> +A subset of the following attributes are visible, depending on the measurement >> +mode of the chip. >> + >> +in[1-4]_input Voltage at V[1-4] pin in millivolt >> +temp2_input External temperature sensor TR1 in millidegrees Celcius >> +temp3_input External temperature sensor TR2 in millidegrees Celcius >> +curr1_input Current in mA across V1-V2 assuming a 1mOhm sense resistor >> +curr2_input Current in mA across V3-V4 assuming a 1mOhm sense resistor >> + >> The "curr*_input" measurements actually report the voltage drop across the >> input pins in microvolts. This is equivalent to the current through a 1mOhm >> sense resistor. Divide the reported value by the actual sense resistor value >> in mOhm to get the actual value. >> - >> -in0_input Voltage at Vcc pin in millivolt (range 2.5V to 5V) >> -temp1_input Internal chip temperature in millidegrees Celcius >> -curr1_input Current in mA across v1-v2 assuming a 1mOhm sense resistor. >> -curr2_input Current in mA across v3-v4 assuming a 1mOhm sense resistor. >> diff --git a/drivers/hwmon/Kconfig b/drivers/hwmon/Kconfig >> index 45cef3d..f7096ca 100644 >> --- a/drivers/hwmon/Kconfig >> +++ b/drivers/hwmon/Kconfig >> @@ -699,15 +699,12 @@ config SENSORS_LTC2945 >> be called ltc2945. >> >> config SENSORS_LTC2990 >> - tristate "Linear Technology LTC2990 (current monitoring mode only)" >> + tristate "Linear Technology LTC2990" >> depends on I2C >> help >> If you say yes here you get support for Linear Technology LTC2990 >> I2C System Monitor. The LTC2990 supports a combination of voltage, >> - current and temperature monitoring, but in addition to the Vcc supply >> - voltage and chip temperature, this driver currently only supports >> - reading two currents by measuring two differential voltages across >> - series resistors. >> + current and temperature monitoring. >> >> This driver can also be built as a module. If so, the module will >> be called ltc2990. >> diff --git a/drivers/hwmon/ltc2990.c b/drivers/hwmon/ltc2990.c >> index 0ec4102..e8d36f5 100644 >> --- a/drivers/hwmon/ltc2990.c >> +++ b/drivers/hwmon/ltc2990.c >> @@ -6,11 +6,7 @@ >> * >> * License: GPLv2 >> * >> - * This driver assumes the chip is wired as a dual current monitor, and >> - * reports the voltage drop across two series resistors. It also reports >> - * the chip's internal temperature and Vcc power supply voltage. >> - * >> - * Value conversion refactored >> + * Value conversion refactored and support for all measurement modes added >> * by Tom Levens <tom.levens@cern.ch> >> */ >> >> @@ -21,6 +17,7 @@ >> #include <linux/i2c.h> >> #include <linux/kernel.h> >> #include <linux/module.h> >> +#include <linux/of.h> >> >> #define LTC2990_STATUS 0x00 >> #define LTC2990_CONTROL 0x01 >> @@ -35,32 +32,96 @@ >> #define LTC2990_CONTROL_KELVIN BIT(7) >> #define LTC2990_CONTROL_SINGLE BIT(6) >> #define LTC2990_CONTROL_MEASURE_ALL (0x3 << 3) >> -#define LTC2990_CONTROL_MODE_CURRENT 0x06 >> -#define LTC2990_CONTROL_MODE_VOLTAGE 0x07 >> +#define LTC2990_CONTROL_MODE_DEFAULT 0x06 > > I think LTC2990_CONTROL_MODE_CURRENT was better - it describes the actual mode. > Changing the define to _DEFAULT does not really improve code readability. > >> +#define LTC2990_CONTROL_MODE_MAX 0x07 >> + >> +#define LTC2990_IN0 BIT(0) >> +#define LTC2990_IN1 BIT(1) >> +#define LTC2990_IN2 BIT(2) >> +#define LTC2990_IN3 BIT(3) >> +#define LTC2990_IN4 BIT(4) >> +#define LTC2990_CURR1 BIT(5) >> +#define LTC2990_CURR2 BIT(6) >> +#define LTC2990_TEMP1 BIT(7) >> +#define LTC2990_TEMP2 BIT(8) >> +#define LTC2990_TEMP3 BIT(9) >> + >> +static const int ltc2990_attrs_ena[] = { >> + LTC2990_IN1 | LTC2990_IN2 | LTC2990_TEMP3, >> + LTC2990_CURR1 | LTC2990_TEMP3, >> + LTC2990_CURR1 | LTC2990_IN3 | LTC2990_IN4, >> + LTC2990_TEMP2 | LTC2990_IN3 | LTC2990_IN4, >> + LTC2990_TEMP2 | LTC2990_CURR2, >> + LTC2990_TEMP2 | LTC2990_TEMP3, >> + LTC2990_CURR1 | LTC2990_CURR2, >> + LTC2990_IN1 | LTC2990_IN2 | LTC2990_IN3 | LTC2990_IN4 >> +}; >> + >> +struct ltc2990_data { >> + struct i2c_client *i2c; >> + u32 mode; >> +}; >> >> /* Return the converted value from the given register in uV or mC */ >> -static int ltc2990_get_value(struct i2c_client *i2c, u8 reg, s32 *result) >> +static int ltc2990_get_value(struct i2c_client *i2c, int index, s32 *result) >> { >> s32 val; >> + u8 reg; >> + >> + switch (index) { >> + case LTC2990_IN0: >> + reg = LTC2990_VCC_MSB; >> + break; >> + case LTC2990_IN1: >> + case LTC2990_CURR1: >> + case LTC2990_TEMP2: >> + reg = LTC2990_V1_MSB; >> + break; >> + case LTC2990_IN2: >> + reg = LTC2990_V2_MSB; >> + break; >> + case LTC2990_IN3: >> + case LTC2990_CURR2: >> + case LTC2990_TEMP3: >> + reg = LTC2990_V3_MSB; >> + break; >> + case LTC2990_IN4: >> + reg = LTC2990_V4_MSB; >> + break; >> + case LTC2990_TEMP1: >> + reg = LTC2990_TINT_MSB; >> + break; >> + default: >> + return -EINVAL; >> + } >> >> val = i2c_smbus_read_word_swapped(i2c, reg); >> if (unlikely(val < 0)) >> return val; >> >> - switch (reg) { >> - case LTC2990_TINT_MSB: >> - /* internal temp, 0.0625 degrees/LSB, 13-bit */ >> + switch (index) { >> + case LTC2990_TEMP1: >> + case LTC2990_TEMP2: >> + case LTC2990_TEMP3: >> + /* temp, 0.0625 degrees/LSB, 13-bit */ >> *result = sign_extend32(val, 12) * 1000 / 16; >> break; >> - case LTC2990_V1_MSB: >> - case LTC2990_V3_MSB: >> - /* Vx-Vy, 19.42uV/LSB. Depends on mode. */ >> + case LTC2990_CURR1: >> + case LTC2990_CURR2: >> + /* Vx-Vy, 19.42uV/LSB */ >> *result = sign_extend32(val, 14) * 1942 / 100; >> break; >> - case LTC2990_VCC_MSB: >> - /* Vcc, 305.18μV/LSB, 2.5V offset */ >> + case LTC2990_IN0: >> + /* Vcc, 305.18uV/LSB, 2.5V offset */ >> *result = sign_extend32(val, 14) * 30518 / (100 * 1000) + 2500; >> break; >> + case LTC2990_IN1: >> + case LTC2990_IN2: >> + case LTC2990_IN3: >> + case LTC2990_IN4: >> + /* Vx: 305.18uV/LSB */ >> + *result = sign_extend32(val, 14) * 30518 / (100 * 1000); >> + break; >> default: >> return -EINVAL; /* won't happen, keep compiler happy */ >> } >> @@ -72,48 +133,104 @@ static ssize_t ltc2990_show_value(struct device *dev, >> struct device_attribute *da, char *buf) >> { >> struct sensor_device_attribute *attr = to_sensor_dev_attr(da); >> + struct ltc2990_data *data = dev_get_drvdata(dev); >> s32 value; >> int ret; >> >> - ret = ltc2990_get_value(dev_get_drvdata(dev), attr->index, &value); >> + ret = ltc2990_get_value(data->i2c, attr->index, &value); >> if (unlikely(ret < 0)) >> return ret; >> >> return snprintf(buf, PAGE_SIZE, "%d\n", value); >> } >> >> +static umode_t ltc2990_attrs_visible(struct kobject *kobj, >> + struct attribute *a, int n) >> +{ >> + struct device *dev = container_of(kobj, struct device, kobj); >> + struct ltc2990_data *data = dev_get_drvdata(dev); >> + struct device_attribute *da = >> + container_of(a, struct device_attribute, attr); >> + struct sensor_device_attribute *attr = to_sensor_dev_attr(da); >> + >> + if (attr->index == LTC2990_TEMP1 || >> + attr->index == LTC2990_IN0 || >> + attr->index & ltc2990_attrs_ena[data->mode]) >> + return a->mode; >> + else > > Unnecessary else > >> + return 0; >> +} >> + >> static SENSOR_DEVICE_ATTR(temp1_input, S_IRUGO, ltc2990_show_value, NULL, >> - LTC2990_TINT_MSB); >> + LTC2990_TEMP1); >> +static SENSOR_DEVICE_ATTR(temp2_input, S_IRUGO, ltc2990_show_value, NULL, >> + LTC2990_TEMP2); >> +static SENSOR_DEVICE_ATTR(temp3_input, S_IRUGO, ltc2990_show_value, NULL, >> + LTC2990_TEMP3); >> static SENSOR_DEVICE_ATTR(curr1_input, S_IRUGO, ltc2990_show_value, NULL, >> - LTC2990_V1_MSB); >> + LTC2990_CURR1); >> static SENSOR_DEVICE_ATTR(curr2_input, S_IRUGO, ltc2990_show_value, NULL, >> - LTC2990_V3_MSB); >> + LTC2990_CURR2); >> static SENSOR_DEVICE_ATTR(in0_input, S_IRUGO, ltc2990_show_value, NULL, >> - LTC2990_VCC_MSB); >> + LTC2990_IN0); >> +static SENSOR_DEVICE_ATTR(in1_input, S_IRUGO, ltc2990_show_value, NULL, >> + LTC2990_IN1); >> +static SENSOR_DEVICE_ATTR(in2_input, S_IRUGO, ltc2990_show_value, NULL, >> + LTC2990_IN2); >> +static SENSOR_DEVICE_ATTR(in3_input, S_IRUGO, ltc2990_show_value, NULL, >> + LTC2990_IN3); >> +static SENSOR_DEVICE_ATTR(in4_input, S_IRUGO, ltc2990_show_value, NULL, >> + LTC2990_IN4); >> >> static struct attribute *ltc2990_attrs[] = { >> &sensor_dev_attr_temp1_input.dev_attr.attr, >> + &sensor_dev_attr_temp2_input.dev_attr.attr, >> + &sensor_dev_attr_temp3_input.dev_attr.attr, >> &sensor_dev_attr_curr1_input.dev_attr.attr, >> &sensor_dev_attr_curr2_input.dev_attr.attr, >> &sensor_dev_attr_in0_input.dev_attr.attr, >> + &sensor_dev_attr_in1_input.dev_attr.attr, >> + &sensor_dev_attr_in2_input.dev_attr.attr, >> + &sensor_dev_attr_in3_input.dev_attr.attr, >> + &sensor_dev_attr_in4_input.dev_attr.attr, >> NULL, >> }; >> -ATTRIBUTE_GROUPS(ltc2990); >> + >> +static const struct attribute_group ltc2990_group = { >> + .attrs = ltc2990_attrs, >> + .is_visible = ltc2990_attrs_visible, >> +}; >> +__ATTRIBUTE_GROUPS(ltc2990); >> >> static int ltc2990_i2c_probe(struct i2c_client *i2c, >> const struct i2c_device_id *id) >> { >> int ret; >> struct device *hwmon_dev; >> + struct ltc2990_data *data; >> + struct device_node *of_node = i2c->dev.of_node; >> >> if (!i2c_check_functionality(i2c->adapter, I2C_FUNC_SMBUS_BYTE_DATA | >> I2C_FUNC_SMBUS_WORD_DATA)) >> return -ENODEV; >> >> - /* Setup continuous mode, current monitor */ >> + data = devm_kzalloc(&i2c->dev, sizeof(struct ltc2990_data), GFP_KERNEL); >> + if (unlikely(!data)) >> + return -ENOMEM; >> + data->i2c = i2c; >> + >> + if (!of_node || of_property_read_u32(of_node, "lltc,mode", &data->mode)) >> + data->mode = LTC2990_CONTROL_MODE_DEFAULT; > > Iam arguing with myself if we should still do this or if we should read the mode > from the chip instead if it isn't provided (after all, it may have been > initialized > by the BIOS/ROMMON). > > Mike, would that break your application, or can you specify the mode in > devicetree ? OMFG, I just spent half the day implementing the exact same thing, and only after sumbmitting it, I stumbled upon this thread. I must be getting old. A well, seems I implemented things a bit differently, so you get to pick and choose which patch was better. Whatever happened to this patch though? It didn't make it to mainline, otherwise I'd have found it sooner... > > Thanks, > Guenter > >> + >> + if (data->mode > LTC2990_CONTROL_MODE_MAX) { >> + dev_err(&i2c->dev, "Error: Invalid mode %d.\n", data->mode); >> + return -EINVAL; >> + } >> + >> + /* Setup continuous mode */ >> ret = i2c_smbus_write_byte_data(i2c, LTC2990_CONTROL, >> LTC2990_CONTROL_MEASURE_ALL | >> - LTC2990_CONTROL_MODE_CURRENT); >> + data->mode); >> if (ret < 0) { >> dev_err(&i2c->dev, "Error: Failed to set control mode.\n"); >> return ret; >> @@ -127,7 +244,7 @@ static int ltc2990_i2c_probe(struct i2c_client *i2c, >> >> hwmon_dev = devm_hwmon_device_register_with_groups(&i2c->dev, >> i2c->name, >> - i2c, >> + data, >> ltc2990_groups); >> >> return PTR_ERR_OR_ZERO(hwmon_dev); >> > Kind regards, Mike Looijmans System Expert TOPIC Products Materiaalweg 4, NL-5681 RJ Best Postbus 440, NL-5680 AK Best Telefoon: +31 (0) 499 33 69 79 E-mail: mike.looijmans@topicproducts.com Website: www.topicproducts.com Please consider the environment before printing this e-mail -- To unsubscribe from this list: send the line "unsubscribe linux-hwmon" in the body of a message to majordomo@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html
On Wed, Jun 28, 2017 at 04:24:03PM +0200, Mike Looijmans wrote: > On 17-11-16 17:56, Guenter Roeck wrote: > >On 11/17/2016 04:10 AM, Tom Levens wrote: > >>Updated version of the ltc2990 driver which supports all measurement > >>modes available in the chip. The mode can be set through a devicetree > >>attribute. > > > >property > > > >> > >>Signed-off-by: Tom Levens <tom.levens@cern.ch> > >>--- > >> > >>Changes since v1: > >> * Refactored value conversion (patch 1/3) > >> * Split the devicetree binding into separate patch (patch 2/3) > >> * Specifying an invalid mode now returns -EINVAL, previously this > >> only issued a warning and used the default value > >> * Removed the "mode" sysfs attribute, as the mode of the chip is > >> hardware specific and should not be user configurable. This allows much > >> simpler code as a result. > >> > >> Documentation/hwmon/ltc2990 | 24 ++++--- > >> drivers/hwmon/Kconfig | 7 +-- > >> drivers/hwmon/ltc2990.c | 167 ++++++++++++++++++++++++++++++++++++------- > >> 3 files changed, 159 insertions(+), 39 deletions(-) > >> > >>diff --git a/Documentation/hwmon/ltc2990 b/Documentation/hwmon/ltc2990 > >>index c25211e..3ed68f6 100644 > >>--- a/Documentation/hwmon/ltc2990 > >>+++ b/Documentation/hwmon/ltc2990 > >>@@ -8,6 +8,7 @@ Supported chips: > >> Datasheet: http://www.linear.com/product/ltc2990 > >> > >> Author: Mike Looijmans <mike.looijmans@topic.nl> > >>+ Tom Levens <tom.levens@cern.ch> > >> > >> > >> Description > >>@@ -16,10 +17,8 @@ Description > >> LTC2990 is a Quad I2C Voltage, Current and Temperature Monitor. > >> The chip's inputs can measure 4 voltages, or two inputs together (1+2 and 3+4) > >> can be combined to measure a differential voltage, which is typically used to > >>-measure current through a series resistor, or a temperature. > >>- > >>-This driver currently uses the 2x differential mode only. In order to support > >>-other modes, the driver will need to be expanded. > >>+measure current through a series resistor, or a temperature with an external > >>+diode. > >> > >> > >> Usage Notes > >>@@ -32,12 +31,19 @@ devices explicitly. > >> Sysfs attributes > >> ---------------- > >> > >>+in0_input Voltage at Vcc pin in millivolt (range 2.5V to 5V) > >>+temp1_input Internal chip temperature in millidegrees Celcius > >>+ > >>+A subset of the following attributes are visible, depending on the measurement > >>+mode of the chip. > >>+ > >>+in[1-4]_input Voltage at V[1-4] pin in millivolt > >>+temp2_input External temperature sensor TR1 in millidegrees Celcius > >>+temp3_input External temperature sensor TR2 in millidegrees Celcius > >>+curr1_input Current in mA across V1-V2 assuming a 1mOhm sense resistor > >>+curr2_input Current in mA across V3-V4 assuming a 1mOhm sense resistor > >>+ > >> The "curr*_input" measurements actually report the voltage drop across the > >> input pins in microvolts. This is equivalent to the current through a 1mOhm > >> sense resistor. Divide the reported value by the actual sense resistor value > >> in mOhm to get the actual value. > >>- > >>-in0_input Voltage at Vcc pin in millivolt (range 2.5V to 5V) > >>-temp1_input Internal chip temperature in millidegrees Celcius > >>-curr1_input Current in mA across v1-v2 assuming a 1mOhm sense resistor. > >>-curr2_input Current in mA across v3-v4 assuming a 1mOhm sense resistor. > >>diff --git a/drivers/hwmon/Kconfig b/drivers/hwmon/Kconfig > >>index 45cef3d..f7096ca 100644 > >>--- a/drivers/hwmon/Kconfig > >>+++ b/drivers/hwmon/Kconfig > >>@@ -699,15 +699,12 @@ config SENSORS_LTC2945 > >> be called ltc2945. > >> > >> config SENSORS_LTC2990 > >>- tristate "Linear Technology LTC2990 (current monitoring mode only)" > >>+ tristate "Linear Technology LTC2990" > >> depends on I2C > >> help > >> If you say yes here you get support for Linear Technology LTC2990 > >> I2C System Monitor. The LTC2990 supports a combination of voltage, > >>- current and temperature monitoring, but in addition to the Vcc supply > >>- voltage and chip temperature, this driver currently only supports > >>- reading two currents by measuring two differential voltages across > >>- series resistors. > >>+ current and temperature monitoring. > >> > >> This driver can also be built as a module. If so, the module will > >> be called ltc2990. > >>diff --git a/drivers/hwmon/ltc2990.c b/drivers/hwmon/ltc2990.c > >>index 0ec4102..e8d36f5 100644 > >>--- a/drivers/hwmon/ltc2990.c > >>+++ b/drivers/hwmon/ltc2990.c > >>@@ -6,11 +6,7 @@ > >> * > >> * License: GPLv2 > >> * > >>- * This driver assumes the chip is wired as a dual current monitor, and > >>- * reports the voltage drop across two series resistors. It also reports > >>- * the chip's internal temperature and Vcc power supply voltage. > >>- * > >>- * Value conversion refactored > >>+ * Value conversion refactored and support for all measurement modes added > >> * by Tom Levens <tom.levens@cern.ch> > >> */ > >> > >>@@ -21,6 +17,7 @@ > >> #include <linux/i2c.h> > >> #include <linux/kernel.h> > >> #include <linux/module.h> > >>+#include <linux/of.h> > >> > >> #define LTC2990_STATUS 0x00 > >> #define LTC2990_CONTROL 0x01 > >>@@ -35,32 +32,96 @@ > >> #define LTC2990_CONTROL_KELVIN BIT(7) > >> #define LTC2990_CONTROL_SINGLE BIT(6) > >> #define LTC2990_CONTROL_MEASURE_ALL (0x3 << 3) > >>-#define LTC2990_CONTROL_MODE_CURRENT 0x06 > >>-#define LTC2990_CONTROL_MODE_VOLTAGE 0x07 > >>+#define LTC2990_CONTROL_MODE_DEFAULT 0x06 > > > >I think LTC2990_CONTROL_MODE_CURRENT was better - it describes the actual mode. > >Changing the define to _DEFAULT does not really improve code readability. > > > >>+#define LTC2990_CONTROL_MODE_MAX 0x07 > >>+ > >>+#define LTC2990_IN0 BIT(0) > >>+#define LTC2990_IN1 BIT(1) > >>+#define LTC2990_IN2 BIT(2) > >>+#define LTC2990_IN3 BIT(3) > >>+#define LTC2990_IN4 BIT(4) > >>+#define LTC2990_CURR1 BIT(5) > >>+#define LTC2990_CURR2 BIT(6) > >>+#define LTC2990_TEMP1 BIT(7) > >>+#define LTC2990_TEMP2 BIT(8) > >>+#define LTC2990_TEMP3 BIT(9) > >>+ > >>+static const int ltc2990_attrs_ena[] = { > >>+ LTC2990_IN1 | LTC2990_IN2 | LTC2990_TEMP3, > >>+ LTC2990_CURR1 | LTC2990_TEMP3, > >>+ LTC2990_CURR1 | LTC2990_IN3 | LTC2990_IN4, > >>+ LTC2990_TEMP2 | LTC2990_IN3 | LTC2990_IN4, > >>+ LTC2990_TEMP2 | LTC2990_CURR2, > >>+ LTC2990_TEMP2 | LTC2990_TEMP3, > >>+ LTC2990_CURR1 | LTC2990_CURR2, > >>+ LTC2990_IN1 | LTC2990_IN2 | LTC2990_IN3 | LTC2990_IN4 > >>+}; > >>+ > >>+struct ltc2990_data { > >>+ struct i2c_client *i2c; > >>+ u32 mode; > >>+}; > >> > >> /* Return the converted value from the given register in uV or mC */ > >>-static int ltc2990_get_value(struct i2c_client *i2c, u8 reg, s32 *result) > >>+static int ltc2990_get_value(struct i2c_client *i2c, int index, s32 *result) > >> { > >> s32 val; > >>+ u8 reg; > >>+ > >>+ switch (index) { > >>+ case LTC2990_IN0: > >>+ reg = LTC2990_VCC_MSB; > >>+ break; > >>+ case LTC2990_IN1: > >>+ case LTC2990_CURR1: > >>+ case LTC2990_TEMP2: > >>+ reg = LTC2990_V1_MSB; > >>+ break; > >>+ case LTC2990_IN2: > >>+ reg = LTC2990_V2_MSB; > >>+ break; > >>+ case LTC2990_IN3: > >>+ case LTC2990_CURR2: > >>+ case LTC2990_TEMP3: > >>+ reg = LTC2990_V3_MSB; > >>+ break; > >>+ case LTC2990_IN4: > >>+ reg = LTC2990_V4_MSB; > >>+ break; > >>+ case LTC2990_TEMP1: > >>+ reg = LTC2990_TINT_MSB; > >>+ break; > >>+ default: > >>+ return -EINVAL; > >>+ } > >> > >> val = i2c_smbus_read_word_swapped(i2c, reg); > >> if (unlikely(val < 0)) > >> return val; > >> > >>- switch (reg) { > >>- case LTC2990_TINT_MSB: > >>- /* internal temp, 0.0625 degrees/LSB, 13-bit */ > >>+ switch (index) { > >>+ case LTC2990_TEMP1: > >>+ case LTC2990_TEMP2: > >>+ case LTC2990_TEMP3: > >>+ /* temp, 0.0625 degrees/LSB, 13-bit */ > >> *result = sign_extend32(val, 12) * 1000 / 16; > >> break; > >>- case LTC2990_V1_MSB: > >>- case LTC2990_V3_MSB: > >>- /* Vx-Vy, 19.42uV/LSB. Depends on mode. */ > >>+ case LTC2990_CURR1: > >>+ case LTC2990_CURR2: > >>+ /* Vx-Vy, 19.42uV/LSB */ > >> *result = sign_extend32(val, 14) * 1942 / 100; > >> break; > >>- case LTC2990_VCC_MSB: > >>- /* Vcc, 305.18μV/LSB, 2.5V offset */ > >>+ case LTC2990_IN0: > >>+ /* Vcc, 305.18uV/LSB, 2.5V offset */ > >> *result = sign_extend32(val, 14) * 30518 / (100 * 1000) + 2500; > >> break; > >>+ case LTC2990_IN1: > >>+ case LTC2990_IN2: > >>+ case LTC2990_IN3: > >>+ case LTC2990_IN4: > >>+ /* Vx: 305.18uV/LSB */ > >>+ *result = sign_extend32(val, 14) * 30518 / (100 * 1000); > >>+ break; > >> default: > >> return -EINVAL; /* won't happen, keep compiler happy */ > >> } > >>@@ -72,48 +133,104 @@ static ssize_t ltc2990_show_value(struct device *dev, > >> struct device_attribute *da, char *buf) > >> { > >> struct sensor_device_attribute *attr = to_sensor_dev_attr(da); > >>+ struct ltc2990_data *data = dev_get_drvdata(dev); > >> s32 value; > >> int ret; > >> > >>- ret = ltc2990_get_value(dev_get_drvdata(dev), attr->index, &value); > >>+ ret = ltc2990_get_value(data->i2c, attr->index, &value); > >> if (unlikely(ret < 0)) > >> return ret; > >> > >> return snprintf(buf, PAGE_SIZE, "%d\n", value); > >> } > >> > >>+static umode_t ltc2990_attrs_visible(struct kobject *kobj, > >>+ struct attribute *a, int n) > >>+{ > >>+ struct device *dev = container_of(kobj, struct device, kobj); > >>+ struct ltc2990_data *data = dev_get_drvdata(dev); > >>+ struct device_attribute *da = > >>+ container_of(a, struct device_attribute, attr); > >>+ struct sensor_device_attribute *attr = to_sensor_dev_attr(da); > >>+ > >>+ if (attr->index == LTC2990_TEMP1 || > >>+ attr->index == LTC2990_IN0 || > >>+ attr->index & ltc2990_attrs_ena[data->mode]) > >>+ return a->mode; > >>+ else > > > >Unnecessary else > > > >>+ return 0; > >>+} > >>+ > >> static SENSOR_DEVICE_ATTR(temp1_input, S_IRUGO, ltc2990_show_value, NULL, > >>- LTC2990_TINT_MSB); > >>+ LTC2990_TEMP1); > >>+static SENSOR_DEVICE_ATTR(temp2_input, S_IRUGO, ltc2990_show_value, NULL, > >>+ LTC2990_TEMP2); > >>+static SENSOR_DEVICE_ATTR(temp3_input, S_IRUGO, ltc2990_show_value, NULL, > >>+ LTC2990_TEMP3); > >> static SENSOR_DEVICE_ATTR(curr1_input, S_IRUGO, ltc2990_show_value, NULL, > >>- LTC2990_V1_MSB); > >>+ LTC2990_CURR1); > >> static SENSOR_DEVICE_ATTR(curr2_input, S_IRUGO, ltc2990_show_value, NULL, > >>- LTC2990_V3_MSB); > >>+ LTC2990_CURR2); > >> static SENSOR_DEVICE_ATTR(in0_input, S_IRUGO, ltc2990_show_value, NULL, > >>- LTC2990_VCC_MSB); > >>+ LTC2990_IN0); > >>+static SENSOR_DEVICE_ATTR(in1_input, S_IRUGO, ltc2990_show_value, NULL, > >>+ LTC2990_IN1); > >>+static SENSOR_DEVICE_ATTR(in2_input, S_IRUGO, ltc2990_show_value, NULL, > >>+ LTC2990_IN2); > >>+static SENSOR_DEVICE_ATTR(in3_input, S_IRUGO, ltc2990_show_value, NULL, > >>+ LTC2990_IN3); > >>+static SENSOR_DEVICE_ATTR(in4_input, S_IRUGO, ltc2990_show_value, NULL, > >>+ LTC2990_IN4); > >> > >> static struct attribute *ltc2990_attrs[] = { > >> &sensor_dev_attr_temp1_input.dev_attr.attr, > >>+ &sensor_dev_attr_temp2_input.dev_attr.attr, > >>+ &sensor_dev_attr_temp3_input.dev_attr.attr, > >> &sensor_dev_attr_curr1_input.dev_attr.attr, > >> &sensor_dev_attr_curr2_input.dev_attr.attr, > >> &sensor_dev_attr_in0_input.dev_attr.attr, > >>+ &sensor_dev_attr_in1_input.dev_attr.attr, > >>+ &sensor_dev_attr_in2_input.dev_attr.attr, > >>+ &sensor_dev_attr_in3_input.dev_attr.attr, > >>+ &sensor_dev_attr_in4_input.dev_attr.attr, > >> NULL, > >> }; > >>-ATTRIBUTE_GROUPS(ltc2990); > >>+ > >>+static const struct attribute_group ltc2990_group = { > >>+ .attrs = ltc2990_attrs, > >>+ .is_visible = ltc2990_attrs_visible, > >>+}; > >>+__ATTRIBUTE_GROUPS(ltc2990); > >> > >> static int ltc2990_i2c_probe(struct i2c_client *i2c, > >> const struct i2c_device_id *id) > >> { > >> int ret; > >> struct device *hwmon_dev; > >>+ struct ltc2990_data *data; > >>+ struct device_node *of_node = i2c->dev.of_node; > >> > >> if (!i2c_check_functionality(i2c->adapter, I2C_FUNC_SMBUS_BYTE_DATA | > >> I2C_FUNC_SMBUS_WORD_DATA)) > >> return -ENODEV; > >> > >>- /* Setup continuous mode, current monitor */ > >>+ data = devm_kzalloc(&i2c->dev, sizeof(struct ltc2990_data), GFP_KERNEL); > >>+ if (unlikely(!data)) > >>+ return -ENOMEM; > >>+ data->i2c = i2c; > >>+ > >>+ if (!of_node || of_property_read_u32(of_node, "lltc,mode", &data->mode)) > >>+ data->mode = LTC2990_CONTROL_MODE_DEFAULT; > > > >Iam arguing with myself if we should still do this or if we should read the mode > >from the chip instead if it isn't provided (after all, it may have been > >initialized > >by the BIOS/ROMMON). > > > >Mike, would that break your application, or can you specify the mode in > >devicetree ? > > OMFG, I just spent half the day implementing the exact same thing, and only > after sumbmitting it, I stumbled upon this thread. I must be getting old. > > A well, seems I implemented things a bit differently, so you get to pick and > choose which patch was better. > As I just replied to your patch, there is no question here. The compatible statement refers to chip compatibility; one can not use it to also provide configuration information. > Whatever happened to this patch though? It didn't make it to mainline, > otherwise I'd have found it sooner... > I'll have to look it up, but I guess I didn't get an updated version. Guenter > > > > >Thanks, > >Guenter > > > >>+ > >>+ if (data->mode > LTC2990_CONTROL_MODE_MAX) { > >>+ dev_err(&i2c->dev, "Error: Invalid mode %d.\n", data->mode); > >>+ return -EINVAL; > >>+ } > >>+ > >>+ /* Setup continuous mode */ > >> ret = i2c_smbus_write_byte_data(i2c, LTC2990_CONTROL, > >> LTC2990_CONTROL_MEASURE_ALL | > >>- LTC2990_CONTROL_MODE_CURRENT); > >>+ data->mode); > >> if (ret < 0) { > >> dev_err(&i2c->dev, "Error: Failed to set control mode.\n"); > >> return ret; > >>@@ -127,7 +244,7 @@ static int ltc2990_i2c_probe(struct i2c_client *i2c, > >> > >> hwmon_dev = devm_hwmon_device_register_with_groups(&i2c->dev, > >> i2c->name, > >>- i2c, > >>+ data, > >> ltc2990_groups); > >> > >> return PTR_ERR_OR_ZERO(hwmon_dev); > >> > > > > > > Kind regards, > > Mike Looijmans > System Expert > > TOPIC Products > Materiaalweg 4, NL-5681 RJ Best > Postbus 440, NL-5680 AK Best > Telefoon: +31 (0) 499 33 69 79 > E-mail: mike.looijmans@topicproducts.com > Website: www.topicproducts.com > > Please consider the environment before printing this e-mail > > > -- To unsubscribe from this list: send the line "unsubscribe linux-hwmon" in the body of a message to majordomo@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html
On Wed, 28 Jun 2017, Guenter Roeck wrote: > On Wed, Jun 28, 2017 at 04:24:03PM +0200, Mike Looijmans wrote: >> On 17-11-16 17:56, Guenter Roeck wrote: >>> On 11/17/2016 04:10 AM, Tom Levens wrote: >>>> Updated version of the ltc2990 driver which supports all measurement >>>> modes available in the chip. The mode can be set through a devicetree >>>> attribute. >>> >>> property >>> >>>> >>>> Signed-off-by: Tom Levens <tom.levens@cern.ch> >>>> --- >>>> >>>> Changes since v1: >>>> * Refactored value conversion (patch 1/3) >>>> * Split the devicetree binding into separate patch (patch 2/3) >>>> * Specifying an invalid mode now returns -EINVAL, previously this >>>> only issued a warning and used the default value >>>> * Removed the "mode" sysfs attribute, as the mode of the chip is >>>> hardware specific and should not be user configurable. This allows much >>>> simpler code as a result. >>>> >>>> Documentation/hwmon/ltc2990 | 24 ++++--- >>>> drivers/hwmon/Kconfig | 7 +-- >>>> drivers/hwmon/ltc2990.c | 167 ++++++++++++++++++++++++++++++++++++------- >>>> 3 files changed, 159 insertions(+), 39 deletions(-) >>>> >>>> diff --git a/Documentation/hwmon/ltc2990 b/Documentation/hwmon/ltc2990 >>>> index c25211e..3ed68f6 100644 >>>> --- a/Documentation/hwmon/ltc2990 >>>> +++ b/Documentation/hwmon/ltc2990 >>>> @@ -8,6 +8,7 @@ Supported chips: >>>> Datasheet: http://www.linear.com/product/ltc2990 >>>> >>>> Author: Mike Looijmans <mike.looijmans@topic.nl> >>>> + Tom Levens <tom.levens@cern.ch> >>>> >>>> >>>> Description >>>> @@ -16,10 +17,8 @@ Description >>>> LTC2990 is a Quad I2C Voltage, Current and Temperature Monitor. >>>> The chip's inputs can measure 4 voltages, or two inputs together (1+2 and 3+4) >>>> can be combined to measure a differential voltage, which is typically used to >>>> -measure current through a series resistor, or a temperature. >>>> - >>>> -This driver currently uses the 2x differential mode only. In order to support >>>> -other modes, the driver will need to be expanded. >>>> +measure current through a series resistor, or a temperature with an external >>>> +diode. >>>> >>>> >>>> Usage Notes >>>> @@ -32,12 +31,19 @@ devices explicitly. >>>> Sysfs attributes >>>> ---------------- >>>> >>>> +in0_input Voltage at Vcc pin in millivolt (range 2.5V to 5V) >>>> +temp1_input Internal chip temperature in millidegrees Celcius >>>> + >>>> +A subset of the following attributes are visible, depending on the measurement >>>> +mode of the chip. >>>> + >>>> +in[1-4]_input Voltage at V[1-4] pin in millivolt >>>> +temp2_input External temperature sensor TR1 in millidegrees Celcius >>>> +temp3_input External temperature sensor TR2 in millidegrees Celcius >>>> +curr1_input Current in mA across V1-V2 assuming a 1mOhm sense resistor >>>> +curr2_input Current in mA across V3-V4 assuming a 1mOhm sense resistor >>>> + >>>> The "curr*_input" measurements actually report the voltage drop across the >>>> input pins in microvolts. This is equivalent to the current through a 1mOhm >>>> sense resistor. Divide the reported value by the actual sense resistor value >>>> in mOhm to get the actual value. >>>> - >>>> -in0_input Voltage at Vcc pin in millivolt (range 2.5V to 5V) >>>> -temp1_input Internal chip temperature in millidegrees Celcius >>>> -curr1_input Current in mA across v1-v2 assuming a 1mOhm sense resistor. >>>> -curr2_input Current in mA across v3-v4 assuming a 1mOhm sense resistor. >>>> diff --git a/drivers/hwmon/Kconfig b/drivers/hwmon/Kconfig >>>> index 45cef3d..f7096ca 100644 >>>> --- a/drivers/hwmon/Kconfig >>>> +++ b/drivers/hwmon/Kconfig >>>> @@ -699,15 +699,12 @@ config SENSORS_LTC2945 >>>> be called ltc2945. >>>> >>>> config SENSORS_LTC2990 >>>> - tristate "Linear Technology LTC2990 (current monitoring mode only)" >>>> + tristate "Linear Technology LTC2990" >>>> depends on I2C >>>> help >>>> If you say yes here you get support for Linear Technology LTC2990 >>>> I2C System Monitor. The LTC2990 supports a combination of voltage, >>>> - current and temperature monitoring, but in addition to the Vcc supply >>>> - voltage and chip temperature, this driver currently only supports >>>> - reading two currents by measuring two differential voltages across >>>> - series resistors. >>>> + current and temperature monitoring. >>>> >>>> This driver can also be built as a module. If so, the module will >>>> be called ltc2990. >>>> diff --git a/drivers/hwmon/ltc2990.c b/drivers/hwmon/ltc2990.c >>>> index 0ec4102..e8d36f5 100644 >>>> --- a/drivers/hwmon/ltc2990.c >>>> +++ b/drivers/hwmon/ltc2990.c >>>> @@ -6,11 +6,7 @@ >>>> * >>>> * License: GPLv2 >>>> * >>>> - * This driver assumes the chip is wired as a dual current monitor, and >>>> - * reports the voltage drop across two series resistors. It also reports >>>> - * the chip's internal temperature and Vcc power supply voltage. >>>> - * >>>> - * Value conversion refactored >>>> + * Value conversion refactored and support for all measurement modes added >>>> * by Tom Levens <tom.levens@cern.ch> >>>> */ >>>> >>>> @@ -21,6 +17,7 @@ >>>> #include <linux/i2c.h> >>>> #include <linux/kernel.h> >>>> #include <linux/module.h> >>>> +#include <linux/of.h> >>>> >>>> #define LTC2990_STATUS 0x00 >>>> #define LTC2990_CONTROL 0x01 >>>> @@ -35,32 +32,96 @@ >>>> #define LTC2990_CONTROL_KELVIN BIT(7) >>>> #define LTC2990_CONTROL_SINGLE BIT(6) >>>> #define LTC2990_CONTROL_MEASURE_ALL (0x3 << 3) >>>> -#define LTC2990_CONTROL_MODE_CURRENT 0x06 >>>> -#define LTC2990_CONTROL_MODE_VOLTAGE 0x07 >>>> +#define LTC2990_CONTROL_MODE_DEFAULT 0x06 >>> >>> I think LTC2990_CONTROL_MODE_CURRENT was better - it describes the actual mode. >>> Changing the define to _DEFAULT does not really improve code readability. >>> >>>> +#define LTC2990_CONTROL_MODE_MAX 0x07 >>>> + >>>> +#define LTC2990_IN0 BIT(0) >>>> +#define LTC2990_IN1 BIT(1) >>>> +#define LTC2990_IN2 BIT(2) >>>> +#define LTC2990_IN3 BIT(3) >>>> +#define LTC2990_IN4 BIT(4) >>>> +#define LTC2990_CURR1 BIT(5) >>>> +#define LTC2990_CURR2 BIT(6) >>>> +#define LTC2990_TEMP1 BIT(7) >>>> +#define LTC2990_TEMP2 BIT(8) >>>> +#define LTC2990_TEMP3 BIT(9) >>>> + >>>> +static const int ltc2990_attrs_ena[] = { >>>> + LTC2990_IN1 | LTC2990_IN2 | LTC2990_TEMP3, >>>> + LTC2990_CURR1 | LTC2990_TEMP3, >>>> + LTC2990_CURR1 | LTC2990_IN3 | LTC2990_IN4, >>>> + LTC2990_TEMP2 | LTC2990_IN3 | LTC2990_IN4, >>>> + LTC2990_TEMP2 | LTC2990_CURR2, >>>> + LTC2990_TEMP2 | LTC2990_TEMP3, >>>> + LTC2990_CURR1 | LTC2990_CURR2, >>>> + LTC2990_IN1 | LTC2990_IN2 | LTC2990_IN3 | LTC2990_IN4 >>>> +}; >>>> + >>>> +struct ltc2990_data { >>>> + struct i2c_client *i2c; >>>> + u32 mode; >>>> +}; >>>> >>>> /* Return the converted value from the given register in uV or mC */ >>>> -static int ltc2990_get_value(struct i2c_client *i2c, u8 reg, s32 *result) >>>> +static int ltc2990_get_value(struct i2c_client *i2c, int index, s32 *result) >>>> { >>>> s32 val; >>>> + u8 reg; >>>> + >>>> + switch (index) { >>>> + case LTC2990_IN0: >>>> + reg = LTC2990_VCC_MSB; >>>> + break; >>>> + case LTC2990_IN1: >>>> + case LTC2990_CURR1: >>>> + case LTC2990_TEMP2: >>>> + reg = LTC2990_V1_MSB; >>>> + break; >>>> + case LTC2990_IN2: >>>> + reg = LTC2990_V2_MSB; >>>> + break; >>>> + case LTC2990_IN3: >>>> + case LTC2990_CURR2: >>>> + case LTC2990_TEMP3: >>>> + reg = LTC2990_V3_MSB; >>>> + break; >>>> + case LTC2990_IN4: >>>> + reg = LTC2990_V4_MSB; >>>> + break; >>>> + case LTC2990_TEMP1: >>>> + reg = LTC2990_TINT_MSB; >>>> + break; >>>> + default: >>>> + return -EINVAL; >>>> + } >>>> >>>> val = i2c_smbus_read_word_swapped(i2c, reg); >>>> if (unlikely(val < 0)) >>>> return val; >>>> >>>> - switch (reg) { >>>> - case LTC2990_TINT_MSB: >>>> - /* internal temp, 0.0625 degrees/LSB, 13-bit */ >>>> + switch (index) { >>>> + case LTC2990_TEMP1: >>>> + case LTC2990_TEMP2: >>>> + case LTC2990_TEMP3: >>>> + /* temp, 0.0625 degrees/LSB, 13-bit */ >>>> *result = sign_extend32(val, 12) * 1000 / 16; >>>> break; >>>> - case LTC2990_V1_MSB: >>>> - case LTC2990_V3_MSB: >>>> - /* Vx-Vy, 19.42uV/LSB. Depends on mode. */ >>>> + case LTC2990_CURR1: >>>> + case LTC2990_CURR2: >>>> + /* Vx-Vy, 19.42uV/LSB */ >>>> *result = sign_extend32(val, 14) * 1942 / 100; >>>> break; >>>> - case LTC2990_VCC_MSB: >>>> - /* Vcc, 305.18μV/LSB, 2.5V offset */ >>>> + case LTC2990_IN0: >>>> + /* Vcc, 305.18uV/LSB, 2.5V offset */ >>>> *result = sign_extend32(val, 14) * 30518 / (100 * 1000) + 2500; >>>> break; >>>> + case LTC2990_IN1: >>>> + case LTC2990_IN2: >>>> + case LTC2990_IN3: >>>> + case LTC2990_IN4: >>>> + /* Vx: 305.18uV/LSB */ >>>> + *result = sign_extend32(val, 14) * 30518 / (100 * 1000); >>>> + break; >>>> default: >>>> return -EINVAL; /* won't happen, keep compiler happy */ >>>> } >>>> @@ -72,48 +133,104 @@ static ssize_t ltc2990_show_value(struct device *dev, >>>> struct device_attribute *da, char *buf) >>>> { >>>> struct sensor_device_attribute *attr = to_sensor_dev_attr(da); >>>> + struct ltc2990_data *data = dev_get_drvdata(dev); >>>> s32 value; >>>> int ret; >>>> >>>> - ret = ltc2990_get_value(dev_get_drvdata(dev), attr->index, &value); >>>> + ret = ltc2990_get_value(data->i2c, attr->index, &value); >>>> if (unlikely(ret < 0)) >>>> return ret; >>>> >>>> return snprintf(buf, PAGE_SIZE, "%d\n", value); >>>> } >>>> >>>> +static umode_t ltc2990_attrs_visible(struct kobject *kobj, >>>> + struct attribute *a, int n) >>>> +{ >>>> + struct device *dev = container_of(kobj, struct device, kobj); >>>> + struct ltc2990_data *data = dev_get_drvdata(dev); >>>> + struct device_attribute *da = >>>> + container_of(a, struct device_attribute, attr); >>>> + struct sensor_device_attribute *attr = to_sensor_dev_attr(da); >>>> + >>>> + if (attr->index == LTC2990_TEMP1 || >>>> + attr->index == LTC2990_IN0 || >>>> + attr->index & ltc2990_attrs_ena[data->mode]) >>>> + return a->mode; >>>> + else >>> >>> Unnecessary else >>> >>>> + return 0; >>>> +} >>>> + >>>> static SENSOR_DEVICE_ATTR(temp1_input, S_IRUGO, ltc2990_show_value, NULL, >>>> - LTC2990_TINT_MSB); >>>> + LTC2990_TEMP1); >>>> +static SENSOR_DEVICE_ATTR(temp2_input, S_IRUGO, ltc2990_show_value, NULL, >>>> + LTC2990_TEMP2); >>>> +static SENSOR_DEVICE_ATTR(temp3_input, S_IRUGO, ltc2990_show_value, NULL, >>>> + LTC2990_TEMP3); >>>> static SENSOR_DEVICE_ATTR(curr1_input, S_IRUGO, ltc2990_show_value, NULL, >>>> - LTC2990_V1_MSB); >>>> + LTC2990_CURR1); >>>> static SENSOR_DEVICE_ATTR(curr2_input, S_IRUGO, ltc2990_show_value, NULL, >>>> - LTC2990_V3_MSB); >>>> + LTC2990_CURR2); >>>> static SENSOR_DEVICE_ATTR(in0_input, S_IRUGO, ltc2990_show_value, NULL, >>>> - LTC2990_VCC_MSB); >>>> + LTC2990_IN0); >>>> +static SENSOR_DEVICE_ATTR(in1_input, S_IRUGO, ltc2990_show_value, NULL, >>>> + LTC2990_IN1); >>>> +static SENSOR_DEVICE_ATTR(in2_input, S_IRUGO, ltc2990_show_value, NULL, >>>> + LTC2990_IN2); >>>> +static SENSOR_DEVICE_ATTR(in3_input, S_IRUGO, ltc2990_show_value, NULL, >>>> + LTC2990_IN3); >>>> +static SENSOR_DEVICE_ATTR(in4_input, S_IRUGO, ltc2990_show_value, NULL, >>>> + LTC2990_IN4); >>>> >>>> static struct attribute *ltc2990_attrs[] = { >>>> &sensor_dev_attr_temp1_input.dev_attr.attr, >>>> + &sensor_dev_attr_temp2_input.dev_attr.attr, >>>> + &sensor_dev_attr_temp3_input.dev_attr.attr, >>>> &sensor_dev_attr_curr1_input.dev_attr.attr, >>>> &sensor_dev_attr_curr2_input.dev_attr.attr, >>>> &sensor_dev_attr_in0_input.dev_attr.attr, >>>> + &sensor_dev_attr_in1_input.dev_attr.attr, >>>> + &sensor_dev_attr_in2_input.dev_attr.attr, >>>> + &sensor_dev_attr_in3_input.dev_attr.attr, >>>> + &sensor_dev_attr_in4_input.dev_attr.attr, >>>> NULL, >>>> }; >>>> -ATTRIBUTE_GROUPS(ltc2990); >>>> + >>>> +static const struct attribute_group ltc2990_group = { >>>> + .attrs = ltc2990_attrs, >>>> + .is_visible = ltc2990_attrs_visible, >>>> +}; >>>> +__ATTRIBUTE_GROUPS(ltc2990); >>>> >>>> static int ltc2990_i2c_probe(struct i2c_client *i2c, >>>> const struct i2c_device_id *id) >>>> { >>>> int ret; >>>> struct device *hwmon_dev; >>>> + struct ltc2990_data *data; >>>> + struct device_node *of_node = i2c->dev.of_node; >>>> >>>> if (!i2c_check_functionality(i2c->adapter, I2C_FUNC_SMBUS_BYTE_DATA | >>>> I2C_FUNC_SMBUS_WORD_DATA)) >>>> return -ENODEV; >>>> >>>> - /* Setup continuous mode, current monitor */ >>>> + data = devm_kzalloc(&i2c->dev, sizeof(struct ltc2990_data), GFP_KERNEL); >>>> + if (unlikely(!data)) >>>> + return -ENOMEM; >>>> + data->i2c = i2c; >>>> + >>>> + if (!of_node || of_property_read_u32(of_node, "lltc,mode", &data->mode)) >>>> + data->mode = LTC2990_CONTROL_MODE_DEFAULT; >>> >>> Iam arguing with myself if we should still do this or if we should read the mode >>> from the chip instead if it isn't provided (after all, it may have been >>> initialized >>> by the BIOS/ROMMON). >>> >>> Mike, would that break your application, or can you specify the mode in >>> devicetree ? >> >> OMFG, I just spent half the day implementing the exact same thing, and only >> after sumbmitting it, I stumbled upon this thread. I must be getting old. >> >> A well, seems I implemented things a bit differently, so you get to pick and >> choose which patch was better. >> > > As I just replied to your patch, there is no question here. The compatible > statement refers to chip compatibility; one can not use it to also provide > configuration information. > >> Whatever happened to this patch though? It didn't make it to mainline, >> otherwise I'd have found it sooner... >> > I'll have to look it up, but I guess I didn't get an updated version. As far as I remember I had a working V3 of this patch, but it is entirely possible that it was never submitted as I have been busy with other projects recently. I'll dig it out and check that it is complete. Cheers, > Guenter > >> >>> >>> Thanks, >>> Guenter >>> >>>> + >>>> + if (data->mode > LTC2990_CONTROL_MODE_MAX) { >>>> + dev_err(&i2c->dev, "Error: Invalid mode %d.\n", data->mode); >>>> + return -EINVAL; >>>> + } >>>> + >>>> + /* Setup continuous mode */ >>>> ret = i2c_smbus_write_byte_data(i2c, LTC2990_CONTROL, >>>> LTC2990_CONTROL_MEASURE_ALL | >>>> - LTC2990_CONTROL_MODE_CURRENT); >>>> + data->mode); >>>> if (ret < 0) { >>>> dev_err(&i2c->dev, "Error: Failed to set control mode.\n"); >>>> return ret; >>>> @@ -127,7 +244,7 @@ static int ltc2990_i2c_probe(struct i2c_client *i2c, >>>> >>>> hwmon_dev = devm_hwmon_device_register_with_groups(&i2c->dev, >>>> i2c->name, >>>> - i2c, >>>> + data, >>>> ltc2990_groups); >>>> >>>> return PTR_ERR_OR_ZERO(hwmon_dev); >>>> >>> >> >> >> >> Kind regards, >> >> Mike Looijmans >> System Expert >> >> TOPIC Products >> Materiaalweg 4, NL-5681 RJ Best >> Postbus 440, NL-5680 AK Best >> Telefoon: +31 (0) 499 33 69 79 >> E-mail: mike.looijmans@topicproducts.com >> Website: www.topicproducts.com >> >> Please consider the environment before printing this e-mail >> >> >> >
On Wed, Jun 28, 2017 at 05:29:38PM +0200, Tom Levens wrote: > [ ... ] > > > >>Whatever happened to this patch though? It didn't make it to mainline, > >>otherwise I'd have found it sooner... > >> > >I'll have to look it up, but I guess I didn't get an updated version. > > As far as I remember I had a working V3 of this patch, but it is entirely > possible that it was never submitted as I have been busy with other projects > recently. I'll dig it out and check that it is complete. > FWIW, I don't see it at https://patchwork.kernel.org/project/linux-hwmon/list/?submitter=171225&state=* Maybe you were waiting for a reply from Rob. Either case, it might make sense to only provide valid modes, ie to abstract the mode bits from the hardware, such as 0 - internal temp only 1 - Tr1 2 - V1 3 - V1-V2 4 - Tr2 5 - V3 6 - V3-V4 7 to 14 - per bit 0..2 Guenter -- To unsubscribe from this list: send the line "unsubscribe linux-hwmon" in the body of a message to majordomo@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html
On Wed, 28 Jun 2017, Guenter Roeck wrote: > On Wed, Jun 28, 2017 at 05:29:38PM +0200, Tom Levens wrote: >> > [ ... ] > >>> >>>> Whatever happened to this patch though? It didn't make it to mainline, >>>> otherwise I'd have found it sooner... >>>> >>> I'll have to look it up, but I guess I didn't get an updated version. >> >> As far as I remember I had a working V3 of this patch, but it is entirely >> possible that it was never submitted as I have been busy with other projects >> recently. I'll dig it out and check that it is complete. >> > FWIW, I don't see it at > https://patchwork.kernel.org/project/linux-hwmon/list/?submitter=171225&state=* > > Maybe you were waiting for a reply from Rob. Either case, it might make > sense to only provide valid modes, ie to abstract the mode bits from the > hardware, such as > > 0 - internal temp only > 1 - Tr1 > 2 - V1 > 3 - V1-V2 > 4 - Tr2 > 5 - V3 > 6 - V3-V4 > 7 to 14 - per bit 0..2 > > Guenter > You are right, there was still an open question about how best to handle the mode selection in DT. In the latest version of my patch I have it implemented as an array for setting the two values, for example: lltc,meas-mode = <7 3>; This sets bits [2..0] = 7 and [4..3] = 3. Of course these could be split into two DT properties, but I was unsure what to name them as both fields are called "mode" in the datasheet and "mode-43"/"mode-20" (or similar) is ugly. With regards to your proposal, it is not clear to me whether the modes which have the same result are exactly equivalent. Does disabling a measurement with the mode[4..3] bits really leaves the part in a safe state for all possible HW connections? With this doubt in my head, I would prefer to keep the option available to the user to select any specific mode. But I am open to suggestions. Mike, if you would like to test it, the latest version of my code is here: https://github.com/levens/ltc2990/blob/dev/drivers/hwmon/ltc2990.c Cheers, -- To unsubscribe from this list: send the line "unsubscribe linux-hwmon" in the body of a message to majordomo@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html
On 28-06-17 19:02, Tom Levens wrote: > > On Wed, 28 Jun 2017, Guenter Roeck wrote: > >> On Wed, Jun 28, 2017 at 05:29:38PM +0200, Tom Levens wrote: >>> >> [ ... ] >> >>>> >>>>> Whatever happened to this patch though? It didn't make it to mainline, >>>>> otherwise I'd have found it sooner... >>>>> >>>> I'll have to look it up, but I guess I didn't get an updated version. >>> >>> As far as I remember I had a working V3 of this patch, but it is >>> entirely >>> possible that it was never submitted as I have been busy with other >>> projects >>> recently. I'll dig it out and check that it is complete. >>> >> FWIW, I don't see it at >> https://patchwork.kernel.org/project/linux-hwmon/list/?submitter=171225&state=* >> >> >> Maybe you were waiting for a reply from Rob. Either case, it might make >> sense to only provide valid modes, ie to abstract the mode bits from the >> hardware, such as >> >> 0 - internal temp only >> 1 - Tr1 >> 2 - V1 >> 3 - V1-V2 >> 4 - Tr2 >> 5 - V3 >> 6 - V3-V4 >> 7 to 14 - per bit 0..2 >> >> Guenter >> > > You are right, there was still an open question about how best to handle > the mode selection in DT. > > In the latest version of my patch I have it implemented as an array for > setting the two values, for example: > > lltc,meas-mode = <7 3>; > > This sets bits [2..0] = 7 and [4..3] = 3. Of course these could be split > into two DT properties, but I was unsure what to name them as both > fields are called "mode" in the datasheet and "mode-43"/"mode-20" (or > similar) is ugly. > > With regards to your proposal, it is not clear to me whether the modes > which have the same result are exactly equivalent. Does disabling a > measurement with the mode[4..3] bits really leaves the part in a safe > state for all possible HW connections? With this doubt in my head, I > would prefer to keep the option available to the user to select any > specific mode. But I am open to suggestions. Well, the input restrictions always apply, so disabling V3 measurement doesn't imply that you can apply 20V to that input safely now. I'd suggest to set unused input to plain voltage measurement. That is "passive" and safe for external components. So I'd suggest just setting the mode as per device datasheet, I can see no real advantage in abstracting it away and forcing users to read yet another document to get it right, e.g.: lltc,mode = <6>; As for the input disabling, since I doubt anyone would use it (why purchase a 4-channel device and use only 2), just add two booleans, e.g. "disable-inputs-34" and "disable-inputs-12" which set the command bits appropriately, and change the mode such that the disabled inputs are voltage readout only. A case could even be made for changing mode at runtime. This allows using it to measure both current and voltage on two inputs, by reading V1, and V3, and then switch mode to obtain (accurate) V1-V2 and V4-V3. That might be a viable way to handle not setting the mode at all. If the mode can be selected via sysfs, the driver can keep the device in a "safe" mode until the mode has been selected. > Mike, if you would like to test it, the latest version of my code is here: > > https://github.com/levens/ltc2990/blob/dev/drivers/hwmon/ltc2990.c Sure, I even have a board with 2 of these devices now :)
On Wed, Jun 28, 2017 at 07:33:33PM +0200, Mike Looijmans wrote: > On 28-06-17 19:02, Tom Levens wrote: > > > >On Wed, 28 Jun 2017, Guenter Roeck wrote: > > > >>On Wed, Jun 28, 2017 at 05:29:38PM +0200, Tom Levens wrote: > >>> > >>[ ... ] > >> > >>>> > >>>>>Whatever happened to this patch though? It didn't make it to mainline, > >>>>>otherwise I'd have found it sooner... > >>>>> > >>>>I'll have to look it up, but I guess I didn't get an updated version. > >>> > >>>As far as I remember I had a working V3 of this patch, but it is > >>>entirely > >>>possible that it was never submitted as I have been busy with other > >>>projects > >>>recently. I'll dig it out and check that it is complete. > >>> > >>FWIW, I don't see it at > >>https://patchwork.kernel.org/project/linux-hwmon/list/?submitter=171225&state=* > >> > >> > >>Maybe you were waiting for a reply from Rob. Either case, it might make > >>sense to only provide valid modes, ie to abstract the mode bits from the > >>hardware, such as > >> > >>0 - internal temp only > >>1 - Tr1 > >>2 - V1 > >>3 - V1-V2 > >>4 - Tr2 > >>5 - V3 > >>6 - V3-V4 > >>7 to 14 - per bit 0..2 > >> > >>Guenter > >> > > > >You are right, there was still an open question about how best to handle > >the mode selection in DT. > > > >In the latest version of my patch I have it implemented as an array for > >setting the two values, for example: > > > > lltc,meas-mode = <7 3>; > > > >This sets bits [2..0] = 7 and [4..3] = 3. Of course these could be split > >into two DT properties, but I was unsure what to name them as both fields > >are called "mode" in the datasheet and "mode-43"/"mode-20" (or similar) is > >ugly. > > > >With regards to your proposal, it is not clear to me whether the modes > >which have the same result are exactly equivalent. Does disabling a > >measurement with the mode[4..3] bits really leaves the part in a safe > >state for all possible HW connections? With this doubt in my head, I would > >prefer to keep the option available to the user to select any specific > >mode. But I am open to suggestions. > > Well, the input restrictions always apply, so disabling V3 measurement > doesn't imply that you can apply 20V to that input safely now. > > I'd suggest to set unused input to plain voltage measurement. That is > "passive" and safe for external components. > > So I'd suggest just setting the mode as per device datasheet, I can see no > real advantage in abstracting it away and forcing users to read yet another > document to get it right, e.g.: > > lltc,mode = <6>; > > As for the input disabling, since I doubt anyone would use it (why purchase > a 4-channel device and use only 2), just add two booleans, e.g. > "disable-inputs-34" and "disable-inputs-12" which set the command bits > appropriately, and change the mode such that the disabled inputs are voltage > readout only. > > A case could even be made for changing mode at runtime. This allows using it > to measure both current and voltage on two inputs, by reading V1, and V3, > and then switch mode to obtain (accurate) V1-V2 and V4-V3. > The hwmon subsystem doesn't really currently support that, and you'll likely confuse userspace as well. > That might be a viable way to handle not setting the mode at all. If the > mode can be selected via sysfs, the driver can keep the device in a "safe" > mode until the mode has been selected. > You'll have to present me with a really good use case for me to agree with that approach. Guenter > > >Mike, if you would like to test it, the latest version of my code is here: > > > >https://github.com/levens/ltc2990/blob/dev/drivers/hwmon/ltc2990.c > > Sure, I even have a board with 2 of these devices now :) > > -- > Mike Looijmans > > > Kind regards, > > Mike Looijmans > System Expert > > TOPIC Products > Materiaalweg 4, NL-5681 RJ Best > Postbus 440, NL-5680 AK Best > Telefoon: +31 (0) 499 33 69 79 > E-mail: mike.looijmans@topicproducts.com > Website: www.topicproducts.com > > Please consider the environment before printing this e-mail > > > -- To unsubscribe from this list: send the line "unsubscribe linux-hwmon" in the body of a message to majordomo@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html
On 28-06-17 19:02, Tom Levens wrote: > > On Wed, 28 Jun 2017, Guenter Roeck wrote: > >> On Wed, Jun 28, 2017 at 05:29:38PM +0200, Tom Levens wrote: >>> >> [ ... ] >> >>>> >>>>> Whatever happened to this patch though? It didn't make it to mainline, >>>>> otherwise I'd have found it sooner... >>>>> >>>> I'll have to look it up, but I guess I didn't get an updated version. >>> >>> As far as I remember I had a working V3 of this patch, but it is entirely >>> possible that it was never submitted as I have been busy with other projects >>> recently. I'll dig it out and check that it is complete. >>> >> FWIW, I don't see it at >> https://patchwork.kernel.org/project/linux-hwmon/list/?submitter=171225&state=* >> >> Maybe you were waiting for a reply from Rob. Either case, it might make >> sense to only provide valid modes, ie to abstract the mode bits from the >> hardware, such as >> >> 0 - internal temp only >> 1 - Tr1 >> 2 - V1 >> 3 - V1-V2 >> 4 - Tr2 >> 5 - V3 >> 6 - V3-V4 >> 7 to 14 - per bit 0..2 >> >> Guenter >> > > You are right, there was still an open question about how best to handle the > mode selection in DT. > > In the latest version of my patch I have it implemented as an array for > setting the two values, for example: > > lltc,meas-mode = <7 3>; > > This sets bits [2..0] = 7 and [4..3] = 3. Of course these could be split into > two DT properties, but I was unsure what to name them as both fields are > called "mode" in the datasheet and "mode-43"/"mode-20" (or similar) is ugly. > > With regards to your proposal, it is not clear to me whether the modes which > have the same result are exactly equivalent. Does disabling a measurement with > the mode[4..3] bits really leaves the part in a safe state for all possible HW > connections? With this doubt in my head, I would prefer to keep the option > available to the user to select any specific mode. But I am open to suggestions. > > Mike, if you would like to test it, the latest version of my code is here: > > https://github.com/levens/ltc2990/blob/dev/drivers/hwmon/ltc2990.c > I pulled your patches, added two lines to the devicetree and it works fine, I'm seeing plausible results: root@topic-miami:/sys/class/hwmon# grep . */* hwmon0/name:battery-gauge hwmon0/temp1_input:291700 hwmon1/in0_input:3258 hwmon1/in1_input:1824 hwmon1/in2_input:1916 hwmon1/in3_input:1512 hwmon1/in4_input:2066 hwmon1/name:ltc2990 hwmon1/temp1_input:28062 hwmon1/uevent:OF_NAME=ltc2990 hwmon1/uevent:OF_FULLNAME=/gpios-i2c@50/ltc2990@4c hwmon1/uevent:OF_COMPATIBLE_0=lltc,ltc2990 hwmon1/uevent:OF_COMPATIBLE_N=1 hwmon2/curr1_input:1825 hwmon2/curr2_input:349 hwmon2/in0_input:3244 hwmon2/name:ltc2990 hwmon2/temp1_input:31500 hwmon2/uevent:OF_NAME=ltc2990 hwmon2/uevent:OF_FULLNAME=/gpios-i2c@52/ltc2990@4C hwmon2/uevent:OF_COMPATIBLE_0=ltc2990 hwmon2/uevent:OF_COMPATIBLE_N=1 As you can see, two ltc2990 on two busses, one measuring current and one measuring 4 independent voltages. I also like your implementation of the mode setting better, it's much simpler than mine. And you've solved a bug in the temperature reading as well. Kind regards, Mike Looijmans System Expert TOPIC Products Materiaalweg 4, NL-5681 RJ Best Postbus 440, NL-5680 AK Best Telefoon: +31 (0) 499 33 69 79 E-mail: mike.looijmans@topicproducts.com Website: www.topicproducts.com Please consider the environment before printing this e-mail -- To unsubscribe from this list: send the line "unsubscribe linux-hwmon" in the body of a message to majordomo@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html
On Thu, 29 Jun 2017, Mike Looijmans wrote: > On 28-06-17 19:02, Tom Levens wrote: >> >> On Wed, 28 Jun 2017, Guenter Roeck wrote: >> >> > On Wed, Jun 28, 2017 at 05:29:38PM +0200, Tom Levens wrote: >> > > >> > [ ... ] >> > >> > > > >> > > > > Whatever happened to this patch though? It didn't make it to >> > > > > mainline, >> > > > > otherwise I'd have found it sooner... >> > > > > >> > > > I'll have to look it up, but I guess I didn't get an updated >> > > > version. >> > > >> > > As far as I remember I had a working V3 of this patch, but it is >> > > entirely >> > > possible that it was never submitted as I have been busy with other >> > > projects >> > > recently. I'll dig it out and check that it is complete. >> > > >> > FWIW, I don't see it at >> > https://patchwork.kernel.org/project/linux-hwmon/list/?submitter=171225&state=* >> > >> > Maybe you were waiting for a reply from Rob. Either case, it might make >> > sense to only provide valid modes, ie to abstract the mode bits from the >> > hardware, such as >> > >> > 0 - internal temp only >> > 1 - Tr1 >> > 2 - V1 >> > 3 - V1-V2 >> > 4 - Tr2 >> > 5 - V3 >> > 6 - V3-V4 >> > 7 to 14 - per bit 0..2 >> > >> > Guenter >> > >> >> You are right, there was still an open question about how best to handle >> the mode selection in DT. >> >> In the latest version of my patch I have it implemented as an array for >> setting the two values, for example: >> >> lltc,meas-mode = <7 3>; >> >> This sets bits [2..0] = 7 and [4..3] = 3. Of course these could be split >> into two DT properties, but I was unsure what to name them as both fields >> are called "mode" in the datasheet and "mode-43"/"mode-20" (or similar) is >> ugly. >> >> With regards to your proposal, it is not clear to me whether the modes >> which have the same result are exactly equivalent. Does disabling a >> measurement with the mode[4..3] bits really leaves the part in a safe >> state for all possible HW connections? With this doubt in my head, I would >> prefer to keep the option available to the user to select any specific >> mode. But I am open to suggestions. >> >> Mike, if you would like to test it, the latest version of my code is here: >> >> https://github.com/levens/ltc2990/blob/dev/drivers/hwmon/ltc2990.c >> > > I pulled your patches, added two lines to the devicetree and it works fine, > I'm seeing plausible results: > root@topic-miami:/sys/class/hwmon# grep . */* > hwmon0/name:battery-gauge > hwmon0/temp1_input:291700 > hwmon1/in0_input:3258 > hwmon1/in1_input:1824 > hwmon1/in2_input:1916 > hwmon1/in3_input:1512 > hwmon1/in4_input:2066 > hwmon1/name:ltc2990 > hwmon1/temp1_input:28062 > hwmon1/uevent:OF_NAME=ltc2990 > hwmon1/uevent:OF_FULLNAME=/gpios-i2c@50/ltc2990@4c > hwmon1/uevent:OF_COMPATIBLE_0=lltc,ltc2990 > hwmon1/uevent:OF_COMPATIBLE_N=1 > hwmon2/curr1_input:1825 > hwmon2/curr2_input:349 > hwmon2/in0_input:3244 > hwmon2/name:ltc2990 > hwmon2/temp1_input:31500 > hwmon2/uevent:OF_NAME=ltc2990 > hwmon2/uevent:OF_FULLNAME=/gpios-i2c@52/ltc2990@4C > hwmon2/uevent:OF_COMPATIBLE_0=ltc2990 > hwmon2/uevent:OF_COMPATIBLE_N=1 > > > As you can see, two ltc2990 on two busses, one measuring current and one > measuring 4 independent voltages. > > I also like your implementation of the mode setting better, it's much simpler > than mine. And you've solved a bug in the temperature reading as well. Excellent, I am glad it also works for you! In this case, I would propose that I submit the V3 of the patch for further discussion. Thanks, > > Kind regards, > > Mike Looijmans > System Expert > > TOPIC Products > Materiaalweg 4, NL-5681 RJ Best > Postbus 440, NL-5680 AK Best > Telefoon: +31 (0) 499 33 69 79 > E-mail: mike.looijmans@topicproducts.com > Website: www.topicproducts.com > > Please consider the environment before printing this e-mail > > > > -- To unsubscribe from this list: send the line "unsubscribe linux-hwmon" in the body of a message to majordomo@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html
diff --git a/Documentation/hwmon/ltc2990 b/Documentation/hwmon/ltc2990 index c25211e..3ed68f6 100644 --- a/Documentation/hwmon/ltc2990 +++ b/Documentation/hwmon/ltc2990 @@ -8,6 +8,7 @@ Supported chips: Datasheet: http://www.linear.com/product/ltc2990 Author: Mike Looijmans <mike.looijmans@topic.nl> + Tom Levens <tom.levens@cern.ch> Description @@ -16,10 +17,8 @@ Description LTC2990 is a Quad I2C Voltage, Current and Temperature Monitor. The chip's inputs can measure 4 voltages, or two inputs together (1+2 and 3+4) can be combined to measure a differential voltage, which is typically used to -measure current through a series resistor, or a temperature. - -This driver currently uses the 2x differential mode only. In order to support -other modes, the driver will need to be expanded. +measure current through a series resistor, or a temperature with an external +diode. Usage Notes @@ -32,12 +31,19 @@ devices explicitly. Sysfs attributes ---------------- +in0_input Voltage at Vcc pin in millivolt (range 2.5V to 5V) +temp1_input Internal chip temperature in millidegrees Celcius + +A subset of the following attributes are visible, depending on the measurement +mode of the chip. + +in[1-4]_input Voltage at V[1-4] pin in millivolt +temp2_input External temperature sensor TR1 in millidegrees Celcius +temp3_input External temperature sensor TR2 in millidegrees Celcius +curr1_input Current in mA across V1-V2 assuming a 1mOhm sense resistor +curr2_input Current in mA across V3-V4 assuming a 1mOhm sense resistor + The "curr*_input" measurements actually report the voltage drop across the input pins in microvolts. This is equivalent to the current through a 1mOhm sense resistor. Divide the reported value by the actual sense resistor value in mOhm to get the actual value. - -in0_input Voltage at Vcc pin in millivolt (range 2.5V to 5V) -temp1_input Internal chip temperature in millidegrees Celcius -curr1_input Current in mA across v1-v2 assuming a 1mOhm sense resistor. -curr2_input Current in mA across v3-v4 assuming a 1mOhm sense resistor. diff --git a/drivers/hwmon/Kconfig b/drivers/hwmon/Kconfig index 45cef3d..f7096ca 100644 --- a/drivers/hwmon/Kconfig +++ b/drivers/hwmon/Kconfig @@ -699,15 +699,12 @@ config SENSORS_LTC2945 be called ltc2945. config SENSORS_LTC2990 - tristate "Linear Technology LTC2990 (current monitoring mode only)" + tristate "Linear Technology LTC2990" depends on I2C help If you say yes here you get support for Linear Technology LTC2990 I2C System Monitor. The LTC2990 supports a combination of voltage, - current and temperature monitoring, but in addition to the Vcc supply - voltage and chip temperature, this driver currently only supports - reading two currents by measuring two differential voltages across - series resistors. + current and temperature monitoring. This driver can also be built as a module. If so, the module will be called ltc2990. diff --git a/drivers/hwmon/ltc2990.c b/drivers/hwmon/ltc2990.c index 0ec4102..e8d36f5 100644 --- a/drivers/hwmon/ltc2990.c +++ b/drivers/hwmon/ltc2990.c @@ -6,11 +6,7 @@ * * License: GPLv2 * - * This driver assumes the chip is wired as a dual current monitor, and - * reports the voltage drop across two series resistors. It also reports - * the chip's internal temperature and Vcc power supply voltage. - * - * Value conversion refactored + * Value conversion refactored and support for all measurement modes added * by Tom Levens <tom.levens@cern.ch> */ @@ -21,6 +17,7 @@ #include <linux/i2c.h> #include <linux/kernel.h> #include <linux/module.h> +#include <linux/of.h> #define LTC2990_STATUS 0x00 #define LTC2990_CONTROL 0x01 @@ -35,32 +32,96 @@ #define LTC2990_CONTROL_KELVIN BIT(7) #define LTC2990_CONTROL_SINGLE BIT(6) #define LTC2990_CONTROL_MEASURE_ALL (0x3 << 3) -#define LTC2990_CONTROL_MODE_CURRENT 0x06 -#define LTC2990_CONTROL_MODE_VOLTAGE 0x07 +#define LTC2990_CONTROL_MODE_DEFAULT 0x06 +#define LTC2990_CONTROL_MODE_MAX 0x07 + +#define LTC2990_IN0 BIT(0) +#define LTC2990_IN1 BIT(1) +#define LTC2990_IN2 BIT(2) +#define LTC2990_IN3 BIT(3) +#define LTC2990_IN4 BIT(4) +#define LTC2990_CURR1 BIT(5) +#define LTC2990_CURR2 BIT(6) +#define LTC2990_TEMP1 BIT(7) +#define LTC2990_TEMP2 BIT(8) +#define LTC2990_TEMP3 BIT(9) + +static const int ltc2990_attrs_ena[] = { + LTC2990_IN1 | LTC2990_IN2 | LTC2990_TEMP3, + LTC2990_CURR1 | LTC2990_TEMP3, + LTC2990_CURR1 | LTC2990_IN3 | LTC2990_IN4, + LTC2990_TEMP2 | LTC2990_IN3 | LTC2990_IN4, + LTC2990_TEMP2 | LTC2990_CURR2, + LTC2990_TEMP2 | LTC2990_TEMP3, + LTC2990_CURR1 | LTC2990_CURR2, + LTC2990_IN1 | LTC2990_IN2 | LTC2990_IN3 | LTC2990_IN4 +}; + +struct ltc2990_data { + struct i2c_client *i2c; + u32 mode; +}; /* Return the converted value from the given register in uV or mC */ -static int ltc2990_get_value(struct i2c_client *i2c, u8 reg, s32 *result) +static int ltc2990_get_value(struct i2c_client *i2c, int index, s32 *result) { s32 val; + u8 reg; + + switch (index) { + case LTC2990_IN0: + reg = LTC2990_VCC_MSB; + break; + case LTC2990_IN1: + case LTC2990_CURR1: + case LTC2990_TEMP2: + reg = LTC2990_V1_MSB; + break; + case LTC2990_IN2: + reg = LTC2990_V2_MSB; + break; + case LTC2990_IN3: + case LTC2990_CURR2: + case LTC2990_TEMP3: + reg = LTC2990_V3_MSB; + break; + case LTC2990_IN4: + reg = LTC2990_V4_MSB; + break; + case LTC2990_TEMP1: + reg = LTC2990_TINT_MSB; + break; + default: + return -EINVAL; + } val = i2c_smbus_read_word_swapped(i2c, reg); if (unlikely(val < 0)) return val; - switch (reg) { - case LTC2990_TINT_MSB: - /* internal temp, 0.0625 degrees/LSB, 13-bit */ + switch (index) { + case LTC2990_TEMP1: + case LTC2990_TEMP2: + case LTC2990_TEMP3: + /* temp, 0.0625 degrees/LSB, 13-bit */ *result = sign_extend32(val, 12) * 1000 / 16; break; - case LTC2990_V1_MSB: - case LTC2990_V3_MSB: - /* Vx-Vy, 19.42uV/LSB. Depends on mode. */ + case LTC2990_CURR1: + case LTC2990_CURR2: + /* Vx-Vy, 19.42uV/LSB */ *result = sign_extend32(val, 14) * 1942 / 100; break; - case LTC2990_VCC_MSB: - /* Vcc, 305.18μV/LSB, 2.5V offset */ + case LTC2990_IN0: + /* Vcc, 305.18uV/LSB, 2.5V offset */ *result = sign_extend32(val, 14) * 30518 / (100 * 1000) + 2500; break; + case LTC2990_IN1: + case LTC2990_IN2: + case LTC2990_IN3: + case LTC2990_IN4: + /* Vx: 305.18uV/LSB */ + *result = sign_extend32(val, 14) * 30518 / (100 * 1000); + break; default: return -EINVAL; /* won't happen, keep compiler happy */ } @@ -72,48 +133,104 @@ static ssize_t ltc2990_show_value(struct device *dev, struct device_attribute *da, char *buf) { struct sensor_device_attribute *attr = to_sensor_dev_attr(da); + struct ltc2990_data *data = dev_get_drvdata(dev); s32 value; int ret; - ret = ltc2990_get_value(dev_get_drvdata(dev), attr->index, &value); + ret = ltc2990_get_value(data->i2c, attr->index, &value); if (unlikely(ret < 0)) return ret; return snprintf(buf, PAGE_SIZE, "%d\n", value); } +static umode_t ltc2990_attrs_visible(struct kobject *kobj, + struct attribute *a, int n) +{ + struct device *dev = container_of(kobj, struct device, kobj); + struct ltc2990_data *data = dev_get_drvdata(dev); + struct device_attribute *da = + container_of(a, struct device_attribute, attr); + struct sensor_device_attribute *attr = to_sensor_dev_attr(da); + + if (attr->index == LTC2990_TEMP1 || + attr->index == LTC2990_IN0 || + attr->index & ltc2990_attrs_ena[data->mode]) + return a->mode; + else + return 0; +} + static SENSOR_DEVICE_ATTR(temp1_input, S_IRUGO, ltc2990_show_value, NULL, - LTC2990_TINT_MSB); + LTC2990_TEMP1); +static SENSOR_DEVICE_ATTR(temp2_input, S_IRUGO, ltc2990_show_value, NULL, + LTC2990_TEMP2); +static SENSOR_DEVICE_ATTR(temp3_input, S_IRUGO, ltc2990_show_value, NULL, + LTC2990_TEMP3); static SENSOR_DEVICE_ATTR(curr1_input, S_IRUGO, ltc2990_show_value, NULL, - LTC2990_V1_MSB); + LTC2990_CURR1); static SENSOR_DEVICE_ATTR(curr2_input, S_IRUGO, ltc2990_show_value, NULL, - LTC2990_V3_MSB); + LTC2990_CURR2); static SENSOR_DEVICE_ATTR(in0_input, S_IRUGO, ltc2990_show_value, NULL, - LTC2990_VCC_MSB); + LTC2990_IN0); +static SENSOR_DEVICE_ATTR(in1_input, S_IRUGO, ltc2990_show_value, NULL, + LTC2990_IN1); +static SENSOR_DEVICE_ATTR(in2_input, S_IRUGO, ltc2990_show_value, NULL, + LTC2990_IN2); +static SENSOR_DEVICE_ATTR(in3_input, S_IRUGO, ltc2990_show_value, NULL, + LTC2990_IN3); +static SENSOR_DEVICE_ATTR(in4_input, S_IRUGO, ltc2990_show_value, NULL, + LTC2990_IN4); static struct attribute *ltc2990_attrs[] = { &sensor_dev_attr_temp1_input.dev_attr.attr, + &sensor_dev_attr_temp2_input.dev_attr.attr, + &sensor_dev_attr_temp3_input.dev_attr.attr, &sensor_dev_attr_curr1_input.dev_attr.attr, &sensor_dev_attr_curr2_input.dev_attr.attr, &sensor_dev_attr_in0_input.dev_attr.attr, + &sensor_dev_attr_in1_input.dev_attr.attr, + &sensor_dev_attr_in2_input.dev_attr.attr, + &sensor_dev_attr_in3_input.dev_attr.attr, + &sensor_dev_attr_in4_input.dev_attr.attr, NULL, }; -ATTRIBUTE_GROUPS(ltc2990); + +static const struct attribute_group ltc2990_group = { + .attrs = ltc2990_attrs, + .is_visible = ltc2990_attrs_visible, +}; +__ATTRIBUTE_GROUPS(ltc2990); static int ltc2990_i2c_probe(struct i2c_client *i2c, const struct i2c_device_id *id) { int ret; struct device *hwmon_dev; + struct ltc2990_data *data; + struct device_node *of_node = i2c->dev.of_node; if (!i2c_check_functionality(i2c->adapter, I2C_FUNC_SMBUS_BYTE_DATA | I2C_FUNC_SMBUS_WORD_DATA)) return -ENODEV; - /* Setup continuous mode, current monitor */ + data = devm_kzalloc(&i2c->dev, sizeof(struct ltc2990_data), GFP_KERNEL); + if (unlikely(!data)) + return -ENOMEM; + data->i2c = i2c; + + if (!of_node || of_property_read_u32(of_node, "lltc,mode", &data->mode)) + data->mode = LTC2990_CONTROL_MODE_DEFAULT; + + if (data->mode > LTC2990_CONTROL_MODE_MAX) { + dev_err(&i2c->dev, "Error: Invalid mode %d.\n", data->mode); + return -EINVAL; + } + + /* Setup continuous mode */ ret = i2c_smbus_write_byte_data(i2c, LTC2990_CONTROL, LTC2990_CONTROL_MEASURE_ALL | - LTC2990_CONTROL_MODE_CURRENT); + data->mode); if (ret < 0) { dev_err(&i2c->dev, "Error: Failed to set control mode.\n"); return ret; @@ -127,7 +244,7 @@ static int ltc2990_i2c_probe(struct i2c_client *i2c, hwmon_dev = devm_hwmon_device_register_with_groups(&i2c->dev, i2c->name, - i2c, + data, ltc2990_groups); return PTR_ERR_OR_ZERO(hwmon_dev);
Updated version of the ltc2990 driver which supports all measurement modes available in the chip. The mode can be set through a devicetree attribute. Signed-off-by: Tom Levens <tom.levens@cern.ch> --- Changes since v1: * Refactored value conversion (patch 1/3) * Split the devicetree binding into separate patch (patch 2/3) * Specifying an invalid mode now returns -EINVAL, previously this only issued a warning and used the default value * Removed the "mode" sysfs attribute, as the mode of the chip is hardware specific and should not be user configurable. This allows much simpler code as a result. Documentation/hwmon/ltc2990 | 24 ++++--- drivers/hwmon/Kconfig | 7 +-- drivers/hwmon/ltc2990.c | 167 ++++++++++++++++++++++++++++++++++++------- 3 files changed, 159 insertions(+), 39 deletions(-)