-
Notifications
You must be signed in to change notification settings - Fork 16
/
Copy pathadafruit_mlx90393.py
executable file
·640 lines (535 loc) · 20.9 KB
/
adafruit_mlx90393.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
# SPDX-FileCopyrightText: 2018 Kevin Townsend for Adafruit Industries
#
# SPDX-License-Identifier: MIT
"""
`adafruit_mlx90393`
====================================================
This is a breakout for the Adafruit MLX90393 magnetometer sensor breakout.
* Author(s): ktown
Implementation Notes
--------------------
**Hardware:**
* Adafruit `MLX90393 Magnetometer Sensor Breakout Board
<https://www.adafruit.com/product/4022>`_ (Product ID: 4022)
**Software and Dependencies:**
* Adafruit CircuitPython firmware for the supported boards:
https://circuitpython.org/downloads
* Adafruit's Bus Device library:
https://github.com/adafruit/Adafruit_CircuitPython_BusDevice
* Adafruit's Register library:
https://github.com/adafruit/Adafruit_CircuitPython_Register
"""
import struct
import time
from adafruit_bus_device.i2c_device import I2CDevice
from micropython import const
try:
from typing import Tuple
from busio import I2C
from circuitpython_typing import ReadableBuffer
except ImportError:
pass
__version__ = "0.0.0+auto.0"
__repo__ = "https://github.com/adafruit/Adafruit_CircuitPython_MLX90393.git"
_CMD_SB = const(0b00010000) # Start burst mode
_CMD_SW = const(0b00100000) # Start wakeup on change mode
_CMD_SM = const(0b00110000) # Start single-measurement mode
_CMD_RM = const(0b01000000) # Read measurement
_CMD_RR = const(0b01010000) # Read register
_CMD_WR = const(0b01100000) # Write register
_CMD_EX = const(0b10000000) # Exit mode
_CMD_HR = const(0b11010000) # Memory recall
_CMD_HS = const(0b11100000) # Memory store
_CMD_RT = const(0b11110000) # Reset
_CMD_NOP = const(0x00) # NOP
_CMD_AXIS_ALL = const(0xE) # X+Y+Z axis bits for commands
_CMD_TEMP = const(0x01) # Temperature bit for commands
_CMD_REG_CONF1 = const(0x00) # Gain
_CMD_REG_CONF2 = const(0x01) # Burst, comm mode, temperature compensation
_CMD_REG_CONF3 = const(0x02) # Oversampling, Filter, Resolution
_CMD_REG_CONF4 = const(0x03) # Sensitivity drift
_CMD_REG_CONF5 = const(0x04) # X-axis Offset Correction
_CMD_REG_CONF6 = const(0x05) # Y-axis Offset Correction
_CMD_REG_CONF7 = const(0x06) # Z-axis Offset Correction
# Gain settings
GAIN_5X = 0x0
GAIN_4X = 0x1
GAIN_3X = 0x2
GAIN_2_5X = 0x3
GAIN_2X = 0x4
GAIN_1_67X = 0x5
GAIN_1_33X = 0x6
GAIN_1X = 0x7
_GAIN_SHIFT = const(4)
# Resolution settings
RESOLUTION_16 = 0x0
RESOLUTION_17 = 0x1
RESOLUTION_18 = 0x2
RESOLUTION_19 = 0x3
# Filter settings
FILTER_0 = 0x0
FILTER_1 = 0x1
FILTER_2 = 0x2
FILTER_3 = 0x3
FILTER_4 = 0x4
FILTER_5 = 0x5
FILTER_6 = 0x6
FILTER_7 = 0x7
# Oversampling settings
OSR_0 = 0x0
OSR_1 = 0x1
OSR_2 = 0x2
OSR_3 = 0x3
# Hall plate spinning rate adjustment
# Must be 0x0C (default) or 0x00
_HALLCONF = const(0x0C)
STATUS_OK = 0x3
# The lookup table below allows you to convert raw sensor data to uT
# using the appropriate (gain/resolution-dependant) lsb-per-uT
# coefficient below. Note that the z axis has a different coefficient
# than the x and y axis.
_LSB_LOOKUP = (
# HALLCONF = 0x0C (default)
(
# 5x gain: res0(xy)(z), res1(xy)(z), res2(xy)(z), res3(xy)(z)
((0.751, 1.210), (1.502, 2.420), (3.004, 4.840), (6.009, 9.680)),
# 4x gain: res0(xy)(z), res1(xy)(z), res2(xy)(z), res3(xy)(z)
((0.601, 0.968), (1.202, 1.936), (2.403, 3.872), (4.840, 7.744)),
# 3x gain: res0(xy)(z), res1(xy)(z), res2(xy)(z), res3(xy)(z)
((0.451, 0.726), (0.901, 1.452), (1.803, 2.904), (3.605, 5.808)),
# 2.5x gain: res0(xy)(z), res1(xy)(z), res2(xy)(z), res3(xy)(z)
((0.376, 0.605), (0.751, 1.210), (1.502, 2.420), (3.004, 4.840)),
# 2x gain: res0(xy)(z), res1(xy)(z), res2(xy)(z), res3(xy)(z)
((0.300, 0.484), (0.601, 0.968), (1.202, 1.936), (2.403, 3.872)),
# 1.667x gain: res0(xy)(z), res1(xy)(z), res2(xy)(z), res3(xy)(z)
((0.250, 0.403), (0.501, 0.807), (1.001, 1.613), (2.003, 3.227)),
# 1.333x gain: res0(xy)(z), res1(xy)(z), res2(xy)(z), res3(xy)(z)
((0.200, 0.323), (0.401, 0.645), (0.801, 1.291), (1.602, 2.581)),
# 1x gain: res0(xy)(z), res1(xy)(z), res2(xy)(z), res3(xy)(z)
((0.150, 0.242), (0.300, 0.484), (0.601, 0.968), (1.202, 1.936)),
),
# HALLCONF = 0x00
(
# GAIN_SEL = 0, 5x gain
((0.787, 1.267), (1.573, 2.534), (3.146, 5.068), (6.292, 10.137)),
# GAIN_SEL = 1, 4x gain
((0.629, 1.014), (1.258, 2.027), (2.517, 4.055), (5.034, 8.109)),
# GAIN_SEL = 2, 3x gain
((0.472, 0.760), (0.944, 1.521), (1.888, 3.041), (3.775, 6.082)),
# GAIN_SEL = 3, 2.5x gain
((0.393, 0.634), (0.787, 1.267), (1.573, 2.534), (3.146, 5.068)),
# GAIN_SEL = 4, 2x gain
((0.315, 0.507), (0.629, 1.014), (1.258, 2.027), (2.517, 4.055)),
# GAIN_SEL = 5, 1.667x gain
((0.262, 0.422), (0.524, 0.845), (1.049, 1.689), (2.097, 3.379)),
# GAIN_SEL = 6, 1.333x gain
((0.210, 0.338), (0.419, 0.676), (0.839, 1.352), (1.678, 2.703)),
# GAIN_SEL = 7, 1x gain
((0.157, 0.253), (0.315, 0.507), (0.629, 1.014), (1.258, 2.027)),
),
)
# Lookup table for conversion times for different filter and
# oversampling settings. Values taken from datasheet.
_TCONV_LOOKUP = (
# OSR = 0 1 2 3
(1.27, 1.84, 3.00, 5.30), # DIG_FILT = 0
(1.46, 2.23, 3.76, 6.84), # DIG_FILT = 1
(1.84, 3.00, 5.30, 9.91), # DIG_FILT = 2
(2.61, 4.53, 8.37, 16.05), # DIG_FILT = 3
(4.15, 7.60, 14.52, 28.34), # DIG_FILT = 4
(7.22, 13.75, 26.80, 52.92), # DIG_FILT = 5
(13.36, 26.04, 51.38, 102.07), # DIG_FILT = 6
(25.65, 50.61, 100.53, 200.37), # DIF_FILT = 7
)
class MLX90393: # pylint: disable=too-many-instance-attributes
"""
Driver for the MLX90393 magnetometer.
:param i2c_bus: The I2C bus the device is connected to
:param int address: The I2C device address. Defaults to :const:`0x0C`
:param int gain: The gain level to apply. Defaults to :const:`GAIN_1X`
:param int resolution: The resolution level to use. Defaults to :const:`RESOLUTION_16`
:param int filt: The filter to use. Defaults to :const:`FILTER_7`
:param int oversampling: The oversampleing setting to use. Defaults to :const:`OSR_3`
:param bool debug: Enable debug output. Defaults to `False`
**Quickstart: Importing and using the device**
Here is an example of using the :class:`MLX90393` class.
First you will need to import the libraries to use the sensor
.. code-block:: python
import board
import adafruit_mlx90393
Once this is done you can define your `board.I2C` object and define your sensor object
.. code-block:: python
i2c = board.I2C() # uses board.SCL and board.SDA
SENSOR = adafruit_mlx90393.MLX90393(i2c)
Now you have access to the :attr:`magnetic` attribute
.. code-block:: python
MX, MY, MZ = SENSOR.magnetic
"""
def __init__( # pylint: disable=too-many-arguments
self,
i2c_bus: I2C,
address: int = 0x0C,
gain: int = GAIN_1X,
resolution: int = RESOLUTION_16,
filt: int = FILTER_7,
oversampling: int = OSR_3,
temperature_compensation: bool = False,
offset: int = 0,
debug: bool = False,
) -> None:
self.i2c_device = I2CDevice(i2c_bus, address)
self._debug = debug
self._status_last = 0
self._res_x = self._res_y = self._res_z = resolution
self._filter = filt
self._osr = oversampling
self._gain_current = gain
self._temperature_compensation = temperature_compensation
self._off_x = self._off_y = self._off_z = offset
# Put the device in a known state to start
self.reset()
# Set resolution to the supplied level
self.resolution_x = self._res_x
self.resolution_y = self._res_y
self.resolution_z = self._res_z
# Set filter to the supplied level
self.filter = self._filter
# Set oversampling to the supplied level
self.oversampling = self._osr
# Set gain to the supplied level
self.gain = self._gain_current
self.temperature_compensation = self._temperature_compensation
# Set offsets to supplied level
self.offset_x = self._off_x
self.offset_y = self._off_y
self.offset_z = self._off_z
def _transceive(self, payload: ReadableBuffer, rxlen: int = 0) -> bytearray:
"""
Writes the specified 'payload' to the sensor
Returns the results of the write attempt.
:param ReadableBuffer payload: The byte array to write to the sensor
:param int rxlen: numbers of bytes to read back. Defaults to :const:`0`
"""
# Read the response (+1 to account for the mandatory status byte!)
data = bytearray(rxlen + 1)
if len(payload) == 1:
# Transceive with repeated start
with self.i2c_device as i2c:
i2c.write_then_readinto(payload, data)
else:
# Write 'value' to the specified register
# TODO: Check this. It's weird that the write is accepted but the read is naked.
with self.i2c_device as i2c:
i2c.write(payload)
while True:
try:
i2c.readinto(data)
if data[0]:
break
except OSError:
pass
# Track status byte
self._status_last = data[0]
# Unpack data (status byte, big-endian 16-bit register value)
if self._debug:
print("\t[{}]".format(time.monotonic()))
print("\t Writing :", [hex(b) for b in payload])
print("\tResponse :", [hex(b) for b in data])
print("\t Status :", hex(data[0]))
return data
@property
def last_status(self) -> int:
"""
The last status byte received from the sensor.
"""
return self._status_last
@property
def gain(self) -> int:
"""
The gain setting for the device.
"""
return self._gain_current
@gain.setter
def gain(self, value: int) -> None:
if value > GAIN_1X or value < GAIN_5X:
raise ValueError("Invalid GAIN setting")
if self._debug:
print("\tSetting gain: {}".format(value))
self._gain_current = value
self._transceive(
bytes(
[
_CMD_WR,
0x00,
self._gain_current << _GAIN_SHIFT | _HALLCONF,
(_CMD_REG_CONF1 & 0x3F) << 2,
]
)
)
@property
def resolution_x(self) -> int:
"""The X axis resolution."""
return self._res_x
@resolution_x.setter
def resolution_x(self, resolution: int) -> None:
self._set_resolution(0, resolution)
self._res_x = resolution
@property
def resolution_y(self) -> int:
"""The Y axis resolution."""
return self._res_y
@resolution_y.setter
def resolution_y(self, resolution: int) -> None:
self._set_resolution(1, resolution)
self._res_y = resolution
@property
def resolution_z(self) -> int:
"""The Z axis resolution."""
return self._res_z
@resolution_z.setter
def resolution_z(self, resolution: int) -> None:
self._set_resolution(2, resolution)
self._res_z = resolution
def _set_resolution(self, axis: int, resolution: int) -> None:
if resolution not in (
RESOLUTION_16,
RESOLUTION_17,
RESOLUTION_18,
RESOLUTION_19,
):
raise ValueError("Incorrect resolution setting.")
shift = (5, 7, 9)[axis]
mask = (0xFF9F, 0xFE7F, 0xF9FF)[axis]
reg = self.read_reg(_CMD_REG_CONF3)
reg &= mask
reg |= (resolution & 0x3) << shift
self.write_reg(_CMD_REG_CONF3, reg)
@property
def filter(self) -> int:
"""The filter level."""
return self._filter
@filter.setter
def filter(self, level: int) -> None:
if level not in range(8):
raise ValueError("Incorrect filter level.")
reg = self.read_reg(_CMD_REG_CONF3)
reg &= 0xFFE3
reg |= (level & 0x7) << 2
self.write_reg(_CMD_REG_CONF3, reg)
self._filter = level
@property
def oversampling(self) -> int:
"""The oversampling level."""
return self._osr
@oversampling.setter
def oversampling(self, level: int) -> None:
if level not in range(4):
raise ValueError("Incorrect oversampling level.")
reg = self.read_reg(_CMD_REG_CONF3)
reg &= 0xFFFC
reg |= level & 0x3
self.write_reg(_CMD_REG_CONF3, reg)
self._osr = level
@property
def temperature_compensation(self) -> bool:
"""The temperature compensation setting"""
return self._temperature_compensation
@temperature_compensation.setter
def temperature_compensation(self, temperature_compensation: bool) -> None:
reg = self.read_reg(_CMD_REG_CONF2)
t_cmp_bit = 10
reg &= ~(1 << t_cmp_bit)
reg |= temperature_compensation << t_cmp_bit
self.write_reg(_CMD_REG_CONF2, reg)
self._temperature_compensation = temperature_compensation
@property
def offset_x(self) -> int:
"""The X axis offset."""
return self._off_x
@offset_x.setter
def offset_x(self, offset: int) -> None:
self._set_offset(0, offset)
self._off_x = offset
@property
def offset_y(self) -> int:
"""The Y axis offset."""
return self._off_y
@offset_y.setter
def offset_y(self, offset: int) -> None:
self._set_offset(1, offset)
self._off_y = offset
@property
def offset_z(self) -> int:
"""The Z axis offset."""
return self._off_z
@offset_z.setter
def offset_z(self, offset: int) -> None:
self._set_offset(2, offset)
self._off_z = offset
def _set_offset(self, axis: int, offset: int) -> None:
if offset < 0x0000 or offset > 0xFFFF:
raise ValueError("Incorrect offset setting.")
if axis == 0:
self.write_reg(_CMD_REG_CONF5, offset)
elif axis == 1:
self.write_reg(_CMD_REG_CONF6, offset)
elif axis == 2:
self.write_reg(_CMD_REG_CONF7, offset)
else:
raise ValueError("Incorrect axis setting.")
def display_status(self) -> None:
"""
Prints out the content of the last status byte in a human-readable
format.
"""
avail = 0
if self._status_last & 0b11 > 0:
avail = 2 * (self._status_last & 0b11) + 2
print("STATUS register = 0x{0:02X}".format(self._status_last))
print("BURST Mode :", (self._status_last & (1 << 7)) > 0)
print("WOC Mode :", (self._status_last & (1 << 6)) > 0)
print("SM Mode :", (self._status_last & (1 << 5)) > 0)
print("Error :", (self._status_last & (1 << 4)) > 0)
print("Single error detection :", (self._status_last & (1 << 3)) > 0)
print("Reset status :", (self._status_last & (1 << 2)) > 0)
print("Response bytes available :", avail)
def read_reg(self, reg: int) -> int:
"""
Gets the current value of the specified register.
:param int reg: The register to read
"""
# Read register
payload = bytes([_CMD_RR, reg << 2])
data = bytearray(3)
with self.i2c_device as i2c:
i2c.write_then_readinto(payload, data)
# Unpack data (status byte, big-endian 16-bit register value)
self._status_last, val = struct.unpack(">BH", data)
if self._debug:
print("\t[{}]".format(time.monotonic()))
print("\t Writing :", [hex(b) for b in payload])
print("\tResponse :", [hex(b) for b in data])
print("\t Status :", hex(data[0]))
return val
def write_reg(self, reg: int, value: int) -> None:
"""
Writes the 16-bit value to the supplied register.
:param int reg: The register to write to
:param int value: The value to write to the register
"""
self._transceive(
bytes(
[
_CMD_WR,
value >> 8, # high byte
value & 0xFF, # low byte
reg << 2, # the register
]
)
)
def reset(self) -> None:
"""
Performs a software reset of the sensor.
"""
self._transceive(bytes([_CMD_EX]))
if self._debug:
print("Resetting sensor")
time.sleep(0.002)
self._transceive(bytes([_CMD_RT]))
# burn a read post reset
try:
self.magnetic
except OSError:
pass
return self._status_last
@property
def read_data(self) -> Tuple[int, int, int]:
"""
Reads a single X/Y/Z sample from the magnetometer.
"""
resolutions = {self.resolution_x, self.resolution_y, self.resolution_z}
valid_tcomp_resolutions = {RESOLUTION_16, RESOLUTION_17}
if self._temperature_compensation and not resolutions.issubset(
valid_tcomp_resolutions
):
resolutions_output = f"""Current Resolutions:
\tresolution_x: {self.resolution_x}
\tresolution_y: {self.resolution_y}
\tresolution_z: {self.resolution_z}"""
raise ValueError(
"All resolutions must be RESOLUTION_16 or RESOLUTION_17"
f" if temperature compensation is enabled.\n {resolutions_output}"
)
# Set conversion delay based on filter and oversampling
delay = _TCONV_LOOKUP[self._filter][self._osr] / 1000 # per datasheet
delay *= 1.1 # plus a little
# Set the device to single measurement mode
self._transceive(bytes([_CMD_SM | _CMD_AXIS_ALL]))
# Insert a delay since we aren't using INTs for DRDY
time.sleep(delay)
# Read the 'XYZ' data
data = self._transceive(bytes([_CMD_RM | _CMD_AXIS_ALL]), 6)
# Unpack status and raw int values
self._status_last = data[0]
m_x = self._unpack_axis_data(self._res_x, data[1:3])
m_y = self._unpack_axis_data(self._res_y, data[3:5])
m_z = self._unpack_axis_data(self._res_z, data[5:7])
# Return the raw int values if requested
return m_x, m_y, m_z
# pylint: disable=no-self-use
def _unpack_axis_data(self, resolution: int, data: ReadableBuffer) -> int:
# see datasheet
if resolution == RESOLUTION_19:
(value,) = struct.unpack(">H", data)
value -= 0x4000
elif resolution == RESOLUTION_18:
(value,) = struct.unpack(">H", data)
value -= 0x8000
elif self.temperature_compensation:
(value,) = struct.unpack(">H", data)
value -= 0x8000
else:
value = struct.unpack(">h", data)[0]
return value
@property
def magnetic(self) -> Tuple[float, float, float]:
"""
The processed magnetometer sensor values.
A 3-tuple of X, Y, Z axis values in microteslas that are signed floats.
"""
x, y, z = self.read_data
# Check for valid HALLCONF value and set _LSB_LOOKUP index
if _HALLCONF == 0x0C:
hallconf_index = 0
elif _HALLCONF == 0x00:
hallconf_index = 1
else:
raise ValueError("Incorrect HALLCONF value, must be '0x0C' or '0x00'.")
# Convert the raw integer values to uT based on gain and resolution
x *= _LSB_LOOKUP[hallconf_index][self._gain_current][self._res_x][0]
y *= _LSB_LOOKUP[hallconf_index][self._gain_current][self._res_y][0]
z *= _LSB_LOOKUP[hallconf_index][self._gain_current][self._res_z][1]
return x, y, z
@property
def temperature(self) -> float:
"""
Reads a single temperature sample from the magnetometer.
Temperature value in Celsius
"""
# Read the temperature reference from register 0x24
treference = self.read_reg(0x24)
# Value taken from maximum time of temperature conversion on the datasheet section 12.
# maximum time for temperature conversion = 1603 us
delay = 0.1
# Set the device to single measurement mode
self._transceive(bytes([_CMD_SM | _CMD_TEMP]))
time.sleep(delay)
# Read the 'temp' data
data = self._transceive(bytes([_CMD_RM | _CMD_TEMP]), 2)
# Unpack status and raw int values
self._status_last = data[0]
# from https://www.melexis.com/-/media/files/documents/
# application-notes/mlx90393-temperature-compensation-application-note-melexis.pdf
tvalue = struct.unpack(">H", data[1:3])[0]
# See previous link for conversion formula
return 35 + ((tvalue - treference) / 45.2)