Re: [PATCH 2/2] iio: humidity: Add support for ENS21x

From: Krzysztof Kozlowski
Date: Wed Jul 10 2024 - 06:17:27 EST


On 09/07/2024 18:36, Joshua Felmeden wrote:
> Add support for ENS210/ENS210A/ENS211/ENS212/ENS213A/ENS215.
>
> The ENS21x is a family of temperature and relative humidity sensors with
> accuracies tailored to the needs of specific applications.
>
> Signed-off-by: Joshua Felmeden <jfelmeden@xxxxxxxxxxxxxxxxxxxx>
> ---
> drivers/iio/humidity/Kconfig | 11 ++
> drivers/iio/humidity/Makefile | 1 +
> drivers/iio/humidity/ens21x.c | 348 ++++++++++++++++++++++++++++++++++++++++++
> 3 files changed, 360 insertions(+)
>


> +
> +static int ens21x_probe(struct i2c_client *client)
> +{
> + const struct i2c_device_id *id = i2c_client_get_device_id(client);
> + const struct of_device_id *match;
> + struct ens21x_dev *dev_data;
> + struct iio_dev *indio_dev;
> + uint16_t part_id_le, part_id;
> + int ret, tries;
> +
> + if (!i2c_check_functionality(client->adapter,
> + I2C_FUNC_SMBUS_WRITE_BYTE_DATA |
> + I2C_FUNC_SMBUS_WRITE_BYTE |
> + I2C_FUNC_SMBUS_READ_I2C_BLOCK)) {
> + dev_err(&client->dev,
> + "adapter does not support some i2c transactions\n");
> + return -EOPNOTSUPP;
> + }
> +
> + match = of_match_device(ens21x_of_match, &client->dev);
> + if (!match) {
> + dev_err(&client->dev, "failed to get match data\n");

That's odd. This should never happen, so printing error suggests
something is odd in your driver.

There is anyway helper for getting match data from i2c/of cases.


> + return -ENODEV;
> + }
> +
> + indio_dev = devm_iio_device_alloc(&client->dev, sizeof(*dev_data));
> + if (!indio_dev)
> + return -ENOMEM;
> +
> + dev_data = iio_priv(indio_dev);
> + i2c_set_clientdata(client, indio_dev);
> + dev_data->client = client;
> + mutex_init(&dev_data->lock);
> +
> + /* reset device */
> + ret = i2c_smbus_write_byte_data(client, ENS21X_REG_SYS_CTRL,
> + ENS21X_SYS_CTRL_SYS_RESET);
> + if (ret)
> + return ret;
> +
> + /* wait for device to become active */
> + usleep_range(4000, 5000);
> +
> + /* disable low power mode */
> + ret = i2c_smbus_write_byte_data(client, ENS21X_REG_SYS_CTRL, 0x00);
> + if (ret)
> + return ret;
> +
> + /* wait for device to become active */
> + tries = 10;
> + while (tries-- > 0) {
> + msleep(20);
> + ret = i2c_smbus_read_byte_data(client, ENS21X_REG_SYS_STAT);
> + if (ret < 0)
> + return ret;
> + if (ret & ENS21X_SYS_STAT_SYS_ACTIVE)
> + break;
> + }
> + if (tries < 0) {
> + dev_err(&client->dev,
> + "timeout waiting for ens21x to become active\n");
> + return -EIO;
> + }
> +
> + /* get part_id */
> + part_id_le = i2c_smbus_read_word_data(client, ENS21X_REG_PART_ID);
> + if (part_id_le < 0)
> + return part_id_le;
> + part_id = le16_to_cpu(part_id_le);
> +
> + if (part_id != id->driver_data) {
> + dev_err(&client->dev,
> + "Part ID does not match (0x%04x != 0x%04lx)\n", part_id,
> + id->driver_data);
> + return -ENODEV;
> + }
> +
> + /* reenable low power */
> + ret = i2c_smbus_write_byte_data(client, ENS21X_REG_SYS_CTRL,
> + ENS21X_SYS_CTRL_LOW_POWER_ENABLE);
> + if (ret)
> + return ret;
> +
> + dev_data->part_id = part_id;
> +
> + indio_dev->name = id->name;
> + indio_dev->modes = INDIO_DIRECT_MODE;
> + indio_dev->channels = ens21x_channels;
> + indio_dev->num_channels = ARRAY_SIZE(ens21x_channels);
> + indio_dev->info = &ens21x_info;
> +
> + return devm_iio_device_register(&client->dev, indio_dev);
> +}
> +
> +
> +static const struct of_device_id ens21x_of_match[] = {
> + { .compatible = "sciosense,ens210" },
> + { .compatible = "sciosense,ens210a" },
> + { .compatible = "sciosense,ens211" },
> + { .compatible = "sciosense,ens212" },
> + { .compatible = "sciosense,ens213a" },
> + { .compatible = "sciosense,ens215" },

Mismathed with i2c_device_id. These should have the same data. Also,
keep the tables next to each other (here).



Best regards,
Krzysztof