Commit 35b651f3 authored by Crestez Dan Leonard's avatar Crestez Dan Leonard Committed by Jonathan Cameron

max44000: Initial support for proximity reading

The proximity sensor relies on sending pulses to an external IR led and
it is disabled by default on powerup. The driver will enable it with a
default power setting.
Signed-off-by: default avatarCrestez Dan Leonard <leonard.crestez@intel.com>
Signed-off-by: default avatarJonathan Cameron <jic23@kernel.org>
parent b9567e66
...@@ -59,6 +59,11 @@ ...@@ -59,6 +59,11 @@
*/ */
#define MAX44000_REG_CFG_RX_DEFAULT 0xf0 #define MAX44000_REG_CFG_RX_DEFAULT 0xf0
/* REG_TX bits */
#define MAX44000_LED_CURRENT_MASK 0xf
#define MAX44000_LED_CURRENT_MAX 11
#define MAX44000_LED_CURRENT_DEFAULT 6
#define MAX44000_ALSDATA_OVERFLOW 0x4000 #define MAX44000_ALSDATA_OVERFLOW 0x4000
struct max44000_data { struct max44000_data {
...@@ -75,6 +80,11 @@ static const struct iio_chan_spec max44000_channels[] = { ...@@ -75,6 +80,11 @@ static const struct iio_chan_spec max44000_channels[] = {
.info_mask_separate = BIT(IIO_CHAN_INFO_RAW), .info_mask_separate = BIT(IIO_CHAN_INFO_RAW),
.info_mask_shared_by_type = BIT(IIO_CHAN_INFO_SCALE), .info_mask_shared_by_type = BIT(IIO_CHAN_INFO_SCALE),
}, },
{
.type = IIO_PROXIMITY,
.info_mask_separate = BIT(IIO_CHAN_INFO_RAW),
.info_mask_shared_by_type = BIT(IIO_CHAN_INFO_SCALE),
},
}; };
static int max44000_read_alsval(struct max44000_data *data) static int max44000_read_alsval(struct max44000_data *data)
...@@ -104,11 +114,23 @@ static int max44000_read_alsval(struct max44000_data *data) ...@@ -104,11 +114,23 @@ static int max44000_read_alsval(struct max44000_data *data)
return regval; return regval;
} }
static int max44000_write_led_current_raw(struct max44000_data *data, int val)
{
/* Maybe we should clamp the value instead? */
if (val < 0 || val > MAX44000_LED_CURRENT_MAX)
return -ERANGE;
if (val >= 8)
val += 4;
return regmap_write_bits(data->regmap, MAX44000_REG_CFG_TX,
MAX44000_LED_CURRENT_MASK, val);
}
static int max44000_read_raw(struct iio_dev *indio_dev, static int max44000_read_raw(struct iio_dev *indio_dev,
struct iio_chan_spec const *chan, struct iio_chan_spec const *chan,
int *val, int *val2, long mask) int *val, int *val2, long mask)
{ {
struct max44000_data *data = iio_priv(indio_dev); struct max44000_data *data = iio_priv(indio_dev);
unsigned int regval;
int ret; int ret;
switch (mask) { switch (mask) {
...@@ -123,6 +145,15 @@ static int max44000_read_raw(struct iio_dev *indio_dev, ...@@ -123,6 +145,15 @@ static int max44000_read_raw(struct iio_dev *indio_dev,
*val = ret; *val = ret;
return IIO_VAL_INT; return IIO_VAL_INT;
case IIO_PROXIMITY:
mutex_lock(&data->lock);
ret = regmap_read(data->regmap, MAX44000_REG_PRX_DATA, &regval);
mutex_unlock(&data->lock);
if (ret < 0)
return ret;
*val = regval;
return IIO_VAL_INT;
default: default:
return -EINVAL; return -EINVAL;
} }
...@@ -260,14 +291,26 @@ static int max44000_probe(struct i2c_client *client, ...@@ -260,14 +291,26 @@ static int max44000_probe(struct i2c_client *client,
*/ */
/* Reset ALS scaling bits */ /* Reset ALS scaling bits */
ret = regmap_write(data->regmap, MAX44000_REG_CFG_RX, MAX44000_REG_CFG_RX_DEFAULT); ret = regmap_write(data->regmap, MAX44000_REG_CFG_RX,
MAX44000_REG_CFG_RX_DEFAULT);
if (ret < 0) {
dev_err(&client->dev, "failed to write default CFG_RX: %d\n",
ret);
return ret;
}
/*
* By default the LED pulse used for the proximity sensor is disabled.
* Set a middle value so that we get some sort of valid data by default.
*/
ret = max44000_write_led_current_raw(data, MAX44000_LED_CURRENT_DEFAULT);
if (ret < 0) { if (ret < 0) {
dev_err(&client->dev, "failed to write default CFG_RX: %d\n", ret); dev_err(&client->dev, "failed to write init config: %d\n", ret);
return ret; return ret;
} }
/* Reset CFG bits to ALS-only mode and no interrupts */ /* Reset CFG bits to ALS_PRX mode which allows easy reading of both values. */
reg = MAX44000_CFG_TRIM | MAX44000_CFG_MODE_ALS_GIR; reg = MAX44000_CFG_TRIM | MAX44000_CFG_MODE_ALS_PRX;
ret = regmap_write(data->regmap, MAX44000_REG_CFG_MAIN, reg); ret = regmap_write(data->regmap, MAX44000_REG_CFG_MAIN, reg);
if (ret < 0) { if (ret < 0) {
dev_err(&client->dev, "failed to write init config: %d\n", ret); dev_err(&client->dev, "failed to write init config: %d\n", ret);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment