2023-07-27 13:05:48 +02:00
|
|
|
|
|
|
|
[![Arduino CI](https://github.com/RobTillaart/MTP40F/workflows/Arduino%20CI/badge.svg)](https://github.com/marketplace/actions/arduino_ci)
|
|
|
|
[![Arduino-lint](https://github.com/RobTillaart/MTP40F/actions/workflows/arduino-lint.yml/badge.svg)](https://github.com/RobTillaart/MTP40F/actions/workflows/arduino-lint.yml)
|
2023-11-14 16:44:34 +01:00
|
|
|
[![JSON check](https://github.com/RobTillaart/MTP40F/actions/workflows/jsoncheck.yml/badge.svg)](https://github.com/RobTillaart/MTP40F/actions/workflows/jsoncheck.yml)
|
|
|
|
[![GitHub issues](https://img.shields.io/github/issues/RobTillaart/MTP40F.svg)](https://github.com/RobTillaart/MTP40F/issues)
|
|
|
|
|
2023-07-27 13:05:48 +02:00
|
|
|
[![License: MIT](https://img.shields.io/badge/license-MIT-green.svg)](https://github.com/RobTillaart/MTP40F/blob/master/LICENSE)
|
|
|
|
[![GitHub release](https://img.shields.io/github/release/RobTillaart/MTP40F.svg?maxAge=3600)](https://github.com/RobTillaart/MTP40F/releases)
|
2023-11-14 16:44:34 +01:00
|
|
|
[![PlatformIO Registry](https://badges.registry.platformio.org/packages/robtillaart/library/MTP40F.svg)](https://registry.platformio.org/libraries/robtillaart/MTP40F)
|
2023-07-27 13:05:48 +02:00
|
|
|
|
|
|
|
|
|
|
|
# MTP40F
|
|
|
|
|
|
|
|
Arduino library for MTP40F CO2 sensor.
|
|
|
|
|
|
|
|
|
|
|
|
## Description
|
|
|
|
|
|
|
|
The library for the MTP40F CO2 sensor is **experimental** as not all functionality is tested.
|
|
|
|
It is based upon my https://github.com/RobTillaart/MTP40C library.
|
|
|
|
|
|
|
|
The MTP40F sensor is an NDIR (Non Dispersive InfraRed) CO2 sensor.
|
2023-08-04 10:16:10 +02:00
|
|
|
It returns the CO2 level in PPM = parts per million, and in normal
|
|
|
|
outside air it should be between 400 - 420.
|
2023-07-27 13:05:48 +02:00
|
|
|
|
|
|
|
The sensor communicates over a 9600 baud serial (TTL) interface with a microprocessor or PC.
|
|
|
|
This implies that calls which can take up to 25 bytes can take as much as about 20 milliseconds.
|
|
|
|
|
|
|
|
On the other hand this low baud rate implies it will work over relative long distances.
|
|
|
|
This signal quality over longer distances is not investigated.
|
|
|
|
|
|
|
|
The MTP40F has more interface options, I2C, PWM and ALARM.
|
|
|
|
This library does not support these other interfaces for now.
|
|
|
|
However minimal examples are added to have a starter but these
|
|
|
|
need to be tested if and how well these work.
|
|
|
|
|
|
|
|
|
2023-08-04 10:16:10 +02:00
|
|
|
#### Warnings
|
|
|
|
|
|
|
|
During tests with an UNO the communication over Software Serial did
|
|
|
|
fail a few times.
|
|
|
|
Therefore it is important to **always check return values**
|
|
|
|
to make your project more robust.
|
|
|
|
|
|
|
|
During tests it became clear that the sensor needs time to process
|
|
|
|
commands e.g. **setSelfCalibration()**.
|
|
|
|
|
|
|
|
The CRC of the sensor responses are not verified by the library (yet).
|
|
|
|
|
|
|
|
|
2023-11-14 16:44:34 +01:00
|
|
|
#### Hardware interface MTP40-F
|
2023-07-27 13:05:48 +02:00
|
|
|
|
|
|
|
Has TTL level RS232, I2C and PWM IO.
|
|
|
|
|
|
|
|
```
|
2023-08-04 10:16:10 +02:00
|
|
|
TOPVIEW MTP40-F
|
2023-07-27 13:05:48 +02:00
|
|
|
+-------------+
|
|
|
|
| |
|
|
|
|
VCC (3v3 out) 5 --| |-- 1 Vin
|
2023-11-14 16:44:34 +01:00
|
|
|
TX / SDA 6 --| |-- 2 GND
|
|
|
|
RX / SCL 7 --| |-- 3 ALARM
|
|
|
|
R/T 8 --| |-- 4 PWM / I2C
|
2023-07-27 13:05:48 +02:00
|
|
|
GND 9 --| |
|
|
|
|
| |
|
|
|
|
+-------------+
|
|
|
|
```
|
|
|
|
|
2023-11-14 16:44:34 +01:00
|
|
|
| Pin | Name | Description |
|
|
|
|
|:-----:|:-----------|:-----------------------------|
|
|
|
|
| 1 | Vin | 4.2V - 5.5V |
|
|
|
|
| 2 | GND | idem |
|
|
|
|
| 3 | ALARM | HIGH above 1000 PPM, LOW below 800 PPM (hysteresis) |
|
|
|
|
| 4 | PWM/I2C | PWM out |
|
|
|
|
| 5 | VCC_O | 3V3 out for serial |
|
|
|
|
| 6 | TX / SDA | Transmit 9600 baud or SDA |
|
|
|
|
| 7 | RX / SCL | Receive 9600 baud or SCL |
|
|
|
|
| 8 | R/T | select Serial or I2C |
|
|
|
|
| 9 | GND | idem |
|
2023-07-27 13:05:48 +02:00
|
|
|
|
|
|
|
|
|
|
|
#### Links
|
|
|
|
|
|
|
|
- https://www.co2.earth/ - current outdoor CO2 level can be used for calibrating.
|
|
|
|
- https://keelingcurve.ucsd.edu/ - historical outdoor CO2 level.
|
|
|
|
- https://github.com/RobTillaart/MTP40C
|
2023-08-04 10:16:10 +02:00
|
|
|
- https://github.com/RobTillaart/MHZCO2
|
|
|
|
- https://github.com/RobTillaart/Cozir
|
2023-07-27 13:05:48 +02:00
|
|
|
|
|
|
|
|
|
|
|
## Interface
|
|
|
|
|
|
|
|
```cpp
|
|
|
|
#include "MTP40F.h"
|
|
|
|
```
|
|
|
|
|
2023-08-04 10:16:10 +02:00
|
|
|
#### Constructor
|
2023-07-27 13:05:48 +02:00
|
|
|
|
2023-08-04 10:16:10 +02:00
|
|
|
- **MTP40F(Stream \* str)** constructor. Should get a Serial port as parameter
|
|
|
|
e.g. \&Serial, \&Serial1 or a software Serial port.
|
|
|
|
That Serial port must connect to the sensor, check datasheet for pins.
|
|
|
|
- **bool begin()** initialize the internal settings for the device.
|
2023-07-27 13:05:48 +02:00
|
|
|
- **uint8_t getType()** returns type, see below.
|
|
|
|
Return 255 for the MTP40 base class.
|
|
|
|
|
|
|
|
| Type | Model | Notes |
|
|
|
|
|:------:|:--------:|:--------:|
|
|
|
|
| 2 | MTP40C | for ref only
|
|
|
|
| 3 | MTP40D | for ref only
|
|
|
|
| 5 | MTP40F |
|
|
|
|
| 255 | MTP40 | for ref only
|
|
|
|
|
|
|
|
|
|
|
|
#### CO2 Measurement
|
|
|
|
|
2023-08-04 10:16:10 +02:00
|
|
|
- **uint16_t getGasConcentration()** returns the CO2 concentration in PPM.
|
2023-07-27 13:05:48 +02:00
|
|
|
The function returns **MTP40_INVALID_GAS_LEVEL** if the request fails.
|
2023-08-04 10:16:10 +02:00
|
|
|
- **void suppressError(bool se)** sets or clears a flag that replaces the
|
|
|
|
error value with the last read value if the request fails.
|
2023-07-27 13:05:48 +02:00
|
|
|
This is useful when plotting the values and one do not want a sudden spike.
|
|
|
|
One can still check **lastError()** to see if the value was OK.
|
|
|
|
- **bool getSuppressError()** gets the value of the suppress flag.
|
|
|
|
- **int lastError()** returns last error set by **getGasConcentration()**
|
|
|
|
or by **getAirPressureReference()**
|
|
|
|
Reading resets internal error to MTP40F_OK;
|
|
|
|
|
|
|
|
|
2023-08-04 10:16:10 +02:00
|
|
|
#### Timeout communication
|
2023-07-27 13:05:48 +02:00
|
|
|
|
|
|
|
The library can set a maximum timeout in the communication with the sensor.
|
2023-08-04 10:16:10 +02:00
|
|
|
Normally this is not needed to set as the default of 100 milliseconds is
|
|
|
|
long enough for even the longest command.
|
|
|
|
This timeout is needed if the sensor did not process the command correctly,
|
|
|
|
preventing the host to wait indefinitely.
|
|
|
|
- **void setTimeout(uint32_t timeOut = 100)** sets the timeout.
|
2023-07-27 13:05:48 +02:00
|
|
|
If no parameter is given a default timeout of 100 milliseconds is set.
|
|
|
|
- **uint32_t getTimeout()** get the value set above or the default.
|
|
|
|
Value returned is time in milliseconds.
|
|
|
|
|
|
|
|
|
|
|
|
## Calibration
|
|
|
|
|
2023-08-04 10:16:10 +02:00
|
|
|
Please read datasheet before using these functions to understand
|
|
|
|
the process of calibration.
|
2023-07-27 13:05:48 +02:00
|
|
|
|
2023-08-04 10:16:10 +02:00
|
|
|
Note the outdoor calibration CO2 level differs per day and one should
|
|
|
|
check a local airport or weather station for a good reference.
|
2023-07-27 13:05:48 +02:00
|
|
|
|
|
|
|
The University of San Diego keeps track of CO2 for a long time now.
|
|
|
|
See - https://keelingcurve.ucsd.edu/
|
|
|
|
|
|
|
|
|
|
|
|
#### Air pressure calibration
|
|
|
|
|
|
|
|
- **float getAirPressureReference()** returns the air pressure reference from the device.
|
|
|
|
Returns **MTP40F_INVALID_AIR_PRESSURE** in case request fails.
|
2023-08-04 10:16:10 +02:00
|
|
|
Default is 1013 (note no decimals) == 1 ATM.
|
|
|
|
- **bool setAirPressureReference(float apr = 1013)** to calibrate the air pressure.
|
|
|
|
Default value = 1013 hPa (1 Atm).
|
2023-07-27 13:05:48 +02:00
|
|
|
One can calibrate the sensor with an external device.
|
2023-08-04 10:16:10 +02:00
|
|
|
Value for air pressure should normally between 700 and 1100.
|
|
|
|
The function returns false if the parameter is out of range or if the request fails.
|
2023-07-27 13:05:48 +02:00
|
|
|
|
2023-08-04 10:16:10 +02:00
|
|
|
For pressure conversion - https://github.com/RobTillaart/pressure
|
2023-07-27 13:05:48 +02:00
|
|
|
|
|
|
|
|
2023-08-04 10:16:10 +02:00
|
|
|
#### SinglePointCorrection calibration
|
2023-07-27 13:05:48 +02:00
|
|
|
|
2023-08-04 10:16:10 +02:00
|
|
|
It takes a relative short time (few minutes) to calibrate the sensor in a known
|
|
|
|
gas concentration.
|
|
|
|
|
|
|
|
- **bool setSinglePointCorrection(uint32_t spc)** takes several minutes. See datasheet.
|
|
|
|
The parameter **spc** should be between 400 and 2000 (assumption).
|
|
|
|
The function returns false if the parameter is out of range or if the request fails.
|
|
|
|
- **bool getSinglePointCorrectionReady()** To see if setting the **SPC** has
|
|
|
|
finished or not.
|
2023-07-27 13:05:48 +02:00
|
|
|
The call also fails if the request fails.
|
|
|
|
|
2023-08-04 10:16:10 +02:00
|
|
|
As far as known the **SPC** value can not be retrieved from the sensor.
|
2023-07-27 13:05:48 +02:00
|
|
|
|
|
|
|
|
|
|
|
#### Self calibration
|
|
|
|
|
2023-08-04 10:16:10 +02:00
|
|
|
Self calibration is a process in which the sensor takes the minimum values over
|
|
|
|
a longer period between 24 - 720 hours as the reference for minimum outdoor values.
|
2023-07-27 13:05:48 +02:00
|
|
|
Note that 720 hours is 30 days / 1 month.
|
2023-08-04 10:16:10 +02:00
|
|
|
Default seems to be 168 hours == 1 week.
|
2023-07-27 13:05:48 +02:00
|
|
|
|
|
|
|
- **bool openSelfCalibration()** start the self calibration cycle.
|
|
|
|
- **bool closeSelfCalibration()** stop the self calibration cycle.
|
|
|
|
- **uint8_t getSelfCalibrationStatus()** Returns if the selfCalibration is open or closed.
|
2023-08-04 10:16:10 +02:00
|
|
|
The function returned 0x00 for CLOSED and 0xFF for OPEN.
|
|
|
|
- **bool setSelfCalibrationHours(uint16_t hours = 168)** Sets the number of hours
|
|
|
|
between self calibration moments.
|
|
|
|
Valid values for hours are 24 - 720 (1 day up to 1 month).
|
|
|
|
Default value = 168 hours = 1 week.
|
2023-07-27 13:05:48 +02:00
|
|
|
- **uint16_t getSelfCalibrationHours()** returns the value set above.
|
|
|
|
|
2023-08-04 10:16:10 +02:00
|
|
|
Note: read datasheet!
|
|
|
|
|
2023-07-27 13:05:48 +02:00
|
|
|
|
|
|
|
## Future
|
|
|
|
|
|
|
|
#### Must
|
2023-08-04 10:16:10 +02:00
|
|
|
|
2023-07-27 13:05:48 +02:00
|
|
|
- update documentation
|
|
|
|
- reorder
|
|
|
|
|
2023-08-04 10:16:10 +02:00
|
|
|
#### Should
|
|
|
|
|
|
|
|
- **setSinglePointCorrection(spc)** investigate parameter.
|
|
|
|
datasheet states 0x2000 but that is 8192 which is rather strange.
|
|
|
|
Assumption 2000 decimal is meant.
|
2023-07-27 13:05:48 +02:00
|
|
|
|
|
|
|
#### Could
|
2023-08-04 10:16:10 +02:00
|
|
|
|
2023-07-27 13:05:48 +02:00
|
|
|
- serial bus with multiple devices? => diodes
|
|
|
|
- separate document
|
|
|
|
- move all code from .h to .cpp file
|
|
|
|
- reuse command buffer as return buffer?
|
|
|
|
- saves a bit.
|
2023-08-04 10:16:10 +02:00
|
|
|
- **getAirPressureReference()** could be smarter
|
2023-07-27 13:05:48 +02:00
|
|
|
- always same value
|
|
|
|
- from cache
|
|
|
|
- dirty flag (-1);
|
2023-08-04 10:16:10 +02:00
|
|
|
- ERROR handling.
|
|
|
|
- some functions returning bool should return int
|
|
|
|
to handle errors better MTP40F_OK or ERROR flag.
|
|
|
|
|
2023-07-27 13:05:48 +02:00
|
|
|
#### Wont
|
|
|
|
|
|
|
|
- store SPC point in the class?
|
|
|
|
|
2023-11-14 16:44:34 +01:00
|
|
|
|
|
|
|
## Support
|
|
|
|
|
|
|
|
If you appreciate my libraries, you can support the development and maintenance.
|
|
|
|
Improve the quality of the libraries by providing issues and Pull Requests, or
|
|
|
|
donate through PayPal or GitHub sponsors.
|
|
|
|
|
|
|
|
Thank you,
|
|
|
|
|