aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorMatt Ranostay <mranostay@gmail.com>2016-08-25 02:44:47 -0400
committerJonathan Cameron <jic23@kernel.org>2016-08-29 12:46:52 -0400
commit8376882f167be87aa44acc105020417532ac3c51 (patch)
treeb78d670a092d7b1fa3baabbfeee66a36ab7b37b5
parent8c9e7b1bf42af845f83350e5141fbc4181dc7f98 (diff)
iio: chemical: vz89x: abstract chip configuration
Abstract chip configuration data to allow supporting multiple variants of the VZ89 chemical sensor line. Signed-off-by: Matt Ranostay <mranostay@gmail.com> Signed-off-by: Jonathan Cameron <jic23@kernel.org>
-rw-r--r--drivers/iio/chemical/vz89x.c80
1 files changed, 62 insertions, 18 deletions
diff --git a/drivers/iio/chemical/vz89x.c b/drivers/iio/chemical/vz89x.c
index 652649da500f..aa6ebc084e19 100644
--- a/drivers/iio/chemical/vz89x.c
+++ b/drivers/iio/chemical/vz89x.c
@@ -19,25 +19,45 @@
19#include <linux/mutex.h> 19#include <linux/mutex.h>
20#include <linux/init.h> 20#include <linux/init.h>
21#include <linux/i2c.h> 21#include <linux/i2c.h>
22#include <linux/of.h>
23#include <linux/of_device.h>
22 24
23#include <linux/iio/iio.h> 25#include <linux/iio/iio.h>
24#include <linux/iio/sysfs.h> 26#include <linux/iio/sysfs.h>
25 27
26#define VZ89X_REG_MEASUREMENT 0x09 28#define VZ89X_REG_MEASUREMENT 0x09
27#define VZ89X_REG_MEASUREMENT_SIZE 6 29#define VZ89X_REG_MEASUREMENT_RD_SIZE 6
30#define VZ89X_REG_MEASUREMENT_WR_SIZE 3
28 31
29#define VZ89X_VOC_CO2_IDX 0 32#define VZ89X_VOC_CO2_IDX 0
30#define VZ89X_VOC_SHORT_IDX 1 33#define VZ89X_VOC_SHORT_IDX 1
31#define VZ89X_VOC_TVOC_IDX 2 34#define VZ89X_VOC_TVOC_IDX 2
32#define VZ89X_VOC_RESISTANCE_IDX 3 35#define VZ89X_VOC_RESISTANCE_IDX 3
33 36
37enum {
38 VZ89X,
39};
40
41struct vz89x_chip_data;
42
34struct vz89x_data { 43struct vz89x_data {
35 struct i2c_client *client; 44 struct i2c_client *client;
45 const struct vz89x_chip_data *chip;
36 struct mutex lock; 46 struct mutex lock;
37 int (*xfer)(struct vz89x_data *data, u8 cmd); 47 int (*xfer)(struct vz89x_data *data, u8 cmd);
38 48
39 unsigned long last_update; 49 unsigned long last_update;
40 u8 buffer[VZ89X_REG_MEASUREMENT_SIZE]; 50 u8 buffer[VZ89X_REG_MEASUREMENT_RD_SIZE];
51};
52
53struct vz89x_chip_data {
54 bool (*valid)(struct vz89x_data *data);
55 const struct iio_chan_spec *channels;
56 u8 num_channels;
57
58 u8 cmd;
59 u8 read_size;
60 u8 write_size;
41}; 61};
42 62
43static const struct iio_chan_spec vz89x_channels[] = { 63static const struct iio_chan_spec vz89x_channels[] = {
@@ -93,16 +113,17 @@ static const struct attribute_group vz89x_attrs_group = {
93 * always zero, and by also confirming the VOC_short isn't zero. 113 * always zero, and by also confirming the VOC_short isn't zero.
94 */ 114 */
95 115
96static int vz89x_measurement_is_valid(struct vz89x_data *data) 116static bool vz89x_measurement_is_valid(struct vz89x_data *data)
97{ 117{
98 if (data->buffer[VZ89X_VOC_SHORT_IDX] == 0) 118 if (data->buffer[VZ89X_VOC_SHORT_IDX] == 0)
99 return 1; 119 return 1;
100 120
101 return !!(data->buffer[VZ89X_REG_MEASUREMENT_SIZE - 1] > 0); 121 return !!(data->buffer[data->chip->read_size - 1] > 0);
102} 122}
103 123
104static int vz89x_i2c_xfer(struct vz89x_data *data, u8 cmd) 124static int vz89x_i2c_xfer(struct vz89x_data *data, u8 cmd)
105{ 125{
126 const struct vz89x_chip_data *chip = data->chip;
106 struct i2c_client *client = data->client; 127 struct i2c_client *client = data->client;
107 struct i2c_msg msg[2]; 128 struct i2c_msg msg[2];
108 int ret; 129 int ret;
@@ -110,12 +131,12 @@ static int vz89x_i2c_xfer(struct vz89x_data *data, u8 cmd)
110 131
111 msg[0].addr = client->addr; 132 msg[0].addr = client->addr;
112 msg[0].flags = client->flags; 133 msg[0].flags = client->flags;
113 msg[0].len = 3; 134 msg[0].len = chip->write_size;
114 msg[0].buf = (char *) &buf; 135 msg[0].buf = (char *) &buf;
115 136
116 msg[1].addr = client->addr; 137 msg[1].addr = client->addr;
117 msg[1].flags = client->flags | I2C_M_RD; 138 msg[1].flags = client->flags | I2C_M_RD;
118 msg[1].len = VZ89X_REG_MEASUREMENT_SIZE; 139 msg[1].len = chip->read_size;
119 msg[1].buf = (char *) &data->buffer; 140 msg[1].buf = (char *) &data->buffer;
120 141
121 ret = i2c_transfer(client->adapter, msg, 2); 142 ret = i2c_transfer(client->adapter, msg, 2);
@@ -133,7 +154,7 @@ static int vz89x_smbus_xfer(struct vz89x_data *data, u8 cmd)
133 if (ret < 0) 154 if (ret < 0)
134 return ret; 155 return ret;
135 156
136 for (i = 0; i < VZ89X_REG_MEASUREMENT_SIZE; i++) { 157 for (i = 0; i < data->chip->read_size; i++) {
137 ret = i2c_smbus_read_byte(client); 158 ret = i2c_smbus_read_byte(client);
138 if (ret < 0) 159 if (ret < 0)
139 return ret; 160 return ret;
@@ -145,17 +166,18 @@ static int vz89x_smbus_xfer(struct vz89x_data *data, u8 cmd)
145 166
146static int vz89x_get_measurement(struct vz89x_data *data) 167static int vz89x_get_measurement(struct vz89x_data *data)
147{ 168{
169 const struct vz89x_chip_data *chip = data->chip;
148 int ret; 170 int ret;
149 171
150 /* sensor can only be polled once a second max per datasheet */ 172 /* sensor can only be polled once a second max per datasheet */
151 if (!time_after(jiffies, data->last_update + HZ)) 173 if (!time_after(jiffies, data->last_update + HZ))
152 return 0; 174 return 0;
153 175
154 ret = data->xfer(data, VZ89X_REG_MEASUREMENT); 176 ret = data->xfer(data, chip->cmd);
155 if (ret < 0) 177 if (ret < 0)
156 return ret; 178 return ret;
157 179
158 ret = vz89x_measurement_is_valid(data); 180 ret = chip->valid(data);
159 if (ret) 181 if (ret)
160 return -EAGAIN; 182 return -EAGAIN;
161 183
@@ -232,11 +254,32 @@ static const struct iio_info vz89x_info = {
232 .driver_module = THIS_MODULE, 254 .driver_module = THIS_MODULE,
233}; 255};
234 256
257static const struct vz89x_chip_data vz89x_chips[] = {
258 {
259 .valid = vz89x_measurement_is_valid,
260
261 .cmd = VZ89X_REG_MEASUREMENT,
262 .read_size = VZ89X_REG_MEASUREMENT_RD_SIZE,
263 .write_size = VZ89X_REG_MEASUREMENT_WR_SIZE,
264
265 .channels = vz89x_channels,
266 .num_channels = ARRAY_SIZE(vz89x_channels),
267 },
268};
269
270static const struct of_device_id vz89x_dt_ids[] = {
271 { .compatible = "sgx,vz89x", .data = (void *) VZ89X },
272 { }
273};
274MODULE_DEVICE_TABLE(of, vz89x_dt_ids);
275
235static int vz89x_probe(struct i2c_client *client, 276static int vz89x_probe(struct i2c_client *client,
236 const struct i2c_device_id *id) 277 const struct i2c_device_id *id)
237{ 278{
238 struct iio_dev *indio_dev; 279 struct iio_dev *indio_dev;
239 struct vz89x_data *data; 280 struct vz89x_data *data;
281 const struct of_device_id *of_id;
282 int chip_id;
240 283
241 indio_dev = devm_iio_device_alloc(&client->dev, sizeof(*data)); 284 indio_dev = devm_iio_device_alloc(&client->dev, sizeof(*data));
242 if (!indio_dev) 285 if (!indio_dev)
@@ -251,8 +294,15 @@ static int vz89x_probe(struct i2c_client *client,
251 else 294 else
252 return -EOPNOTSUPP; 295 return -EOPNOTSUPP;
253 296
297 of_id = of_match_device(vz89x_dt_ids, &client->dev);
298 if (!of_id)
299 chip_id = id->driver_data;
300 else
301 chip_id = (unsigned long)of_id->data;
302
254 i2c_set_clientdata(client, indio_dev); 303 i2c_set_clientdata(client, indio_dev);
255 data->client = client; 304 data->client = client;
305 data->chip = &vz89x_chips[chip_id];
256 data->last_update = jiffies - HZ; 306 data->last_update = jiffies - HZ;
257 mutex_init(&data->lock); 307 mutex_init(&data->lock);
258 308
@@ -261,24 +311,18 @@ static int vz89x_probe(struct i2c_client *client,
261 indio_dev->name = dev_name(&client->dev); 311 indio_dev->name = dev_name(&client->dev);
262 indio_dev->modes = INDIO_DIRECT_MODE; 312 indio_dev->modes = INDIO_DIRECT_MODE;
263 313
264 indio_dev->channels = vz89x_channels; 314 indio_dev->channels = data->chip->channels;
265 indio_dev->num_channels = ARRAY_SIZE(vz89x_channels); 315 indio_dev->num_channels = data->chip->num_channels;
266 316
267 return devm_iio_device_register(&client->dev, indio_dev); 317 return devm_iio_device_register(&client->dev, indio_dev);
268} 318}
269 319
270static const struct i2c_device_id vz89x_id[] = { 320static const struct i2c_device_id vz89x_id[] = {
271 { "vz89x", 0 }, 321 { "vz89x", VZ89X },
272 { } 322 { }
273}; 323};
274MODULE_DEVICE_TABLE(i2c, vz89x_id); 324MODULE_DEVICE_TABLE(i2c, vz89x_id);
275 325
276static const struct of_device_id vz89x_dt_ids[] = {
277 { .compatible = "sgx,vz89x" },
278 { }
279};
280MODULE_DEVICE_TABLE(of, vz89x_dt_ids);
281
282static struct i2c_driver vz89x_driver = { 326static struct i2c_driver vz89x_driver = {
283 .driver = { 327 .driver = {
284 .name = "vz89x", 328 .name = "vz89x",