On 14/06/15 12:28, Daniel Baluta wrote: > On Sun, Jun 14, 2015 at 1:51 PM, Jonathan Cameron <jic23@xxxxxxxxxx> wrote: >> On 10/06/15 15:21, Daniel Baluta wrote: >>> This patch adds support for ROHM RPR0521 ambient light and proximity >>> sensor. It offers raw readings for intensity and proximity. >>> >>> Signed-off-by: Daniel Baluta <daniel.baluta@xxxxxxxxx> >> Looks good to me. A few comments inline. Only one I'd really >> bother making a change for is to be consistent using genmask >> for all the masks rather than just some of them. > > Agree. I will fix this. > >> >> Another thought is whether the regmap field stuff would give some >> gains in this driver. What do you think? >> > > Lately, I've seen that there is a strong preference for using regmap. > Personally, I like > it because I don't have to deal with deal with bit operations. > > Here, the only advantage seems to be for caching gain values. Also, one future > possible benefit of regmap would be easier porting for SPI bus (if added). > > So, if you don't mind I prefer leaving it like this. That's fine. Using the field stuff mostly only gives a small gain in clarity and that only once you have looked up how it works :) Jonathan > >> Anyhow, if this would make the merge window I'd have taken it now >> but seeing as we just missed anyway, lets tidy up these loose ends >> and give others a little more time to comment. >> >> Thanks, >> >> Jonathan >> >>> --- >>> Changes since v2: (after Peter's review) >>> * replaced calibscale with scale >>> * several style fixes (new lines, spellings, etc) >>> * fixed proximity interpretation. Now we have the >>> following relation: high proximity <-> low distance >>> * will send a follow up patch to clarify the ABI description >>> and to fix all other drivers that misinterpret this. >>> >>> drivers/iio/light/Kconfig | 11 + >>> drivers/iio/light/Makefile | 1 + >>> drivers/iio/light/rpr0521.c | 622 ++++++++++++++++++++++++++++++++++++++++++++ >>> 3 files changed, 634 insertions(+) >>> create mode 100644 drivers/iio/light/rpr0521.c >>> >>> diff --git a/drivers/iio/light/Kconfig b/drivers/iio/light/Kconfig >>> index e6198b7..cbc4677 100644 >>> --- a/drivers/iio/light/Kconfig >>> +++ b/drivers/iio/light/Kconfig >>> @@ -168,6 +168,17 @@ config JSA1212 >>> To compile this driver as a module, choose M here: >>> the module will be called jsa1212. >>> >>> +config RPR0521 >>> + tristate "ROHM RPR0521 ALS and proximity sensor driver" >>> + depends on I2C >>> + select REGMAP_I2C >>> + help >>> + Say Y here if you want to build support for ROHM's RPR0521 >>> + ambient light and proximity sensor device. >>> + >>> + To compile this driver as a module, choose M here: >>> + the module will be called rpr0521. >>> + >>> config SENSORS_LM3533 >>> tristate "LM3533 ambient light sensor" >>> depends on MFD_LM3533 >>> diff --git a/drivers/iio/light/Makefile b/drivers/iio/light/Makefile >>> index e2d50fd..adf9723 100644 >>> --- a/drivers/iio/light/Makefile >>> +++ b/drivers/iio/light/Makefile >>> @@ -19,6 +19,7 @@ obj-$(CONFIG_ISL29125) += isl29125.o >>> obj-$(CONFIG_JSA1212) += jsa1212.o >>> obj-$(CONFIG_SENSORS_LM3533) += lm3533-als.o >>> obj-$(CONFIG_LTR501) += ltr501.o >>> +obj-$(CONFIG_RPR0521) += rpr0521.o >>> obj-$(CONFIG_SENSORS_TSL2563) += tsl2563.o >>> obj-$(CONFIG_STK3310) += stk3310.o >>> obj-$(CONFIG_TCS3414) += tcs3414.o >>> diff --git a/drivers/iio/light/rpr0521.c b/drivers/iio/light/rpr0521.c >>> new file mode 100644 >>> index 0000000..a75032ea >>> --- /dev/null >>> +++ b/drivers/iio/light/rpr0521.c >>> @@ -0,0 +1,622 @@ >>> +/* >>> + * RPR-0521 ROHM Ambient Light and Proximity Sensor >>> + * >>> + * Copyright (c) 2015, Intel Corporation. >>> + * >>> + * This file is subject to the terms and conditions of version 2 of >>> + * the GNU General Public License. See the file COPYING in the main >>> + * directory of this archive for more details. >>> + * >>> + * IIO driver for RPR-0521RS (7-bit I2C slave address 0x38). >>> + * >>> + * TODO: illuminance channel, PM support, buffer >>> + */ >>> + >>> +#include <linux/module.h> >>> +#include <linux/init.h> >>> +#include <linux/i2c.h> >>> +#include <linux/regmap.h> >>> +#include <linux/delay.h> >>> +#include <linux/acpi.h> >>> + >>> +#include <linux/iio/iio.h> >>> +#include <linux/iio/sysfs.h> >>> +#include <linux/pm_runtime.h> >>> + >>> +#define RPR0521_REG_SYSTEM_CTRL 0x40 >>> +#define RPR0521_REG_MODE_CTRL 0x41 >>> +#define RPR0521_REG_ALS_CTRL 0x42 >>> +#define RPR0521_REG_PXS_CTRL 0x43 >>> +#define RPR0521_REG_PXS_DATA 0x44 /* 16-bit, little endian */ >>> +#define RPR0521_REG_ALS_DATA0 0x46 /* 16-bit, little endian */ >>> +#define RPR0521_REG_ALS_DATA1 0x48 /* 16-bit, little endian */ >>> +#define RPR0521_REG_ID 0x92 >>> + >>> +#define RPR0521_MODE_ALS_MASK BIT(7) >>> +#define RPR0521_MODE_PXS_MASK BIT(6) >>> +#define RPR0521_MODE_MEAS_TIME_MASK GENMASK(3, 0) >>> +#define RPR0521_ALS_DATA0_GAIN_MASK (BIT(4) | BIT(5)) >> Ideal would probably be genmask for these other masks as well. > > Ok. > >>> +#define RPR0521_ALS_DATA0_GAIN_SHIFT 4 >>> +#define RPR0521_ALS_DATA1_GAIN_MASK (BIT(2) | BIT(3)) >>> +#define RPR0521_ALS_DATA1_GAIN_SHIFT 2 >>> +#define RPR0521_PXS_GAIN_MASK (BIT(4) | BIT(5)) >>> +#define RPR0521_PXS_GAIN_SHIFT 4 >>> + >>> +#define RPR0521_MODE_ALS_ENABLE BIT(7) >>> +#define RPR0521_MODE_ALS_DISABLE 0x00 >>> +#define RPR0521_MODE_PXS_ENABLE BIT(6) >>> +#define RPR0521_MODE_PXS_DISABLE 0x00 >>> + >>> +#define RPR0521_PXS_MAX_VAL (BIT(13) - 1) >>> +#define RPR0521_MANUFACT_ID 0xE0 >>> +#define RPR0521_DEFAULT_MEAS_TIME 0x06 /* ALS - 100ms, PXS - 100ms */ >>> + >>> +#define RPR0521_DRV_NAME "RPR0521" >>> +#define RPR0521_REGMAP_NAME "rpr0521_regmap" >>> + >>> +#define RPR0521_SLEEP_DELAY_MS 2000 >>> + >>> +#define RPR0521_ALS_SCALE_AVAIL "0.007812 0.015625 0.5 1" >>> +#define RPR0521_PXS_SCALE_AVAIL "0.125 0.5 1" >>> + >>> +struct rpr0521_gain { >>> + int scale; >>> + int uscale; >>> +}; >>> + >>> +static const struct rpr0521_gain rpr0521_als_gain[4] = { >>> + {1, 0}, /* x1 */ >>> + {0, 500000}, /* x2 */ >>> + {0, 15625}, /* x64 */ >>> + {0, 7812}, /* x128 */ >>> +}; >>> + >>> +static const struct rpr0521_gain rpr0521_pxs_gain[3] = { >>> + {1, 0}, /* x1 */ >>> + {0, 500000}, /* x2 */ >>> + {0, 125000}, /* x4 */ >>> +}; >>> + >>> +enum rpr0521_channel { >>> + RPR0521_CHAN_ALS_DATA0, >>> + RPR0521_CHAN_ALS_DATA1, >>> + RPR0521_CHAN_PXS, >>> +}; >>> + >>> +struct rpr0521_reg_desc { >>> + u8 address; >>> + u8 device_mask; >>> +}; >>> + >>> +static const struct rpr0521_reg_desc rpr0521_data_reg[] = { >>> + [RPR0521_CHAN_ALS_DATA0] = { >>> + .address = RPR0521_REG_ALS_DATA0, >>> + .device_mask = RPR0521_MODE_ALS_MASK, >>> + }, >>> + [RPR0521_CHAN_ALS_DATA1] = { >>> + .address = RPR0521_REG_ALS_DATA1, >>> + .device_mask = RPR0521_MODE_ALS_MASK, >>> + }, >>> + [RPR0521_CHAN_PXS] = { >>> + .address = RPR0521_REG_PXS_DATA, >>> + .device_mask = RPR0521_MODE_PXS_MASK, >>> + }, >>> +}; >>> + >> I'd have been tempted to roll the above reg_desc and this >> gain_info structures together and just have a chan_info >> structure containing them both. Still more a matter of >> taste than anything else so I'm fine with what you have here >> as well! > > I will have a closer look when sending v4. > >>> +static const struct rpr0521_gain_info { >>> + u8 reg; >>> + u8 mask; >>> + u8 shift; >>> + const struct rpr0521_gain *gain; >>> + int size; >>> +} rpr0521_gain[] = { >>> + [RPR0521_CHAN_ALS_DATA0] = { >>> + .reg = RPR0521_REG_ALS_CTRL, >>> + .mask = RPR0521_ALS_DATA0_GAIN_MASK, >>> + .shift = RPR0521_ALS_DATA0_GAIN_SHIFT, >>> + .gain = rpr0521_als_gain, >>> + .size = ARRAY_SIZE(rpr0521_als_gain), >>> + }, >>> + [RPR0521_CHAN_ALS_DATA1] = { >>> + .reg = RPR0521_REG_ALS_CTRL, >>> + .mask = RPR0521_ALS_DATA1_GAIN_MASK, >>> + .shift = RPR0521_ALS_DATA1_GAIN_SHIFT, >>> + .gain = rpr0521_als_gain, >>> + .size = ARRAY_SIZE(rpr0521_als_gain), >>> + }, >>> + [RPR0521_CHAN_PXS] = { >>> + .reg = RPR0521_REG_PXS_CTRL, >>> + .mask = RPR0521_PXS_GAIN_MASK, >>> + .shift = RPR0521_PXS_GAIN_SHIFT, >>> + .gain = rpr0521_pxs_gain, >>> + .size = ARRAY_SIZE(rpr0521_pxs_gain), >>> + }, >>> +}; >>> + >>> +struct rpr0521_data { >>> + struct i2c_client *client; >>> + >>> + /* protect device params updates (e.g state, gain) */ >>> + struct mutex lock; >>> + >>> + /* device active status */ >>> + bool als_dev_en; >>> + bool pxs_dev_en; >>> + >>> + /* optimize runtime pm ops - enable device only if needed */ >>> + bool als_ps_need_en; >>> + bool pxs_ps_need_en; >>> + >>> + struct regmap *regmap; >>> +}; >>> + >>> +static IIO_CONST_ATTR(in_intensity_scale_available, RPR0521_ALS_SCALE_AVAIL); >>> +static IIO_CONST_ATTR(in_proximity_scale_available, RPR0521_PXS_SCALE_AVAIL); >>> + >>> +static struct attribute *rpr0521_attributes[] = { >>> + &iio_const_attr_in_intensity_scale_available.dev_attr.attr, >>> + &iio_const_attr_in_proximity_scale_available.dev_attr.attr, >>> + NULL, >>> +}; >>> + >>> +static const struct attribute_group rpr0521_attribute_group = { >>> + .attrs = rpr0521_attributes, >>> +}; >>> + >>> +static const struct iio_chan_spec rpr0521_channels[] = { >>> + { >>> + .type = IIO_INTENSITY, >>> + .modified = 1, >>> + .address = RPR0521_CHAN_ALS_DATA0, >>> + .channel2 = IIO_MOD_LIGHT_BOTH, >>> + .info_mask_separate = BIT(IIO_CHAN_INFO_RAW) | >>> + BIT(IIO_CHAN_INFO_SCALE), >>> + }, >>> + { >>> + .type = IIO_INTENSITY, >>> + .modified = 1, >>> + .address = RPR0521_CHAN_ALS_DATA1, >>> + .channel2 = IIO_MOD_LIGHT_IR, >>> + .info_mask_separate = BIT(IIO_CHAN_INFO_RAW) | >>> + BIT(IIO_CHAN_INFO_SCALE), >>> + }, >>> + { >>> + .type = IIO_PROXIMITY, >>> + .address = RPR0521_CHAN_PXS, >>> + .info_mask_separate = BIT(IIO_CHAN_INFO_RAW) | >>> + BIT(IIO_CHAN_INFO_SCALE), >>> + } >>> +}; >>> + >>> +static int rpr0521_als_enable(struct rpr0521_data *data, u8 status) >>> +{ >>> + int ret; >>> + >>> + ret = regmap_update_bits(data->regmap, RPR0521_REG_MODE_CTRL, >>> + RPR0521_MODE_ALS_MASK, >>> + status); >>> + if (ret < 0) >>> + return ret; >>> + >>> + data->als_dev_en = true; >>> + >>> + return 0; >>> +} >>> + >>> +static int rpr0521_pxs_enable(struct rpr0521_data *data, u8 status) >>> +{ >>> + int ret; >>> + >>> + ret = regmap_update_bits(data->regmap, RPR0521_REG_MODE_CTRL, >>> + RPR0521_MODE_PXS_MASK, >>> + status); >>> + if (ret < 0) >>> + return ret; >>> + >>> + data->pxs_dev_en = true; >>> + >>> + return 0; >>> +} >>> + >>> +/** >>> + * rpr0521_set_power_state - handles runtime PM state and sensors enabled status >>> + * >>> + * @data: rpr0521 device private data >>> + * @on: state to be set for devices in @device_mask >>> + * @device_mask: bitmask specifying for which device we need to update @on state >>> + * >>> + * We rely on rpr0521_runtime_resume to enable our @device_mask devices, but >>> + * if (for example) PXS was enabled (pxs_dev_en = true) by a previous call to >>> + * rpr0521_runtime_resume and we want to enable ALS we MUST set ALS enable >>> + * bit of RPR0521_REG_MODE_CTRL here because rpr0521_runtime_resume will not >>> + * be called twice. >>> + */ >>> +static int rpr0521_set_power_state(struct rpr0521_data *data, bool on, >>> + u8 device_mask) >>> +{ >>> +#ifdef CONFIG_PM >>> + int ret; >>> + u8 update_mask = 0; >>> + >>> + if (device_mask & RPR0521_MODE_ALS_MASK) { >>> + if (on && !data->als_ps_need_en && data->pxs_dev_en) >>> + update_mask |= RPR0521_MODE_ALS_MASK; >>> + else >>> + data->als_ps_need_en = on; >>> + } >>> + >>> + if (device_mask & RPR0521_MODE_PXS_MASK) { >>> + if (on && !data->pxs_ps_need_en && data->als_dev_en) >>> + update_mask |= RPR0521_MODE_PXS_MASK; >>> + else >>> + data->pxs_ps_need_en = on; >>> + } >>> + >>> + if (update_mask) { >>> + ret = regmap_update_bits(data->regmap, RPR0521_REG_MODE_CTRL, >>> + update_mask, update_mask); >>> + if (ret < 0) >>> + return ret; >>> + } >>> + >>> + if (on) { >>> + ret = pm_runtime_get_sync(&data->client->dev); >>> + } else { >>> + pm_runtime_mark_last_busy(&data->client->dev); >>> + ret = pm_runtime_put_autosuspend(&data->client->dev); >>> + } >>> + if (ret < 0) { >>> + dev_err(&data->client->dev, >>> + "Failed: rpr0521_set_power_state for %d, ret %d\n", >>> + on, ret); >>> + if (on) >>> + pm_runtime_put_noidle(&data->client->dev); >>> + >>> + return ret; >>> + } >>> +#endif >>> + return 0; >>> +} >>> + >>> +static int rpr0521_get_gain_index(const struct rpr0521_gain *gainv, int size, >>> + int val, int val2) >>> +{ >>> + int i; >>> + >>> + for (i = 0; i < size; i++) >>> + if (val == gainv[i].scale && val2 == gainv[i].uscale) >>> + return i; >>> + >>> + return -EINVAL; >>> +} >>> + >>> +static int rpr0521_get_gain(struct rpr0521_data *data, int chan, >>> + int *val, int *val2) >>> +{ >>> + int ret, reg, idx; >>> + >>> + ret = regmap_read(data->regmap, rpr0521_gain[chan].reg, ®); >>> + if (ret < 0) >>> + return ret; >>> + >>> + idx = (rpr0521_gain[chan].mask & reg) >> rpr0521_gain[chan].shift; >>> + *val = rpr0521_gain[chan].gain[idx].scale; >>> + *val2 = rpr0521_gain[chan].gain[idx].uscale; >>> + >>> + return 0; >>> +} >>> + >>> +static int rpr0521_set_gain(struct rpr0521_data *data, int chan, >>> + int val, int val2) >>> +{ >>> + int idx; >>> + >>> + idx = rpr0521_get_gain_index(rpr0521_gain[chan].gain, >>> + rpr0521_gain[chan].size, val, val2); >> Marginal benefit in having the separate get_gain_index function as >> only used here and very simple at that. Maybe, just worth while.. > > Agree with this. > >>> + if (idx < 0) >>> + return idx; >>> + >>> + return regmap_update_bits(data->regmap, rpr0521_gain[chan].reg, >>> + rpr0521_gain[chan].mask, >>> + idx << rpr0521_gain[chan].shift); >>> +} >>> + >>> +static int rpr0521_read_raw(struct iio_dev *indio_dev, >>> + struct iio_chan_spec const *chan, int *val, >>> + int *val2, long mask) >>> +{ >>> + struct rpr0521_data *data = iio_priv(indio_dev); >>> + int ret; >>> + u8 device_mask; >>> + __le16 raw_data; >>> + >>> + switch (mask) { >>> + case IIO_CHAN_INFO_RAW: >>> + if (chan->type != IIO_INTENSITY && chan->type != IIO_PROXIMITY) >>> + return -EINVAL; >>> + >>> + device_mask = rpr0521_data_reg[chan->address].device_mask; >>> + >>> + mutex_lock(&data->lock); >>> + ret = rpr0521_set_power_state(data, true, device_mask); >>> + if (ret < 0) { >> Fine as is, but perhaps you would have been better pulling this section >> out to a separate function then using the traditional goto type error >> handling to undo the relevant elements as you go rather than having repeats >> of this mutex unlock for example. Not quite a complex enough case for >> me to insist on it here though :) >>> + mutex_unlock(&data->lock); >>> + return ret; > > I see, it seems ok as it is. I'll try to see if using goto makes code easier > to read. > > thanks, > Daniel. > -- To unsubscribe from this list: send the line "unsubscribe linux-iio" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html