2021-06-27 13:04:39 +02:00
|
|
|
//
|
|
|
|
// FILE: SGP30.cpp
|
|
|
|
// AUTHOR: Rob Tillaart
|
2021-12-28 11:10:52 +01:00
|
|
|
// VERSION: 0.1.5
|
2021-06-27 13:04:39 +02:00
|
|
|
// DATE: 2021-06-24
|
|
|
|
// PURPOSE: SGP30 library for Arduino
|
|
|
|
// URL: https://github.com/RobTillaart/SGP30
|
|
|
|
// https://www.adafruit.com/product/3709
|
|
|
|
//
|
|
|
|
// HISTORY:
|
|
|
|
// 0.1.0 2021-06-24 initial version
|
|
|
|
// 0.1.1 2021-06-26 add get/setBaseline ++
|
|
|
|
// 0.1.2 2021-06-26 experimental add units H2 + Ethanol
|
|
|
|
// 0.1.3 2021-06-26 add get/setTVOCbaseline()
|
2021-07-01 22:13:51 +02:00
|
|
|
// 0.1.4 2021-07-01 add CRC checking
|
2021-12-28 11:10:52 +01:00
|
|
|
// 0.1.5 2021-12-28 update library,json, readme, license, minor edits
|
2021-06-27 13:04:39 +02:00
|
|
|
|
|
|
|
|
|
|
|
#include "SGP30.h"
|
|
|
|
|
|
|
|
|
|
|
|
/////////////////////////////////////////////////////
|
|
|
|
//
|
|
|
|
// CONSTRUCTOR
|
|
|
|
//
|
|
|
|
SGP30::SGP30(TwoWire *wire)
|
|
|
|
{
|
|
|
|
_address = 0x58;
|
|
|
|
_wire = wire;
|
|
|
|
|
|
|
|
_tvoc = 0;
|
|
|
|
_co2 = 0;
|
|
|
|
_h2 = 0;
|
|
|
|
_ethanol = 0;
|
|
|
|
|
|
|
|
_lastTime = 0;
|
|
|
|
_error = SGP30_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#if defined (ESP8266) || defined(ESP32)
|
|
|
|
bool SGP30::begin(uint8_t dataPin, uint8_t clockPin)
|
|
|
|
{
|
|
|
|
_wire = &Wire;
|
|
|
|
if ((dataPin < 255) && (clockPin < 255))
|
|
|
|
{
|
|
|
|
_wire->begin(dataPin, clockPin);
|
|
|
|
} else {
|
|
|
|
_wire->begin();
|
|
|
|
}
|
|
|
|
if (! isConnected()) return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
bool SGP30::begin()
|
|
|
|
{
|
|
|
|
_wire->begin();
|
|
|
|
if (! isConnected()) return false;
|
|
|
|
_init();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool SGP30::isConnected()
|
|
|
|
{
|
|
|
|
_wire->beginTransmission(_address);
|
|
|
|
return ( _wire->endTransmission() == 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// INITIAL VERSION - needs optimization
|
|
|
|
bool SGP30::getID()
|
|
|
|
{
|
|
|
|
_command(0x3682);
|
|
|
|
delay(1);
|
|
|
|
if (_wire->requestFrom(_address, (uint8_t)9) == 9)
|
|
|
|
{
|
|
|
|
for (uint8_t i = 0, j = 0; i < 3; i++)
|
|
|
|
{
|
|
|
|
_id[j++] = _wire->read();
|
|
|
|
_id[j++] = _wire->read();
|
2021-07-01 22:13:51 +02:00
|
|
|
uint8_t crc = _wire->read();
|
|
|
|
uint16_t val = _id[j-2] * 256 + _id[j-1];
|
|
|
|
if (_CRC8(val) != crc)
|
|
|
|
{
|
|
|
|
_error = SGP30_ERROR_CRC;
|
|
|
|
return false;
|
|
|
|
}
|
2021-06-27 13:04:39 +02:00
|
|
|
}
|
2021-07-01 22:13:51 +02:00
|
|
|
_error = SGP30_OK;
|
2021-06-27 13:04:39 +02:00
|
|
|
return true;
|
|
|
|
}
|
2021-07-01 22:13:51 +02:00
|
|
|
_error = SGP30_ERROR_I2C;
|
2021-06-27 13:04:39 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// expect to return 0x0022
|
|
|
|
uint16_t SGP30::getFeatureSet()
|
|
|
|
{
|
|
|
|
_command(0x202F);
|
|
|
|
delay(10);
|
|
|
|
uint16_t rv = 0;
|
|
|
|
if (_wire->requestFrom(_address, (uint8_t)3) == 3)
|
|
|
|
{
|
|
|
|
rv = _wire->read() << 8;
|
|
|
|
rv += _wire->read();
|
2021-07-01 22:13:51 +02:00
|
|
|
uint8_t crc = _wire->read();
|
|
|
|
if (_CRC8(rv) != crc)
|
|
|
|
{
|
|
|
|
_error = SGP30_ERROR_CRC;
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
_error = SGP30_OK;
|
|
|
|
return rv;
|
2021-06-27 13:04:39 +02:00
|
|
|
}
|
2021-07-01 22:13:51 +02:00
|
|
|
_error = SGP30_ERROR_I2C;
|
|
|
|
return 0xFFFF;
|
2021-06-27 13:04:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// WARNING: resets all devices on I2C bus.
|
|
|
|
void SGP30::GenericReset()
|
|
|
|
{
|
|
|
|
_command(0x0006);
|
|
|
|
_init();
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
bool SGP30::measureTest()
|
|
|
|
{
|
|
|
|
uint16_t rv = 0;
|
|
|
|
_command(0x2032);
|
|
|
|
delay(220); // Page 11
|
|
|
|
if (_wire->requestFrom(_address, (uint8_t)3) == 3)
|
|
|
|
{
|
|
|
|
rv = _wire->read() << 8;
|
|
|
|
rv += _wire->read();
|
2021-07-01 22:13:51 +02:00
|
|
|
uint8_t crc = _wire->read();
|
|
|
|
if (_CRC8(rv) != crc)
|
|
|
|
{
|
|
|
|
_error = SGP30_ERROR_CRC;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
_error = SGP30_OK;
|
2021-06-27 13:04:39 +02:00
|
|
|
return rv == 0xD400;
|
|
|
|
}
|
2021-07-01 22:13:51 +02:00
|
|
|
_error = SGP30_ERROR_I2C;
|
2021-06-27 13:04:39 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/////////////////////////////////////////////////////
|
|
|
|
//
|
|
|
|
// MEASUREMENT
|
|
|
|
//
|
|
|
|
|
|
|
|
// SYNCHRONUOUS MODUS
|
|
|
|
bool SGP30::measure(bool all) // this is the workhorse
|
|
|
|
{
|
|
|
|
// 1 second between measurements. P.09
|
|
|
|
if (millis() - _lastTime < 1000) return false;
|
|
|
|
_lastTime = millis();
|
|
|
|
|
|
|
|
request();
|
|
|
|
delay(12);
|
|
|
|
read();
|
|
|
|
|
|
|
|
if (not all) return true;
|
|
|
|
|
|
|
|
requestRaw();
|
|
|
|
delay(25);
|
|
|
|
readRaw();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// A-SYNCHRONUOUS INTERFACE
|
|
|
|
void SGP30::request()
|
|
|
|
{
|
|
|
|
_lastRequest = millis();
|
|
|
|
_command(0x2008);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool SGP30::read()
|
|
|
|
{
|
|
|
|
if (_lastRequest == 0) return false;
|
|
|
|
if (millis() - _lastRequest < 13) return false; // P11
|
|
|
|
_lastRequest = 0;
|
|
|
|
|
|
|
|
if (_wire->requestFrom(_address, (uint8_t)6) != 6)
|
|
|
|
{
|
2021-07-01 22:13:51 +02:00
|
|
|
_error = SGP30_ERROR_I2C;
|
2021-06-27 13:04:39 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
_co2 = _wire->read() << 8;
|
|
|
|
_co2 += _wire->read();
|
2021-07-01 22:13:51 +02:00
|
|
|
uint8_t crc = _wire->read();
|
|
|
|
if (_CRC8(_co2) != crc)
|
|
|
|
{
|
|
|
|
_error = SGP30_ERROR_CRC;
|
|
|
|
return false;
|
|
|
|
}
|
2021-06-27 13:04:39 +02:00
|
|
|
_tvoc = _wire->read() << 8;
|
|
|
|
_tvoc += _wire->read();
|
2021-07-01 22:13:51 +02:00
|
|
|
crc = _wire->read();
|
|
|
|
if (_CRC8(_tvoc) != crc)
|
|
|
|
{
|
|
|
|
_error = SGP30_ERROR_CRC;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
_error = SGP30_OK;
|
2021-06-27 13:04:39 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void SGP30::requestRaw()
|
|
|
|
{
|
|
|
|
_lastRequest = millis();
|
|
|
|
_command(0x2050);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool SGP30::readRaw()
|
|
|
|
{
|
|
|
|
if (_lastRequest == 0) return false;
|
|
|
|
if (millis() - _lastRequest < 26) return false; // P11
|
|
|
|
_lastRequest = 0;
|
|
|
|
|
|
|
|
if (_wire->requestFrom(_address, (uint8_t)6) != 6)
|
|
|
|
{
|
2021-07-01 22:13:51 +02:00
|
|
|
_error = SGP30_ERROR_I2C;
|
2021-06-27 13:04:39 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
_h2 = _wire->read() << 8;
|
|
|
|
_h2 += _wire->read();
|
2021-07-01 22:13:51 +02:00
|
|
|
uint8_t crc = _wire->read();
|
|
|
|
if (_CRC8(_h2) != crc)
|
|
|
|
{
|
|
|
|
_error = SGP30_ERROR_CRC;
|
|
|
|
return false;
|
|
|
|
}
|
2021-06-27 13:04:39 +02:00
|
|
|
_ethanol = _wire->read() << 8;
|
|
|
|
_ethanol += _wire->read();
|
2021-07-01 22:13:51 +02:00
|
|
|
crc = _wire->read();
|
|
|
|
if (_CRC8(_ethanol) != crc)
|
|
|
|
{
|
|
|
|
_error = SGP30_ERROR_CRC;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
_error = SGP30_OK;
|
2021-06-27 13:04:39 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// experimental - datasheet P2
|
|
|
|
// 1.953125e-3 = 1/512
|
|
|
|
float SGP30::getH2()
|
|
|
|
{
|
|
|
|
float cref = 0.5; // ppm
|
|
|
|
return cref * exp((_srefH2 - _h2) * 1.953125e-3);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
float SGP30::getEthanol()
|
|
|
|
{
|
|
|
|
float cref = 0.4; // ppm
|
|
|
|
return cref * exp((_srefEth - _ethanol) * 1.953125e-3);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/////////////////////////////////////////////////////
|
|
|
|
//
|
|
|
|
// CALIBRATION
|
|
|
|
//
|
|
|
|
|
|
|
|
// T in °C
|
|
|
|
// RH == RelativeHumidity
|
|
|
|
float SGP30::setRelHumidity(float T, float RH) // P10
|
|
|
|
{
|
|
|
|
// page 10 datasheet
|
|
|
|
// AH = AbsoluteHumidity
|
|
|
|
// uint16_t AH = 216.7 * RH/100 * 6.117 * exp((17.62 * T)/(243.12 + T)) / (273.15 + T);
|
2021-12-28 11:10:52 +01:00
|
|
|
float absoluteHumidity = (2.167 * 6.112) * RH ;
|
|
|
|
absoluteHumidity *= exp((17.62 * T)/(243.12 + T));
|
|
|
|
absoluteHumidity /= (273.15 + T);
|
2021-06-27 13:04:39 +02:00
|
|
|
|
2021-12-28 11:10:52 +01:00
|
|
|
setAbsHumidity(absoluteHumidity);
|
|
|
|
return absoluteHumidity;
|
2021-06-27 13:04:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-12-28 11:10:52 +01:00
|
|
|
void SGP30::setAbsHumidity(float absoluteHumidity)
|
2021-06-27 13:04:39 +02:00
|
|
|
{
|
2021-12-28 11:10:52 +01:00
|
|
|
uint16_t AH = absoluteHumidity;
|
|
|
|
uint8_t tmp = (absoluteHumidity - AH) * 256;
|
2021-06-27 13:04:39 +02:00
|
|
|
AH = (AH << 8) | tmp;
|
|
|
|
|
|
|
|
_command(0x2061, AH); // P 11
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void SGP30::setBaseline(uint16_t CO2, uint16_t TVOC)
|
|
|
|
{
|
|
|
|
_command(0x201E, CO2, TVOC);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool SGP30::getBaseline(uint16_t *CO2, uint16_t *TVOC)
|
|
|
|
{
|
|
|
|
_command(0x2015);
|
|
|
|
if (_wire->requestFrom(_address, (uint8_t)6) != 6)
|
|
|
|
{
|
2021-07-01 22:13:51 +02:00
|
|
|
_error = SGP30_ERROR_I2C;
|
2021-06-27 13:04:39 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
*CO2 = _wire->read() << 8;
|
|
|
|
*CO2 += _wire->read();
|
2021-07-01 22:13:51 +02:00
|
|
|
uint8_t crc = _wire->read();
|
|
|
|
if (_CRC8(*CO2) != crc)
|
|
|
|
{
|
|
|
|
_error = SGP30_ERROR_CRC;
|
|
|
|
return false;
|
|
|
|
}
|
2021-06-27 13:04:39 +02:00
|
|
|
*TVOC = _wire->read() << 8;
|
|
|
|
*TVOC += _wire->read();
|
2021-07-01 22:13:51 +02:00
|
|
|
crc = _wire->read();
|
|
|
|
if (_CRC8(*TVOC) != crc)
|
|
|
|
{
|
|
|
|
_error = SGP30_ERROR_CRC;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
_error = SGP30_OK;
|
2021-06-27 13:04:39 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void SGP30::setTVOCBaseline(uint16_t TVOC)
|
|
|
|
{
|
|
|
|
_command(0x2077, TVOC);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool SGP30::getTVOCBaseline(uint16_t *TVOC)
|
|
|
|
{
|
|
|
|
_command(0x20B3);
|
|
|
|
if (_wire->requestFrom(_address, (uint8_t)3) != 3)
|
|
|
|
{
|
2021-07-01 22:13:51 +02:00
|
|
|
_error = SGP30_ERROR_I2C;
|
2021-06-27 13:04:39 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
*TVOC = _wire->read() << 8;
|
|
|
|
*TVOC += _wire->read();
|
2021-07-01 22:13:51 +02:00
|
|
|
uint8_t crc = _wire->read();
|
|
|
|
if (_CRC8(*TVOC) != crc)
|
|
|
|
{
|
|
|
|
_error = SGP30_ERROR_CRC;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
_error = SGP30_OK;
|
2021-06-27 13:04:39 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/////////////////////////////////////////////////////
|
|
|
|
//
|
|
|
|
// MISC
|
|
|
|
//
|
|
|
|
int SGP30::lastError()
|
|
|
|
{
|
|
|
|
int error = _error;
|
|
|
|
_error = SGP30_OK;
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/////////////////////////////////////////////////////
|
|
|
|
//
|
|
|
|
// PRIVATE
|
|
|
|
//
|
|
|
|
int SGP30::_command(uint16_t cmd)
|
|
|
|
{
|
|
|
|
_wire->beginTransmission(_address);
|
|
|
|
_wire->write(cmd >> 8);
|
|
|
|
_wire->write(cmd & 0xFF);
|
|
|
|
_error = _wire->endTransmission();
|
|
|
|
return _error;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int SGP30::_command(uint16_t cmd, uint16_t v1)
|
|
|
|
{
|
|
|
|
_wire->beginTransmission(_address);
|
|
|
|
_wire->write(cmd >> 8);
|
|
|
|
_wire->write(cmd & 0xFF);
|
|
|
|
_wire->write(v1 >> 8);
|
|
|
|
_wire->write(v1 & 0xFF);
|
|
|
|
_wire->write(_CRC8(v1));
|
|
|
|
_error = _wire->endTransmission();
|
|
|
|
return _error;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int SGP30::_command(uint16_t cmd, uint16_t v1, uint16_t v2)
|
|
|
|
{
|
|
|
|
_wire->beginTransmission(_address);
|
|
|
|
_wire->write(cmd >> 8);
|
|
|
|
_wire->write(cmd & 0xFF);
|
|
|
|
_wire->write(v1 >> 8);
|
|
|
|
_wire->write(v1 & 0xFF);
|
|
|
|
_wire->write(_CRC8(v1));
|
|
|
|
_wire->write(v2 >> 8);
|
|
|
|
_wire->write(v2 & 0xFF);
|
|
|
|
_wire->write(_CRC8(v2));
|
|
|
|
_error = _wire->endTransmission();
|
|
|
|
return _error;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// for sending command - CRC lib.
|
|
|
|
// always 2 bytes
|
|
|
|
uint8_t SGP30::_CRC8(uint16_t data)
|
|
|
|
{
|
|
|
|
uint8_t val[2];
|
|
|
|
val[0] = data >> 8;
|
|
|
|
val[1] = data & 0xFF;
|
|
|
|
|
|
|
|
uint8_t crc = 0xFF; // start value
|
|
|
|
for(uint8_t i = 0; i < 2; i++)
|
|
|
|
{
|
|
|
|
crc ^= val[i];
|
|
|
|
for (uint8_t b = 8; b > 0; b--)
|
|
|
|
{
|
|
|
|
if (crc & 0x80)
|
|
|
|
crc = (crc << 1) ^ 0x31; // polynomial
|
|
|
|
else
|
|
|
|
crc <<= 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return crc;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
void SGP30::_init()
|
|
|
|
{
|
|
|
|
_command(0x2003);
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
// -- END OF FILE --
|
2021-12-28 11:10:52 +01:00
|
|
|
|