402 lines
7.5 KiB
C++
Raw Normal View History

2021-08-13 13:08:52 +02:00
//
// FILE: AGS02MA.cpp
2022-04-25 14:17:43 +02:00
// AUTHOR: Rob Tillaart, Viktor Balint, Beanow
2021-08-13 13:08:52 +02:00
// DATE: 2021-08-12
2023-09-25 17:36:00 +02:00
// VERSION: 0.3.4
2023-01-21 14:28:43 +01:00
// PURPOSE: Arduino library for AGS02MA TVOC sensor
2021-08-13 13:08:52 +02:00
// URL: https://github.com/RobTillaart/AGS02MA
#include "AGS02MA.h"
2023-01-21 14:28:43 +01:00
// REGISTERS
2021-08-13 13:08:52 +02:00
#define AGS02MA_DATA 0x00
#define AGS02MA_CALIBRATION 0x01
#define AGS02MA_VERSION 0x11
#define AGS02MA_SLAVE_ADDRESS 0x21
AGS02MA::AGS02MA(const uint8_t deviceAddress, TwoWire *wire)
{
2021-09-24 12:31:48 +02:00
_address = deviceAddress;
_wire = wire;
reset();
2021-08-13 13:08:52 +02:00
}
#if defined (ESP8266) || defined(ESP32)
bool AGS02MA::begin(uint8_t dataPin, uint8_t clockPin)
{
2023-09-25 17:36:00 +02:00
_startTime = millis(); // PREHEAT
2021-08-13 13:08:52 +02:00
if ((dataPin < 255) && (clockPin < 255))
{
_wire->begin(dataPin, clockPin);
} else {
_wire->begin();
}
return isConnected();
}
#endif
bool AGS02MA::begin()
{
2023-01-21 14:28:43 +01:00
_startTime = millis(); // PREHEAT TIMING
2021-08-13 13:08:52 +02:00
_wire->begin();
return isConnected();
}
bool AGS02MA::isConnected()
{
#if defined (__AVR__)
2023-01-21 14:28:43 +01:00
// TWBR = 255; // == 30.4 KHz with TWSR = 0x00
2023-09-25 17:36:00 +02:00
TWBR = 78; // == 25.0 KHZ
TWSR = 0x01; // pre-scaler = 4
2021-08-13 13:08:52 +02:00
#else
_wire->setClock(AGS02MA_I2C_CLOCK);
#endif
2023-01-21 14:28:43 +01:00
2021-08-13 13:08:52 +02:00
_wire->beginTransmission(_address);
2021-08-15 19:35:40 +02:00
bool rv = ( _wire->endTransmission(true) == 0);
2023-01-21 14:28:43 +01:00
2022-08-12 12:15:08 +02:00
#if defined (__AVR__)
TWSR = 0x00;
#endif
2021-08-15 19:35:40 +02:00
_wire->setClock(_I2CResetSpeed);
2021-08-13 13:08:52 +02:00
return rv;
}
2021-09-24 12:31:48 +02:00
void AGS02MA::reset()
{
_I2CResetSpeed = 100000;
_startTime = millis();
_lastRead = 0;
_lastPPB = 0;
_mode = 255;
_status = AGS02MA_OK;
_error = AGS02MA_OK;
}
2021-08-13 13:08:52 +02:00
bool AGS02MA::setAddress(const uint8_t deviceAddress)
{
2021-09-24 12:31:48 +02:00
if ((deviceAddress < 10) or (deviceAddress > 119)) return false;
2021-08-13 13:08:52 +02:00
_buffer[2] = _buffer[0] = deviceAddress;
_buffer[3] = _buffer[1] = 0xFF ^ deviceAddress;
_buffer[4] = _CRC8(_buffer, 4);
if (_writeRegister(AGS02MA_SLAVE_ADDRESS))
{
_address = deviceAddress;
}
return isConnected();
}
uint8_t AGS02MA::getSensorVersion()
{
uint8_t version = 0xFF;
if (_readRegister(AGS02MA_VERSION))
{
2021-09-24 12:31:48 +02:00
// for (int i = 0; i < 5; i++)
// {
// Serial.print(_buffer[i]);
// Serial.print('\t');
// }
// Serial.println();
// unclear what the other bytes have for information.
// datasheet names these 3 bytes as KEEP.
// BUFFER VALUE MEANING
// buffer [0] == 20 year ?
// buffer [1] == 07 month ?
// buffer [2] == 28 day ?
// buffer [3] == 117 VERSION
// buffer [4] == CRC
2021-08-13 13:08:52 +02:00
version = _buffer[3];
if (_CRC8(_buffer, 5) != 0)
{
2021-09-24 12:31:48 +02:00
_error = AGS02MA_ERROR_CRC;
2021-08-13 13:08:52 +02:00
}
}
return version;
}
2022-01-22 10:17:50 +01:00
uint32_t AGS02MA::getSensorDate()
{
uint32_t date = 0xFFFFFFFF;
if (_readRegister(AGS02MA_VERSION))
{
date = 0x20;
date <<= 8;
date += _bin2bcd(_buffer[0]);
date <<= 8;
date += _bin2bcd(_buffer[1]);
date <<= 8;
date += _bin2bcd(_buffer[2]);
2023-01-21 14:28:43 +01:00
// version = _buffer[3];
2022-01-22 10:17:50 +01:00
if (_CRC8(_buffer, 5) != 0)
{
_error = AGS02MA_ERROR_CRC;
}
}
return date;
}
2021-08-13 13:08:52 +02:00
bool AGS02MA::setPPBMode()
{
_buffer[0] = 0x00;
_buffer[1] = 0xFF;
_buffer[2] = 0x00;
_buffer[3] = 0xFF;
_buffer[4] = 0x30;
if (_writeRegister(AGS02MA_DATA))
{
_mode = 0;
return true;
}
return false;
}
bool AGS02MA::setUGM3Mode()
{
_buffer[0] = 0x02;
_buffer[1] = 0xFD;
_buffer[2] = 0x02;
_buffer[3] = 0xFD;
_buffer[4] = 0x00;
if (_writeRegister(AGS02MA_DATA))
{
_mode = 1;
return true;
}
return false;
}
uint32_t AGS02MA::readPPB()
{
2021-09-24 12:31:48 +02:00
uint32_t val = _readSensor();
if (_error == AGS02MA_OK)
2021-08-13 13:08:52 +02:00
{
2021-09-24 12:31:48 +02:00
_lastRead = millis();
_lastPPB = val;
}
else
{
val = _lastPPB;
2021-08-13 13:08:52 +02:00
}
return val;
}
uint32_t AGS02MA::readUGM3()
{
2021-09-24 12:31:48 +02:00
uint32_t val = _readSensor();
if (_error == AGS02MA_OK)
2021-08-13 13:08:52 +02:00
{
2021-09-24 12:31:48 +02:00
_lastRead = millis();
_lastUGM3 = val;
}
else
{
val = _lastUGM3;
2021-08-13 13:08:52 +02:00
}
return val;
}
2022-04-25 14:17:43 +02:00
bool AGS02MA::manualZeroCalibration(uint16_t value)
2021-08-13 13:08:52 +02:00
{
_buffer[0] = 0x00;
_buffer[1] = 0x0C;
2022-04-25 14:17:43 +02:00
_buffer[2] = (uint8_t) (value >> 8);
_buffer[3] = (uint8_t) (value & 0x00FF);
_buffer[4] = _CRC8(_buffer, 4);
2021-08-13 13:08:52 +02:00
return _writeRegister(AGS02MA_CALIBRATION);
}
2022-04-25 14:17:43 +02:00
bool AGS02MA::getZeroCalibrationData(AGS02MA::ZeroCalibrationData &data) {
if (!_readRegister(AGS02MA_CALIBRATION))
{
return false;
}
if (_CRC8(_buffer, 5) != 0)
{
_error = AGS02MA_ERROR_CRC;
return false;
}
_error = AGS02MA_OK;
2022-10-27 10:47:11 +02:00
// Don't pollute the struct given to us, until we've handled all error cases.
2022-04-25 14:17:43 +02:00
data.status = _getDataMSB();
data.value = _getDataLSB();
return true;
}
2021-08-13 13:08:52 +02:00
int AGS02MA::lastError()
{
int e = _error;
2022-10-27 10:47:11 +02:00
_error = AGS02MA_OK; // reset error after read
2021-08-13 13:08:52 +02:00
return e;
}
2022-08-12 12:15:08 +02:00
2022-05-02 13:50:27 +02:00
bool AGS02MA::readRegister(uint8_t address, AGS02MA::RegisterData &reg) {
if (!_readRegister(address))
{
return false;
}
_error = AGS02MA_OK;
2022-10-27 10:47:11 +02:00
// Don't pollute the struct given to us, until we've handled all error cases.
2022-05-02 13:50:27 +02:00
reg.data[0] = _buffer[0];
reg.data[1] = _buffer[1];
reg.data[2] = _buffer[2];
reg.data[3] = _buffer[3];
reg.crc = _buffer[4];
reg.crcValid = _CRC8(_buffer, 5) == 0;
return true;
}
2021-08-13 13:08:52 +02:00
2023-01-21 14:28:43 +01:00
2021-08-13 13:08:52 +02:00
/////////////////////////////////////////////////////////
//
2022-10-27 10:47:11 +02:00
// PRIVATE
2021-08-13 13:08:52 +02:00
//
2021-09-24 12:31:48 +02:00
uint32_t AGS02MA::_readSensor()
{
_error = AGS02MA_ERROR_READ;
2022-01-22 10:17:50 +01:00
uint32_t value = 0;
2021-09-24 12:31:48 +02:00
if (_readRegister(AGS02MA_DATA))
{
_error = AGS02MA_OK;
_status = _buffer[0];
2022-01-22 10:17:50 +01:00
if (_status & 0x01)
{
_error = AGS02MA_ERROR_NOT_READY;
}
value = _buffer[1] * 65536UL;
value += _buffer[2] * 256;
value += _buffer[3];
2021-09-24 12:31:48 +02:00
if (_CRC8(_buffer, 5) != 0)
{
_error = AGS02MA_ERROR_CRC;
}
}
2022-01-22 10:17:50 +01:00
return value;
2021-09-24 12:31:48 +02:00
}
2022-04-25 14:17:43 +02:00
2021-09-24 12:31:48 +02:00
2021-08-13 13:08:52 +02:00
bool AGS02MA::_readRegister(uint8_t reg)
{
2021-09-24 12:31:48 +02:00
while (millis() - _lastRegTime < 30) yield();
2021-08-13 13:08:52 +02:00
#if defined (__AVR__)
2023-01-21 14:28:43 +01:00
// TWBR = 255; // == 30.4 KHz with TWSR = 0x00
2023-09-25 17:36:00 +02:00
TWBR = 78; // == 25.0 KHZ
TWSR = 0x01; // pre-scaler = 4
2021-08-13 13:08:52 +02:00
#else
_wire->setClock(AGS02MA_I2C_CLOCK);
#endif
2023-01-21 14:28:43 +01:00
2021-08-13 13:08:52 +02:00
_wire->beginTransmission(_address);
_wire->write(reg);
2021-08-15 19:35:40 +02:00
_error = _wire->endTransmission(true);
2022-08-12 12:15:08 +02:00
// TODO investigate async interface
2021-08-15 19:35:40 +02:00
delay(30);
2022-08-12 12:15:08 +02:00
2021-08-13 13:08:52 +02:00
if (_wire->requestFrom(_address, (uint8_t)5) != 5)
{
2021-09-24 12:31:48 +02:00
_error = AGS02MA_ERROR_READ;
2022-08-12 12:15:08 +02:00
#if defined (__AVR__)
2023-09-25 17:36:00 +02:00
TWSR = 0x00; // reset pre-scaler = 1
2022-08-12 12:15:08 +02:00
#endif
2021-08-15 19:35:40 +02:00
_wire->setClock(_I2CResetSpeed);
2021-08-13 13:08:52 +02:00
return false;
}
2021-09-24 12:31:48 +02:00
for (uint8_t i = 0; i < 5; i++)
2021-08-13 13:08:52 +02:00
{
_buffer[i] = _wire->read();
}
2022-08-12 12:15:08 +02:00
#if defined (__AVR__)
2023-09-25 17:36:00 +02:00
TWSR = 0x00; // reset pre-scaler = 1
2022-08-12 12:15:08 +02:00
#endif
2021-08-15 19:35:40 +02:00
_wire->setClock(_I2CResetSpeed);
2021-08-13 13:08:52 +02:00
return true;
}
bool AGS02MA::_writeRegister(uint8_t reg)
{
2021-09-24 12:31:48 +02:00
while (millis() - _lastRegTime < 30) yield();
_lastRegTime = millis();
2021-08-13 13:08:52 +02:00
#if defined (__AVR__)
2023-01-21 14:28:43 +01:00
// TWBR = 255; // == 30.4 KHz with TWSR = 0x00
2023-09-25 17:36:00 +02:00
TWBR = 78; // == 25.0 KHZ
TWSR = 0x01; // pre-scaler = 4
2021-08-13 13:08:52 +02:00
#else
_wire->setClock(AGS02MA_I2C_CLOCK);
#endif
_wire->beginTransmission(_address);
_wire->write(reg);
2021-09-24 12:31:48 +02:00
for (uint8_t i = 0; i < 5; i++)
2021-08-13 13:08:52 +02:00
{
_wire->write(_buffer[i]);
}
2021-08-15 19:35:40 +02:00
_error = _wire->endTransmission(true);
2022-08-12 12:15:08 +02:00
#if defined (__AVR__)
TWSR = 0x00;
#endif
2021-08-15 19:35:40 +02:00
_wire->setClock(_I2CResetSpeed);
2021-08-13 13:08:52 +02:00
return (_error == 0);
}
2023-01-21 14:28:43 +01:00
2022-04-25 14:17:43 +02:00
uint16_t AGS02MA::_getDataMSB()
{
return (_buffer[0] << 8) + _buffer[1];
}
2023-01-21 14:28:43 +01:00
2022-04-25 14:17:43 +02:00
uint16_t AGS02MA::_getDataLSB()
{
return (_buffer[2] << 8) + _buffer[3];
}
2021-08-13 13:08:52 +02:00
2023-01-21 14:28:43 +01:00
2021-08-13 13:08:52 +02:00
uint8_t AGS02MA::_CRC8(uint8_t * buf, uint8_t size)
{
2022-10-27 10:47:11 +02:00
uint8_t crc = 0xFF; // start value
2021-08-13 13:08:52 +02:00
for (uint8_t b = 0; b < size; b++)
{
crc ^= buf[b];
for (uint8_t i = 0; i < 8; i++)
{
if (crc & 0x80) crc = (crc << 1) ^ 0x31;
else crc = (crc << 1);
}
}
return crc;
}
2022-01-22 10:17:50 +01:00
uint8_t AGS02MA::_bin2bcd (uint8_t value)
{
return value + 6 * (value / 10);
}
2023-01-21 14:28:43 +01:00
// -- END OF FILE --