354 lines
7.1 KiB
C++
Raw Normal View History

2019-02-18 14:01:41 +01:00
//
// FILE: SHT31.cpp
// AUTHOR: Rob Tillaart
2021-09-20 11:41:38 +02:00
// VERSION: 0.3.4
2019-02-18 14:01:41 +01:00
// DATE: 2019-02-08
2020-11-27 11:33:55 +01:00
// PURPOSE: Arduino library for the SHT31 temperature and humidity sensor
2019-02-18 14:01:41 +01:00
// https://www.adafruit.com/product/2857
2020-11-27 11:33:55 +01:00
// URL: https://github.com/RobTillaart/SHT31
2019-02-18 14:01:41 +01:00
//
2021-01-29 12:31:58 +01:00
// HISTORY:
// 0.1.0 2019-02-08 initial version
// 0.1.1 2019-02-18 add description readStatus(),
// async interface
// 0.1.2 2019-03-05 fix issue #123 - error in humidity
// stable version
// 0.2.0 2020-05-12 made humidity & temperature private;
// support ESP32 I2C
// 0.2.1 2020-06-19 fix library.json
// 0.2.2 2020-07-05 fix compiling for ESP
// 0.2.3 2020-07-19 added CRC by PetrDa (thanks); cleanup
// 0.2.4 2020-11-29 added isConnected()
// 0.2.5 2020-12-02 added isHeaterOn() + unittest + arduino-ci
// 0.2.6 2021-01-01 patch version 0.2.6
//
// 0.3.0 2021-01-04 arduino-ci
2021-05-28 13:45:10 +02:00
// 0.3.1 2021-05-27 arduino-lint fixes
2021-08-06 19:16:44 +02:00
// 0.3.2 2021-08-05 expose raw data from sensor
2021-08-24 16:01:55 +02:00
// 0.3.3 2021-08-24 fix #22 prevent heater to switch on too fast.
// update readme
2021-09-20 11:41:38 +02:00
// 0.3.4 2021-09-19 update build-CI
2020-11-27 11:33:55 +01:00
2019-02-18 14:01:41 +01:00
#include "SHT31.h"
2021-01-29 12:31:58 +01:00
2020-11-27 11:33:55 +01:00
// SUPPORTED COMMANDS - single shot mode only
2019-02-18 14:01:41 +01:00
#define SHT31_READ_STATUS 0xF32D
#define SHT31_CLEAR_STATUS 0x3041
2020-11-27 11:33:55 +01:00
2019-03-05 18:45:36 +01:00
#define SHT31_SOFT_RESET 0x30A2
#define SHT31_HARD_RESET 0x0006
2020-11-27 11:33:55 +01:00
2021-01-29 12:31:58 +01:00
#define SHT31_MEASUREMENT_FAST 0x2416 // page 10 datasheet
#define SHT31_MEASUREMENT_SLOW 0x2400 // no clock stretching
2020-11-27 11:33:55 +01:00
2019-02-18 14:01:41 +01:00
#define SHT31_HEAT_ON 0x306D
#define SHT31_HEAT_OFF 0x3066
2021-08-24 16:01:55 +02:00
#define SHT31_HEATER_TIMEOUT 180000UL // milliseconds
2019-02-18 14:01:41 +01:00
SHT31::SHT31()
{
2021-08-24 16:01:55 +02:00
_address = 0;
2021-08-06 19:16:44 +02:00
_lastRead = 0;
rawTemperature = 0;
rawHumidity = 0;
2021-08-24 16:01:55 +02:00
_heatTimeout = 0;
2021-08-06 19:16:44 +02:00
_heaterStart = 0;
2021-08-24 16:01:55 +02:00
_heaterStop = 0;
_heaterOn = false;
2021-08-06 19:16:44 +02:00
_error = SHT31_OK;
2019-02-18 14:01:41 +01:00
}
2021-01-29 12:31:58 +01:00
2020-11-27 11:33:55 +01:00
#if defined(ESP8266) || defined(ESP32)
bool SHT31::begin(const uint8_t address, const uint8_t dataPin, const uint8_t clockPin)
2019-02-18 14:01:41 +01:00
{
2021-01-29 12:31:58 +01:00
if ((address != 0x44) && (address != 0x45))
{
return false;
}
2021-08-24 16:01:55 +02:00
_address = address;
2020-11-27 11:33:55 +01:00
_wire = &Wire;
if ((dataPin < 255) && (clockPin < 255))
{
_wire->begin(dataPin, clockPin);
} else {
_wire->begin();
}
2021-01-29 12:31:58 +01:00
return reset();
2019-02-18 14:01:41 +01:00
}
2020-11-27 11:33:55 +01:00
#endif
2019-02-18 14:01:41 +01:00
2021-01-29 12:31:58 +01:00
2020-11-27 11:33:55 +01:00
bool SHT31::begin(const uint8_t address)
2019-02-18 14:01:41 +01:00
{
2020-11-27 11:33:55 +01:00
return begin(address, &Wire);
}
2019-02-18 14:01:41 +01:00
2021-01-29 12:31:58 +01:00
2020-11-27 11:33:55 +01:00
bool SHT31::begin(const uint8_t address, TwoWire *wire)
{
2021-01-29 12:31:58 +01:00
if ((address != 0x44) && (address != 0x45))
{
return false;
}
2021-08-24 16:01:55 +02:00
_address = address;
_wire = wire;
2020-11-27 11:33:55 +01:00
_wire->begin();
2021-01-29 12:31:58 +01:00
return reset();
2020-11-27 11:33:55 +01:00
}
2019-02-18 14:01:41 +01:00
2021-01-29 12:31:58 +01:00
2020-11-27 11:33:55 +01:00
bool SHT31::read(bool fast)
{
2021-01-29 12:31:58 +01:00
if (writeCmd(fast ? SHT31_MEASUREMENT_FAST : SHT31_MEASUREMENT_SLOW) == false)
{
return false;
}
2020-11-27 11:33:55 +01:00
delay(fast ? 4 : 15); // table 4 datasheet
return readData(fast);
2019-02-18 14:01:41 +01:00
}
2021-01-29 12:31:58 +01:00
bool SHT31::isConnected()
{
2021-08-24 16:01:55 +02:00
_wire->beginTransmission(_address);
2021-01-29 12:31:58 +01:00
int rv = _wire->endTransmission();
if (rv != 0) _error = SHT31_ERR_NOT_CONNECT;
return (rv == 0);
}
#ifdef doc
// bit - description
// ==================
// 15 Alert pending status
// '0': no pending alerts
// '1': at least one pending alert - default
// 14 Reserved 0
// 13 Heater status
// '0 : Heater OFF - default
// '1 : Heater ON
// 12 Reserved '0
// 11 Humidity tracking alert
// '0 : no alert - default
// '1 : alert
// 10 Temp tracking alert
// '0 : no alert - default
// '1 : alert
// 9:5 Reserved '00000
// 4 System reset detected
// '0': no reset since last clear status register command
// '1': reset detected (hard or soft reset command or supply fail) - default
// 3:2 Reserved 00
// 1 Command status
// '0': last cmd executed successfully
// '1': last cmd not processed. Invalid or failed checksum
// 0 Write data checksum status
// '0': checksum of last write correct
// '1': checksum of last write transfer failed
#endif
2019-02-18 14:01:41 +01:00
uint16_t SHT31::readStatus()
{
2021-01-29 12:31:58 +01:00
uint8_t status[3] = { 0, 0, 0 };
// page 13 datasheet
if (writeCmd(SHT31_READ_STATUS) == false)
{
return 0xFFFF;
}
// 16 bit status + CRC
if (readBytes(3, (uint8_t*) &status[0]) == false)
{
return 0xFFFF;
}
if (status[2] != crc8(status, 2))
{
_error = SHT31_ERR_CRC_STATUS;
return 0xFFFF;
}
return (uint16_t) (status[0] << 8) + status[1];
}
bool SHT31::reset(bool hard)
{
bool b = writeCmd(hard ? SHT31_HARD_RESET : SHT31_SOFT_RESET);
if (b == false)
{
return false;
2020-11-27 11:33:55 +01:00
}
delay(1); // table 4 datasheet
2021-01-29 12:31:58 +01:00
return true;
2020-11-27 11:33:55 +01:00
}
2021-01-29 12:31:58 +01:00
2020-11-27 11:33:55 +01:00
void SHT31::setHeatTimeout(uint8_t seconds)
{
2021-08-24 16:01:55 +02:00
_heatTimeout = seconds;
if (_heatTimeout > 180) _heatTimeout = 180;
2019-02-18 14:01:41 +01:00
}
2021-01-29 12:31:58 +01:00
bool SHT31::heatOn()
2019-02-18 14:01:41 +01:00
{
2021-08-24 16:01:55 +02:00
if (isHeaterOn()) return true;
if ((_heaterStop > 0) && (millis() - _heaterStop < SHT31_HEATER_TIMEOUT))
{
_error = SHT31_ERR_HEATER_COOLDOWN;
return false;
}
2021-01-29 12:31:58 +01:00
if (writeCmd(SHT31_HEAT_ON) == false)
{
2021-08-24 16:01:55 +02:00
_error = SHT31_ERR_HEATER_ON;
2021-01-29 12:31:58 +01:00
return false;
}
2020-11-27 11:33:55 +01:00
_heaterStart = millis();
2021-08-24 16:01:55 +02:00
_heaterOn = true;
2021-01-29 12:31:58 +01:00
return true;
2019-02-18 14:01:41 +01:00
}
2021-01-29 12:31:58 +01:00
bool SHT31::heatOff()
2019-02-18 14:01:41 +01:00
{
2021-08-24 16:01:55 +02:00
// always switch off the heater - ignore _heaterOn flag.
2021-01-29 12:31:58 +01:00
if (writeCmd(SHT31_HEAT_OFF) == false)
{
_error = SHT31_ERR_HEATER_OFF; // can be serious!
return false;
}
2021-08-24 16:01:55 +02:00
_heaterStop = millis();
_heaterOn = false;
2021-01-29 12:31:58 +01:00
return true;
2020-11-27 11:33:55 +01:00
}
2021-01-29 12:31:58 +01:00
bool SHT31::isHeaterOn()
2020-11-27 11:33:55 +01:00
{
2021-08-24 16:01:55 +02:00
if (_heaterOn == false)
2021-01-29 12:31:58 +01:00
{
return false;
}
2020-11-27 11:33:55 +01:00
// did not exceed time out
2021-08-24 16:01:55 +02:00
if (millis() - _heaterStart < (_heatTimeout * 1000UL))
2021-01-29 12:31:58 +01:00
{
return true;
}
2021-08-24 16:01:55 +02:00
heatOff();
2020-11-27 11:33:55 +01:00
return false;
2019-02-18 14:01:41 +01:00
}
2021-01-29 12:31:58 +01:00
bool SHT31::requestData()
2019-03-05 18:45:36 +01:00
{
2021-01-29 12:31:58 +01:00
if (writeCmd(SHT31_MEASUREMENT_SLOW) == false)
{
return false;
}
2019-03-05 18:45:36 +01:00
_lastRequest = millis();
2021-01-29 12:31:58 +01:00
return true;
2019-03-05 18:45:36 +01:00
}
2021-01-29 12:31:58 +01:00
2019-03-05 18:45:36 +01:00
bool SHT31::dataReady()
{
2021-01-29 12:31:58 +01:00
return ((millis() - _lastRequest) > 15); // TODO MAGIC NR
2019-03-05 18:45:36 +01:00
}
2021-01-29 12:31:58 +01:00
2020-11-27 11:33:55 +01:00
bool SHT31::readData(bool fast)
2019-03-05 18:45:36 +01:00
{
uint8_t buffer[6];
2021-01-29 12:31:58 +01:00
if (readBytes(6, (uint8_t*) &buffer[0]) == false)
{
return false;
}
2019-03-05 18:45:36 +01:00
2020-11-27 11:33:55 +01:00
if (!fast)
{
2021-01-29 12:31:58 +01:00
if (buffer[2] != crc8(buffer, 2))
{
_error = SHT31_ERR_CRC_TEMP;
return false;
}
if (buffer[5] != crc8(buffer + 3, 2))
2020-11-27 11:33:55 +01:00
{
2021-01-29 12:31:58 +01:00
_error = SHT31_ERR_CRC_HUM;
2020-11-27 11:33:55 +01:00
return false;
}
}
2021-08-06 19:16:44 +02:00
rawTemperature = (buffer[0] << 8) + buffer[1];
2021-08-24 16:01:55 +02:00
rawHumidity = (buffer[3] << 8) + buffer[4];
2019-03-05 18:45:36 +01:00
_lastRead = millis();
2020-11-27 11:33:55 +01:00
return true;
2019-03-05 18:45:36 +01:00
}
2021-01-29 12:31:58 +01:00
int SHT31::getError()
{
int rv = _error;
_error = SHT31_OK;
return rv;
}
2019-02-18 14:01:41 +01:00
//////////////////////////////////////////////////////////
2021-01-29 12:31:58 +01:00
uint8_t SHT31::crc8(const uint8_t *data, uint8_t len)
2020-11-27 11:33:55 +01:00
{
// CRC-8 formula from page 14 of SHT spec pdf
const uint8_t POLY(0x31);
uint8_t crc(0xFF);
2021-01-29 12:31:58 +01:00
for (uint8_t j = len; j; --j)
2020-11-27 11:33:55 +01:00
{
crc ^= *data++;
2021-01-29 12:31:58 +01:00
for (uint8_t i = 8; i; --i)
2020-11-27 11:33:55 +01:00
{
crc = (crc & 0x80) ? (crc << 1) ^ POLY : (crc << 1);
}
}
return crc;
}
2021-01-29 12:31:58 +01:00
bool SHT31::writeCmd(uint16_t cmd)
2019-02-18 14:01:41 +01:00
{
2021-08-24 16:01:55 +02:00
_wire->beginTransmission(_address);
2020-11-27 11:33:55 +01:00
_wire->write(cmd >> 8 );
_wire->write(cmd & 0xFF);
2021-01-29 12:31:58 +01:00
if (_wire->endTransmission() != 0)
{
_error = SHT31_ERR_WRITECMD;
return false;
}
return true;
2019-02-18 14:01:41 +01:00
}
2021-01-29 12:31:58 +01:00
bool SHT31::readBytes(uint8_t n, uint8_t *val)
2019-02-18 14:01:41 +01:00
{
2021-08-24 16:01:55 +02:00
int rv = _wire->requestFrom(_address, (uint8_t) n);
2021-01-29 12:31:58 +01:00
if (rv == n)
2019-02-18 14:01:41 +01:00
{
2021-01-29 12:31:58 +01:00
for (uint8_t i = 0; i < n; i++)
{
val[i] = _wire->read();
}
return true;
2019-02-18 14:01:41 +01:00
}
2021-01-29 12:31:58 +01:00
_error = SHT31_ERR_READBYTES;
return false;
2019-02-18 14:01:41 +01:00
}
2020-11-27 11:33:55 +01:00
// -- END OF FILE --