1 /*
2 * Copyright (c) 2017 Intel Corporation
3 * Copyright (c) 2018 Phytec Messtechnik GmbH
4 *
5 *SPDX-License-Identifier: Apache-2.0
6 */
7
8 #define DT_DRV_COMPAT avago_apds9960
9
10 /* @file
11 * @brief driver for APDS9960 ALS/RGB/gesture/proximity sensor
12 */
13
14 #include <zephyr/device.h>
15 #include <zephyr/drivers/sensor.h>
16 #include <zephyr/drivers/i2c.h>
17 #include <zephyr/pm/device.h>
18 #include <zephyr/sys/__assert.h>
19 #include <zephyr/sys/byteorder.h>
20 #include <zephyr/init.h>
21 #include <zephyr/kernel.h>
22 #include <string.h>
23 #include <zephyr/logging/log.h>
24
25 #include "apds9960.h"
26
27 LOG_MODULE_REGISTER(APDS9960, CONFIG_SENSOR_LOG_LEVEL);
28
apds9960_handle_cb(struct apds9960_data * drv_data)29 static void apds9960_handle_cb(struct apds9960_data *drv_data)
30 {
31 apds9960_setup_int(drv_data->dev->config, false);
32
33 #ifdef CONFIG_APDS9960_TRIGGER
34 k_work_submit(&drv_data->work);
35 #else
36 k_sem_give(&drv_data->data_sem);
37 #endif
38 }
39
apds9960_gpio_callback(const struct device * dev,struct gpio_callback * cb,uint32_t pins)40 static void apds9960_gpio_callback(const struct device *dev,
41 struct gpio_callback *cb, uint32_t pins)
42 {
43 struct apds9960_data *drv_data =
44 CONTAINER_OF(cb, struct apds9960_data, gpio_cb);
45
46 apds9960_handle_cb(drv_data);
47 }
48
apds9960_sample_fetch(const struct device * dev,enum sensor_channel chan)49 static int apds9960_sample_fetch(const struct device *dev,
50 enum sensor_channel chan)
51 {
52 const struct apds9960_config *config = dev->config;
53 struct apds9960_data *data = dev->data;
54 uint8_t tmp;
55
56 if (chan != SENSOR_CHAN_ALL) {
57 LOG_ERR("Unsupported sensor channel");
58 return -ENOTSUP;
59 }
60
61 #ifndef CONFIG_APDS9960_TRIGGER
62 apds9960_setup_int(config, true);
63
64 #ifdef CONFIG_APDS9960_ENABLE_ALS
65 tmp = APDS9960_ENABLE_PON | APDS9960_ENABLE_AIEN;
66 #else
67 tmp = APDS9960_ENABLE_PON | APDS9960_ENABLE_PIEN;
68 #endif
69 if (i2c_reg_update_byte_dt(&config->i2c,
70 APDS9960_ENABLE_REG, tmp, tmp)) {
71 LOG_ERR("Power on bit not set.");
72 return -EIO;
73 }
74
75 k_sem_take(&data->data_sem, K_FOREVER);
76 #endif
77
78 if (i2c_reg_read_byte_dt(&config->i2c,
79 APDS9960_STATUS_REG, &tmp)) {
80 return -EIO;
81 }
82
83 LOG_DBG("status: 0x%x", tmp);
84 if (tmp & APDS9960_STATUS_PINT) {
85 if (i2c_reg_read_byte_dt(&config->i2c,
86 APDS9960_PDATA_REG, &data->pdata)) {
87 return -EIO;
88 }
89 }
90
91 if (tmp & APDS9960_STATUS_AINT) {
92 if (i2c_burst_read_dt(&config->i2c,
93 APDS9960_CDATAL_REG,
94 (uint8_t *)&data->sample_crgb,
95 sizeof(data->sample_crgb))) {
96 return -EIO;
97 }
98
99 }
100
101 #ifndef CONFIG_APDS9960_TRIGGER
102 if (i2c_reg_update_byte_dt(&config->i2c,
103 APDS9960_ENABLE_REG,
104 APDS9960_ENABLE_PON,
105 0)) {
106 return -EIO;
107 }
108 #endif
109
110 if (i2c_reg_write_byte_dt(&config->i2c,
111 APDS9960_AICLEAR_REG, 0)) {
112 return -EIO;
113 }
114
115 return 0;
116 }
117
apds9960_channel_get(const struct device * dev,enum sensor_channel chan,struct sensor_value * val)118 static int apds9960_channel_get(const struct device *dev,
119 enum sensor_channel chan,
120 struct sensor_value *val)
121 {
122 struct apds9960_data *data = dev->data;
123
124 switch (chan) {
125 #ifdef CONFIG_APDS9960_ENABLE_ALS
126 case SENSOR_CHAN_LIGHT:
127 val->val1 = sys_le16_to_cpu(data->sample_crgb[0]);
128 val->val2 = 0;
129 break;
130 case SENSOR_CHAN_RED:
131 val->val1 = sys_le16_to_cpu(data->sample_crgb[1]);
132 val->val2 = 0;
133 break;
134 case SENSOR_CHAN_GREEN:
135 val->val1 = sys_le16_to_cpu(data->sample_crgb[2]);
136 val->val2 = 0;
137 break;
138 case SENSOR_CHAN_BLUE:
139 val->val1 = sys_le16_to_cpu(data->sample_crgb[3]);
140 val->val2 = 0;
141 break;
142 #endif
143 case SENSOR_CHAN_PROX:
144 val->val1 = data->pdata;
145 val->val2 = 0;
146 break;
147 default:
148 return -ENOTSUP;
149 }
150
151 return 0;
152 }
153
apds9960_proxy_setup(const struct device * dev)154 static int apds9960_proxy_setup(const struct device *dev)
155 {
156 const struct apds9960_config *config = dev->config;
157
158 if (i2c_reg_write_byte_dt(&config->i2c,
159 APDS9960_POFFSET_UR_REG,
160 APDS9960_DEFAULT_POFFSET_UR)) {
161 LOG_ERR("Default offset UR not set ");
162 return -EIO;
163 }
164
165 if (i2c_reg_write_byte_dt(&config->i2c,
166 APDS9960_POFFSET_DL_REG,
167 APDS9960_DEFAULT_POFFSET_DL)) {
168 LOG_ERR("Default offset DL not set ");
169 return -EIO;
170 }
171
172 if (i2c_reg_write_byte_dt(&config->i2c,
173 APDS9960_PPULSE_REG,
174 config->ppcount)) {
175 LOG_ERR("Default pulse count not set ");
176 return -EIO;
177 }
178
179 if (i2c_reg_update_byte_dt(&config->i2c,
180 APDS9960_CONTROL_REG,
181 APDS9960_CONTROL_LDRIVE,
182 APDS9960_DEFAULT_LDRIVE)) {
183 LOG_ERR("LED Drive Strength not set");
184 return -EIO;
185 }
186
187 if (i2c_reg_update_byte_dt(&config->i2c,
188 APDS9960_CONFIG2_REG,
189 APDS9960_PLED_BOOST_300,
190 config->pled_boost)) {
191 LOG_ERR("LED Drive Strength not set");
192 return -EIO;
193 }
194
195 if (i2c_reg_update_byte_dt(&config->i2c,
196 APDS9960_CONTROL_REG, APDS9960_CONTROL_PGAIN,
197 (config->pgain & APDS9960_PGAIN_8X))) {
198 LOG_ERR("Gain is not set");
199 return -EIO;
200 }
201
202 if (i2c_reg_write_byte_dt(&config->i2c,
203 APDS9960_PILT_REG, APDS9960_DEFAULT_PILT)) {
204 LOG_ERR("Low threshold not set");
205 return -EIO;
206 }
207
208 if (i2c_reg_write_byte_dt(&config->i2c,
209 APDS9960_PIHT_REG, APDS9960_DEFAULT_PIHT)) {
210 LOG_ERR("High threshold not set");
211 return -EIO;
212 }
213
214 if (i2c_reg_update_byte_dt(&config->i2c,
215 APDS9960_ENABLE_REG, APDS9960_ENABLE_PEN,
216 APDS9960_ENABLE_PEN)) {
217 LOG_ERR("Proximity mode is not enabled");
218 return -EIO;
219 }
220
221 return 0;
222 }
223
224 #ifdef CONFIG_APDS9960_ENABLE_ALS
apds9960_ambient_setup(const struct device * dev)225 static int apds9960_ambient_setup(const struct device *dev)
226 {
227 const struct apds9960_config *config = dev->config;
228 uint16_t th;
229
230 /* ADC value */
231 if (i2c_reg_write_byte_dt(&config->i2c,
232 APDS9960_ATIME_REG, APDS9960_DEFAULT_ATIME)) {
233 LOG_ERR("Default integration time not set for ADC");
234 return -EIO;
235 }
236
237 /* ALS Gain */
238 if (i2c_reg_update_byte_dt(&config->i2c,
239 APDS9960_CONTROL_REG,
240 APDS9960_CONTROL_AGAIN,
241 (config->again & APDS9960_AGAIN_64X))) {
242 LOG_ERR("Ambient Gain is not set");
243 return -EIO;
244 }
245
246 th = sys_cpu_to_le16(APDS9960_DEFAULT_AILT);
247 if (i2c_burst_write_dt(&config->i2c,
248 APDS9960_INT_AILTL_REG,
249 (uint8_t *)&th, sizeof(th))) {
250 LOG_ERR("ALS low threshold not set");
251 return -EIO;
252 }
253
254 th = sys_cpu_to_le16(APDS9960_DEFAULT_AIHT);
255 if (i2c_burst_write_dt(&config->i2c,
256 APDS9960_INT_AIHTL_REG,
257 (uint8_t *)&th, sizeof(th))) {
258 LOG_ERR("ALS low threshold not set");
259 return -EIO;
260 }
261
262 /* Enable ALS */
263 if (i2c_reg_update_byte_dt(&config->i2c,
264 APDS9960_ENABLE_REG, APDS9960_ENABLE_AEN,
265 APDS9960_ENABLE_AEN)) {
266 LOG_ERR("ALS is not enabled");
267 return -EIO;
268 }
269
270 return 0;
271 }
272 #endif
273
apds9960_sensor_setup(const struct device * dev)274 static int apds9960_sensor_setup(const struct device *dev)
275 {
276 const struct apds9960_config *config = dev->config;
277 uint8_t chip_id;
278
279 if (i2c_reg_read_byte_dt(&config->i2c,
280 APDS9960_ID_REG, &chip_id)) {
281 LOG_ERR("Failed reading chip id");
282 return -EIO;
283 }
284
285 if (!((chip_id == APDS9960_ID_1) || (chip_id == APDS9960_ID_2))) {
286 LOG_ERR("Invalid chip id 0x%x", chip_id);
287 return -EIO;
288 }
289
290 /* Disable all functions and interrupts */
291 if (i2c_reg_write_byte_dt(&config->i2c,
292 APDS9960_ENABLE_REG, 0)) {
293 LOG_ERR("ENABLE register is not cleared");
294 return -EIO;
295 }
296
297 if (i2c_reg_write_byte_dt(&config->i2c,
298 APDS9960_AICLEAR_REG, 0)) {
299 return -EIO;
300 }
301
302 /* Disable gesture interrupt */
303 if (i2c_reg_write_byte_dt(&config->i2c,
304 APDS9960_GCONFIG4_REG, 0)) {
305 LOG_ERR("GCONFIG4 register is not cleared");
306 return -EIO;
307 }
308
309 if (i2c_reg_write_byte_dt(&config->i2c,
310 APDS9960_WTIME_REG, APDS9960_DEFAULT_WTIME)) {
311 LOG_ERR("Default wait time not set");
312 return -EIO;
313 }
314
315 if (i2c_reg_write_byte_dt(&config->i2c,
316 APDS9960_CONFIG1_REG,
317 APDS9960_DEFAULT_CONFIG1)) {
318 LOG_ERR("Default WLONG not set");
319 return -EIO;
320 }
321
322 if (i2c_reg_write_byte_dt(&config->i2c,
323 APDS9960_CONFIG2_REG,
324 APDS9960_DEFAULT_CONFIG2)) {
325 LOG_ERR("Configuration Register Two not set");
326 return -EIO;
327 }
328
329 if (i2c_reg_write_byte_dt(&config->i2c,
330 APDS9960_CONFIG3_REG,
331 APDS9960_DEFAULT_CONFIG3)) {
332 LOG_ERR("Configuration Register Three not set");
333 return -EIO;
334 }
335
336 if (i2c_reg_write_byte_dt(&config->i2c,
337 APDS9960_PERS_REG,
338 APDS9960_DEFAULT_PERS)) {
339 LOG_ERR("Interrupt persistence not set");
340 return -EIO;
341 }
342
343 if (apds9960_proxy_setup(dev)) {
344 LOG_ERR("Failed to setup proximity functionality");
345 return -EIO;
346 }
347
348 #ifdef CONFIG_APDS9960_ENABLE_ALS
349 if (apds9960_ambient_setup(dev)) {
350 LOG_ERR("Failed to setup ambient light functionality");
351 return -EIO;
352 }
353 #endif
354
355 return 0;
356 }
357
apds9960_init_interrupt(const struct device * dev)358 static int apds9960_init_interrupt(const struct device *dev)
359 {
360 const struct apds9960_config *config = dev->config;
361 struct apds9960_data *drv_data = dev->data;
362
363 if (!gpio_is_ready_dt(&config->int_gpio)) {
364 LOG_ERR("%s: device %s is not ready", dev->name,
365 config->int_gpio.port->name);
366 return -ENODEV;
367 }
368
369 gpio_pin_configure_dt(&config->int_gpio, GPIO_INPUT | config->int_gpio.dt_flags);
370
371 gpio_init_callback(&drv_data->gpio_cb,
372 apds9960_gpio_callback,
373 BIT(config->int_gpio.pin));
374
375 if (gpio_add_callback(config->int_gpio.port, &drv_data->gpio_cb) < 0) {
376 LOG_DBG("Failed to set gpio callback!");
377 return -EIO;
378 }
379
380 drv_data->dev = dev;
381
382 #ifdef CONFIG_APDS9960_TRIGGER
383 drv_data->work.handler = apds9960_work_cb;
384 if (i2c_reg_update_byte_dt(&config->i2c,
385 APDS9960_ENABLE_REG,
386 APDS9960_ENABLE_PON,
387 APDS9960_ENABLE_PON)) {
388 LOG_ERR("Power on bit not set.");
389 return -EIO;
390 }
391
392 #else
393 k_sem_init(&drv_data->data_sem, 0, K_SEM_MAX_LIMIT);
394 #endif
395 apds9960_setup_int(config, true);
396
397 if (gpio_pin_get_dt(&config->int_gpio) > 0) {
398 apds9960_handle_cb(drv_data);
399 }
400
401 return 0;
402 }
403
404 #ifdef CONFIG_PM_DEVICE
apds9960_pm_action(const struct device * dev,enum pm_device_action action)405 static int apds9960_pm_action(const struct device *dev,
406 enum pm_device_action action)
407 {
408 const struct apds9960_config *config = dev->config;
409 int ret = 0;
410
411 switch (action) {
412 case PM_DEVICE_ACTION_RESUME:
413 if (i2c_reg_update_byte_dt(&config->i2c,
414 APDS9960_ENABLE_REG,
415 APDS9960_ENABLE_PON,
416 APDS9960_ENABLE_PON)) {
417 ret = -EIO;
418 }
419 break;
420 case PM_DEVICE_ACTION_SUSPEND:
421 if (i2c_reg_update_byte_dt(&config->i2c,
422 APDS9960_ENABLE_REG,
423 APDS9960_ENABLE_PON, 0)) {
424 ret = -EIO;
425 }
426
427 if (i2c_reg_write_byte_dt(&config->i2c,
428 APDS9960_AICLEAR_REG, 0)) {
429 ret = -EIO;
430 }
431 break;
432 default:
433 return -ENOTSUP;
434 }
435
436 return ret;
437 }
438 #endif
439
apds9960_init(const struct device * dev)440 static int apds9960_init(const struct device *dev)
441 {
442 const struct apds9960_config *config = dev->config;
443 struct apds9960_data *data = dev->data;
444
445 /* Initialize time 5.7ms */
446 k_sleep(K_MSEC(6));
447
448 if (!device_is_ready(config->i2c.bus)) {
449 LOG_ERR("Bus device is not ready");
450 return -EINVAL;
451 }
452
453 (void)memset(data->sample_crgb, 0, sizeof(data->sample_crgb));
454 data->pdata = 0U;
455
456 if (apds9960_sensor_setup(dev) < 0) {
457 LOG_ERR("Failed to setup device!");
458 return -EIO;
459 }
460
461 if (apds9960_init_interrupt(dev) < 0) {
462 LOG_ERR("Failed to initialize interrupt!");
463 return -EIO;
464 }
465
466 return 0;
467 }
468
469 static DEVICE_API(sensor, apds9960_driver_api) = {
470 .sample_fetch = &apds9960_sample_fetch,
471 .channel_get = &apds9960_channel_get,
472 #ifdef CONFIG_APDS9960_TRIGGER
473 .attr_set = apds9960_attr_set,
474 .trigger_set = apds9960_trigger_set,
475 #endif
476 };
477
478 static const struct apds9960_config apds9960_config = {
479 .i2c = I2C_DT_SPEC_INST_GET(0),
480 .int_gpio = GPIO_DT_SPEC_INST_GET(0, int_gpios),
481 #if CONFIG_APDS9960_PGAIN_8X
482 .pgain = APDS9960_PGAIN_8X,
483 #elif CONFIG_APDS9960_PGAIN_4X
484 .pgain = APDS9960_PGAIN_4X,
485 #elif CONFIG_APDS9960_PGAIN_2X
486 .pgain = APDS9960_PGAIN_2X,
487 #else
488 .pgain = APDS9960_PGAIN_1X,
489 #endif
490 #if CONFIG_APDS9960_AGAIN_64X
491 .again = APDS9960_AGAIN_64X,
492 #elif CONFIG_APDS9960_AGAIN_16X
493 .again = APDS9960_AGAIN_16X,
494 #elif CONFIG_APDS9960_AGAIN_4X
495 .again = APDS9960_AGAIN_4X,
496 #else
497 .again = APDS9960_AGAIN_1X,
498 #endif
499 #if CONFIG_APDS9960_PPULSE_LENGTH_32US
500 .ppcount = APDS9960_PPULSE_LENGTH_32US |
501 (CONFIG_APDS9960_PPULSE_COUNT - 1),
502 #elif CONFIG_APDS9960_PPULSE_LENGTH_16US
503 .ppcount = APDS9960_PPULSE_LENGTH_16US |
504 (CONFIG_APDS9960_PPULSE_COUNT - 1),
505 #elif CONFIG_APDS9960_PPULSE_LENGTH_8US
506 .ppcount = APDS9960_PPULSE_LENGTH_8US |
507 (CONFIG_APDS9960_PPULSE_COUNT - 1),
508 #else
509 .ppcount = APDS9960_PPULSE_LENGTH_4US |
510 (CONFIG_APDS9960_PPULSE_COUNT - 1),
511 #endif
512 #if CONFIG_APDS9960_PLED_BOOST_300PCT
513 .pled_boost = APDS9960_PLED_BOOST_300,
514 #elif CONFIG_APDS9960_PLED_BOOST_200PCT
515 .pled_boost = APDS9960_PLED_BOOST_200,
516 #elif CONFIG_APDS9960_PLED_BOOST_150PCT
517 .pled_boost = APDS9960_PLED_BOOST_150,
518 #else
519 .pled_boost = APDS9960_PLED_BOOST_100,
520 #endif
521 };
522
523 static struct apds9960_data apds9960_data;
524
525 PM_DEVICE_DT_INST_DEFINE(0, apds9960_pm_action);
526
527 SENSOR_DEVICE_DT_INST_DEFINE(0, apds9960_init,
528 PM_DEVICE_DT_INST_GET(0), &apds9960_data, &apds9960_config,
529 POST_KERNEL, CONFIG_SENSOR_INIT_PRIORITY, &apds9960_driver_api);
530