1 /*
2 * Copyright (c) 2018 Alexander Wachter.
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
7 #define DT_DRV_COMPAT ams_ens210
8
9 #include <zephyr/device.h>
10 #include <zephyr/drivers/i2c.h>
11 #include <zephyr/kernel.h>
12 #include <zephyr/sys/byteorder.h>
13 #include <zephyr/sys/util.h>
14 #include <zephyr/drivers/sensor.h>
15 #include <zephyr/sys/__assert.h>
16 #include <zephyr/logging/log.h>
17 #include "ens210.h"
18
19 LOG_MODULE_REGISTER(ENS210, CONFIG_SENSOR_LOG_LEVEL);
20
21 #ifdef CONFIG_ENS210_CRC_CHECK
ens210_crc7(uint32_t bitstream)22 static uint32_t ens210_crc7(uint32_t bitstream)
23 {
24 uint32_t polynomial = (ENS210_CRC7_POLY << (ENS210_CRC7_DATA_WIDTH - 1));
25 uint32_t bit = ENS210_CRC7_DATA_MSB << ENS210_CRC7_WIDTH;
26 uint32_t val = (bitstream << ENS210_CRC7_WIDTH) | ENS210_CRC7_IVEC;
27
28 while (bit & (ENS210_CRC7_DATA_MASK << ENS210_CRC7_WIDTH)) {
29 if (bit & val) {
30 val ^= polynomial;
31 }
32
33 bit >>= 1;
34 polynomial >>= 1;
35 }
36
37 return val;
38 }
39 #endif /* CONFIG_ENS210_CRC_CHECK */
40
41 #if defined(CONFIG_ENS210_TEMPERATURE_SINGLE) \
42 || defined(CONFIG_ENS210_HUMIDITY_SINGLE)
ens210_measure(const struct device * dev,enum sensor_channel chan)43 static int ens210_measure(const struct device *dev, enum sensor_channel chan)
44 {
45 struct ens210_data *drv_data = dev->data;
46 const struct ens210_config *config = dev->config;
47 uint8_t buf;
48 int ret;
49 const struct ens210_sens_start sense_start = {
50 .t_start = ENS210_T_START && (chan == SENSOR_CHAN_ALL
51 || chan == SENSOR_CHAN_AMBIENT_TEMP),
52 .h_start = ENS210_H_START && (chan == SENSOR_CHAN_ALL
53 || chan == SENSOR_CHAN_HUMIDITY)
54 };
55
56 /* Start measuring */
57 ret = i2c_reg_write_byte_dt(&config->i2c, ENS210_REG_SENS_START, *(uint8_t *)&sense_start);
58
59 if (ret < 0) {
60 LOG_ERR("Failed to set SENS_START to 0x%x",
61 *(uint8_t *)&sense_start);
62 return -EIO;
63 }
64
65 /* Wait for measurement to be completed */
66 do {
67 k_sleep(K_MSEC(2));
68 ret = i2c_reg_read_byte_dt(&config->i2c, ENS210_REG_SENS_START, &buf);
69
70 if (ret < 0) {
71 LOG_ERR("Failed to read SENS_STAT");
72 }
73 } while (buf & *(uint8_t *)&sense_start);
74
75 return ret;
76 }
77 #endif /* Single shot mode */
78
ens210_sample_fetch(const struct device * dev,enum sensor_channel chan)79 static int ens210_sample_fetch(const struct device *dev,
80 enum sensor_channel chan)
81 {
82 struct ens210_data *drv_data = dev->data;
83 const struct ens210_config *config = dev->config;
84 struct ens210_value_data data[2];
85 int ret, cnt;
86
87 #ifdef CONFIG_ENS210_CRC_CHECK
88 uint32_t temp_valid, humidity_valid;
89 #endif /* CONFIG_ENS210_CRC_CHECK */
90
91 __ASSERT_NO_MSG(chan == SENSOR_CHAN_ALL
92 || chan == SENSOR_CHAN_AMBIENT_TEMP
93 || chan == SENSOR_CHAN_HUMIDITY);
94
95 #if defined(CONFIG_ENS210_TEMPERATURE_SINGLE) \
96 || defined(CONFIG_ENS210_HUMIDITY_SINGLE)
97 ret = ens210_measure(dev, chan);
98 if (ret < 0) {
99 LOG_ERR("Failed to measure");
100 return ret;
101 }
102 #endif /* Single shot mode */
103
104 for (cnt = 0; cnt <= CONFIG_ENS210_MAX_READ_RETRIES; cnt++) {
105 ret = i2c_burst_read_dt(&config->i2c, ENS210_REG_T_VAL, (uint8_t *)&data,
106 sizeof(data));
107 if (ret < 0) {
108 LOG_ERR("Failed to read data");
109 continue;
110 }
111
112 /* Get temperature value */
113 if (chan == SENSOR_CHAN_ALL || chan == SENSOR_CHAN_AMBIENT_TEMP) {
114
115 if (!data[0].valid) {
116 LOG_WRN("Temperature not valid");
117 continue;
118 }
119
120 #ifdef CONFIG_ENS210_CRC_CHECK
121 temp_valid = data[0].val |
122 (data[0].valid << (sizeof(data[0].val) * 8));
123
124 if (ens210_crc7(temp_valid) != data[0].crc7) {
125 LOG_WRN("Temperature CRC error");
126 continue;
127 }
128 #endif /* CONFIG_ENS210_CRC_CHECK */
129
130 drv_data->temp = data[0];
131 }
132
133 /* Get humidity value */
134 if (chan == SENSOR_CHAN_ALL || chan == SENSOR_CHAN_HUMIDITY) {
135
136 if (!data[1].valid) {
137 LOG_WRN("Humidity not valid");
138 continue;
139 }
140
141 #ifdef CONFIG_ENS210_CRC_CHECK
142 humidity_valid = data[1].val |
143 (data[1].valid << (sizeof(data[1].val) * 8));
144
145 if (ens210_crc7(humidity_valid) != data[1].crc7) {
146 LOG_WRN("Humidity CRC error");
147 continue;
148 }
149 #endif /* CONFIG_ENS210_CRC_CHECK */
150
151 drv_data->humidity = data[1];
152 }
153
154 return 0;
155 }
156
157 return -EIO;
158 }
159
ens210_channel_get(const struct device * dev,enum sensor_channel chan,struct sensor_value * val)160 static int ens210_channel_get(const struct device *dev,
161 enum sensor_channel chan,
162 struct sensor_value *val)
163 {
164 struct ens210_data *drv_data = dev->data;
165 int32_t temp_frac;
166 int32_t humidity_frac;
167
168 switch (chan) {
169 case SENSOR_CHAN_AMBIENT_TEMP:
170 /* Temperature is in 1/64 Kelvin. Subtract 273.15 for Celsius */
171 temp_frac = sys_le16_to_cpu(drv_data->temp.val) * (1000000 / 64);
172 temp_frac -= 273150000;
173
174 val->val1 = temp_frac / 1000000;
175 val->val2 = temp_frac % 1000000;
176 break;
177 case SENSOR_CHAN_HUMIDITY:
178 humidity_frac = sys_le16_to_cpu(drv_data->humidity.val) *
179 (1000000 / 512);
180 val->val1 = humidity_frac / 1000000;
181 val->val2 = humidity_frac % 1000000;
182
183 break;
184 default:
185 return -ENOTSUP;
186 }
187
188 return 0;
189 }
190
ens210_sys_reset(const struct device * dev)191 static int ens210_sys_reset(const struct device *dev)
192 {
193 const struct ens210_config *config = dev->config;
194
195 const struct ens210_sys_ctrl sys_ctrl = {
196 .low_power = 0,
197 .reset = 1
198 };
199 int ret;
200
201 ret = i2c_reg_write_byte_dt(&config->i2c, ENS210_REG_SYS_CTRL, *(uint8_t *)&sys_ctrl);
202 if (ret < 0) {
203 LOG_ERR("Failed to set SYS_CTRL to 0x%x", *(uint8_t *)&sys_ctrl);
204 }
205 return ret;
206 }
207
ens210_sys_enable(const struct device * dev,uint8_t low_power)208 static int ens210_sys_enable(const struct device *dev, uint8_t low_power)
209 {
210 const struct ens210_config *config = dev->config;
211
212 const struct ens210_sys_ctrl sys_ctrl = {
213 .low_power = low_power,
214 .reset = 0
215 };
216 int ret;
217
218 ret = i2c_reg_write_byte_dt(&config->i2c, ENS210_REG_SYS_CTRL, *(uint8_t *)&sys_ctrl);
219 if (ret < 0) {
220 LOG_ERR("Failed to set SYS_CTRL to 0x%x", *(uint8_t *)&sys_ctrl);
221 }
222 return ret;
223 }
224
ens210_wait_boot(const struct device * dev)225 static int ens210_wait_boot(const struct device *dev)
226 {
227 const struct ens210_config *config = dev->config;
228
229 int cnt;
230 int ret;
231 struct ens210_sys_stat sys_stat;
232
233 for (cnt = 0; cnt <= CONFIG_ENS210_MAX_STAT_RETRIES; cnt++) {
234 ret = i2c_reg_read_byte_dt(&config->i2c, ENS210_REG_SYS_STAT, (uint8_t *)&sys_stat);
235
236 if (ret < 0) {
237 k_sleep(K_MSEC(1));
238 continue;
239 }
240
241 if (sys_stat.sys_active) {
242 return 0;
243 }
244
245 if (cnt == 0) {
246 ens210_sys_reset(dev);
247 }
248
249 ens210_sys_enable(dev, 0);
250
251 k_sleep(K_MSEC(2));
252 }
253
254 if (ret < 0) {
255 LOG_ERR("Failed to read SYS_STATE");
256 }
257
258
259 LOG_ERR("Sensor is not in active state");
260 return -EIO;
261 }
262
263 static DEVICE_API(sensor, en210_driver_api) = {
264 .sample_fetch = ens210_sample_fetch,
265 .channel_get = ens210_channel_get,
266 };
267
ens210_init(const struct device * dev)268 static int ens210_init(const struct device *dev)
269 {
270 const struct ens210_config *config = dev->config;
271 const struct ens210_sens_run sense_run = {
272 .t_run = ENS210_T_RUN,
273 .h_run = ENS210_H_RUN
274 };
275
276 #if defined(CONFIG_ENS210_TEMPERATURE_CONTINUOUS) \
277 || defined(CONFIG_ENS210_HUMIDITY_CONTINUOUS)
278 const struct ens210_sens_start sense_start = {
279 .t_start = ENS210_T_RUN,
280 .h_start = ENS210_H_RUN
281 };
282 #endif
283
284 int ret;
285 uint16_t part_id;
286
287 if (!device_is_ready(config->i2c.bus)) {
288 LOG_ERR("I2C bus device not ready");
289 return -ENODEV;
290 }
291
292 /* Wait until the device is ready. */
293 ret = ens210_wait_boot(dev);
294 if (ret < 0) {
295 return -EIO;
296 }
297
298 /* Check Hardware ID. This is only possible after device is ready
299 * and active
300 */
301 ret = i2c_burst_read_dt(&config->i2c, ENS210_REG_PART_ID, (uint8_t *)&part_id,
302 sizeof(part_id));
303 if (ret < 0) {
304 LOG_ERR("Failed to read Part ID register");
305 return -EIO;
306 }
307
308 if (part_id != ENS210_PART_ID) {
309 LOG_ERR("Part ID does not match. Want 0x%x, got 0x%x",
310 ENS210_PART_ID, part_id);
311 return -EIO;
312 }
313
314 /* Enable low power mode */
315 if ((ENS210_T_RUN | ENS210_H_RUN) == 0) {
316 ens210_sys_enable(dev, 1);
317 }
318
319 /* Set measurement mode*/
320 ret = i2c_reg_write_byte_dt(&config->i2c, ENS210_REG_SENS_RUN, *(uint8_t *)&sense_run);
321 if (ret < 0) {
322 LOG_ERR("Failed to set SENS_RUN to 0x%x",
323 *(uint8_t *)&sense_run);
324 return -EIO;
325 }
326
327 #if defined(CONFIG_ENS210_TEMPERATURE_CONTINUOUS) \
328 || defined(CONFIG_ENS210_HUMIDITY_CONTINUOUS)
329 /* Start measuring */
330 ret = i2c_reg_write_byte_dt(&config->i2c, ENS210_REG_SENS_START, *(uint8_t *)&sense_start);
331 if (ret < 0) {
332 LOG_ERR("Failed to set SENS_START to 0x%x",
333 *(uint8_t *)&sense_start);
334 return -EIO;
335 }
336 #endif
337 return 0;
338 }
339
340 #define ENS210_DEFINE(inst) \
341 static struct ens210_data ens210_data_##inst; \
342 \
343 static const struct ens210_config ens210_config_##inst = { \
344 .i2c = I2C_DT_SPEC_INST_GET(inst), \
345 }; \
346 \
347 SENSOR_DEVICE_DT_INST_DEFINE(inst, ens210_init, NULL, \
348 &ens210_data_##inst, &ens210_config_##inst, POST_KERNEL, \
349 CONFIG_SENSOR_INIT_PRIORITY, &en210_driver_api); \
350
351 DT_INST_FOREACH_STATUS_OKAY(ENS210_DEFINE)
352