Re: [PATCH V2] iio: light: driver for Vishay VEML6040

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



On Mon, 13 May 2024 14:33:39 +0000
Arthur Becker <arthur.becker@xxxxxxxxxx> wrote:

> Implements driver for the Vishay VEML6040 rgbw light sensor.
> 
> Included functionality: setting the integration time and reading the raw
> values for the four channels
> 
> Not yet implemented: setting the measurements to 'Manual Force Mode' (Auto
> measurements off, and adding a measurement trigger)
> 
> Datasheet: https://www.vishay.com/docs/84276/veml6040.pdf
> signed-off-by: Arthur Becker <arthur.becker@xxxxxxxxxx>
> ---
> V1 -> V2: Addressed review comments. DT-bindings in separate patch

Needs to be in the same series - with a cover-letter.
That explains the confusion going on wrt to the binding patch.
I was assuming this was unintentional mail server stuff but
from this comment I guess it's a misunderstanding.

Various comments inline. Pretty much all minor style things
as in general the driver is looking nice.

Jonathan

> diff --git a/drivers/iio/light/veml6040.c b/drivers/iio/light/veml6040.c
> new file mode 100644
> index 000000000000..9ce807d5484a
> --- /dev/null
> +++ b/drivers/iio/light/veml6040.c
> @@ -0,0 +1,307 @@
> +#include <linux/bitfield.h>
> +#include <linux/err.h>
> +#include <linux/i2c.h>
> +#include <linux/iio/iio.h>
> +#include <linux/iio/sysfs.h>
> +#include <linux/module.h>
> +#include <linux/regmap.h>
> +

/*
 * VEML...

> +/* VEML6040 Configuration Registers
> + *
> + * SD: Shutdown
> + * AF: Auto / Force Mode (Auto Measurements On:0, Off:1)
> + * TR: Trigger Measurement (when AF Bit is set)
> + * IT: Integration Time
> + */
> +#define VEML6040_CONF_REG_RW 0x000
> +#define VEML6040_CONF_SD_MSK BIT(0)
> +#define VEML6040_CONF_AF_MSK BIT(1)
> +#define VEML6040_CONF_TR_MSK BIT(2)
> +#define VEML6040_CONF_IT_MSK GENMASK(6, 4)
> +#define VEML6040_CONF_IT_40 0
> +#define VEML6040_CONF_IT_80 1
> +#define VEML6040_CONF_IT_160 2
> +#define VEML6040_CONF_IT_320 3
> +#define VEML6040_CONF_IT_640 4
> +#define VEML6040_CONF_IT_1280 5
> +
> +/* VEML6040 Read Only Registers */
> +#define VEML6040_REG_R_RO 0x08
> +#define VEML6040_REG_G_RO 0x09
> +#define VEML6040_REG_B_RO 0x0A
> +#define VEML6040_REG_W_RO 0x0B
I'd be tempted to drop the _RW / _RO postfix.
It's a nice thing in more complex drives, but feels unnecessary
for a device with a small interface like this where the register usage
kind of makes it obvious anyway.
Channel registers are RO because writing them makes not sense and
config is RW because it's for configuration.

If you really like it then I don't mind that much.

