From: Guenter Roeck <linux@roeck-us.net>
To: Nishanth Menon <nm@ti.com>, Jean Delvare <jdelvare@suse.com>
Cc: linux-kernel@vger.kernel.org, lm-sensors@lm-sensors.org,
linux-omap@vger.kernel.org, beagleboard-x15@googlegroups.com,
Eduardo Valentin <edubezval@gmail.com>
Subject: Re: [PATCH] hwmon: (tmp102) Force wait for conversion time for the first valid data
Date: Mon, 30 Nov 2015 21:50:11 -0800 [thread overview]
Message-ID: <565D3513.7050905@roeck-us.net> (raw)
In-Reply-To: <1448943955-2385-1-git-send-email-nm@ti.com>
On 11/30/2015 08:25 PM, Nishanth Menon wrote:
> TMP102 works based on conversions done periodically. However, as per
> the TMP102 data sheet[1] the first conversion is triggered immediately
> after we program the configuration register. The temperature data
> registers do not reflect proper data until the first conversion is
> complete (in our case HZ/4).
>
> The driver currently sets the last_update to be jiffies - HZ, just
> after the configuration is complete. When tmp102 driver registers
> with the thermal framework, it immediately tries to read the sensor
> temperature data. This takes place even before the conversion on the
> TMP102 is complete and results in an invalid temperature read.
>
> Depending on the value read, this may cause thermal framework to
> assume that a critical temperature event has occurred and attempts to
> shutdown the system.
>
> Instead of causing an invalid mid-conversion value to be read
> erroneously, we mark the last_update to be in-line with the current
> jiffies. This allows the tmp102_update_device function to skip update
> until the required conversion time is complete. Further, we ensure to
> return -EAGAIN result instead of returning spurious temperature (such
> as 0C) values to the caller to prevent any wrong decisions made with
> such values.
>
> A simpler alternative approach could be to sleep in the probe for the
> duration required, but that will result in latency that is undesirable
> that can delay boot sequence un-necessarily.
>
A really simpler solution would be to mark when the device is ready
to be accessed in the probe function, and go to sleep for the remaining time
in the update function if necessary. This would not affect the probe function,
avoid the somewhat awkward -EAGAIN, avoid overloading the value cache, and only
sleep if necessary and as long as needed.
> [1] http://www.ti.com/lit/ds/symlink/tmp102.pdf
>
> Cc: Eduardo Valentin <edubezval@gmail.com>
> Reported-by: Aparna Balasubramanian <aparnab@ti.com>
> Reported-by: Elvita Lobo <elvita@ti.com>
> Reported-by: Yan Liu <yan-liu@ti.com>
> Signed-off-by: Nishanth Menon <nm@ti.com>
> ---
>
> Example case (from Beagleboard-x15 using an older kernel revision):
> http://pastebin.ubuntu.com/13591711/
> Notice the thermal shutdown trigger:
> thermal thermal_zone3: critical temperature reached(108 C),shutting down
>
> drivers/hwmon/tmp102.c | 19 ++++++++++++++++++-
> 1 file changed, 18 insertions(+), 1 deletion(-)
>
> diff --git a/drivers/hwmon/tmp102.c b/drivers/hwmon/tmp102.c
> index 65482624ea2c..145f69108f23 100644
> --- a/drivers/hwmon/tmp102.c
> +++ b/drivers/hwmon/tmp102.c
> @@ -50,6 +50,9 @@
> #define TMP102_TLOW_REG 0x02
> #define TMP102_THIGH_REG 0x03
>
> +/* TMP102 range is -55 to 150C -> we use -128 as a default invalid value */
> +#define TMP102_NOTREADY -128
> +
This is a bit misleading, and also not correct, since the temperature is stored in
milli-degrees C, so a value of -128 reflects -0.128 degreees C. While that value
will not be seen in practice, it is still not a good idea to use it for this purpose.
Even though the chip temperature range is -55 .. 150 C, that doesn't mean
it never returns a value outside that range, for example if nothing is connected
to an external sensor or if something is broken.
You should use a value outside the value range, ie outside
[-128,000 .. 127,999 ] to detect the "not ready" condition.
> struct tmp102 {
> struct i2c_client *client;
> struct device *hwmon_dev;
> @@ -102,6 +105,12 @@ static int tmp102_read_temp(void *dev, int *temp)
> {
> struct tmp102 *tmp102 = tmp102_update_device(dev);
>
> + /* Is it too early even to return a conversion? */
> + if (tmp102->temp[0] == TMP102_NOTREADY) {
> + dev_dbg(dev, "%s: Conversion not ready yet..\n", __func__);
> + return -EAGAIN;
Does this cause a hard loop in the calling code, or will the thermal code
delay before it reads again ?
If it causes a hard loop, it may be better to go to sleep if needed
when reading the data, as suggested above.
> + }
> +
> *temp = tmp102->temp[0];
>
> return 0;
> @@ -114,6 +123,10 @@ static ssize_t tmp102_show_temp(struct device *dev,
> struct sensor_device_attribute *sda = to_sensor_dev_attr(attr);
> struct tmp102 *tmp102 = tmp102_update_device(dev);
>
> + /* Is it too early even to return a read? */
> + if (tmp102->temp[sda->index] == TMP102_NOTREADY)
> + return -EAGAIN;
> +
> return sprintf(buf, "%d\n", tmp102->temp[sda->index]);
> }
>
> @@ -207,7 +220,11 @@ static int tmp102_probe(struct i2c_client *client,
> status = -ENODEV;
> goto fail_restore_config;
> }
> - tmp102->last_update = jiffies - HZ;
> + tmp102->last_update = jiffies;
> + /* Mark that we are not ready with data until conversion is complete */
> + tmp102->temp[0] = TMP102_NOTREADY;
> + tmp102->temp[1] = TMP102_NOTREADY;
> + tmp102->temp[2] = TMP102_NOTREADY;
> mutex_init(&tmp102->lock);
>
> hwmon_dev = hwmon_device_register_with_groups(dev, client->name,
>
next prev parent reply other threads:[~2015-12-01 5:50 UTC|newest]
Thread overview: 8+ messages / expand[flat|nested] mbox.gz Atom feed top
2015-12-01 4:25 [PATCH] hwmon: (tmp102) Force wait for conversion time for the first valid data Nishanth Menon
2015-12-01 5:50 ` Guenter Roeck [this message]
2015-12-01 13:47 ` Nishanth Menon
2015-12-01 14:21 ` Nishanth Menon
2015-12-01 15:09 ` Guenter Roeck
2015-12-01 15:14 ` Nishanth Menon
2015-12-01 16:10 ` [PATCH V2] " Nishanth Menon
2015-12-01 21:06 ` Guenter Roeck
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=565D3513.7050905@roeck-us.net \
--to=linux@roeck-us.net \
--cc=beagleboard-x15@googlegroups.com \
--cc=edubezval@gmail.com \
--cc=jdelvare@suse.com \
--cc=linux-kernel@vger.kernel.org \
--cc=linux-omap@vger.kernel.org \
--cc=lm-sensors@lm-sensors.org \
--cc=nm@ti.com \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).