From a6b27860eb6c499b50c0390c315340aa745c42f4 Mon Sep 17 00:00:00 2001 From: Nicola <62206011+n-elia@users.noreply.github.com> Date: Tue, 24 May 2022 14:57:04 +0200 Subject: [PATCH] Improves error handling, removes default I2C configuration. Updates readme readability, improves example script. Bumps to v0.3.5. --- README.md | 209 ++++++++++++++++++++++--------------------- example/main.py | 42 ++++++--- max30102/max30102.py | 54 ++++++----- setup.py | 2 +- 4 files changed, 162 insertions(+), 145 deletions(-) diff --git a/README.md b/README.md index 58ee0bd..caf95ad 100644 --- a/README.md +++ b/README.md @@ -1,52 +1,28 @@ [![Upload Python Package](https://github.com/n-elia/MAX30102-MicroPython-driver/actions/workflows/python-publish.yml/badge.svg)](https://github.com/n-elia/MAX30102-MicroPython-driver/actions/workflows/python-publish.yml) [![PyPI version](https://badge.fury.io/py/micropython-max30102.svg)](https://badge.fury.io/py/micropython-max30102) -![PyPI - Downloads](https://img.shields.io/pypi/dm/micropython-max30102?color=blue&label=upip%20installations) + + # Maxim MAX30102 MicroPython driver A port of the SparkFun driver for Maxim MAX30102 sensor to MicroPython. It _should_ work for MAX30105, too. If you have the chance to test this library with a MAX30105, please leave your feedback in the Discussions section. -## Aknowledgements - -This work is a lot based on: - -- [SparkFun MAX3010x Sensor Library](https://github.com/sparkfun/SparkFun_MAX3010x_Sensor_Library "GitHub | SparkFun MAX3010x Sensor Library") - - Written by **Peter Jansen** and **Nathan Seidle** (SparkFun) - This is a library written for the Maxim MAX30105 Optical Smoke Detector - It should also work with the MAX30102. However, the MAX30102 does not have a Green LED. - These sensors use I2C to communicate, as well as a single (optional) - interrupt line that is not currently supported in this driver. - Written by Peter Jansen and Nathan Seidle (SparkFun) - BSD license, all text above must be included in any redistribution. - -- [esp32-micropython](https://github.com/kandizzy/esp32-micropython/blob/master/PPG/ppg/MAX30105.py "GitHub | esp32-micropython") - - A port of the library to MicroPython by **kandizzy** ## Disclaimer This work is not intended to be used in professional environments, and there are no guarantees on its functionalities. Please do not rely on it for medical purposes or professional usage. -## Repository organisation -- Driver: `./max30102` -- Example: `./example` +## Usage -## Changelog -- v0.3.4 - - The package has been refactored to be compliant to PEP standards. -- v0.3.3 - - Made a PyPi package. Now you can install this package with upip. - - Tested with Raspberry Pi Pico and non-genuine sensors. -- v0.3 - - Tested with TinyPico board (based on ESP32-D4) and genuine Maxim MAX30102 sensor. +Driver usage is quite straightforward. You just need to import the library, and to set up a `SoftI2C` instance. + +A full example is provided in `/example` directory. -## How to import the library and run the example -Important note: the library will load the default TinyPico ESP32 board I2C configuration (SDA Pin 21, SCL Pin 22, 400kHz speed). If you're using a different board, follow the instructions given below, in *Setup and configuration* section. +### 1 - Including this library into your project -### Including this library into your project (**network-enabled MicroPython ports**) -To include the library into a network-enabled MicroPython project, it's sufficient to install the package: +#### 1a - **network-enabled MicroPython ports** +To include the library into a network-enabled MicroPython project, it's sufficient to install the package using `upip`: ```python import upip @@ -57,68 +33,58 @@ Make sure that your firmware runs these lines **after** an Internet connection h To run the example in `./example` folder, please set your WiFi credentials in `boot.py` and then upload `./example` content into your microcontroller. If you prefer, you can perform a manual install as explained below. -### Including this library into your project (**manual way**) +#### 1b - **manual way** (no Internet access required) -To directly include the library into a MicroPython project, it's sufficient to copy `max30102/circular_buffer.py` and `max30102/max30102.py` next to your `main.py` file, or into a `lib` directory. Then, import the constructor as follows: +To directly include the library into a MicroPython project, it's sufficient to copy `max30102/circular_buffer.py` and `max30102/max30102.py` next to your `main.py` file, into a `lib` directory. Then, import the constructor as follows: ```python from max30102 import MAX30102 ``` -To run the example in `./example` folder, copy `max30102/circular_buffer.py` and `max30102/max30102.py` into the `./example/lib` directory. Then, upload the `./example` directory content into your microcontroller. -The content of the board root should appear as follows: -``` -/ -└── root.py -└── main.py -└── lib - └── circular_buffer.py - └── max30102.py -``` -After the upload, press the reset button of your board are you're good to go. +To run the example in `./example` folder, copy `max30102/circular_buffer.py` and `max30102/max30102.py` into the `./example/lib` directory. Then, upload the `./example` directory content into your microcontroller. After the upload, press the reset button of your board are you're good to go. -### Setup and configuration -#### I2C pins -When creating a sensor instance, if you leave the arguments empty, the library will load the default TinyPico ESP32 board I2C configuration (SDA Pin 21, SCL Pin 22, 400kHz speed). +### 2 - I2C setup and sensor configuration -If you have a different board, you can set different I2C pins as shown in the following example: +#### I2C connection -```python -# Default config (ESP32): -sensor = MAX30102() +Create a `SoftI2C` instance as in the following example: -# Alternative: +```python from machine import SoftI2C, Pin + +my_SDA_pin = 21 # I2C SDA pin number here! +my_SCL_pin = 22 # I2C SCL pin number here! +my_i2c_freq = 400000 # I2C frequency (Hz) here! + i2c = SoftI2C(sda=Pin(my_SDA_pin), scl=Pin(my_SCL_pin), - freq=100000) -sensor = MAX30102(i2cHexAddress = 0x57, i2c = i2cInstance) + freq=my_i2c_freq) + +sensor = MAX30102(i2c=i2c) ``` +The I2C pin numbers depend on the board that you are using, and how you wired the sensor to it. + #### Sensor setup -Then, the sensor has to be setup. The library provides a method to setup the sensor at once. Leaving the arguments empty, makes the library load their default values. -> Default configuration values: -> -> Led mode: 2 (RED + IR) -> -> ADC range: 16384 -> -> Sample rate: 400 Hz -> -> Led power: maximum (50.0mA - Presence detection of ~12 inch) -> -> Averaged samples: 8 +The library provides a method to setup the sensor at once. Leaving the arguments empty, makes the library load the default values. + +> **Default configuration values:** > -> pulse width: 411 +> _Led mode_: 2 (RED + IR) +> _ADC range_: 16384 +> _Sample rate_: 400 Hz +> _Led power_: maximum (50.0mA - Presence detection of ~12 inch) +> _Averaged samples_: 8 +> _Pulse width_: 411 ```python # Setup with default values sensor.setup_sensor() # Alternative example: -setup_sensor(self, LED_MODE=2, ADC_RANGE=16384, SAMPLE_RATE=400) +setup_sensor(self, led_mode=2, adc_range=16384, sample_rate=400) ``` The library provides the methods to change the configuration parameters one by one, too. Remember that the `setup_sensor()` method has still to be called before modifying the single parameters. @@ -165,12 +131,14 @@ LED_POWER = MAX30105_PULSEAMP_MEDIUM sensor.set_active_leds_amplitude(LED_POWER) ``` -#### Data acquisition +### 3 - Data acquisition The sensor will store all the readings into a FIFO register (FIFO_DATA). Based on the number of active LEDs and other configuration paramenters, the sensor instance will read data from that register, putting it into the_storage_. The_storage_ is a circular buffer, that can be read using the provided methods. The `check()` method polls the sensor to check if new samples are available in the FIFO queue. If data is available, it will be read and put into the _storage_. We can access those samples using the provided methods such as `popRedFromStorage()`. +#### Read data from sensor + As a consequence, this is an example on how the library can be used to read data from the sensor: ```python @@ -190,7 +158,7 @@ while (True): print(red_sample, ",", ir_sample) ``` -#### Data acquisition rate +#### Notes on data acquisition rate Considering the sensor configuration, two main parameters will affect the data throughput of the sensor itself: @@ -213,27 +181,27 @@ However, there are some limitations on sensor side and on micropocessor side tha # (Assuming that the sensor instance has been already set-up) from utime import ticks_diff, ticks_ms -# Starting time of the acquisition -t_start = ticks_ms() -# Number of samples that has been collected -samples_n = 0 - -while (True): - sensor.check() - - if (sensor.available()): - # Access the storage FIFO and gather the readings (integers) - red_sample = sensor.pop_red_from_storage() - ir_sample = sensor.pop_ir_from_storage() - - # We can compute the real frequency at which we receive data - if (compute_frequency): - samples_n = samples_n + 1 - if (ticks_diff(ticks_ms(), t_start) > 999): - f_HZ = samples_n / 1 - samples_n = 0 - t_start = ticks_ms() - print("Acquisition frequency = ", f_HZ) +t_start = ticks_us() # Starting time of the acquisition +samples_n = 0 # Number of samples that have been collected + +while True: + sensor.check() + if sensor.available(): + red_reading = sensor.pop_red_from_storage() + ir_reading = sensor.pop_ir_from_storage() + + # Print the acquired data (so that it can be plotted with a Serial Plotter) + print(red_reading, ",", ir_reading) + + # Compute the real frequency at which we receive data (with microsecond precision) + if compute_frequency: + if ticks_diff(ticks_us(), t_start) >= 999999: + f_HZ = samples_n + samples_n = 0 + print("acquisition frequency = ", f_HZ) + t_start = ticks_us() + else: + samples_n = samples_n + 1 ``` #### Die temperature reading @@ -248,23 +216,62 @@ print("Die temperature: ", temperature_C, "°C") Note: as stated in the [datasheet](https://datasheets.maximintegrated.com/en/ds/MAX30102.pdf), the internal die temperature sensor is intended for calibrating the temperature dependence of the SpO2 subsystem. It has an inherent resolution of 0.0625°C, but be aware that the accuracy is ±1°C. -#### Realtime plot over Serial -The example proposed in this repository ([main.py](../src/main.py)) contains a print statement in this form: `print(red_reading, ",", IR_reading)`. If you open the Arduino IDE, and connect your board to it, then you will be able to open the *serial plotter* (Ctrl+Maiusc+L) and see a real-time plot of your readings (need help? take a look [here](https://learn.sparkfun.com/tutorials/max30105-particle-and-pulse-ox-sensor-hookup-guide/all)). +## Changelog + +- v0.3.5 + - A `SoftI2C` instance is now required by the constructor. + - The constructor now raises `RuntimeError` when the sensor is not found on I2C bus. + - The example has been updated to intercept the errors thrown by the constructor. + - The example has been updated to estimate real acquisition frequency with a precision of 1 microsecond. + - The readme has been re-organized to improve readability. +- v0.3.4 + - The package has been refactored to be compliant to PEP standards. +- v0.3.3 + - Made a PyPi package. Now you can install this package with upip. + - Tested with Raspberry Pi Pico and non-genuine sensors. +- v0.3 + - Tested with TinyPico board (based on ESP32-D4) and genuine Maxim MAX30102 sensor. + + +## Aknowledgements + +This work is a lot based on: + +- [SparkFun MAX3010x Sensor Library](https://github.com/sparkfun/SparkFun_MAX3010x_Sensor_Library "GitHub | SparkFun MAX3010x Sensor Library") + + Written by **Peter Jansen** and **Nathan Seidle** (SparkFun) + This is a library written for the Maxim MAX30105 Optical Smoke Detector + It should also work with the MAX30102. However, the MAX30102 does not have a Green LED. + These sensors use I2C to communicate, as well as a single (optional) + interrupt line that is not currently supported in this driver. + Written by Peter Jansen and Nathan Seidle (SparkFun) + BSD license, all text above must be included in any redistribution. + +- [esp32-micropython](https://github.com/kandizzy/esp32-micropython/blob/master/PPG/ppg/MAX30105.py "GitHub | esp32-micropython") + + A port of the library to MicroPython by **kandizzy** + + +## Other useful things + +### Realtime plot over Serial + +The example proposed in this repository ([main.py](./example/main.py)) contains a print statement in a CSV-like format: `print(red_reading, ",", IR_reading)`. If you open Arduino IDE and connect your board, then you will be able to open the *serial plotter* (Ctrl+Maiusc+L) and see a real-time plot of your readings (need some help? take a look [here](https://learn.sparkfun.com/tutorials/max30105-particle-and-pulse-ox-sensor-hookup-guide/all)). + For instance, this is an example of my heartbeat taken on the index finger: ![Serial Plotter picture](./img/arduino-IDE-serial-plotter-heartbeat.png "Serial Plotter picture") ### Tested platforms -The example works well on TinyPico (ESP32-D4 board) running 'tinypico-20210418-v1.15.bin' MicroPython firmware, connected to a genuine Maxim 30102 breakout board ([MAXREFDES117#](https://www.maximintegrated.com/en/design/reference-design-center/system-board/6300.html)). +The library has been tested on TinyPico (ESP32-D4 board) running 'tinypico-20210418-v1.15.bin' MicroPython firmware, connected to a genuine Maxim 30102 breakout board ([MAXREFDES117#](https://www.maximintegrated.com/en/design/reference-design-center/system-board/6300.html)). Tested ([thanks to ebolisa](https://github.com/n-elia/MAX30102-MicroPython-driver/issues/4)) and working on Raspberry Pi Pico + non-Maxim breakout board. -### Other things that it's worth mentioning - -- There is an issue involving chinese clones of the Maxim MAX30102: some of them appear to have the red and IR registers inverted (or maybe the LEDs swapped) (see [here](https://github.com/aromring/MAX30102_by_RF/issues/13)). You can easily check if your sensor is inverted by putting it in LED mode 1: only the red LED should work. If you see the IR LED (use your phone camera to check), then you have to collect IR samples as red ones and viceversa. +### Sensor clones -- The sensor has to be wired to the I2C pins of your board +There is an issue involving chinese clones of the Maxim MAX30102: some of them appear to have the red and IR registers inverted (or maybe the LEDs swapped) (see [here](https://github.com/aromring/MAX30102_by_RF/issues/13)). You can easily check if your sensor is inverted by putting it in LED mode 1: only the red LED should work. If you see the IR LED (use your phone camera to check), then you have to collect IR samples as red ones and viceversa. -- If you're looking for algorithms for extracting heartrate and SPO2 from your RAW data, take a look [here](https://github.com/aromring/MAX30102_by_RF) and [here](https://github.com/kandizzy/esp32-micropython/tree/master/PPG) +### Heartrate and SPO2 estimation +If you're looking for algorithms for extracting heartrate and SPO2 from your RAW data, take a look [here](https://github.com/aromring/MAX30102_by_RF) and [here](https://github.com/kandizzy/esp32-micropython/tree/master/PPG) diff --git a/example/main.py b/example/main.py index 903bc90..748912a 100644 --- a/example/main.py +++ b/example/main.py @@ -1,15 +1,28 @@ # main.py -from machine import sleep -from utime import ticks_diff, ticks_ms +from machine import sleep, SoftI2C, Pin +from utime import ticks_diff, ticks_us from max30102 import MAX30102 if __name__ == '__main__': - # Sensor instance. If left empty, loads default ESP32 I2C configuration - sensor = MAX30102() - # Alternatively (for other boards): - # sensor = MAX30102(i2cHexAddress = 0x57) - # sensor = MAX30102(i2cHexAddress = 0x57, i2c = i2cInstance) + # I2C software instance + i2c = SoftI2C(sda=Pin(22), # Here, use your I2C SDA pin + scl=Pin(21), # Here, use your I2C SCL pin + freq=400000) # Fast: 400kHz, slow: 100kHz + + # Examples of working I2C configurations: + # Board | SDA pin | SCL pin + # ------------------------------------------ + # ESP32 D1 Mini | 22 | 21 + # TinyPico ESP32 | 21 | 22 + # Raspberry Pi Pico | 16 | 17 + + # Sensor instance + try: + sensor = MAX30102(i2c=i2c) # An I2C instance is required + except RuntimeError as err: + print("Error occurred while sensor initialization:", err) + raise SystemExit(err) # The default sensor configuration is: # Led mode: 2 (RED + IR) @@ -42,8 +55,8 @@ print("Starting data acquisition from RED & IR registers...", '\n') sleep(1) - t_start = ticks_ms() # Starting time of the acquisition - samples_n = 0 # Number of samples that has been collected + t_start = ticks_us() # Starting time of the acquisition + samples_n = 0 # Number of samples that have been collected while True: # The check() method has to be continuously polled, to check if @@ -60,11 +73,12 @@ # Print the acquired data (so that it can be plotted with a Serial Plotter) print(red_reading, ",", ir_reading) - # We can compute the real frequency at which we receive data + # Compute the real frequency at which we receive data if compute_frequency: - samples_n = samples_n + 1 - if ticks_diff(ticks_ms(), t_start) > 999: - f_HZ = samples_n / 1 + if ticks_diff(ticks_us(), t_start) >= 999999: + f_HZ = samples_n samples_n = 0 - t_start = ticks_ms() print("acquisition frequency = ", f_HZ) + t_start = ticks_us() + else: + samples_n = samples_n + 1 \ No newline at end of file diff --git a/max30102/max30102.py b/max30102/max30102.py index 466d6ab..0eac627 100644 --- a/max30102/max30102.py +++ b/max30102/max30102.py @@ -11,23 +11,18 @@ # - https://github.com/kandizzy/esp32-micropython/blob/master/PPG/ppg/MAX30105.py # A port of the library to MicroPython by kandizzy # -# With this driver, I want to give almost full access to Maxim MAX30102 sensor -# functionalities. -# This code is being tested on TinyPico Board with Maxim genuine sensors. +# This driver aims at giving almost full access to Maxim MAX30102 functionalities. # n-elia -from machine import Pin, SoftI2C +import uerrno +from machine import SoftI2C from ustruct import unpack from utime import sleep_ms, ticks_diff, ticks_ms from circular_buffer import CircularBuffer -# These I2C default settings work for TinyPico (ESP32-based board) -MAX3010X_I2C_ADDRESS = 0x57 -I2C_SPEED_FAST = 400000 # 400kHz speed -I2C_SPEED_NORMAL = 100000 # 100kHz speed -I2C_DEF_SDA_PIN = 21 -I2C_DEF_SCL_PIN = 22 +# I2C address (7-bit address) +MAX3010X_I2C_ADDRESS = 0x57 # Right-shift of 0xAE, 0xAF # Status Registers MAX30105_INT_STAT_1 = 0x00 @@ -175,12 +170,10 @@ def __init__(self): # Sensor class class MAX30102(object): def __init__(self, - i2cHexAddress=MAX3010X_I2C_ADDRESS, - i2c=SoftI2C(sda=Pin(I2C_DEF_SDA_PIN), - scl=Pin(I2C_DEF_SCL_PIN), - freq=I2C_SPEED_FAST) + i2c: SoftI2C, + i2c_hex_address=MAX3010X_I2C_ADDRESS, ): - self._address = i2cHexAddress + self._address = i2c_hex_address self._i2c = i2c self._active_leds = None self._pulse_width = None @@ -196,42 +189,45 @@ def __init__(self, try: self._i2c.readfrom(self._address, 1) except OSError as error: - raise SystemExit(error) + if error.errno == uerrno.ENODEV: + raise RuntimeError("Sensor not found on I2C bus.") + else: + raise RuntimeError(f"Error while reading from I2C bus: OSError code {error.errno}") if not (self.check_part_id()): - raise SystemExit() + raise RuntimeError("I2C device ID not corresponding to MAX30102 or MAX30105") # Sensor setup method - def setup_sensor(self, LED_MODE=2, ADC_RANGE=16384, SAMPLE_RATE=400, - LED_POWER=MAX30105_PULSE_AMP_HIGH, SAMPLE_AVG=8, - PULSE_WIDTH=411): + def setup_sensor(self, led_mode=2, adc_range=16384, sample_rate=400, + led_power=MAX30105_PULSE_AMP_HIGH, sample_avg=8, + pulse_width=411): # Reset the sensor's registers from previous configurations self.soft_reset() # Set the number of samples to be averaged by the chip to 8 - self.set_fifo_average(SAMPLE_AVG) + self.set_fifo_average(sample_avg) # Allow FIFO queues to wrap/roll over self.enable_fifo_rollover() # Set the LED mode to the default value of 2 (RED + IR) # Note: the 3rd mode is available only with MAX30105 - self.set_led_mode(LED_MODE) + self.set_led_mode(led_mode) # Set the ADC range to default value of 16384 - self.set_adc_range(ADC_RANGE) + self.set_adc_range(adc_range) # Set the sample rate to the default value of 400 - self.set_sample_rate(SAMPLE_RATE) + self.set_sample_rate(sample_rate) # Set the Pulse Width to the default value of 411 - self.set_pulse_width(PULSE_WIDTH) + self.set_pulse_width(pulse_width) # Set the LED brightness to the default value of 'low' - self.set_pulse_amplitude_red(LED_POWER) - self.set_pulse_amplitude_it(LED_POWER) - self.set_pulse_amplitude_green(LED_POWER) - self.set_pulse_amplitude_proximity(LED_POWER) + self.set_pulse_amplitude_red(led_power) + self.set_pulse_amplitude_it(led_power) + self.set_pulse_amplitude_green(led_power) + self.set_pulse_amplitude_proximity(led_power) # Clears the FIFO self.clear_fifo() diff --git a/setup.py b/setup.py index 10d02d1..5ad6d61 100644 --- a/setup.py +++ b/setup.py @@ -3,7 +3,7 @@ setup( name="micropython-max30102", - version="0.3.4", + version="0.3.5", description="MAX30102 driver for micropython.", long_description=open("README.md").read(), long_description_content_type='text/markdown',