blob: 38095d79a87432087d82fd3756b457c35a6d5d23 [file] [log] [blame]
Daniel Balutaefa86e9f2015-06-17 12:42:51 +03001/*
2 * RPR-0521 ROHM Ambient Light and Proximity Sensor
3 *
4 * Copyright (c) 2015, Intel Corporation.
5 *
6 * This file is subject to the terms and conditions of version 2 of
7 * the GNU General Public License. See the file COPYING in the main
8 * directory of this archive for more details.
9 *
10 * IIO driver for RPR-0521RS (7-bit I2C slave address 0x38).
11 *
Mikko Koivunen484c3142017-05-18 15:12:51 +030012 * TODO: illuminance channel, buffer
Daniel Balutaefa86e9f2015-06-17 12:42:51 +030013 */
14
15#include <linux/module.h>
16#include <linux/init.h>
17#include <linux/i2c.h>
18#include <linux/regmap.h>
19#include <linux/delay.h>
20#include <linux/acpi.h>
21
22#include <linux/iio/iio.h>
23#include <linux/iio/sysfs.h>
24#include <linux/pm_runtime.h>
25
26#define RPR0521_REG_SYSTEM_CTRL 0x40
27#define RPR0521_REG_MODE_CTRL 0x41
28#define RPR0521_REG_ALS_CTRL 0x42
29#define RPR0521_REG_PXS_CTRL 0x43
30#define RPR0521_REG_PXS_DATA 0x44 /* 16-bit, little endian */
31#define RPR0521_REG_ALS_DATA0 0x46 /* 16-bit, little endian */
32#define RPR0521_REG_ALS_DATA1 0x48 /* 16-bit, little endian */
33#define RPR0521_REG_ID 0x92
34
35#define RPR0521_MODE_ALS_MASK BIT(7)
36#define RPR0521_MODE_PXS_MASK BIT(6)
37#define RPR0521_MODE_MEAS_TIME_MASK GENMASK(3, 0)
38#define RPR0521_ALS_DATA0_GAIN_MASK GENMASK(5, 4)
39#define RPR0521_ALS_DATA0_GAIN_SHIFT 4
40#define RPR0521_ALS_DATA1_GAIN_MASK GENMASK(3, 2)
41#define RPR0521_ALS_DATA1_GAIN_SHIFT 2
42#define RPR0521_PXS_GAIN_MASK GENMASK(5, 4)
43#define RPR0521_PXS_GAIN_SHIFT 4
44
45#define RPR0521_MODE_ALS_ENABLE BIT(7)
46#define RPR0521_MODE_ALS_DISABLE 0x00
47#define RPR0521_MODE_PXS_ENABLE BIT(6)
48#define RPR0521_MODE_PXS_DISABLE 0x00
49
50#define RPR0521_MANUFACT_ID 0xE0
51#define RPR0521_DEFAULT_MEAS_TIME 0x06 /* ALS - 100ms, PXS - 100ms */
52
53#define RPR0521_DRV_NAME "RPR0521"
54#define RPR0521_REGMAP_NAME "rpr0521_regmap"
55
56#define RPR0521_SLEEP_DELAY_MS 2000
57
58#define RPR0521_ALS_SCALE_AVAIL "0.007812 0.015625 0.5 1"
59#define RPR0521_PXS_SCALE_AVAIL "0.125 0.5 1"
60
61struct rpr0521_gain {
62 int scale;
63 int uscale;
64};
65
66static const struct rpr0521_gain rpr0521_als_gain[4] = {
67 {1, 0}, /* x1 */
68 {0, 500000}, /* x2 */
69 {0, 15625}, /* x64 */
70 {0, 7812}, /* x128 */
71};
72
73static const struct rpr0521_gain rpr0521_pxs_gain[3] = {
74 {1, 0}, /* x1 */
75 {0, 500000}, /* x2 */
76 {0, 125000}, /* x4 */
77};
78
79enum rpr0521_channel {
80 RPR0521_CHAN_ALS_DATA0,
81 RPR0521_CHAN_ALS_DATA1,
82 RPR0521_CHAN_PXS,
83};
84
85struct rpr0521_reg_desc {
86 u8 address;
87 u8 device_mask;
88};
89
90static const struct rpr0521_reg_desc rpr0521_data_reg[] = {
91 [RPR0521_CHAN_ALS_DATA0] = {
92 .address = RPR0521_REG_ALS_DATA0,
93 .device_mask = RPR0521_MODE_ALS_MASK,
94 },
95 [RPR0521_CHAN_ALS_DATA1] = {
96 .address = RPR0521_REG_ALS_DATA1,
97 .device_mask = RPR0521_MODE_ALS_MASK,
98 },
99 [RPR0521_CHAN_PXS] = {
100 .address = RPR0521_REG_PXS_DATA,
101 .device_mask = RPR0521_MODE_PXS_MASK,
102 },
103};
104
105static const struct rpr0521_gain_info {
106 u8 reg;
107 u8 mask;
108 u8 shift;
109 const struct rpr0521_gain *gain;
110 int size;
111} rpr0521_gain[] = {
112 [RPR0521_CHAN_ALS_DATA0] = {
113 .reg = RPR0521_REG_ALS_CTRL,
114 .mask = RPR0521_ALS_DATA0_GAIN_MASK,
115 .shift = RPR0521_ALS_DATA0_GAIN_SHIFT,
116 .gain = rpr0521_als_gain,
117 .size = ARRAY_SIZE(rpr0521_als_gain),
118 },
119 [RPR0521_CHAN_ALS_DATA1] = {
120 .reg = RPR0521_REG_ALS_CTRL,
121 .mask = RPR0521_ALS_DATA1_GAIN_MASK,
122 .shift = RPR0521_ALS_DATA1_GAIN_SHIFT,
123 .gain = rpr0521_als_gain,
124 .size = ARRAY_SIZE(rpr0521_als_gain),
125 },
126 [RPR0521_CHAN_PXS] = {
127 .reg = RPR0521_REG_PXS_CTRL,
128 .mask = RPR0521_PXS_GAIN_MASK,
129 .shift = RPR0521_PXS_GAIN_SHIFT,
130 .gain = rpr0521_pxs_gain,
131 .size = ARRAY_SIZE(rpr0521_pxs_gain),
132 },
133};
134
135struct rpr0521_data {
136 struct i2c_client *client;
137
138 /* protect device params updates (e.g state, gain) */
139 struct mutex lock;
140
141 /* device active status */
142 bool als_dev_en;
143 bool pxs_dev_en;
144
Mikko Koivunen484c3142017-05-18 15:12:51 +0300145 /* optimize runtime pm ops - enable/disable device only if needed */
Daniel Balutaefa86e9f2015-06-17 12:42:51 +0300146 bool als_ps_need_en;
147 bool pxs_ps_need_en;
Mikko Koivunen484c3142017-05-18 15:12:51 +0300148 bool als_need_dis;
149 bool pxs_need_dis;
Daniel Balutaefa86e9f2015-06-17 12:42:51 +0300150
151 struct regmap *regmap;
152};
153
154static IIO_CONST_ATTR(in_intensity_scale_available, RPR0521_ALS_SCALE_AVAIL);
155static IIO_CONST_ATTR(in_proximity_scale_available, RPR0521_PXS_SCALE_AVAIL);
156
157static struct attribute *rpr0521_attributes[] = {
158 &iio_const_attr_in_intensity_scale_available.dev_attr.attr,
159 &iio_const_attr_in_proximity_scale_available.dev_attr.attr,
160 NULL,
161};
162
163static const struct attribute_group rpr0521_attribute_group = {
164 .attrs = rpr0521_attributes,
165};
166
167static const struct iio_chan_spec rpr0521_channels[] = {
168 {
169 .type = IIO_INTENSITY,
170 .modified = 1,
171 .address = RPR0521_CHAN_ALS_DATA0,
172 .channel2 = IIO_MOD_LIGHT_BOTH,
173 .info_mask_separate = BIT(IIO_CHAN_INFO_RAW) |
174 BIT(IIO_CHAN_INFO_SCALE),
175 },
176 {
177 .type = IIO_INTENSITY,
178 .modified = 1,
179 .address = RPR0521_CHAN_ALS_DATA1,
180 .channel2 = IIO_MOD_LIGHT_IR,
181 .info_mask_separate = BIT(IIO_CHAN_INFO_RAW) |
182 BIT(IIO_CHAN_INFO_SCALE),
183 },
184 {
185 .type = IIO_PROXIMITY,
186 .address = RPR0521_CHAN_PXS,
187 .info_mask_separate = BIT(IIO_CHAN_INFO_RAW) |
188 BIT(IIO_CHAN_INFO_SCALE),
189 }
190};
191
192static int rpr0521_als_enable(struct rpr0521_data *data, u8 status)
193{
194 int ret;
195
196 ret = regmap_update_bits(data->regmap, RPR0521_REG_MODE_CTRL,
197 RPR0521_MODE_ALS_MASK,
198 status);
199 if (ret < 0)
200 return ret;
201
Mikko Koivunenf87fa262017-05-18 15:12:49 +0300202 if (status & RPR0521_MODE_ALS_MASK)
203 data->als_dev_en = true;
204 else
205 data->als_dev_en = false;
Daniel Balutaefa86e9f2015-06-17 12:42:51 +0300206
207 return 0;
208}
209
210static int rpr0521_pxs_enable(struct rpr0521_data *data, u8 status)
211{
212 int ret;
213
214 ret = regmap_update_bits(data->regmap, RPR0521_REG_MODE_CTRL,
215 RPR0521_MODE_PXS_MASK,
216 status);
217 if (ret < 0)
218 return ret;
219
Mikko Koivunenf87fa262017-05-18 15:12:49 +0300220 if (status & RPR0521_MODE_PXS_MASK)
221 data->pxs_dev_en = true;
222 else
223 data->pxs_dev_en = false;
Daniel Balutaefa86e9f2015-06-17 12:42:51 +0300224
225 return 0;
226}
227
228/**
229 * rpr0521_set_power_state - handles runtime PM state and sensors enabled status
230 *
231 * @data: rpr0521 device private data
232 * @on: state to be set for devices in @device_mask
233 * @device_mask: bitmask specifying for which device we need to update @on state
234 *
Mikko Koivunen484c3142017-05-18 15:12:51 +0300235 * Calls for this function must be balanced so that each ON should have matching
236 * OFF. Otherwise pm usage_count gets out of sync.
Daniel Balutaefa86e9f2015-06-17 12:42:51 +0300237 */
238static int rpr0521_set_power_state(struct rpr0521_data *data, bool on,
239 u8 device_mask)
240{
241#ifdef CONFIG_PM
242 int ret;
Daniel Balutaefa86e9f2015-06-17 12:42:51 +0300243
244 if (device_mask & RPR0521_MODE_ALS_MASK) {
Mikko Koivunen484c3142017-05-18 15:12:51 +0300245 data->als_ps_need_en = on;
246 data->als_need_dis = !on;
Daniel Balutaefa86e9f2015-06-17 12:42:51 +0300247 }
248
249 if (device_mask & RPR0521_MODE_PXS_MASK) {
Mikko Koivunen484c3142017-05-18 15:12:51 +0300250 data->pxs_ps_need_en = on;
251 data->pxs_need_dis = !on;
Daniel Balutaefa86e9f2015-06-17 12:42:51 +0300252 }
253
Mikko Koivunen484c3142017-05-18 15:12:51 +0300254 /*
255 * On: _resume() is called only when we are suspended
256 * Off: _suspend() is called after delay if _resume() is not
257 * called before that.
258 * Note: If either measurement is re-enabled before _suspend(),
259 * both stay enabled until _suspend().
260 */
Daniel Balutaefa86e9f2015-06-17 12:42:51 +0300261 if (on) {
262 ret = pm_runtime_get_sync(&data->client->dev);
263 } else {
264 pm_runtime_mark_last_busy(&data->client->dev);
265 ret = pm_runtime_put_autosuspend(&data->client->dev);
266 }
267 if (ret < 0) {
268 dev_err(&data->client->dev,
269 "Failed: rpr0521_set_power_state for %d, ret %d\n",
270 on, ret);
271 if (on)
272 pm_runtime_put_noidle(&data->client->dev);
273
274 return ret;
275 }
Mikko Koivunen484c3142017-05-18 15:12:51 +0300276
277 if (on) {
278 /* If _resume() was not called, enable measurement now. */
279 if (data->als_ps_need_en) {
280 ret = rpr0521_als_enable(data, RPR0521_MODE_ALS_ENABLE);
281 if (ret)
282 return ret;
283 data->als_ps_need_en = false;
284 }
285
286 if (data->pxs_ps_need_en) {
287 ret = rpr0521_pxs_enable(data, RPR0521_MODE_PXS_ENABLE);
288 if (ret)
289 return ret;
290 data->pxs_ps_need_en = false;
291 }
292 }
Daniel Balutaefa86e9f2015-06-17 12:42:51 +0300293#endif
294 return 0;
295}
296
297static int rpr0521_get_gain(struct rpr0521_data *data, int chan,
298 int *val, int *val2)
299{
300 int ret, reg, idx;
301
302 ret = regmap_read(data->regmap, rpr0521_gain[chan].reg, &reg);
303 if (ret < 0)
304 return ret;
305
306 idx = (rpr0521_gain[chan].mask & reg) >> rpr0521_gain[chan].shift;
307 *val = rpr0521_gain[chan].gain[idx].scale;
308 *val2 = rpr0521_gain[chan].gain[idx].uscale;
309
310 return 0;
311}
312
313static int rpr0521_set_gain(struct rpr0521_data *data, int chan,
314 int val, int val2)
315{
316 int i, idx = -EINVAL;
317
318 /* get gain index */
319 for (i = 0; i < rpr0521_gain[chan].size; i++)
320 if (val == rpr0521_gain[chan].gain[i].scale &&
321 val2 == rpr0521_gain[chan].gain[i].uscale) {
322 idx = i;
323 break;
324 }
325
326 if (idx < 0)
327 return idx;
328
329 return regmap_update_bits(data->regmap, rpr0521_gain[chan].reg,
330 rpr0521_gain[chan].mask,
331 idx << rpr0521_gain[chan].shift);
332}
333
334static int rpr0521_read_raw(struct iio_dev *indio_dev,
335 struct iio_chan_spec const *chan, int *val,
336 int *val2, long mask)
337{
338 struct rpr0521_data *data = iio_priv(indio_dev);
339 int ret;
340 u8 device_mask;
341 __le16 raw_data;
342
343 switch (mask) {
344 case IIO_CHAN_INFO_RAW:
345 if (chan->type != IIO_INTENSITY && chan->type != IIO_PROXIMITY)
346 return -EINVAL;
347
348 device_mask = rpr0521_data_reg[chan->address].device_mask;
349
350 mutex_lock(&data->lock);
351 ret = rpr0521_set_power_state(data, true, device_mask);
352 if (ret < 0) {
353 mutex_unlock(&data->lock);
354 return ret;
355 }
356
357 ret = regmap_bulk_read(data->regmap,
358 rpr0521_data_reg[chan->address].address,
Mikko Koivunenae591652017-05-18 15:12:52 +0300359 &raw_data, sizeof(raw_data));
Daniel Balutaefa86e9f2015-06-17 12:42:51 +0300360 if (ret < 0) {
361 rpr0521_set_power_state(data, false, device_mask);
362 mutex_unlock(&data->lock);
363 return ret;
364 }
365
366 ret = rpr0521_set_power_state(data, false, device_mask);
367 mutex_unlock(&data->lock);
368 if (ret < 0)
369 return ret;
370
371 *val = le16_to_cpu(raw_data);
372
373 return IIO_VAL_INT;
374 case IIO_CHAN_INFO_SCALE:
375 mutex_lock(&data->lock);
376 ret = rpr0521_get_gain(data, chan->address, val, val2);
377 mutex_unlock(&data->lock);
378 if (ret < 0)
379 return ret;
380
381 return IIO_VAL_INT_PLUS_MICRO;
382 default:
383 return -EINVAL;
384 }
385}
386
387static int rpr0521_write_raw(struct iio_dev *indio_dev,
388 struct iio_chan_spec const *chan, int val,
389 int val2, long mask)
390{
391 struct rpr0521_data *data = iio_priv(indio_dev);
392 int ret;
393
394 switch (mask) {
395 case IIO_CHAN_INFO_SCALE:
396 mutex_lock(&data->lock);
397 ret = rpr0521_set_gain(data, chan->address, val, val2);
398 mutex_unlock(&data->lock);
399
400 return ret;
401 default:
402 return -EINVAL;
403 }
404}
405
406static const struct iio_info rpr0521_info = {
407 .driver_module = THIS_MODULE,
408 .read_raw = rpr0521_read_raw,
409 .write_raw = rpr0521_write_raw,
410 .attrs = &rpr0521_attribute_group,
411};
412
413static int rpr0521_init(struct rpr0521_data *data)
414{
415 int ret;
416 int id;
417
418 ret = regmap_read(data->regmap, RPR0521_REG_ID, &id);
419 if (ret < 0) {
420 dev_err(&data->client->dev, "Failed to read REG_ID register\n");
421 return ret;
422 }
423
424 if (id != RPR0521_MANUFACT_ID) {
425 dev_err(&data->client->dev, "Wrong id, got %x, expected %x\n",
426 id, RPR0521_MANUFACT_ID);
427 return -ENODEV;
428 }
429
430 /* set default measurement time - 100 ms for both ALS and PS */
431 ret = regmap_update_bits(data->regmap, RPR0521_REG_MODE_CTRL,
432 RPR0521_MODE_MEAS_TIME_MASK,
433 RPR0521_DEFAULT_MEAS_TIME);
434 if (ret) {
435 pr_err("regmap_update_bits returned %d\n", ret);
436 return ret;
437 }
438
Mikko Koivunen484c3142017-05-18 15:12:51 +0300439#ifndef CONFIG_PM
Daniel Balutaefa86e9f2015-06-17 12:42:51 +0300440 ret = rpr0521_als_enable(data, RPR0521_MODE_ALS_ENABLE);
441 if (ret < 0)
442 return ret;
443 ret = rpr0521_pxs_enable(data, RPR0521_MODE_PXS_ENABLE);
444 if (ret < 0)
445 return ret;
Mikko Koivunen484c3142017-05-18 15:12:51 +0300446#endif
Daniel Balutaefa86e9f2015-06-17 12:42:51 +0300447
448 return 0;
449}
450
451static int rpr0521_poweroff(struct rpr0521_data *data)
452{
453 int ret;
454
455 ret = regmap_update_bits(data->regmap, RPR0521_REG_MODE_CTRL,
456 RPR0521_MODE_ALS_MASK |
457 RPR0521_MODE_PXS_MASK,
458 RPR0521_MODE_ALS_DISABLE |
459 RPR0521_MODE_PXS_DISABLE);
460 if (ret < 0)
461 return ret;
462
463 data->als_dev_en = false;
464 data->pxs_dev_en = false;
465
466 return 0;
467}
468
469static bool rpr0521_is_volatile_reg(struct device *dev, unsigned int reg)
470{
471 switch (reg) {
472 case RPR0521_REG_MODE_CTRL:
473 case RPR0521_REG_ALS_CTRL:
474 case RPR0521_REG_PXS_CTRL:
475 return false;
476 default:
477 return true;
478 }
479}
480
481static const struct regmap_config rpr0521_regmap_config = {
482 .name = RPR0521_REGMAP_NAME,
483
484 .reg_bits = 8,
485 .val_bits = 8,
486
487 .max_register = RPR0521_REG_ID,
488 .cache_type = REGCACHE_RBTREE,
489 .volatile_reg = rpr0521_is_volatile_reg,
490};
491
492static int rpr0521_probe(struct i2c_client *client,
493 const struct i2c_device_id *id)
494{
495 struct rpr0521_data *data;
496 struct iio_dev *indio_dev;
497 struct regmap *regmap;
498 int ret;
499
500 indio_dev = devm_iio_device_alloc(&client->dev, sizeof(*data));
501 if (!indio_dev)
502 return -ENOMEM;
503
504 regmap = devm_regmap_init_i2c(client, &rpr0521_regmap_config);
505 if (IS_ERR(regmap)) {
506 dev_err(&client->dev, "regmap_init failed!\n");
507 return PTR_ERR(regmap);
508 }
509
510 data = iio_priv(indio_dev);
511 i2c_set_clientdata(client, indio_dev);
512 data->client = client;
513 data->regmap = regmap;
514
515 mutex_init(&data->lock);
516
517 indio_dev->dev.parent = &client->dev;
518 indio_dev->info = &rpr0521_info;
519 indio_dev->name = RPR0521_DRV_NAME;
520 indio_dev->channels = rpr0521_channels;
521 indio_dev->num_channels = ARRAY_SIZE(rpr0521_channels);
522 indio_dev->modes = INDIO_DIRECT_MODE;
523
524 ret = rpr0521_init(data);
525 if (ret < 0) {
526 dev_err(&client->dev, "rpr0521 chip init failed\n");
527 return ret;
528 }
Daniel Balutaefa86e9f2015-06-17 12:42:51 +0300529
530 ret = pm_runtime_set_active(&client->dev);
531 if (ret < 0)
Mikko Koivunen12d74942017-05-18 15:12:50 +0300532 goto err_poweroff;
Daniel Balutaefa86e9f2015-06-17 12:42:51 +0300533
534 pm_runtime_enable(&client->dev);
535 pm_runtime_set_autosuspend_delay(&client->dev, RPR0521_SLEEP_DELAY_MS);
536 pm_runtime_use_autosuspend(&client->dev);
537
Mikko Koivunen12d74942017-05-18 15:12:50 +0300538 ret = iio_device_register(indio_dev);
539 if (ret)
540 goto err_pm_disable;
541
542 return 0;
543
544err_pm_disable:
545 pm_runtime_disable(&client->dev);
546 pm_runtime_set_suspended(&client->dev);
547 pm_runtime_put_noidle(&client->dev);
548err_poweroff:
549 rpr0521_poweroff(data);
550
551 return ret;
Daniel Balutaefa86e9f2015-06-17 12:42:51 +0300552}
553
554static int rpr0521_remove(struct i2c_client *client)
555{
556 struct iio_dev *indio_dev = i2c_get_clientdata(client);
557
Adriana Reus7d0ead52015-11-05 16:25:29 +0200558 iio_device_unregister(indio_dev);
559
Daniel Balutaefa86e9f2015-06-17 12:42:51 +0300560 pm_runtime_disable(&client->dev);
561 pm_runtime_set_suspended(&client->dev);
562 pm_runtime_put_noidle(&client->dev);
563
Daniel Balutaefa86e9f2015-06-17 12:42:51 +0300564 rpr0521_poweroff(iio_priv(indio_dev));
565
566 return 0;
567}
568
569#ifdef CONFIG_PM
570static int rpr0521_runtime_suspend(struct device *dev)
571{
572 struct iio_dev *indio_dev = i2c_get_clientdata(to_i2c_client(dev));
573 struct rpr0521_data *data = iio_priv(indio_dev);
574 int ret;
575
Daniel Balutaefa86e9f2015-06-17 12:42:51 +0300576 mutex_lock(&data->lock);
Mikko Koivunen484c3142017-05-18 15:12:51 +0300577 /* If measurements are enabled, enable them on resume */
578 if (!data->als_need_dis)
579 data->als_ps_need_en = data->als_dev_en;
580 if (!data->pxs_need_dis)
581 data->pxs_ps_need_en = data->pxs_dev_en;
582
583 /* disable channels and sets {als,pxs}_dev_en to false */
Daniel Balutaefa86e9f2015-06-17 12:42:51 +0300584 ret = rpr0521_poweroff(data);
Mikko Koivunen484c3142017-05-18 15:12:51 +0300585 regcache_mark_dirty(data->regmap);
Daniel Balutaefa86e9f2015-06-17 12:42:51 +0300586 mutex_unlock(&data->lock);
587
588 return ret;
589}
590
591static int rpr0521_runtime_resume(struct device *dev)
592{
593 struct iio_dev *indio_dev = i2c_get_clientdata(to_i2c_client(dev));
594 struct rpr0521_data *data = iio_priv(indio_dev);
595 int ret;
596
Mikko Koivunen484c3142017-05-18 15:12:51 +0300597 regcache_sync(data->regmap);
Daniel Balutaefa86e9f2015-06-17 12:42:51 +0300598 if (data->als_ps_need_en) {
599 ret = rpr0521_als_enable(data, RPR0521_MODE_ALS_ENABLE);
600 if (ret < 0)
601 return ret;
602 data->als_ps_need_en = false;
603 }
604
605 if (data->pxs_ps_need_en) {
606 ret = rpr0521_pxs_enable(data, RPR0521_MODE_PXS_ENABLE);
607 if (ret < 0)
608 return ret;
609 data->pxs_ps_need_en = false;
610 }
Mikko Koivunen484c3142017-05-18 15:12:51 +0300611 msleep(100); //wait for first measurement result
Daniel Balutaefa86e9f2015-06-17 12:42:51 +0300612
613 return 0;
614}
615#endif
616
617static const struct dev_pm_ops rpr0521_pm_ops = {
618 SET_RUNTIME_PM_OPS(rpr0521_runtime_suspend,
619 rpr0521_runtime_resume, NULL)
620};
621
622static const struct acpi_device_id rpr0521_acpi_match[] = {
623 {"RPR0521", 0},
624 { }
625};
626MODULE_DEVICE_TABLE(acpi, rpr0521_acpi_match);
627
628static const struct i2c_device_id rpr0521_id[] = {
629 {"rpr0521", 0},
630 { }
631};
632
633MODULE_DEVICE_TABLE(i2c, rpr0521_id);
634
635static struct i2c_driver rpr0521_driver = {
636 .driver = {
637 .name = RPR0521_DRV_NAME,
638 .pm = &rpr0521_pm_ops,
639 .acpi_match_table = ACPI_PTR(rpr0521_acpi_match),
640 },
641 .probe = rpr0521_probe,
642 .remove = rpr0521_remove,
643 .id_table = rpr0521_id,
644};
645
646module_i2c_driver(rpr0521_driver);
647
648MODULE_AUTHOR("Daniel Baluta <daniel.baluta@intel.com>");
649MODULE_DESCRIPTION("RPR0521 ROHM Ambient Light and Proximity Sensor driver");
650MODULE_LICENSE("GPL v2");