1 /*
2  * Copyright (c) 2022 Badgerd Technologies B.V.
3  * Copyright (c) 2023 Metratec GmbH
4  *
5  * SPDX-License-Identifier: Apache-2.0
6  */
7 
8 #include "bmp581.h"
9 
10 #include <math.h>
11 
12 #include <zephyr/device.h>
13 #include <zephyr/drivers/gpio.h>
14 #include <zephyr/drivers/i2c.h>
15 #include <zephyr/drivers/sensor.h>
16 #include <zephyr/init.h>
17 #include <zephyr/logging/log.h>
18 #include <zephyr/sys/check.h>
19 
20 LOG_MODULE_REGISTER(bmp581, CONFIG_SENSOR_LOG_LEVEL);
21 
22 #if DT_NUM_INST_STATUS_OKAY(DT_DRV_COMPAT) == 0
23 #warning "BMP581 driver enabled without any devices"
24 #endif
25 
26 static int power_up_check(const struct device *dev);
27 static int get_nvm_status(uint8_t *nvm_status, const struct device *dev);
28 static int get_interrupt_status(uint8_t *int_status, const struct device *dev);
29 static int validate_chip_id(struct bmp581_data *drv);
30 static int get_osr_odr_press_config(struct bmp581_osr_odr_press_config *osr_odr_press_cfg,
31 				    const struct device *dev);
32 static int set_osr_config(const struct sensor_value *osr, enum sensor_channel chan,
33 			  const struct device *dev);
34 static int set_odr_config(const struct sensor_value *odr, const struct device *dev);
35 static int soft_reset(const struct device *dev);
36 static int set_iir_config(const struct sensor_value *iir, const struct device *dev);
37 static int get_power_mode(enum bmp5_powermode *powermode, const struct device *dev);
38 static int set_power_mode(enum bmp5_powermode powermode, const struct device *dev);
39 
set_power_mode(enum bmp5_powermode powermode,const struct device * dev)40 static int set_power_mode(enum bmp5_powermode powermode, const struct device *dev)
41 {
42 	struct bmp581_config *conf = (struct bmp581_config *)dev->config;
43 	int ret = BMP5_OK;
44 	uint8_t odr = 0;
45 	enum bmp5_powermode current_powermode;
46 
47 	CHECKIF(dev == NULL) {
48 		return -EINVAL;
49 	}
50 
51 	ret = get_power_mode(&current_powermode, dev);
52 	if (ret != BMP5_OK) {
53 		LOG_ERR("Couldnt set the power mode because something went wrong when getting the "
54 			"current power mode.");
55 		return ret;
56 	}
57 
58 	if (current_powermode != BMP5_POWERMODE_STANDBY) {
59 		/*
60 		 * Device should be set to standby before transitioning to forced mode or normal
61 		 * mode or continuous mode.
62 		 */
63 
64 		ret = i2c_reg_read_byte_dt(&conf->i2c, BMP5_REG_ODR_CONFIG, &odr);
65 		if (ret == BMP5_OK) {
66 			/* Setting deep_dis = 1(BMP5_DEEP_DISABLED) disables the deep standby mode
67 			 */
68 			odr = BMP5_SET_BITSLICE(odr, BMP5_DEEP_DISABLE, BMP5_DEEP_DISABLED);
69 			odr = BMP5_SET_BITS_POS_0(odr, BMP5_POWERMODE, BMP5_POWERMODE_STANDBY);
70 			ret = i2c_reg_write_byte_dt(&conf->i2c, BMP5_REG_ODR_CONFIG, odr);
71 
72 			if (ret != BMP5_OK) {
73 				LOG_DBG("Failed to set power mode to BMP5_POWERMODE_STANDBY.");
74 				return ret;
75 			}
76 		}
77 	}
78 
79 	/* lets update the power mode */
80 	switch (powermode) {
81 	case BMP5_POWERMODE_STANDBY:
82 		/* this change is already done so we can just return */
83 		ret = BMP5_OK;
84 		break;
85 	case BMP5_POWERMODE_DEEP_STANDBY:
86 		LOG_DBG("Setting power mode to DEEP STANDBY is not supported, current power mode "
87 			"is BMP5_POWERMODE_STANDBY.");
88 		ret = -ENOTSUP;
89 		break;
90 	case BMP5_POWERMODE_NORMAL:
91 	case BMP5_POWERMODE_FORCED:
92 	case BMP5_POWERMODE_CONTINUOUS:
93 		odr = BMP5_SET_BITSLICE(odr, BMP5_DEEP_DISABLE, BMP5_DEEP_DISABLED);
94 		odr = BMP5_SET_BITS_POS_0(odr, BMP5_POWERMODE, powermode);
95 		ret = i2c_reg_write_byte_dt(&conf->i2c, BMP5_REG_ODR_CONFIG, odr);
96 		break;
97 	default:
98 		/* invalid power mode */
99 		ret = -EINVAL;
100 		break;
101 	}
102 
103 	return ret;
104 }
105 
get_power_mode(enum bmp5_powermode * powermode,const struct device * dev)106 static int get_power_mode(enum bmp5_powermode *powermode, const struct device *dev)
107 {
108 	struct bmp581_config *conf = (struct bmp581_config *)dev->config;
109 	int ret = BMP5_OK;
110 
111 	CHECKIF(powermode == NULL || dev == NULL) {
112 		return -EINVAL;
113 	}
114 
115 	uint8_t reg = 0;
116 	uint8_t raw_power_mode = 0;
117 
118 	ret = i2c_reg_read_byte_dt(&conf->i2c, BMP5_REG_ODR_CONFIG, &reg);
119 	if (ret != BMP5_OK) {
120 		LOG_DBG("Failed to read odr config to get power mode!");
121 		return ret;
122 	}
123 
124 	raw_power_mode = BMP5_GET_BITS_POS_0(reg, BMP5_POWERMODE);
125 
126 	switch (raw_power_mode) {
127 	case BMP5_POWERMODE_STANDBY: {
128 		/* Getting deep disable status */
129 		uint8_t deep_dis = BMP5_GET_BITSLICE(reg, BMP5_DEEP_DISABLE);
130 
131 		/* Checking deepstandby status only when powermode is in standby mode */
132 
133 		/* If deep_dis = 0(BMP5_DEEP_ENABLED) then deepstandby mode is enabled.
134 		 * If deep_dis = 1(BMP5_DEEP_DISABLED) then deepstandby mode is disabled
135 		 */
136 		if (deep_dis == BMP5_DEEP_ENABLED) {
137 			/* TODO: check if it is really deep standby */
138 			*powermode = BMP5_POWERMODE_DEEP_STANDBY;
139 		} else {
140 			*powermode = BMP5_POWERMODE_STANDBY;
141 		}
142 
143 		break;
144 	}
145 	case BMP5_POWERMODE_NORMAL:
146 		*powermode = BMP5_POWERMODE_NORMAL;
147 		break;
148 	case BMP5_POWERMODE_FORCED:
149 		*powermode = BMP5_POWERMODE_FORCED;
150 		break;
151 	case BMP5_POWERMODE_CONTINUOUS:
152 		*powermode = BMP5_POWERMODE_CONTINUOUS;
153 		break;
154 	default:
155 		/* invalid power mode */
156 		ret = -EINVAL;
157 		LOG_DBG("Something went wrong invalid powermode!");
158 		break;
159 	}
160 
161 	return ret;
162 }
163 
power_up_check(const struct device * dev)164 static int power_up_check(const struct device *dev)
165 {
166 	int8_t rslt = 0;
167 	uint8_t nvm_status = 0;
168 
169 	CHECKIF(dev == NULL) {
170 		return -EINVAL;
171 	}
172 
173 	rslt = get_nvm_status(&nvm_status, dev);
174 
175 	if (rslt == BMP5_OK) {
176 		/* Check if nvm_rdy status = 1 and nvm_err status = 0 to proceed */
177 		if ((nvm_status & BMP5_INT_NVM_RDY) && (!(nvm_status & BMP5_INT_NVM_ERR))) {
178 			rslt = BMP5_OK;
179 		} else {
180 			rslt = -EFAULT;
181 		}
182 	}
183 
184 	return rslt;
185 }
186 
get_interrupt_status(uint8_t * int_status,const struct device * dev)187 static int get_interrupt_status(uint8_t *int_status, const struct device *dev)
188 {
189 	struct bmp581_config *conf = (struct bmp581_config *)dev->config;
190 
191 	CHECKIF(int_status == NULL || dev == NULL) {
192 		return -EINVAL;
193 	}
194 
195 	return i2c_reg_read_byte_dt(&conf->i2c, BMP5_REG_INT_STATUS, int_status);
196 }
197 
get_nvm_status(uint8_t * nvm_status,const struct device * dev)198 static int get_nvm_status(uint8_t *nvm_status, const struct device *dev)
199 {
200 	struct bmp581_config *conf = (struct bmp581_config *)dev->config;
201 
202 	CHECKIF(nvm_status == NULL || dev == NULL) {
203 		return -EINVAL;
204 	}
205 
206 	return i2c_reg_read_byte_dt(&conf->i2c, BMP5_REG_STATUS, nvm_status);
207 }
208 
validate_chip_id(struct bmp581_data * drv)209 static int validate_chip_id(struct bmp581_data *drv)
210 {
211 	int8_t rslt = 0;
212 
213 	CHECKIF(drv == NULL) {
214 		return -EINVAL;
215 	}
216 
217 	if ((drv->chip_id == BMP5_CHIP_ID_PRIM) || (drv->chip_id == BMP5_CHIP_ID_SEC)) {
218 		rslt = BMP5_OK;
219 	} else {
220 		drv->chip_id = 0;
221 		rslt = -ENODEV;
222 	}
223 
224 	return rslt;
225 }
226 
227 /*!
228  *  This API gets the configuration for oversampling of temperature, oversampling of
229  *  pressure and ODR configuration along with pressure enable.
230  */
get_osr_odr_press_config(struct bmp581_osr_odr_press_config * osr_odr_press_cfg,const struct device * dev)231 static int get_osr_odr_press_config(struct bmp581_osr_odr_press_config *osr_odr_press_cfg,
232 				    const struct device *dev)
233 {
234 	struct bmp581_config *conf = (struct bmp581_config *)dev->config;
235 
236 	/* Variable to store the function result */
237 	int8_t rslt = 0;
238 
239 	/* Variable to store OSR and ODR config */
240 	uint8_t reg_data[2] = {0};
241 
242 	CHECKIF(osr_odr_press_cfg == NULL || dev == NULL) {
243 		return -EINVAL;
244 	}
245 
246 	/* Get OSR and ODR configuration in burst read */
247 	rslt = i2c_burst_read_dt(&conf->i2c, BMP5_REG_OSR_CONFIG, reg_data, 2);
248 
249 	if (rslt == BMP5_OK) {
250 		osr_odr_press_cfg->osr_t = BMP5_GET_BITS_POS_0(reg_data[0], BMP5_TEMP_OS);
251 		osr_odr_press_cfg->osr_p = BMP5_GET_BITSLICE(reg_data[0], BMP5_PRESS_OS);
252 		osr_odr_press_cfg->press_en = BMP5_GET_BITSLICE(reg_data[0], BMP5_PRESS_EN);
253 		osr_odr_press_cfg->odr = BMP5_GET_BITSLICE(reg_data[1], BMP5_ODR);
254 	}
255 
256 	return rslt;
257 }
258 
set_osr_config(const struct sensor_value * osr,enum sensor_channel chan,const struct device * dev)259 static int set_osr_config(const struct sensor_value *osr, enum sensor_channel chan,
260 			  const struct device *dev)
261 {
262 	CHECKIF(osr == NULL || dev == NULL) {
263 		return -EINVAL;
264 	}
265 
266 	struct bmp581_data *drv = (struct bmp581_data *)dev->data;
267 	struct bmp581_config *conf = (struct bmp581_config *)dev->config;
268 	int ret = 0;
269 
270 	uint8_t oversampling = osr->val1;
271 	uint8_t press_en = osr->val2 != 0; /* if it is not 0 then pressure is enabled */
272 	uint8_t osr_val = 0;
273 
274 	ret = i2c_reg_read_byte_dt(&conf->i2c, BMP5_REG_OSR_CONFIG, &osr_val);
275 	if (ret == BMP5_OK) {
276 		switch (chan) {
277 		case SENSOR_CHAN_ALL:
278 			osr_val = BMP5_SET_BITS_POS_0(osr_val, BMP5_TEMP_OS, oversampling);
279 			osr_val = BMP5_SET_BITSLICE(osr_val, BMP5_PRESS_OS, oversampling);
280 			osr_val = BMP5_SET_BITSLICE(osr_val, BMP5_PRESS_EN, press_en);
281 			break;
282 		case SENSOR_CHAN_PRESS:
283 			osr_val = BMP5_SET_BITSLICE(osr_val, BMP5_PRESS_OS, oversampling);
284 			osr_val = BMP5_SET_BITSLICE(osr_val, BMP5_PRESS_EN, press_en);
285 			break;
286 		case SENSOR_CHAN_AMBIENT_TEMP:
287 			osr_val = BMP5_SET_BITS_POS_0(osr_val, BMP5_TEMP_OS, oversampling);
288 			break;
289 		default:
290 			ret = -ENOTSUP;
291 			break;
292 		}
293 
294 		if (ret == BMP5_OK) {
295 			ret = i2c_reg_write_byte_dt(&conf->i2c, BMP5_REG_OSR_CONFIG, osr_val);
296 			get_osr_odr_press_config(&drv->osr_odr_press_config, dev);
297 		}
298 	}
299 
300 	return ret;
301 }
302 
set_odr_config(const struct sensor_value * odr,const struct device * dev)303 static int set_odr_config(const struct sensor_value *odr, const struct device *dev)
304 {
305 	CHECKIF(odr == NULL || dev == NULL) {
306 		return -EINVAL;
307 	}
308 
309 	struct bmp581_data *drv = (struct bmp581_data *)dev->data;
310 	struct bmp581_config *conf = (struct bmp581_config *)dev->config;
311 	int ret = 0;
312 	uint8_t odr_val = 0;
313 
314 	ret = i2c_reg_read_byte_dt(&conf->i2c, BMP5_REG_ODR_CONFIG, &odr_val);
315 	if (ret != BMP5_OK) {
316 		return ret;
317 	}
318 	odr_val = BMP5_SET_BITSLICE(odr_val, BMP5_ODR, odr->val1);
319 	ret = i2c_reg_write_byte_dt(&conf->i2c, BMP5_REG_ODR_CONFIG, odr_val);
320 	get_osr_odr_press_config(&drv->osr_odr_press_config, dev);
321 
322 	return ret;
323 }
324 
soft_reset(const struct device * dev)325 static int soft_reset(const struct device *dev)
326 {
327 	struct bmp581_config *conf = (struct bmp581_config *)dev->config;
328 	int ret = 0;
329 	const uint8_t reset_cmd = BMP5_SOFT_RESET_CMD;
330 	uint8_t int_status = 0;
331 
332 	CHECKIF(dev == NULL) {
333 		return -EINVAL;
334 	}
335 
336 	ret = i2c_reg_write_byte_dt(&conf->i2c, BMP5_REG_CMD, reset_cmd);
337 
338 	if (ret == BMP5_OK) {
339 		k_usleep(BMP5_DELAY_US_SOFT_RESET);
340 		ret = get_interrupt_status(&int_status, dev);
341 		if (ret == BMP5_OK) {
342 			if (int_status & BMP5_INT_ASSERTED_POR_SOFTRESET_COMPLETE) {
343 				ret = BMP5_OK;
344 			} else {
345 				ret = -EFAULT;
346 			}
347 		}
348 	} else {
349 		LOG_DBG("Failed perform soft-reset.");
350 	}
351 
352 	return ret;
353 }
354 
bmp581_sample_fetch(const struct device * dev,enum sensor_channel chan)355 static int bmp581_sample_fetch(const struct device *dev, enum sensor_channel chan)
356 {
357 	CHECKIF(dev == NULL) {
358 		return -EINVAL;
359 	}
360 
361 	if (chan != SENSOR_CHAN_ALL) {
362 		return -ENOTSUP;
363 	}
364 
365 	struct bmp581_data *drv = (struct bmp581_data *)dev->data;
366 	struct bmp581_config *conf = (struct bmp581_config *)dev->config;
367 	uint8_t data[6];
368 	int ret = 0;
369 
370 	ret = i2c_burst_read_dt(&conf->i2c, BMP5_REG_TEMP_DATA_XLSB, data, 6);
371 	if (ret == BMP5_OK) {
372 		/* convert raw sensor data to sensor_value. Shift the decimal part by 1 decimal
373 		 * place to compensate for the conversion in sensor_value_to_double()
374 		 */
375 		drv->last_sample.temperature.val1 = data[2];
376 		drv->last_sample.temperature.val2 = (data[1] << 8 | data[0]) * 10;
377 
378 		if (drv->osr_odr_press_config.press_en == BMP5_ENABLE) {
379 			uint32_t raw_pressure = (uint32_t)((uint32_t)(data[5] << 16) |
380 							   (uint16_t)(data[4] << 8) | data[3]);
381 			/* convert raw sensor data to sensor_value. Shift the decimal part by
382 			 * 4 decimal places to compensate for the conversion in
383 			 * sensor_value_to_double()
384 			 */
385 			drv->last_sample.pressure.val1 = raw_pressure >> 6;
386 			drv->last_sample.pressure.val2 = (raw_pressure & BIT_MASK(6)) * 10000;
387 		} else {
388 			drv->last_sample.pressure.val1 = 0;
389 			drv->last_sample.pressure.val2 = 0;
390 		}
391 	}
392 
393 	return ret;
394 }
395 
bmp581_channel_get(const struct device * dev,enum sensor_channel chan,struct sensor_value * val)396 static int bmp581_channel_get(const struct device *dev, enum sensor_channel chan,
397 			      struct sensor_value *val)
398 {
399 	CHECKIF(dev == NULL || val == NULL) {
400 		return -EINVAL;
401 	}
402 
403 	struct bmp581_data *drv = (struct bmp581_data *)dev->data;
404 
405 	switch (chan) {
406 	case SENSOR_CHAN_PRESS:
407 		/* returns pressure in Pa */
408 		*val = drv->last_sample.pressure;
409 		return BMP5_OK;
410 	case SENSOR_CHAN_AMBIENT_TEMP:
411 		/* returns temperature in Celcius */
412 		*val = drv->last_sample.temperature;
413 		return BMP5_OK;
414 	default:
415 		return -ENOTSUP;
416 	}
417 }
418 
set_iir_config(const struct sensor_value * iir,const struct device * dev)419 static int set_iir_config(const struct sensor_value *iir, const struct device *dev)
420 {
421 	struct bmp581_config *conf = (struct bmp581_config *)dev->config;
422 	int ret = BMP5_OK;
423 
424 	CHECKIF((iir == NULL) | (dev == NULL)) {
425 		return -EINVAL;
426 	}
427 
428 	/* Variable to store existing powermode */
429 	enum bmp5_powermode prev_powermode;
430 
431 	ret = get_power_mode(&prev_powermode, dev);
432 	if (ret != BMP5_OK) {
433 		LOG_DBG("Not able to get current power mode.");
434 		return ret;
435 	}
436 	/* IIR configuration is writable only during STANDBY mode(as per datasheet) */
437 	set_power_mode(BMP5_POWERMODE_STANDBY, dev);
438 
439 	/* update IIR config */
440 	uint8_t dsp_config[2];
441 
442 	ret = i2c_burst_read_dt(&conf->i2c, BMP5_REG_DSP_CONFIG, dsp_config, 2);
443 	if (ret != BMP5_OK) {
444 		LOG_DBG("Failed to read dsp config register.");
445 		return ret;
446 	}
447 	/* Put IIR filtered values in data registers */
448 	dsp_config[0] = BMP5_SET_BITSLICE(dsp_config[0], BMP5_SHDW_SET_IIR_TEMP, BMP5_ENABLE);
449 	dsp_config[0] = BMP5_SET_BITSLICE(dsp_config[0], BMP5_SHDW_SET_IIR_PRESS, BMP5_ENABLE);
450 
451 	/* Configure IIR filter */
452 	dsp_config[1] = iir->val1;
453 	dsp_config[1] = BMP5_SET_BITSLICE(dsp_config[1], BMP5_SET_IIR_PRESS, iir->val2);
454 
455 	/* Set IIR configuration */
456 	ret = i2c_burst_write_dt(&conf->i2c, BMP5_REG_DSP_CONFIG, dsp_config, 2);
457 
458 	if (ret != BMP5_OK) {
459 		LOG_DBG("Failed to configure IIR filter.");
460 		return ret;
461 	}
462 
463 	/* Restore previous power mode if it is not standby already */
464 	if (prev_powermode != BMP5_POWERMODE_STANDBY) {
465 		ret = set_power_mode(prev_powermode, dev);
466 	}
467 
468 	return ret;
469 }
470 
bmp581_attr_set(const struct device * dev,enum sensor_channel chan,enum sensor_attribute attr,const struct sensor_value * val)471 static int bmp581_attr_set(const struct device *dev, enum sensor_channel chan,
472 			   enum sensor_attribute attr, const struct sensor_value *val)
473 {
474 	CHECKIF(dev == NULL || val == NULL) {
475 		return -EINVAL;
476 	}
477 
478 	int ret;
479 
480 	switch ((int)attr) {
481 	case SENSOR_ATTR_SAMPLING_FREQUENCY:
482 		ret = set_odr_config(val, dev);
483 		break;
484 	case SENSOR_ATTR_OVERSAMPLING:
485 		ret = set_osr_config(val, chan, dev);
486 		break;
487 	case BMP5_ATTR_POWER_MODE: {
488 		enum bmp5_powermode powermode = (enum bmp5_powermode)val->val1;
489 
490 		ret = set_power_mode(powermode, dev);
491 		break;
492 	}
493 	case BMP5_ATTR_IIR_CONFIG:
494 		ret = set_iir_config(val, dev);
495 		break;
496 	default:
497 		ret = -ENOTSUP;
498 		break;
499 	}
500 	return ret;
501 }
502 
bmp581_init(const struct device * dev)503 static int bmp581_init(const struct device *dev)
504 {
505 	CHECKIF(dev == NULL) {
506 		return -EINVAL;
507 	}
508 
509 	struct bmp581_data *drv = (struct bmp581_data *)dev->data;
510 	struct bmp581_config *conf = (struct bmp581_config *)dev->config;
511 	int ret = -1;
512 
513 	/* Reset the chip id. */
514 	drv->chip_id = 0;
515 	memset(&drv->osr_odr_press_config, 0, sizeof(drv->osr_odr_press_config));
516 	memset(&drv->last_sample, 0, sizeof(drv->last_sample));
517 
518 	soft_reset(dev);
519 
520 	ret = i2c_reg_read_byte_dt(&conf->i2c, BMP5_REG_CHIP_ID, &drv->chip_id);
521 	if (ret != BMP5_OK) {
522 		return ret;
523 	}
524 
525 	if (drv->chip_id != 0) {
526 		ret = power_up_check(dev);
527 		if (ret == BMP5_OK) {
528 			ret = validate_chip_id(drv);
529 			if (ret != BMP5_OK) {
530 				LOG_ERR("Unexpected chip id (%x). Expected (%x or %x)",
531 					drv->chip_id, BMP5_CHIP_ID_PRIM, BMP5_CHIP_ID_SEC);
532 			}
533 		}
534 	} else {
535 		/* that means something went wrong */
536 		LOG_ERR("Unexpected chip id (%x). Expected (%x or %x)", drv->chip_id,
537 			BMP5_CHIP_ID_PRIM, BMP5_CHIP_ID_SEC);
538 		return -EINVAL;
539 	}
540 	return ret;
541 }
542 
543 static DEVICE_API(sensor, bmp581_driver_api) = {
544 	.sample_fetch = bmp581_sample_fetch,
545 	.channel_get = bmp581_channel_get,
546 	.attr_set = bmp581_attr_set,
547 };
548 
549 #define BMP581_CONFIG(i)                                                                           \
550 	static const struct bmp581_config bmp581_config_##i = {                                    \
551 		.i2c = I2C_DT_SPEC_INST_GET(i),                                                    \
552 	}
553 
554 #define BMP581_INIT(i)                                                                             \
555 	static struct bmp581_data bmp581_data_##i;                                                 \
556 	BMP581_CONFIG(i);                                                                          \
557                                                                                                    \
558 	SENSOR_DEVICE_DT_INST_DEFINE(i, bmp581_init, NULL, &bmp581_data_##i, &bmp581_config_##i,   \
559 				     POST_KERNEL, CONFIG_SENSOR_INIT_PRIORITY,                     \
560 				     &bmp581_driver_api);
561 
562 DT_INST_FOREACH_STATUS_OKAY(BMP581_INIT)
563