317 lines
12 KiB
Markdown
Raw Normal View History

2021-01-29 12:31:58 +01:00
[![Arduino CI](https://github.com/RobTillaart/Cozir/workflows/Arduino%20CI/badge.svg)](https://github.com/marketplace/actions/arduino_ci)
2021-10-20 09:58:17 +02:00
[![Arduino-lint](https://github.com/RobTillaart/Cozir/actions/workflows/arduino-lint.yml/badge.svg)](https://github.com/RobTillaart/Cozir/actions/workflows/arduino-lint.yml)
[![JSON check](https://github.com/RobTillaart/Cozir/actions/workflows/jsoncheck.yml/badge.svg)](https://github.com/RobTillaart/Cozir/actions/workflows/jsoncheck.yml)
2021-01-29 12:31:58 +01:00
[![License: MIT](https://img.shields.io/badge/license-MIT-green.svg)](https://github.com/RobTillaart/Cozir/blob/master/LICENSE)
[![GitHub release](https://img.shields.io/github/release/RobTillaart/Cozir.svg?maxAge=3600)](https://github.com/RobTillaart/Cozir/releases)
2020-11-27 11:10:47 +01:00
# Cozir
2022-02-21 19:40:59 +01:00
Arduino library for COZIR range CO2 sensors.
2020-11-27 11:10:47 +01:00
2021-02-03 17:20:20 +01:00
2020-11-27 11:10:47 +01:00
## Description
2021-01-29 12:31:58 +01:00
2022-03-04 15:07:12 +01:00
The COZIR library is **experimental** as not all functionality is tested.
2022-02-21 19:40:59 +01:00
The polling mode as used in the examples is tested in the past by DirtGambit.
**CO2meter.com** sponsored a **COZIR GC0034** to start hands on testing (2022-02).
This sensor does not support all commands, but as the pattern of the commands
is similar the non-tested are expected to work as well.
2021-08-15 19:38:45 +02:00
2022-02-21 19:40:59 +01:00
This version of the library supports only the **Serial** interface.
Preferred is a hardware Serial port to connect the sensor but software Serial
does work too.
2020-11-27 11:10:47 +01:00
2022-02-25 12:16:35 +01:00
The library (since 0.3.4) a separate class to parse the STREAMING data.
See COZIRParser below.
2022-03-04 15:07:12 +01:00
The COZIR class is focussed on polling and sending commands.
2022-02-25 12:16:35 +01:00
2021-02-03 17:20:20 +01:00
#### Notes
2021-10-20 09:58:17 +02:00
2021-08-15 19:38:45 +02:00
- Read the datasheet before using this library.
2022-02-21 19:40:59 +01:00
It helps to understand the working of the COZIR sensor.
- Be aware that not all microprocessors have a SoftwareSerial library or one
limited in performance. Polling at 9600 baud can be pretty blocking.
- the hardware serial based examples needs to be run on a MEGA or a Teensy,
at least a board with more than one Serial port.
- If the device is in **CZR_COMMAND** mode it does not respond too polling calls.
It needs to be set to **CZR_POLLING** mode.
- Not all COZIR devices support all calls of this library.
2021-02-03 17:20:20 +01:00
2020-11-27 11:10:47 +01:00
2022-03-04 15:07:12 +01:00
## Interface Cozir
2021-01-29 12:31:58 +01:00
2021-08-15 19:38:45 +02:00
Read the datasheet (again).
2021-01-29 12:31:58 +01:00
2021-02-03 17:20:20 +01:00
### Constructor and initialisation
2020-11-27 11:10:47 +01:00
2022-02-21 19:40:59 +01:00
- **COZIR(Stream \* str)** constructor, gets a serial stream as reference.
- **void init()** sets operatingMode to **CZR_POLLING**
2021-08-15 19:38:45 +02:00
- **bool isInitialized()** returns true if enough time has passed after the call to **init()** for the sensor.
2022-02-21 19:40:59 +01:00
The sensor needs a few seconds to get correct values.
2021-08-15 19:38:45 +02:00
### Operating mode
- **void setOperatingMode(uint8_t mode)** set the operating mode either to **CZR_COMMAND**, **CZR_POLLING** or **CZR_STREAMING**
2022-02-21 19:40:59 +01:00
- **uint8_t getOperatingMode()** returns the mode set, **CZR_STREAMING** is the factory default.
2021-08-15 19:38:45 +02:00
Please note that **init()** sets the operating mode to **CZR_POLLING**.
2021-02-03 17:20:20 +01:00
2022-02-21 19:40:59 +01:00
### Core functions
The COZIR CO2 sensors all support:
- **uint32_t CO2()** returns the CO2 concentration in PPM (!! might need PPMfactor).
- **uint16_t getPPMFactor()** returns 1, 10, 100.
Normally the value returned is 1 but one should check at the first read and when there is a big jump in values returned.
Also when time interval between reads is large it might be useful to check the PPM factor.
Some COZIR sensors also support:
- **float celsius()** returns temperature of the sensor.
2021-08-15 19:38:45 +02:00
- **float fahrenheit()** idem, 'wrapper' around **celsius()**
2022-02-21 19:40:59 +01:00
- **float humidity()** idem, 'wrapper' around **celsius()**
2021-08-15 19:38:45 +02:00
- **float light()** idem.
2021-02-03 17:20:20 +01:00
### Calibration
2021-08-15 19:38:45 +02:00
Read datasheet before using these functions:
2021-02-03 17:20:20 +01:00
2021-08-15 19:38:45 +02:00
- **uint16_t fineTuneZeroPoint(uint16_t v1, uint16_t v2)**
2022-02-21 19:40:59 +01:00
- **uint16_t calibrateFreshAir()** typically 400 PPM.
2021-08-15 19:38:45 +02:00
- **uint16_t calibrateNitrogen()**
2022-02-21 19:40:59 +01:00
- **uint16_t calibrateKnownGas(uint16_t value)**
2021-02-03 17:20:20 +01:00
#### Calibration NOT Recommended
2021-08-15 19:38:45 +02:00
Following 3 functions are **NOT RECOMMENDED** by the datasheet.
2022-02-21 19:40:59 +01:00
Feel free to uncomment in the code and use at your own risk.
2021-08-15 19:38:45 +02:00
Read datasheet before using these functions:
2021-02-03 17:20:20 +01:00
2021-08-15 19:38:45 +02:00
- **uint16_t calibrateManual(uint16_t value)**
- **uint16_t setSpanCalibrate(uint16_t value)**
- **uint16_t getSpanCalibrate()**
2021-02-03 17:20:20 +01:00
2022-02-21 19:40:59 +01:00
### DigiFilter
2021-02-03 17:20:20 +01:00
2021-08-15 19:38:45 +02:00
use with care, read datasheet before use.
2021-02-03 17:20:20 +01:00
2022-02-21 19:40:59 +01:00
| value | meaning |
|:-----:|:---------------------------------------------------|
| 0 | Special, see datasheet |
| 1 | fast, shows every single sample, raw, can be noisy |
| 32 | default, good average |
| 255 | very slow, max smoothed |
2021-02-03 17:20:20 +01:00
2022-03-04 15:07:12 +01:00
2022-02-21 19:40:59 +01:00
- **void setDigiFilter(uint8_t value)** The larger the value the more smoothed the signal is.
Larger values also means that the output does not follow fast changes.
So depending on your needs you need to find an optimal value for the project.
It might even so that you alternate between smooth and fast or adapt depending on
the actual CO2 value.
- **uint8_t getDigiFilter()** returns set value.
2020-11-27 11:10:47 +01:00
2021-02-03 17:20:20 +01:00
### Streaming MODE
2022-03-04 15:07:12 +01:00
Warning: hardware serial is needed / recommended to improve the capture of all output correctly.
2021-02-03 17:20:20 +01:00
2021-08-15 19:38:45 +02:00
- **void setOutputFields(uint16_t fields)** Sets the fields in the output stream as a 16 bit mask. See table below.
- **void clearOutputFields()** clears all the fields.
- **uint16_t getOutputFields()** returns the 16 bit mask of set output fields.
- **bool inOutputFields(uint16_t field)** returns true if the field is set.
2022-02-21 19:40:59 +01:00
- **void getRecentFields()** After a call to getRecentFields() you must read and parse the serial stream yourself.
2021-08-15 19:38:45 +02:00
The internal buffer of this Class cannot handle the possible large output. Lines can be over 100 bytes long!
The fields must be set as a bit mask, the order of the fields in the output is undetermined.
So one need to parse the output of the sensor carefully.
2022-02-21 19:40:59 +01:00
Note: NOT all sensors support all fields, check the datasheet of the sensor used.
| Field | Value HEX | Value DEC | Notes |
|:------------------|:---------:|:---------:|:----------------|
| CZR_LIGHT | 0X2000 | 8192 | |
| CZR_HUMIDITY | 0X1000 | 4096 | |
| CZR_FILTLED | 0X0800 | 2048 | |
| CZR_RAWLED | 0X0400 | 1024 | |
| CZR_MAXLED | 0X0200 | 512 | |
| CZR_ZEROPOINT | 0X0100 | 256 | |
| CZR_RAWTEMP | 0X0080 | 128 | |
| CZR_FILTTEMP | 0X0040 | 64 | |
| CZR_FILTLEDSIGNAL | 0X0020 | 32 | |
| CZR_RAWLEDSIGNAL | 0X0010 | 16 | |
| CZR_SENSTEMP | 0X0008 | 8 | |
| CZR_FILTCO2 | 0X0004 | 4 | |
| CZR_RAWCO2 | 0X0002 | 2 | |
| CZR_NONE | 0X0001 | 1 | reset |
| | | | |
| CZR_DEFAULT | 0X0006 | 6 | factory default |
| CZR_HTC | 0X1082 | 4226 | shortcut |
| CZR_ALL | 0X3FFE | 16383 | debug |
2022-03-04 15:07:12 +01:00
Default value is CZR_DEFAULT == 6 == CZR_FILTCO2 + CZR_RAWCO2
2021-02-03 17:20:20 +01:00
### EEPROM
2022-02-21 19:40:59 +01:00
Note: not all COZIR devices support EEPROM.
Check datasheet for the capabilities of your device.
2021-02-03 17:20:20 +01:00
2022-02-21 19:40:59 +01:00
In 0.3.0 the EEPROM function have been replaced by specific accessor functions.
Read datasheet about the addresses and their meaning. Use with care.
2021-08-15 19:38:45 +02:00
- **void setAutoCalibrationPreload(uint16_t value)**
- **uint16_t getAutoCalibrationPreload()**
- **void setAutoCalibrationInterval(uint16_t value)**
- **uint16_t getAutoCalibrationInterval()**
- **void setAutoCalibrationOn()**
- **void setAutoCalibrationOff()**
- **bool getAutoCalibration()**
- **void setAutoCalibrationBackgroundConcentration(uint16_t value)**
- **uint16_t getAutoCalibrationBackgroundConcentration()**
- **void setAmbientConcentration(uint16_t value)**
- **uint16_t getAmbientConcentration()**
- **void setBufferClearTime(uint16_t value)**
- **uint16_t getBufferClearTime()**
#### EEPROM addresses used by above functions.
2022-02-21 19:40:59 +01:00
Read datasheet for the details and defaults of sensor at hand.
2021-08-15 19:38:45 +02:00
| Name | Address | Default | Notes |
|:---------|:-------:|:-------:|:---------|
| AHHI | 0x00 | ? | reserved |
| ANLO | 0x01 | ? | reserved |
| ANSOURCE | 0x02 | ? | reserved |
| ACINITHI | 0x03 | 87 | |
| ACINITLO | 0x04 | 192 | |
| ACHI | 0x05 | 94 | |
| ACLO | 0x06 | 128 | |
| ACONOFF | 0x07 | 0 | |
| ACPPMHI | 0x08 | 1 | |
| ACPPMLO | 0x09 | 194 | |
| AMBHI | 0x0A | 1 | |
| AMBLO | 0x0B | 194 | |
| BCHI | 0x0C | 0 | |
| BCLO | 0x0D | 8 | |
### Miscellaneous
- **void getVersionSerial()** requests version over serial.
The user should read (and parse) the serial output as it can become large.
Also the user must reset the operating mode either to **CZR_POLLING** or **CZR_STREAMING**
- **void getConfiguration()** requests configuration over serial.
The user should read (and parse) the serial output as it can become large.
Also the user must reset the operating mode either to **CZR_POLLING** or **CZR_STREAMING**
2022-02-21 19:40:59 +01:00
## Operation
2021-08-15 19:38:45 +02:00
2022-02-21 19:40:59 +01:00
See examples.
2021-02-03 17:20:20 +01:00
2022-03-04 15:07:12 +01:00
## Future
2022-10-30 20:10:57 +01:00
#### must
#### should
2022-03-04 15:07:12 +01:00
- improve documentation
2022-10-30 20:10:57 +01:00
- COZIR Parser
- separate readme.md
- separate .h and .cpp
- separate repo
2022-03-04 15:07:12 +01:00
- add examples
2022-10-30 20:10:57 +01:00
- example COZIR with I2C LCD display?
#### could
2022-03-04 15:07:12 +01:00
- COZIR I2C class for newer generation
~ same functional interface
- multiWire / pin a la PCF8574 lib
2022-10-30 20:10:57 +01:00
#### won't
2022-03-04 15:07:12 +01:00
- add a **setEEPROMFactoryDefault()**?
- unknown if all sensors have same values
- build a Arduino COZIR simulator for testing.
- add other sensors underneath?
----
2022-02-25 12:16:35 +01:00
## COZIRParser
Class to parse the output of a COZIR sensor in stream mode.
2022-03-04 15:07:12 +01:00
2022-02-25 12:16:35 +01:00
## Description
(added in 0.3.4, experimental)
The COZIRparser object has a main function called **nextChar(c)**
It needs to be called with all incoming characters from the sensor.
If a new field is found **nextChar()** returns the field identifier
of the last parsed field (single char) to indicate its value has been
updated, otherwise it returns 0.
The updated value can be accessed with one of the functions, see cozir.h file.
An example **Cozir_stream_parse.ino** is added to show how to use this class.
2022-03-04 15:07:12 +01:00
**NOTE:** to send commands e.g. outputField selection, to the sensor the COZIR
class can be used (which is mostly focussed on polling access).
Alternatively the user sends the low level commands just as a string over serial.
This latter method will save memory especially in a final version of a project.
**NOTE:** The COZIRparser skips the output of the Y, \* and @ command.
These are configuration fields and therefore not part of the **stream mode** fields.
Furthermore not all fields these lines produce are understood.
So parsing these lines is left to the user for now.
**NOTE:** The COZIRparser class does not check for missing characters,
the range of the fields recognized, or other errors. So the values
returned should be handled with care.
## Interface COZIRParser
Read the datasheet (again).
### Constructor and initialisation
- **C0ZIRParser()** constructor
- **void init()** resets all internal variables to 0 except PPM (set to 1)
- **void resetParser()** should / could be called if stream from sensor is
interrupted for long period, e.g. if there are missing characters.
This should prevent the parser to overshoot a value due to concatenating the
value of different (interrupted) measurements or FIELDs.
- **uint8_t nextChar(char c)** all characters coming from the sensor should
be send to the parser by means of **nextChar()**.
Default it will return 0 and the FIELD character is a field has been updated.
Read Datasheet for the characters used.
2022-02-25 12:16:35 +01:00
2022-03-04 15:07:12 +01:00
The remainder of the interface are getters for the different fields.
2022-02-25 12:16:35 +01:00
2022-02-21 19:40:59 +01:00
## Future
2021-02-03 17:20:20 +01:00
2022-02-21 19:40:59 +01:00
- improve documentation
2022-02-25 12:16:35 +01:00
- COZIR Parser a separate readme?
2022-03-04 15:07:12 +01:00
- support splitting output of Y and \* command.
- separate parser COZIRParserY()
- separate parser COZIRParserStar()
- separate parser COZIRParser@()
2022-02-21 19:40:59 +01:00
- add examples
2022-02-25 12:16:35 +01:00
- examples for COZIRParser.
2020-11-27 11:10:47 +01:00