> +static int veml6040_read_raw(struct iio_dev *indio_dev,
> +			     struct iio_chan_spec const *chan, int *val,
> +			     int *val2, long mask)
> +{
> +	int ret, reg, it_index;
> +	struct veml6040_data *data = iio_priv(indio_dev);
> +	struct regmap *regmap = data->regmap;
> +	struct device *dev = &data->client->dev;
> +
> +	switch (mask) {
> +	case IIO_CHAN_INFO_RAW:
> +		switch (chan->channel) {
Put VEML6040_REG_R_R0 etc in
	chan->address when you define the iio_chan_spec array
as then this becomes

		ret = regmap_read(regmap, chan->addr, &reg);
		if (ret) {
			dev_err(dev, "Data read failed: %d\n", ret);
			return ret;
		}
		*val = reg;
		return IIO_VAL_INT;

The address field is there for precisely this kind of case where you just
need a mapping to a specific register address.

> +		case CH_RED:
> +			ret = regmap_read(regmap, VEML6040_REG_R_RO, &reg);
> +			break;
> +		case CH_GREEN:
> +			ret = regmap_read(regmap, VEML6040_REG_G_RO, &reg);
> +			break;
> +		case CH_BLUE:
> +			ret = regmap_read(regmap, VEML6040_REG_B_RO, &reg);
> +			break;
> +		case CH_WHITE:
> +			ret = regmap_read(regmap, VEML6040_REG_W_RO, &reg);
> +			break;
> +		default:
> +			return -EINVAL;
> +		}
> +		if (ret) {
> +			dev_err(dev, "iio-veml6040 - Can't read data %d\n",

No need to list the driver, that's easy to establish from what dev_err includes
anyway.

> +				ret);
> +			return ret;
> +		}
> +		*val = reg;
> +		return IIO_VAL_INT;
> +
> +	case IIO_CHAN_INFO_INT_TIME:
> +		ret = regmap_read(regmap, VEML6040_CONF_REG_RW, &reg);
> +		if (ret) {
> +			dev_err(dev, "iio-veml6040 - Can't read data %d\n",
> +				ret);
> +			return ret;
> +		}
> +		it_index = FIELD_GET(VEML6040_CONF_IT_MSK, reg);
> +		if (it_index >= ARRAY_SIZE(veml6040_int_time_avail)) {
> +			dev_err(dev,
> +				"iio-veml6040 - Invalid Integration Time Set");
> +			return -EINVAL;
> +		}
> +		*val = veml6040_int_time_avail[it_index];
> +		return IIO_VAL_INT;
> +
> +	default:
> +		return -EINVAL;
> +	}
> +}
> +
> +static int veml6040_write_raw(struct iio_dev *indio_dev,
> +			      struct iio_chan_spec const *chan, int val,
> +			      int val2, long mask)
> +{
> +	struct veml6040_data *data = iio_priv(indio_dev);
> +
> +	switch (mask) {
> +	case IIO_CHAN_INFO_INT_TIME:
> +		for (int i = 0; i < ARRAY_SIZE(veml6040_int_time_avail); i++) {
> +			if (veml6040_int_time_avail[i] == val) {
> +				return regmap_update_bits(
> +					data->regmap, VEML6040_CONF_REG_RW,
> +					VEML6040_CONF_IT_MSK,
> +					FIELD_PREP(VEML6040_CONF_IT_MSK, i));
> +			}
Trick to reduce indent - flip logic ;)
			if (veml6040_int_time_avail[i] != val)
				continue;

			return regmap_update_bits(regmap,
						  VEML6040_CONF_REG_RW,
						  VEML6040_CONF_IT_MSK,
						  FIELD_PREP(VEML6040_CONF_IT_MSK, i));
//case where it is worth going past 80 chars for readability.
			
> +		}
> +		return -EINVAL;
> +	default:
> +		return -EINVAL;
> +	}
> +	return 0;

Can't get here so drop this final return.

> +}
> +
> +static int veml6040_read_avail(struct iio_dev *indio_dev,
> +			       struct iio_chan_spec const *chan,
> +			       const int **vals, int *type, int *length,
> +			       long mask)
> +{
> +	switch (mask) {
> +	case IIO_CHAN_INFO_INT_TIME:
> +		*length = ARRAY_SIZE(veml6040_int_time_avail);
> +		*vals = veml6040_int_time_avail;
> +		*type = IIO_VAL_INT;

That makes me suspicious.  The integration times are measured in seconds and
you only have integer values? 
Check all your ABI against the docs in Documentation/ABI/testing/sysfs-bus-iio



> +		return IIO_AVAIL_LIST;
> +
> +	default:
> +		return -EINVAL;
> +	}
> +}

