diff options
author | Matt Ranostay <mranostay@gmail.com> | 2016-08-25 02:44:47 -0400 |
---|---|---|
committer | Jonathan Cameron <jic23@kernel.org> | 2016-08-29 12:46:52 -0400 |
commit | 8376882f167be87aa44acc105020417532ac3c51 (patch) | |
tree | b78d670a092d7b1fa3baabbfeee66a36ab7b37b5 | |
parent | 8c9e7b1bf42af845f83350e5141fbc4181dc7f98 (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.c | 80 |
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 | ||
37 | enum { | ||
38 | VZ89X, | ||
39 | }; | ||
40 | |||
41 | struct vz89x_chip_data; | ||
42 | |||
34 | struct vz89x_data { | 43 | struct 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 | |||
53 | struct 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 | ||
43 | static const struct iio_chan_spec vz89x_channels[] = { | 63 | static 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 | ||
96 | static int vz89x_measurement_is_valid(struct vz89x_data *data) | 116 | static 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 | ||
104 | static int vz89x_i2c_xfer(struct vz89x_data *data, u8 cmd) | 124 | static 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 | ||
146 | static int vz89x_get_measurement(struct vz89x_data *data) | 167 | static 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 | ||
257 | static 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 | |||
270 | static const struct of_device_id vz89x_dt_ids[] = { | ||
271 | { .compatible = "sgx,vz89x", .data = (void *) VZ89X }, | ||
272 | { } | ||
273 | }; | ||
274 | MODULE_DEVICE_TABLE(of, vz89x_dt_ids); | ||
275 | |||
235 | static int vz89x_probe(struct i2c_client *client, | 276 | static 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 | ||
270 | static const struct i2c_device_id vz89x_id[] = { | 320 | static const struct i2c_device_id vz89x_id[] = { |
271 | { "vz89x", 0 }, | 321 | { "vz89x", VZ89X }, |
272 | { } | 322 | { } |
273 | }; | 323 | }; |
274 | MODULE_DEVICE_TABLE(i2c, vz89x_id); | 324 | MODULE_DEVICE_TABLE(i2c, vz89x_id); |
275 | 325 | ||
276 | static const struct of_device_id vz89x_dt_ids[] = { | ||
277 | { .compatible = "sgx,vz89x" }, | ||
278 | { } | ||
279 | }; | ||
280 | MODULE_DEVICE_TABLE(of, vz89x_dt_ids); | ||
281 | |||
282 | static struct i2c_driver vz89x_driver = { | 326 | static struct i2c_driver vz89x_driver = { |
283 | .driver = { | 327 | .driver = { |
284 | .name = "vz89x", | 328 | .name = "vz89x", |