Re: [PATCH 2/2] iio: temperature: tmp006: make sure the chip is powered up in probe

From: Jonathan Cameron
Date: Sat Jul 17 2021 - 14:17:11 EST


On Thu, 24 Jun 2021 11:19:24 +0300
Alexandru Ardelean <aardelean@xxxxxxxxxxx> wrote:

> When the device is probed, there's no guarantee that the device is not in
> power-down mode. This can happen if the driver is unregistered and
> re-probed.
>
> To make sure this doesn't happen, the value of the TMP006_CONFIG register
> (which is read in the probe function and stored in the device's private
> data) is being checked to see if the MOD bits have the correct value.
>
> This is a fix for a somewhat-rare corner case. As it stands, this doesn't
> look like a high priority to go into the Fixes route.
>
> Signed-off-by: Alexandru Ardelean <aardelean@xxxxxxxxxxx>
This one 'looks' right and as it's been on the list a while I'll apply it.
Would be great if anyone can test it though!

Series applied.

Thanks,

Jonathan

> ---
> drivers/iio/temperature/tmp006.c | 33 +++++++++++++++++++++-----------
> 1 file changed, 22 insertions(+), 11 deletions(-)
>
> diff --git a/drivers/iio/temperature/tmp006.c b/drivers/iio/temperature/tmp006.c
> index db1ac6029c27..e4943a0bc9aa 100644
> --- a/drivers/iio/temperature/tmp006.c
> +++ b/drivers/iio/temperature/tmp006.c
> @@ -193,15 +193,23 @@ static bool tmp006_check_identification(struct i2c_client *client)
> return mid == TMP006_MANUFACTURER_MAGIC && did == TMP006_DEVICE_MAGIC;
> }
>
> -static int tmp006_powerdown(struct tmp006_data *data)
> +static int tmp006_power(struct device *dev, bool up)
> {
> + struct iio_dev *indio_dev = i2c_get_clientdata(to_i2c_client(dev));
> + struct tmp006_data *data = iio_priv(indio_dev);
> +
> + if (up)
> + data->config |= TMP006_CONFIG_MOD_MASK;
> + else
> + data->config &= ~TMP006_CONFIG_MOD_MASK;
> +
> return i2c_smbus_write_word_swapped(data->client, TMP006_CONFIG,
> - data->config & ~TMP006_CONFIG_MOD_MASK);
> + data->config);
> }
>
> -static void tmp006_powerdown_cleanup(void *data)
> +static void tmp006_powerdown_cleanup(void *dev)
> {
> - tmp006_powerdown(data);
> + tmp006_power(dev, false);
> }
>
> static int tmp006_probe(struct i2c_client *client,
> @@ -239,7 +247,14 @@ static int tmp006_probe(struct i2c_client *client,
> return ret;
> data->config = ret;
>
> - ret = devm_add_action(&client->dev, tmp006_powerdown_cleanup, data);
> + if ((ret & TMP006_CONFIG_MOD_MASK) != TMP006_CONFIG_MOD_MASK) {
> + ret = tmp006_power(&client->dev, true);
> + if (ret < 0)
> + return ret;
> + }
> +
> + ret = devm_add_action_or_reset(&client->dev, tmp006_powerdown_cleanup,
> + &client->dev);
> if (ret < 0)
> return ret;
>
> @@ -249,16 +264,12 @@ static int tmp006_probe(struct i2c_client *client,
> #ifdef CONFIG_PM_SLEEP
> static int tmp006_suspend(struct device *dev)
> {
> - struct iio_dev *indio_dev = i2c_get_clientdata(to_i2c_client(dev));
> - return tmp006_powerdown(iio_priv(indio_dev));
> + return tmp006_power(dev, false);
> }
>
> static int tmp006_resume(struct device *dev)
> {
> - struct tmp006_data *data = iio_priv(i2c_get_clientdata(
> - to_i2c_client(dev)));
> - return i2c_smbus_write_word_swapped(data->client, TMP006_CONFIG,
> - data->config | TMP006_CONFIG_MOD_MASK);
> + return tmp006_power(dev, true);
> }
> #endif
>