> +
> +static int veml6040_shutdown(struct veml6040_data *data)
> +{
> +	return regmap_update_bits(data->regmap, VEML6040_CONF_REG_RW,
> +				  VEML6040_CONF_SD_MSK, VEML6040_CONF_SD_MSK);
> +}
> +
> +static void veml6040_shutdown_action(void *data)
> +{
> +	veml6040_shutdown(data);

Combine these two functions into one as only called via this one.

> +}
> +
> +static int veml6040_probe(struct i2c_client *client)
> +{

Add a 
	struct device *dev = client->dev;
and use it throughout to avoid looking this up in every
error print etc.

> +	struct veml6040_data *data;
> +	struct iio_dev *indio_dev;
> +	struct regmap *regmap;
> +	int ret;
> +
> +	if (!i2c_check_functionality(client->adapter, I2C_FUNC_I2C)) {
> +		return dev_err_probe(&client->dev, -EOPNOTSUPP,
> +				     "I2C adapter doesn't support plain I2C\n");
> +	}
> +
> +	indio_dev = devm_iio_device_alloc(&client->dev, sizeof(*data));
> +	if (!indio_dev) {
> +		return dev_err_probe(&client->dev, -ENOMEM,
> +				     "IIO device allocation failed\n");
> +	}
> +
> +	regmap = devm_regmap_init_i2c(client, &veml6040_regmap_config);
> +	if (IS_ERR(regmap)) {
> +		return dev_err_probe(&client->dev, PTR_ERR(regmap),
> +				     "Regmap setup failed\n");
> +	}
> +
> +	data = iio_priv(indio_dev);
> +	i2c_set_clientdata(client, indio_dev);
> +	data->client = client;
> +	data->regmap = regmap;
> +
> +	indio_dev->name = "veml6040";
> +	indio_dev->info = &veml6040_info;
> +	indio_dev->channels = veml6040_channels;
> +	indio_dev->num_channels = ARRAY_SIZE(veml6040_channels);
> +	indio_dev->modes = INDIO_DIRECT_MODE;
> +
> +	ret = regmap_update_bits(
> +		data->regmap, VEML6040_CONF_REG_RW, VEML6040_CONF_IT_MSK,
You have regmap locally, so use that instead of via data->regmap.

> +		FIELD_PREP(VEML6040_CONF_IT_MSK, VEML6040_CONF_IT_40));
Unusual wrapping that we tend to only use where there are individual
parameters that can't be easily wrapped.  Preferred as

	ret = regmap_update_bits(regmap, VEML6040_CONF_REG_RW,
				 VEML6040_CONF_IT_MSK,
				 FIELD_PREP(VEML6040_CONF_IT_MASK,
					    VEML6040_CONF_IT_40));

> +
> +	if (ret) {
> +		return dev_err_probe(&client->dev, ret,
> +				     "Could not set Integration Time: %d\n",
> +				     ret);
> +	}
	if (ret)
		return dev_err_probe(dev, ret,
				     "Could not set Integration Time\n");
similar in other cases.


> +
> +	ret = regmap_update_bits(data->regmap, VEML6040_CONF_REG_RW,
> +				 VEML6040_CONF_AF_MSK,
> +				 FIELD_PREP(VEML6040_CONF_AF_MSK, 0));
> +	if (ret) {
> +		return dev_err_probe(&client->dev, ret,
> +				     "Could not set Automode: %d\n", ret);
> +	}
You are writing multiple fields in the same register. Can you just do
a single write? Build the mask and values then a single call to  regmap_update_bits
or just write the remaining couple of bits to their existing defaults so
we know what state they are in.

If there is a reason this needs to be done as multiple writes, then
add some comments to explain the sequencing.

> +
> +	ret = regmap_update_bits(data->regmap, VEML6040_CONF_REG_RW,
> +				 VEML6040_CONF_SD_MSK,
> +				 FIELD_PREP(VEML6040_CONF_SD_MSK, 0));
> +	if (ret) {
> +		return dev_err_probe(&client->dev, ret,
> +				     "Could not set Enable: %d\n", ret);
> +	}

No need for brackets around a single statement like above.
Fix all of these as there are a lot of them!

> +
> +	ret = devm_add_action_or_reset(&client->dev, veml6040_shutdown_action,
> +				       data);
> +	if (ret) {
> +		return dev_err_probe(&client->dev, ret,
> +				     "Could not add shutdown action: %d\n",
> +				     ret);

You should look at what dev_err_probe() does.
Don't print the ret value in your messages - dev_err_probe() does it much
better!

Some of these are also vanishingly unlikely to fail so you could reduce
where you print messages to the ones that are more to do with bus accesses
etc.  This one is an example of that as it can only fail if a very small
allocation fails.

> +	}
> +
> +	return devm_iio_device_register(&client->dev, indio_dev);
> +}
> +
> +static const struct i2c_device_id veml6040_id_table[] = { { "veml6040", 0 },
> +							  {} };
I'd prefer this in the more common formatting of 

static const struct i2c_device_id veml6040_id_table[] = {
	{ "veml6040" }, //note I dropped the 0 as no point in setting that when it's unused.
        { }
};

> +MODULE_DEVICE_TABLE(i2c, veml6040_id_table);
> +





[Index of Archives]     [Linux USB Devel]     [Video for Linux]     [Linux Audio Users]     [Yosemite News]     [Linux Input]     [Linux Kernel]     [Linux SCSI]     [X.org]

  Powered by Linux