1 /* sensor_lsm9ds0_mfd.c - Driver for LSM9DS0 accelerometer, magnetometer
2 * and temperature (MFD) sensor driver
3 */
4
5 /*
6 * Copyright (c) 2016 Intel Corporation
7 *
8 * SPDX-License-Identifier: Apache-2.0
9 */
10
11 #define DT_DRV_COMPAT st_lsm9ds0_mfd
12
13 #include <zephyr/drivers/sensor.h>
14 #include <zephyr/kernel.h>
15 #include <zephyr/device.h>
16 #include <zephyr/init.h>
17 #include <zephyr/drivers/i2c.h>
18 #include <zephyr/sys/byteorder.h>
19 #include <zephyr/drivers/gpio.h>
20 #include <zephyr/logging/log.h>
21
22 #include "lsm9ds0_mfd.h"
23
24 LOG_MODULE_REGISTER(LSM9DS0_MFD, CONFIG_SENSOR_LOG_LEVEL);
25
lsm9ds0_mfd_reboot_memory(const struct device * dev)26 static inline int lsm9ds0_mfd_reboot_memory(const struct device *dev)
27 {
28 const struct lsm9ds0_mfd_config *config = dev->config;
29
30 if (i2c_reg_update_byte_dt(&config->i2c, LSM9DS0_MFD_REG_CTRL_REG0_XM,
31 LSM9DS0_MFD_MASK_CTRL_REG0_XM_BOOT,
32 1 << LSM9DS0_MFD_SHIFT_CTRL_REG0_XM_BOOT) < 0) {
33 return -EIO;
34 }
35
36 k_busy_wait(USEC_PER_MSEC * 50U);
37
38 return 0;
39 }
40
41 #if !defined(LSM9DS0_MFD_ACCEL_DISABLED)
lsm9ds0_mfd_accel_set_odr_raw(const struct device * dev,uint8_t odr)42 static inline int lsm9ds0_mfd_accel_set_odr_raw(const struct device *dev,
43 uint8_t odr)
44 {
45 const struct lsm9ds0_mfd_config *config = dev->config;
46
47 return i2c_reg_update_byte_dt(&config->i2c, LSM9DS0_MFD_REG_CTRL_REG1_XM,
48 LSM9DS0_MFD_MASK_CTRL_REG1_XM_AODR,
49 odr << LSM9DS0_MFD_SHIFT_CTRL_REG1_XM_AODR);
50 }
51
52 #if defined(CONFIG_LSM9DS0_MFD_ACCEL_SAMPLING_RATE_RUNTIME)
53 static const struct {
54 int freq_int;
55 int freq_micro;
56 } lsm9ds0_mfd_accel_odr_map[] = { {0, 0},
57 {3, 125000},
58 {6, 250000},
59 {12, 500000},
60 {25, 0},
61 {50, 0},
62 {100, 0},
63 {200, 0},
64 {400, 0},
65 {800, 0},
66 {1600, 0} };
67
lsm9ds0_mfd_accel_set_odr(const struct device * dev,const struct sensor_value * val)68 static int lsm9ds0_mfd_accel_set_odr(const struct device *dev,
69 const struct sensor_value *val)
70 {
71 uint8_t i;
72
73 for (i = 0U; i < ARRAY_SIZE(lsm9ds0_mfd_accel_odr_map); ++i) {
74 if (val->val1 < lsm9ds0_mfd_accel_odr_map[i].freq_int ||
75 (val->val1 == lsm9ds0_mfd_accel_odr_map[i].freq_int &&
76 val->val2 <= lsm9ds0_mfd_accel_odr_map[i].freq_micro)) {
77 return lsm9ds0_mfd_accel_set_odr_raw(dev, i);
78 }
79 }
80
81 return -ENOTSUP;
82 }
83 #endif
84
lsm9ds0_mfd_accel_set_fs_raw(const struct device * dev,uint8_t fs)85 static inline int lsm9ds0_mfd_accel_set_fs_raw(const struct device *dev,
86 uint8_t fs)
87 {
88 const struct lsm9ds0_mfd_config *config = dev->config;
89
90 if (i2c_reg_update_byte_dt(&config->i2c, LSM9DS0_MFD_REG_CTRL_REG2_XM,
91 LSM9DS0_MFD_MASK_CTRL_REG2_XM_AFS,
92 fs << LSM9DS0_MFD_SHIFT_CTRL_REG2_XM_AFS) < 0) {
93 return -EIO;
94 }
95
96 #if defined(CONFIG_LSM9DS0_MFD_ACCEL_FULL_SCALE_RUNTIME)
97 data->accel_fs = fs;
98 #endif
99
100 return 0;
101 }
102
103 #if defined(CONFIG_LSM9DS0_MFD_ACCEL_FULL_SCALE_RUNTIME)
104 static const struct {
105 int fs;
106 } lsm9ds0_mfd_accel_fs_map[] = { {2},
107 {4},
108 {6},
109 {8},
110 {16} };
111
lsm9ds0_mfd_accel_set_fs(const struct device * dev,int val)112 static int lsm9ds0_mfd_accel_set_fs(const struct device *dev, int val)
113 {
114 uint8_t i;
115
116 for (i = 0U; i < ARRAY_SIZE(lsm9ds0_mfd_accel_fs_map); ++i) {
117 if (val <= lsm9ds0_mfd_accel_fs_map[i].fs) {
118 return lsm9ds0_mfd_accel_set_fs_raw(dev, i);
119 }
120 }
121
122 return -ENOTSUP;
123 }
124 #endif
125 #endif
126
127 #if !defined(LSM9DS0_MFD_MAGN_DISABLED)
lsm9ds0_mfd_magn_set_odr_raw(const struct device * dev,uint8_t odr)128 static inline int lsm9ds0_mfd_magn_set_odr_raw(const struct device *dev,
129 uint8_t odr)
130 {
131 const struct lsm9ds0_mfd_config *config = dev->config;
132
133 return i2c_reg_update_byte_dt(&config->i2c, LSM9DS0_MFD_REG_CTRL_REG5_XM,
134 LSM9DS0_MFD_MASK_CTRL_REG5_XM_M_ODR,
135 odr << LSM9DS0_MFD_SHIFT_CTRL_REG5_XM_M_ODR);
136 }
137
138 #if defined(CONFIG_LSM9DS0_MFD_MAGN_SAMPLING_RATE_RUNTIME)
139 static const struct {
140 int freq_int;
141 int freq_micro;
142 } lsm9ds0_mfd_magn_odr_map[] = { {0, 0},
143 {3, 125000},
144 {6, 250000},
145 {12, 500000},
146 {25, 0},
147 {50, 0},
148 {100, 0} };
149
lsm9ds0_mfd_magn_set_odr(const struct device * dev,const struct sensor_value * val)150 static int lsm9ds0_mfd_magn_set_odr(const struct device *dev,
151 const struct sensor_value *val)
152 {
153 uint8_t i;
154
155 for (i = 0U; i < ARRAY_SIZE(lsm9ds0_mfd_accel_odr_map); ++i) {
156 if (val->val1 < lsm9ds0_mfd_accel_odr_map[i].freq_int ||
157 (val->val1 == lsm9ds0_mfd_accel_odr_map[i].freq_int &&
158 val->val2 <= lsm9ds0_mfd_accel_odr_map[i].freq_micro)) {
159 return lsm9ds0_mfd_magn_set_odr_raw(dev, i);
160 }
161 }
162
163 return -ENOTSUP;
164 }
165 #endif
166
lsm9ds0_mfd_magn_set_fs_raw(const struct device * dev,uint8_t fs)167 static inline int lsm9ds0_mfd_magn_set_fs_raw(const struct device *dev,
168 uint8_t fs)
169 {
170 const struct lsm9ds0_mfd_config *config = dev->config;
171
172 if (i2c_reg_update_byte_dt(&config->i2c, LSM9DS0_MFD_REG_CTRL_REG6_XM,
173 LSM9DS0_MFD_MASK_CTRL_REG6_XM_MFS,
174 fs << LSM9DS0_MFD_SHIFT_CTRL_REG6_XM_MFS) < 0) {
175 return -EIO;
176 }
177
178 #if defined(CONFIG_LSM9DS0_MFD_MAGN_FULL_SCALE_RUNTIME)
179 data->magn_fs = fs;
180 #endif
181
182 return 0;
183 }
184
185 #if defined(CONFIG_LSM9DS0_MFD_MAGN_FULL_SCALE_RUNTIME)
186 static const struct {
187 int fs;
188 } lsm9ds0_mfd_magn_fs_map[] = { {2},
189 {4},
190 {8},
191 {12} };
192
lsm9ds0_mfd_magn_set_fs(const struct device * dev,const struct sensor_value * val)193 static int lsm9ds0_mfd_magn_set_fs(const struct device *dev,
194 const struct sensor_value *val)
195 {
196 uint8_t i;
197
198 for (i = 0U; i < ARRAY_SIZE(lsm9ds0_mfd_magn_fs_map); ++i) {
199 if (val->val1 <= lsm9ds0_mfd_magn_fs_map[i].fs) {
200 return lsm9ds0_mfd_magn_set_fs_raw(dev, i);
201 }
202 }
203
204 return -ENOTSUP;
205 }
206 #endif
207 #endif
208
209 #if !defined(LSM9DS0_MFD_ACCEL_DISABLED)
lsm9ds0_mfd_sample_fetch_accel(const struct device * dev)210 static inline int lsm9ds0_mfd_sample_fetch_accel(const struct device *dev)
211 {
212 struct lsm9ds0_mfd_data *data = dev->data;
213 const struct lsm9ds0_mfd_config *config = dev->config;
214 uint8_t out_l, out_h;
215
216 #if defined(CONFIG_LSM9DS0_MFD_ACCEL_ENABLE_X)
217 if (i2c_reg_read_byte_dt(&config->i2c, LSM9DS0_MFD_REG_OUT_X_L_A, &out_l) < 0 ||
218 i2c_reg_read_byte_dt(&config->i2c, LSM9DS0_MFD_REG_OUT_X_H_A, &out_h) < 0) {
219 LOG_DBG("failed to read accel sample (X axis)");
220 return -EIO;
221 }
222
223 data->sample_accel_x = (int16_t)((uint16_t)(out_l) |
224 ((uint16_t)(out_h) << 8));
225 #endif
226
227 #if defined(CONFIG_LSM9DS0_MFD_ACCEL_ENABLE_Y)
228 if (i2c_reg_read_byte_dt(&config->i2c, LSM9DS0_MFD_REG_OUT_Y_L_A, &out_l) < 0 ||
229 i2c_reg_read_byte_dt(&config->i2c, LSM9DS0_MFD_REG_OUT_Y_H_A, &out_h) < 0) {
230 LOG_DBG("failed to read accel sample (Y axis)");
231 return -EIO;
232 }
233
234 data->sample_accel_y = (int16_t)((uint16_t)(out_l) |
235 ((uint16_t)(out_h) << 8));
236 #endif
237
238 #if defined(CONFIG_LSM9DS0_MFD_ACCEL_ENABLE_Z)
239 if (i2c_reg_read_byte_dt(&config->i2c, LSM9DS0_MFD_REG_OUT_Z_L_A, &out_l) < 0 ||
240 i2c_reg_read_byte_dt(&config->i2c, LSM9DS0_MFD_REG_OUT_Z_H_A, &out_h) < 0) {
241 LOG_DBG("failed to read accel sample (Z axis)");
242 return -EIO;
243 }
244
245 data->sample_accel_z = (int16_t)((uint16_t)(out_l) |
246 ((uint16_t)(out_h) << 8));
247 #endif
248
249 #if defined(CONFIG_LSM9DS0_MFD_ACCEL_FULL_SCALE_RUNTIME)
250 data->sample_accel_fs = data->accel_fs;
251 #endif
252
253 return 0;
254 }
255 #endif
256
257 #if !defined(LSM9DS0_MFD_MAGN_DISABLED)
lsm9ds0_mfd_sample_fetch_magn(const struct device * dev)258 static inline int lsm9ds0_mfd_sample_fetch_magn(const struct device *dev)
259 {
260 struct lsm9ds0_mfd_data *data = dev->data;
261 const struct lsm9ds0_mfd_config *config = dev->config;
262 uint8_t out_l, out_h;
263
264 if (i2c_reg_read_byte_dt(&config->i2c, LSM9DS0_MFD_REG_OUT_X_L_M, &out_l) < 0 ||
265 i2c_reg_read_byte_dt(&config->i2c, LSM9DS0_MFD_REG_OUT_X_H_M, &out_h) < 0) {
266 LOG_DBG("failed to read magn sample (X axis)");
267 return -EIO;
268 }
269
270 data->sample_magn_x = (int16_t)((uint16_t)(out_l) |
271 ((uint16_t)(out_h) << 8));
272
273 if (i2c_reg_read_byte_dt(&config->i2c, LSM9DS0_MFD_REG_OUT_Y_L_M, &out_l) < 0 ||
274 i2c_reg_read_byte_dt(&config->i2c, LSM9DS0_MFD_REG_OUT_Y_H_M, &out_h) < 0) {
275 LOG_DBG("failed to read magn sample (Y axis)");
276 return -EIO;
277 }
278
279 data->sample_magn_y = (int16_t)((uint16_t)(out_l) |
280 ((uint16_t)(out_h) << 8));
281
282 if (i2c_reg_read_byte_dt(&config->i2c, LSM9DS0_MFD_REG_OUT_Z_L_M, &out_l) < 0 ||
283 i2c_reg_read_byte_dt(&config->i2c, LSM9DS0_MFD_REG_OUT_Z_H_M, &out_h) < 0) {
284 LOG_DBG("failed to read magn sample (Z axis)");
285 return -EIO;
286 }
287
288 data->sample_magn_z = (int16_t)((uint16_t)(out_l) |
289 ((uint16_t)(out_h) << 8));
290
291 #if defined(CONFIG_LSM9DS0_MFD_MAGN_FULL_SCALE_RUNTIME)
292 data->sample_magn_fs = data->magn_fs;
293 #endif
294
295 return 0;
296 }
297 #endif
298
299 #if !defined(LSM9DS0_MFD_TEMP_DISABLED)
lsm9ds0_mfd_sample_fetch_temp(const struct device * dev)300 static inline int lsm9ds0_mfd_sample_fetch_temp(const struct device *dev)
301 {
302 struct lsm9ds0_mfd_data *data = dev->data;
303 const struct lsm9ds0_mfd_config *config = dev->config;
304 uint8_t out_l, out_h;
305
306 if (i2c_reg_read_byte_dt(&config->i2c, LSM9DS0_MFD_REG_OUT_TEMP_L_XM, &out_l) < 0 ||
307 i2c_reg_read_byte_dt(&config->i2c, LSM9DS0_MFD_REG_OUT_TEMP_H_XM, &out_h) < 0) {
308 LOG_DBG("failed to read temperature sample\n");
309 return -EIO;
310 }
311
312 data->sample_temp = (int16_t)((uint16_t)(out_l) |
313 ((uint16_t)(out_h) << 8));
314
315 return 0;
316 }
317 #endif
318
lsm9ds0_mfd_sample_fetch_all(const struct device * dev)319 static inline int lsm9ds0_mfd_sample_fetch_all(const struct device *dev)
320 {
321 #if !defined(LSM9DS0_MFD_ACCEL_DISABLED)
322 if (lsm9ds0_mfd_sample_fetch_accel(dev) < 0) {
323 return -EIO;
324 }
325 #endif
326
327 #if !defined(LSM9DS0_MFD_MAGN_DISABLED)
328 if (lsm9ds0_mfd_sample_fetch_magn(dev) < 0) {
329 return -EIO;
330 }
331 #endif
332
333 #if !defined(LSM9DS0_MFD_TEMP_DISABLED)
334 if (lsm9ds0_mfd_sample_fetch_temp(dev) < 0) {
335 return -EIO;
336 }
337 #endif
338
339 return 0;
340 }
341
lsm9ds0_mfd_sample_fetch(const struct device * dev,enum sensor_channel chan)342 static int lsm9ds0_mfd_sample_fetch(const struct device *dev,
343 enum sensor_channel chan)
344 {
345 switch (chan) {
346 #if !defined(LSM9DS0_MFD_ACCEL_DISABLED)
347 case SENSOR_CHAN_ACCEL_XYZ:
348 return lsm9ds0_mfd_sample_fetch_accel(dev);
349 #endif
350 #if !defined(LSM9DS0_MFD_MAGN_DISABLED)
351 case SENSOR_CHAN_MAGN_XYZ:
352 return lsm9ds0_mfd_sample_fetch_magn(dev);
353 #endif
354 #if !defined(LSM9DS0_MFD_TEMP_DISABLED)
355 case SENSOR_CHAN_DIE_TEMP:
356 return lsm9ds0_mfd_sample_fetch_temp(dev);
357 #endif
358 case SENSOR_CHAN_ALL:
359 return lsm9ds0_mfd_sample_fetch_all(dev);
360 default:
361 return -EINVAL;
362 }
363
364 return 0;
365 }
366
367 #if !defined(LSM9DS0_MFD_ACCEL_DISABLED)
lsm9ds0_mfd_convert_accel(struct sensor_value * val,int raw_val,float scale)368 static inline void lsm9ds0_mfd_convert_accel(struct sensor_value *val,
369 int raw_val,
370 float scale)
371 {
372 double dval;
373
374 dval = (double)(raw_val) * (double)scale;
375 val->val1 = (int32_t)dval;
376 val->val2 = ((int32_t)(dval * 1000000)) % 1000000;
377 }
378
lsm9ds0_mfd_get_accel_channel(enum sensor_channel chan,struct sensor_value * val,struct lsm9ds0_mfd_data * data,float scale)379 static inline int lsm9ds0_mfd_get_accel_channel(enum sensor_channel chan,
380 struct sensor_value *val,
381 struct lsm9ds0_mfd_data *data,
382 float scale)
383 {
384 switch (chan) {
385 case SENSOR_CHAN_ACCEL_X:
386 lsm9ds0_mfd_convert_accel(val, data->sample_accel_x, scale);
387 break;
388 case SENSOR_CHAN_ACCEL_Y:
389 lsm9ds0_mfd_convert_accel(val, data->sample_accel_y, scale);
390 break;
391 case SENSOR_CHAN_ACCEL_Z:
392 lsm9ds0_mfd_convert_accel(val, data->sample_accel_z, scale);
393 break;
394 case SENSOR_CHAN_ACCEL_XYZ:
395 lsm9ds0_mfd_convert_accel(val, data->sample_accel_x, scale);
396 lsm9ds0_mfd_convert_accel(val + 1, data->sample_accel_y, scale);
397 lsm9ds0_mfd_convert_accel(val + 2, data->sample_accel_z, scale);
398 break;
399 default:
400 return -ENOTSUP;
401 }
402
403 return 0;
404 }
405
lsm9ds0_mfd_get_accel(const struct device * dev,enum sensor_channel chan,struct sensor_value * val)406 static inline int lsm9ds0_mfd_get_accel(const struct device *dev,
407 enum sensor_channel chan,
408 struct sensor_value *val)
409 {
410 struct lsm9ds0_mfd_data *data = dev->data;
411
412 #if defined(CONFIG_LSM9DS0_MFD_ACCEL_FULL_SCALE_RUNTIME)
413 switch (data->sample_accel_fs) {
414 case 0:
415 return lsm9ds0_mfd_get_accel_channel(chan, val, data,
416 2.0 * 9.807 / 32767.0);
417 case 1:
418 return lsm9ds0_mfd_get_accel_channel(chan, val, data,
419 4.0 * 9.807 / 32767.0);
420 case 2:
421 return lsm9ds0_mfd_get_accel_channel(chan, val, data,
422 6.0 * 9.807 / 32767.0);
423 case 3:
424 return lsm9ds0_mfd_get_accel_channel(chan, val, data,
425 8.0 * 9.807 / 32767.0);
426 case 4:
427 return lsm9ds0_mfd_get_accel_channel(chan, val, data,
428 16.0 * 9.807 / 32767.0);
429 default:
430 return -ENOTSUP;
431 }
432 #elif defined(CONFIG_LSM9DS0_MFD_ACCEL_FULL_SCALE_2)
433 return lsm9ds0_mfd_get_accel_channel(chan, val, data,
434 2.0 * 9.807 / 32767.0);
435 #elif defined(CONFIG_LSM9DS0_MFD_ACCEL_FULL_SCALE_4)
436 return lsm9ds0_mfd_get_accel_channel(chan, val, data,
437 4.0 * 9.807 / 32767.0);
438 #elif defined(CONFIG_LSM9DS0_MFD_ACCEL_FULL_SCALE_6)
439 return lsm9ds0_mfd_get_accel_channel(chan, val, data,
440 6.0 * 9.807 / 32767.0);
441 #elif defined(CONFIG_LSM9DS0_MFD_ACCEL_FULL_SCALE_8)
442 return lsm9ds0_mfd_get_accel_channel(chan, val, data,
443 8.0 * 9.807 / 32767.0);
444 #elif defined(CONFIG_LSM9DS0_MFD_ACCEL_FULL_SCALE_16)
445 return lsm9ds0_mfd_get_accel_channel(chan, val, data,
446 16.0 * 9.807 / 32767.0);
447 #endif
448
449 return 0;
450 }
451 #endif
452
453 #if !defined(LSM9DS0_MFD_MAGN_DISABLED)
lsm9ds0_mfd_convert_magn(struct sensor_value * val,int raw_val,float scale)454 static inline void lsm9ds0_mfd_convert_magn(struct sensor_value *val,
455 int raw_val,
456 float scale)
457 {
458 double dval;
459
460 dval = (double)(raw_val) * (double)scale;
461 val->val1 = (int32_t)dval;
462 val->val2 = ((int32_t)(dval * 1000000)) % 1000000;
463 }
464
lsm9ds0_mfd_get_magn_channel(enum sensor_channel chan,struct sensor_value * val,struct lsm9ds0_mfd_data * data,float scale)465 static inline int lsm9ds0_mfd_get_magn_channel(enum sensor_channel chan,
466 struct sensor_value *val,
467 struct lsm9ds0_mfd_data *data,
468 float scale)
469 {
470 switch (chan) {
471 case SENSOR_CHAN_MAGN_X:
472 lsm9ds0_mfd_convert_magn(val, data->sample_magn_x, scale);
473 break;
474 case SENSOR_CHAN_MAGN_Y:
475 lsm9ds0_mfd_convert_magn(val, data->sample_magn_y, scale);
476 break;
477 case SENSOR_CHAN_MAGN_Z:
478 lsm9ds0_mfd_convert_magn(val, data->sample_magn_z, scale);
479 break;
480 case SENSOR_CHAN_MAGN_XYZ:
481 lsm9ds0_mfd_convert_magn(val, data->sample_magn_x, scale);
482 lsm9ds0_mfd_convert_magn(val + 1, data->sample_magn_y, scale);
483 lsm9ds0_mfd_convert_magn(val + 2, data->sample_magn_z, scale);
484 break;
485 default:
486 return -ENOTSUP;
487 }
488
489 return 0;
490 }
491
lsm9ds0_mfd_get_magn(const struct device * dev,enum sensor_channel chan,struct sensor_value * val)492 static inline int lsm9ds0_mfd_get_magn(const struct device *dev,
493 enum sensor_channel chan,
494 struct sensor_value *val)
495 {
496 struct lsm9ds0_mfd_data *data = dev->data;
497
498 #if defined(CONFIG_LSM9DS0_MFD_MAGN_FULL_SCALE_RUNTIME)
499 switch (data->sample_magn_fs) {
500 case 0:
501 return lsm9ds0_mfd_get_magn_channel(chan, val, data,
502 2.0 / 32767.0);
503 case 1:
504 return lsm9ds0_mfd_get_magn_channel(chan, val, data,
505 4.0 / 32767.0);
506 case 2:
507 return lsm9ds0_mfd_get_magn_channel(chan, val, data,
508 8.0 / 32767.0);
509 case 3:
510 return lsm9ds0_mfd_get_magn_channel(chan, val, data,
511 12.0 / 32767.0);
512 default:
513 return -ENOTSUP;
514 }
515 #elif defined(CONFIG_LSM9DS0_MFD_MAGN_FULL_SCALE_2)
516 return lsm9ds0_mfd_get_magn_channel(chan, val, data, 2.0 / 32767.0);
517 #elif defined(CONFIG_LSM9DS0_MFD_MAGN_FULL_SCALE_4)
518 return lsm9ds0_mfd_get_magn_channel(chan, val, data, 4.0 / 32767.0);
519 #elif defined(CONFIG_LSM9DS0_MFD_MAGN_FULL_SCALE_8)
520 return lsm9ds0_mfd_get_magn_channel(chan, val, data, 8.0 / 32767.0);
521 #elif defined(CONFIG_LSM9DS0_MFD_MAGN_FULL_SCALE_12)
522 return lsm9ds0_mfd_get_magn_channel(chan, val, data, 12.0 / 32767.0);
523 #endif
524
525 return 0;
526 }
527 #endif
528
lsm9ds0_mfd_channel_get(const struct device * dev,enum sensor_channel chan,struct sensor_value * val)529 static int lsm9ds0_mfd_channel_get(const struct device *dev,
530 enum sensor_channel chan,
531 struct sensor_value *val)
532 {
533 #if !defined(LSM9DS0_MFD_TEMP_DISABLED)
534 struct lsm9ds0_mfd_data *data = dev->data;
535 #endif
536
537 switch (chan) {
538 #if !defined(LSM9DS0_MFD_ACCEL_DISABLED)
539 case SENSOR_CHAN_ACCEL_X:
540 case SENSOR_CHAN_ACCEL_Y:
541 case SENSOR_CHAN_ACCEL_Z:
542 case SENSOR_CHAN_ACCEL_XYZ:
543 return lsm9ds0_mfd_get_accel(dev, chan, val);
544 #endif
545 #if !defined(LSM9DS0_MFD_MAGN_DISABLED)
546 case SENSOR_CHAN_MAGN_X:
547 case SENSOR_CHAN_MAGN_Y:
548 case SENSOR_CHAN_MAGN_Z:
549 case SENSOR_CHAN_MAGN_XYZ:
550 return lsm9ds0_mfd_get_magn(dev, chan, val);
551 #endif
552 #if !defined(LSM9DS0_MFD_TEMP_DISABLED)
553 case SENSOR_CHAN_DIE_TEMP:
554 val->val1 = data->sample_temp;
555 val->val2 = 0;
556 return 0;
557 #endif
558 default:
559 return -ENOTSUP;
560 }
561 }
562
563 #if defined(LSM9DS0_MFD_ATTR_SET_ACCEL)
lsm9ds0_mfd_attr_set_accel(const struct device * dev,enum sensor_attribute attr,const struct sensor_value * val)564 static inline int lsm9ds0_mfd_attr_set_accel(const struct device *dev,
565 enum sensor_attribute attr,
566 const struct sensor_value *val)
567 {
568 switch (attr) {
569 #if defined(CONFIG_LSM9DS0_MFD_ACCEL_SAMPLING_RATE_RUNTIME)
570 case SENSOR_ATTR_SAMPLING_FREQUENCY:
571 return lsm9ds0_mfd_accel_set_odr(dev, val);
572 #endif
573 #if defined(CONFIG_LSM9DS0_MFD_ACCEL_FULL_SCALE_RUNTIME)
574 case SENSOR_ATTR_FULL_SCALE:
575 return lsm9ds0_mfd_accel_set_fs(dev, sensor_ms2_to_g(val));
576 #endif
577 default:
578 return -ENOTSUP;
579 }
580
581 return 0;
582 }
583 #endif
584
585 #if defined(LSM9DS0_MFD_ATTR_SET_MAGN)
lsm9ds0_mfd_attr_set_magn(const struct device * dev,enum sensor_attribute attr,const struct sensor_value * val)586 static inline int lsm9ds0_mfd_attr_set_magn(const struct device *dev,
587 enum sensor_attribute attr,
588 const struct sensor_value *val)
589 {
590 switch (attr) {
591 #if defined(CONFIG_LSM9DS0_MFD_MAGN_SAMPLING_RATE_RUNTIME)
592 case SENSOR_ATTR_SAMPLING_FREQUENCY:
593 return lsm9ds0_mfd_magn_set_odr(dev, val);
594 #endif
595 #if defined(CONFIG_LSM9DS0_MFD_MAGN_FULL_SCALE_RUNTIME)
596 case SENSOR_ATTR_FULL_SCALE:
597 return lsm9ds0_mfd_magn_set_fs(dev, val);
598 #endif
599 default:
600 return -ENOTSUP;
601 }
602
603 return 0;
604 }
605 #endif
606
607 #if defined(LSM9DS0_MFD_ATTR_SET)
lsm9ds0_mfd_attr_set(const struct device * dev,enum sensor_channel chan,enum sensor_attribute attr,const struct sensor_value * val)608 static int lsm9ds0_mfd_attr_set(const struct device *dev,
609 enum sensor_channel chan,
610 enum sensor_attribute attr,
611 const struct sensor_value *val)
612 {
613
614 switch (chan) {
615 #if defined(LSM9DS0_MFD_ATTR_SET_ACCEL)
616 case SENSOR_CHAN_ACCEL_X:
617 case SENSOR_CHAN_ACCEL_Y:
618 case SENSOR_CHAN_ACCEL_Z:
619 case SENSOR_CHAN_ACCEL_XYZ:
620 return lsm9ds0_mfd_attr_set_accel(dev, attr, val);
621 #endif
622 #if defined(LSM9DS0_MFD_ATTR_SET_MAGN)
623 case SENSOR_CHAN_MAGN_X:
624 case SENSOR_CHAN_MAGN_Y:
625 case SENSOR_CHAN_MAGN_Z:
626 case SENSOR_CHAN_MAGN_XYZ:
627 return lsm9ds0_mfd_attr_set_magn(dev, attr, val);
628 #endif
629 default:
630 return -ENOTSUP;
631 }
632
633 return 0;
634 }
635 #endif
636
637 static DEVICE_API(sensor, lsm9ds0_mfd_api_funcs) = {
638 .sample_fetch = lsm9ds0_mfd_sample_fetch,
639 .channel_get = lsm9ds0_mfd_channel_get,
640 #if defined(LSM9DS0_MFD_ATTR_SET)
641 .attr_set = lsm9ds0_mfd_attr_set,
642 #endif
643 };
644
lsm9ds0_mfd_init_chip(const struct device * dev)645 static int lsm9ds0_mfd_init_chip(const struct device *dev)
646 {
647 const struct lsm9ds0_mfd_config *config = dev->config;
648 uint8_t chip_id;
649
650 if (lsm9ds0_mfd_reboot_memory(dev) < 0) {
651 LOG_DBG("failed to reset device");
652 return -EIO;
653 }
654
655 if (i2c_reg_read_byte_dt(&config->i2c, LSM9DS0_MFD_REG_WHO_AM_I_XM, &chip_id) < 0) {
656 LOG_DBG("failed reading chip id");
657 return -EIO;
658 }
659
660 if (chip_id != LSM9DS0_MFD_VAL_WHO_AM_I_XM) {
661 LOG_DBG("invalid chip id 0x%x", chip_id);
662 return -EIO;
663 }
664
665 LOG_DBG("chip id 0x%x", chip_id);
666
667 #if !defined(LSM9DS0_MFD_ACCEL_DISABLED)
668 if (i2c_reg_update_byte_dt(&config->i2c, LSM9DS0_MFD_REG_CTRL_REG1_XM,
669 LSM9DS0_MFD_MASK_CTRL_REG1_XM_BDU |
670 LSM9DS0_MFD_MASK_CTRL_REG1_XM_AODR,
671 (1 << LSM9DS0_MFD_SHIFT_CTRL_REG1_XM_BDU) |
672 (LSM9DS0_MFD_ACCEL_DEFAULT_AODR <<
673 LSM9DS0_MFD_SHIFT_CTRL_REG1_XM_AODR))) {
674 LOG_DBG("failed to set AODR and BDU");
675 return -EIO;
676 }
677
678 if (lsm9ds0_mfd_accel_set_fs_raw(dev, LSM9DS0_MFD_ACCEL_DEFAULT_FS)) {
679 LOG_DBG("failed to set accelerometer full-scale");
680 return -EIO;
681 }
682
683 if (i2c_reg_update_byte_dt(&config->i2c, LSM9DS0_MFD_REG_CTRL_REG1_XM,
684 LSM9DS0_MFD_MASK_CTRL_REG1_XM_AXEN |
685 LSM9DS0_MFD_MASK_CTRL_REG1_XM_AYEN |
686 LSM9DS0_MFD_MASK_CTRL_REG1_XM_AZEN,
687 (LSM9DS0_MFD_ACCEL_ENABLE_X <<
688 LSM9DS0_MFD_SHIFT_CTRL_REG1_XM_AXEN) |
689 (LSM9DS0_MFD_ACCEL_ENABLE_Y <<
690 LSM9DS0_MFD_SHIFT_CTRL_REG1_XM_AYEN) |
691 (LSM9DS0_MFD_ACCEL_ENABLE_Z <<
692 LSM9DS0_MFD_SHIFT_CTRL_REG1_XM_AZEN)) < 0) {
693 LOG_DBG("failed to set accelerometer axis enable bits\n");
694 return -EIO;
695 }
696
697 #elif !defined(LSM9DS0_MFD_MAGN_DISABLED)
698 if (i2c_reg_update_byte_dt(&config->i2c, LSM9DS0_MFD_REG_CTRL_REG1_XM,
699 LSM9DS0_MFD_MASK_CTRL_REG1_XM_BDU,
700 1 << LSM9DS0_MFD_SHIFT_CTRL_REG1_XM_BDU) < 0) {
701 LOG_DBG("failed to set BDU\n");
702 return -EIO;
703 }
704 #endif
705
706 #if !defined(LSM9DS0_MFD_MAGN_DISABLED)
707 if (i2c_reg_update_byte_dt(&config->i2c, LSM9DS0_MFD_REG_CTRL_REG7_XM,
708 LSM9DS0_MFD_MASK_CTRL_REG7_XM_MD,
709 (0 << LSM9DS0_MFD_SHIFT_CTRL_REG7_XM_MD)) < 0) {
710 LOG_DBG("failed to power on magnetometer");
711 return -EIO;
712 }
713
714 if (lsm9ds0_mfd_magn_set_odr_raw(dev, LSM9DS0_MFD_MAGN_DEFAULT_M_ODR)) {
715 LOG_DBG("failed to set magnetometer sampling rate");
716 return -EIO;
717 }
718
719 if (lsm9ds0_mfd_magn_set_fs_raw(dev, LSM9DS0_MFD_MAGN_DEFAULT_FS)) {
720 LOG_DBG("failed to set magnetometer full-scale");
721 return -EIO;
722 }
723 #endif
724
725 #if !defined(LSM9DS0_MFD_TEMP_DISABLED)
726 if (i2c_reg_update_byte_dt(&config->i2c, LSM9DS0_MFD_REG_CTRL_REG5_XM,
727 LSM9DS0_MFD_MASK_CTRL_REG5_XM_TEMP_EN,
728 1 << LSM9DS0_MFD_SHIFT_CTRL_REG5_XM_TEMP_EN) < 0) {
729 LOG_DBG("failed to power on temperature sensor");
730 return -EIO;
731 }
732 #endif
733
734 return 0;
735 }
736
lsm9ds0_mfd_init(const struct device * dev)737 int lsm9ds0_mfd_init(const struct device *dev)
738 {
739 const struct lsm9ds0_mfd_config * const config = dev->config;
740
741 if (!device_is_ready(config->i2c.bus)) {
742 LOG_ERR("Bus device is not ready");
743 return -ENODEV;
744 }
745
746 if (lsm9ds0_mfd_init_chip(dev) < 0) {
747 LOG_DBG("failed to initialize chip");
748 return -EIO;
749 }
750
751 return 0;
752 }
753
754 #define LSM9DS0_MFD_DEFINE(inst) \
755 static struct lsm9ds0_mfd_data lsm9ds0_mfd_data_##inst; \
756 \
757 static const struct lsm9ds0_mfd_config lsm9ds0_mfd_config_##inst = { \
758 .i2c = I2C_DT_SPEC_INST_GET(inst), \
759 }; \
760 \
761 SENSOR_DEVICE_DT_INST_DEFINE(inst, lsm9ds0_mfd_init, NULL, \
762 &lsm9ds0_mfd_data_##inst, &lsm9ds0_mfd_config_##inst, POST_KERNEL,\
763 CONFIG_SENSOR_INIT_PRIORITY, &lsm9ds0_mfd_api_funcs); \
764
765 DT_INST_FOREACH_STATUS_OKAY(LSM9DS0_MFD_DEFINE)
766