1 /* HopeRF Electronic HP206C precision barometer and altimeter driver
2 *
3 * Copyright (c) 2016 Intel Corporation
4 *
5 * SPDX-License-Identifier: Apache-2.0
6 *
7 * Datasheet:
8 * http://www.hoperf.com/upload/sensor/HP206C_DataSheet_EN_V2.0.pdf
9 */
10
11 #define DT_DRV_COMPAT hoperf_hp206c
12
13 #include <zephyr/init.h>
14 #include <zephyr/drivers/sensor.h>
15 #include <zephyr/drivers/i2c.h>
16 #include <zephyr/sys/byteorder.h>
17 #include <zephyr/kernel.h>
18 #include <zephyr/logging/log.h>
19
20 #include "hp206c.h"
21
22 LOG_MODULE_REGISTER(HP206C, CONFIG_SENSOR_LOG_LEVEL);
23
hp206c_bus_config(const struct device * dev)24 static inline int hp206c_bus_config(const struct device *dev)
25 {
26 const struct hp206c_device_config *cfg = dev->config;
27 uint32_t i2c_cfg;
28
29 i2c_cfg = I2C_MODE_CONTROLLER | I2C_SPEED_SET(I2C_SPEED_STANDARD);
30
31 return i2c_configure(cfg->i2c.bus, i2c_cfg);
32 }
33
hp206c_read(const struct device * dev,uint8_t cmd,uint8_t * data,uint8_t len)34 static int hp206c_read(const struct device *dev, uint8_t cmd, uint8_t *data,
35 uint8_t len)
36 {
37 const struct hp206c_device_config *cfg = dev->config;
38
39 hp206c_bus_config(dev);
40
41 if (i2c_burst_read_dt(&cfg->i2c, cmd, data, len) < 0) {
42 return -EIO;
43 }
44
45 return 0;
46 }
47
hp206c_read_reg(const struct device * dev,uint8_t reg_addr,uint8_t * reg_val)48 static int hp206c_read_reg(const struct device *dev, uint8_t reg_addr,
49 uint8_t *reg_val)
50 {
51 uint8_t cmd = HP206C_CMD_READ_REG | (reg_addr & HP206C_REG_ADDR_MASK);
52
53 return hp206c_read(dev, cmd, reg_val, 1);
54 }
55
hp206c_write(const struct device * dev,uint8_t cmd,uint8_t * data,uint8_t len)56 static int hp206c_write(const struct device *dev, uint8_t cmd, uint8_t *data,
57 uint8_t len)
58 {
59 const struct hp206c_device_config *cfg = dev->config;
60
61 hp206c_bus_config(dev);
62
63 if (i2c_burst_write_dt(&cfg->i2c, cmd, data, len) < 0) {
64 return -EIO;
65 }
66
67 return 0;
68 }
69
hp206c_write_reg(const struct device * dev,uint8_t reg_addr,uint8_t reg_val)70 static int hp206c_write_reg(const struct device *dev, uint8_t reg_addr,
71 uint8_t reg_val)
72 {
73 uint8_t cmd = HP206C_CMD_WRITE_REG | (reg_addr & HP206C_REG_ADDR_MASK);
74
75 return hp206c_write(dev, cmd, ®_val, 1);
76 }
77
hp206c_cmd_send(const struct device * dev,uint8_t cmd)78 static int hp206c_cmd_send(const struct device *dev, uint8_t cmd)
79 {
80 const struct hp206c_device_config *cfg = dev->config;
81
82 hp206c_bus_config(dev);
83
84 return i2c_write_dt(&cfg->i2c, &cmd, 1);
85 }
86
87 /*
88 * The conversion times in this map were rounded up. The reason for doing that
89 * is merely to spare 24 bytes that, otherwise, would've been taken by having
90 * the times converted to microseconds. The trade-off is 900us added to the
91 * conversion wait time which looks like a good compromise provided the highest
92 * precision computation takes 131.1ms.
93 */
94 static uint8_t hp206c_adc_time_ms[] = {
95 /* conversion time(ms), OSR */
96 132, /* 4096 */
97 66, /* 2048 */
98 34, /* 1024 */
99 17, /* 512 */
100 9, /* 256 */
101 5, /* 128 */
102 };
103
hp206c_osr_set(const struct device * dev,uint16_t osr)104 static int hp206c_osr_set(const struct device *dev, uint16_t osr)
105 {
106 struct hp206c_device_data *hp206c = dev->data;
107 uint8_t i;
108
109 /* the following code translates OSR values to an index */
110 for (i = 0U; i < 6 && BIT(12 - i) != osr; i++) {
111 }
112
113 if (i == 6U) {
114 return -ENOTSUP;
115 }
116
117 hp206c->osr = i;
118
119 return 0;
120 }
121
hp206c_altitude_offs_set(const struct device * dev,int16_t offs)122 static int hp206c_altitude_offs_set(const struct device *dev, int16_t offs)
123 {
124 uint8_t reg_val;
125
126 reg_val = offs & 0xff;
127
128 if (hp206c_write_reg(dev, HP206C_REG_ALT_OFF_LSB, reg_val) < 0) {
129 return -EIO;
130 }
131
132 reg_val = (offs & 0xff00) >> 8;
133
134 if (hp206c_write_reg(dev, HP206C_REG_ALT_OFF_MSB, reg_val) < 0) {
135 return -EIO;
136 }
137
138 return hp206c_write_reg(dev, HP206C_REG_PARA, HP206C_COMPENSATION_EN);
139 }
140
hp206c_attr_set(const struct device * dev,enum sensor_channel chan,enum sensor_attribute attr,const struct sensor_value * val)141 static int hp206c_attr_set(const struct device *dev, enum sensor_channel chan,
142 enum sensor_attribute attr,
143 const struct sensor_value *val)
144 {
145 #ifdef CONFIG_HP206C_OSR_RUNTIME
146 if (attr == SENSOR_ATTR_OVERSAMPLING) {
147 return hp206c_osr_set(dev, val->val1);
148 }
149 #endif
150 #ifdef CONFIG_HP206C_ALT_OFFSET_RUNTIME
151 if (attr == SENSOR_ATTR_OFFSET) {
152 if (chan != SENSOR_CHAN_ALTITUDE) {
153 return -ENOTSUP;
154 }
155
156 return hp206c_altitude_offs_set(dev, val->val1);
157 }
158 #endif
159
160 return -ENOTSUP;
161 }
162
hp206c_wait_dev_ready(const struct device * dev,uint32_t timeout_ms)163 static int hp206c_wait_dev_ready(const struct device *dev,
164 uint32_t timeout_ms)
165 {
166 struct hp206c_device_data *hp206c = dev->data;
167 uint8_t int_src;
168
169 k_timer_start(&hp206c->tmr, K_MSEC(timeout_ms), K_NO_WAIT);
170 k_timer_status_sync(&hp206c->tmr);
171
172 if (hp206c_read_reg(dev, HP206C_REG_INT_SRC, &int_src) < 0) {
173 return -EIO;
174 }
175
176 if (int_src & HP206C_DEV_RDY) {
177 return 0;
178 }
179
180 return -EBUSY;
181 }
182
hp206c_adc_acquire(const struct device * dev,enum sensor_channel chan)183 static int hp206c_adc_acquire(const struct device *dev,
184 enum sensor_channel chan)
185 {
186 struct hp206c_device_data *hp206c = dev->data;
187
188 if (hp206c_cmd_send(dev, HP206C_CMD_ADC_CVT | (hp206c->osr << 2)) < 0) {
189 return -EIO;
190 }
191
192 return hp206c_wait_dev_ready(dev, hp206c_adc_time_ms[hp206c->osr]);
193 }
194
hp206c_buf_convert(uint8_t * buf,bool signed_val)195 static int32_t hp206c_buf_convert(uint8_t *buf, bool signed_val)
196 {
197 int32_t tmp = 0;
198
199 if (signed_val && (buf[0] & 0x08)) {
200 tmp |= (0xff << 24) | (0xf0 << 16);
201 }
202
203 tmp |= ((buf[0] & 0x0f) << 16) | (buf[1] << 8) | buf[2];
204
205 return tmp;
206 }
207
hp206c_val_get(const struct device * dev,uint8_t cmd,struct sensor_value * val)208 static int hp206c_val_get(const struct device *dev,
209 uint8_t cmd, struct sensor_value *val)
210 {
211 uint8_t buf[3];
212 int32_t temp = 0;
213
214 if (hp206c_read(dev, cmd, buf, 3) < 0) {
215 return -EIO;
216 }
217
218 /*
219 * According to documentation, pressure and altitude are 20 bit unsigned
220 * values whereas temperature is a signed.
221 */
222 if (cmd == HP206C_CMD_READ_T) {
223 temp = hp206c_buf_convert(buf, true);
224 } else {
225 temp = hp206c_buf_convert(buf, false);
226 }
227
228 if (cmd == HP206C_CMD_READ_P) {
229 val->val1 = temp / 1000;
230 val->val2 = temp % 1000 * 1000;
231 } else {
232 val->val1 = temp / 100;
233 val->val2 = temp % 100 * 10000;
234 }
235
236 return 0;
237 }
238
hp206c_pressure_get(const struct device * dev,struct sensor_value * val)239 static inline int hp206c_pressure_get(const struct device *dev,
240 struct sensor_value *val)
241 {
242 return hp206c_val_get(dev, HP206C_CMD_READ_P, val);
243 }
244
hp206c_altitude_get(const struct device * dev,struct sensor_value * val)245 static inline int hp206c_altitude_get(const struct device *dev,
246 struct sensor_value *val)
247 {
248 return hp206c_val_get(dev, HP206C_CMD_READ_A, val);
249 }
250
hp206c_temperature_get(const struct device * dev,struct sensor_value * val)251 static inline int hp206c_temperature_get(const struct device *dev,
252 struct sensor_value *val)
253 {
254 return hp206c_val_get(dev, HP206C_CMD_READ_T, val);
255 }
256
hp206c_channel_get(const struct device * dev,enum sensor_channel chan,struct sensor_value * val)257 static int hp206c_channel_get(const struct device *dev,
258 enum sensor_channel chan,
259 struct sensor_value *val)
260 {
261 switch (chan) {
262 case SENSOR_CHAN_AMBIENT_TEMP:
263 return hp206c_temperature_get(dev, val);
264
265 case SENSOR_CHAN_PRESS:
266 return hp206c_pressure_get(dev, val);
267
268 case SENSOR_CHAN_ALTITUDE:
269 return hp206c_altitude_get(dev, val);
270
271 default:
272 return -ENOTSUP;
273 }
274
275 return 0;
276 }
277
278 static DEVICE_API(sensor, hp206c_api) = {
279 .attr_set = hp206c_attr_set,
280 .sample_fetch = hp206c_adc_acquire,
281 .channel_get = hp206c_channel_get,
282 };
283
hp206c_init(const struct device * dev)284 static int hp206c_init(const struct device *dev)
285 {
286 struct hp206c_device_data *hp206c = dev->data;
287 const struct hp206c_device_config *cfg = dev->config;
288
289 if (!device_is_ready(cfg->i2c.bus)) {
290 LOG_ERR("Bus device is not ready");
291 return -EINVAL;
292 }
293
294 /* reset the chip */
295 if (hp206c_cmd_send(dev, HP206C_CMD_SOFT_RST) < 0) {
296 LOG_ERR("Cannot reset chip.");
297 return -EIO;
298 }
299
300 k_timer_init(&hp206c->tmr, NULL, NULL);
301
302 k_busy_wait(500);
303
304 if (hp206c_osr_set(dev, HP206C_DEFAULT_OSR) < 0) {
305 LOG_ERR("OSR value is not supported.");
306 return -ENOTSUP;
307 }
308
309 if (hp206c_altitude_offs_set(dev, HP206C_DEFAULT_ALT_OFFSET) < 0) {
310 return -EIO;
311 }
312
313 return 0;
314 }
315
316 #define HP206C_DEFINE(inst) \
317 static struct hp206c_device_data hp206c_data_##inst; \
318 \
319 static const struct hp206c_device_config hp206c_config_##inst = { \
320 .i2c = I2C_DT_SPEC_INST_GET(inst), \
321 }; \
322 \
323 SENSOR_DEVICE_DT_INST_DEFINE(inst, hp206c_init, NULL, \
324 &hp206c_data_##inst, &hp206c_config_##inst, POST_KERNEL, \
325 CONFIG_SENSOR_INIT_PRIORITY, &hp206c_api); \
326
327 DT_INST_FOREACH_STATUS_OKAY(HP206C_DEFINE)
328