Files
esPicoServer/src/Sensors/DHT20.cpp
T
2025-12-11 21:14:27 +00:00

308 lines
5.8 KiB
C++

/**
* Raspberry Pico / RP2040 library for DHT20 I2C temperature and humidity sensor
* Ported from this Arduino library: https://github.com/RobTillaart/DHT20 by Rob Tillaart
*
* @author Johannes Braun
* @version 0.1
* @url https://codeberg.org/hannenz/pico-dht20
*/
#include "DHT20.h"
#define millis() (to_ms_since_boot(get_absolute_time()))
const uint8_t DHT20_ADDRESS = 0x38;
DHT20::DHT20(int pin_sda, int pin_scl)
{
_temperature = 0;
_humidity = 0;
_humOffset = 0;
_tempOffset = 0;
_status = DHT20_OK;
_lastRequest = 0;
_lastRead = 0;
// Setup / init I2C
i2c_init(i2c_default, 100 * 1000);
gpio_set_function(pin_sda, GPIO_FUNC_I2C);
gpio_set_function(pin_scl, GPIO_FUNC_I2C);
gpio_pull_up(pin_sda);
gpio_pull_up(pin_scl);
}
bool DHT20::begin()
{
return isConnected();
}
bool DHT20::isConnected()
{
return true;
// don't know yet how to translate this,
// some sort of sending a dummy byte and checking the return value?
// _wire->beginTransmission(DHT20_ADDRESS);
// int rv = _wire->endTransmission();
// return rv == 0;
}
uint8_t DHT20::getAddress()
{
return DHT20_ADDRESS;
}
// See datasheet 7.4 Sensor Reading Process, point 1
// use with care.
uint8_t DHT20::resetSensor()
{
uint8_t count = 255;
if ((readStatus() & 0x18) != 0x18)
{
count++;
if (_resetRegister(0x1B))
count++;
if (_resetRegister(0x1C))
count++;
if (_resetRegister(0x1E))
count++;
sleep_ms(10);
}
return count;
}
////////////////////////////////////////////////
//
// READ THE SENSOR
//
int DHT20::read()
{
// do not read too fast == more than once per second.
if (millis() - _lastRead < 1000)
{
return DHT20_ERROR_LASTREAD;
}
int status = requestData();
if (status < 0)
{
return status;
}
// wait for measurement ready
uint32_t start = millis();
while (isMeasuring())
{
if (millis() - start >= 1000)
{
return DHT20_ERROR_READ_TIMEOUT;
}
break;
}
// read the measurement
status = readData();
if (status < 0)
{
return status;
}
// convert it to meaningful data
return convert();
}
int DHT20::requestData()
{
// reset sensor if needed.
resetSensor();
// GET CONNECTION
uint8_t buf[] = {0xAC, 0x33, 0x00};
int rv = i2c_write_blocking(i2c_default, DHT20_ADDRESS, buf, 3, false);
_lastRequest = millis();
return rv;
}
int DHT20::readData()
{
// GET DATA
const uint8_t length = 7;
int bytes = i2c_read_blocking(i2c_default, DHT20_ADDRESS, _bits, length, false);
if (bytes == 0)
return DHT20_ERROR_CONNECT;
if (bytes < length)
return DHT20_MISSING_BYTES;
bool allZero = true;
for (int i = 0; i < bytes; i++)
{
allZero = allZero && (_bits[i] == 0);
}
if (allZero)
return DHT20_ERROR_BYTES_ALL_ZERO;
_lastRead = millis();
return bytes;
}
int DHT20::convert()
{
// CONVERT AND STORE
_status = _bits[0];
uint32_t raw = _bits[1];
raw <<= 8;
raw += _bits[2];
raw <<= 4;
raw += (_bits[3] >> 4);
_humidity = raw * 9.5367431640625e-5; // ==> / 1048576.0 * 100%;
raw = (_bits[3] & 0x0F);
raw <<= 8;
raw += _bits[4];
raw <<= 8;
raw += _bits[5];
_temperature = raw * 1.9073486328125e-4 - 50; // ==> / 1048576.0 * 200 - 50;
// TEST CHECKSUM
uint8_t _crc = _crc8(_bits, 6);
if (_crc != _bits[6])
return DHT20_ERROR_CHECKSUM;
return DHT20_OK;
}
////////////////////////////////////////////////
//
// TEMPERATURE & HUMIDITY & OFFSET
//
float DHT20::getHumidity()
{
return _humidity + _humOffset;
};
float DHT20::getTemperature()
{
return _temperature + _tempOffset;
};
void DHT20::setHumOffset(float offset)
{
_humOffset = offset;
};
void DHT20::setTempOffset(float offset)
{
_tempOffset = offset;
};
float DHT20::getHumOffset()
{
return _humOffset;
};
float DHT20::getTempOffset()
{
return _tempOffset;
};
////////////////////////////////////////////////
//
// STATUS
//
uint8_t DHT20::readStatus()
{
uint8_t byte;
i2c_read_blocking(i2c_default, DHT20_ADDRESS, &byte, 1, false);
return byte;
}
bool DHT20::isCalibrated()
{
return (readStatus() & 0x08) == 0x08;
}
bool DHT20::isMeasuring()
{
return (readStatus() & 0x80) == 0x80;
}
bool DHT20::isIdle()
{
return (readStatus() & 0x80) == 0x00;
}
int DHT20::internalStatus()
{
return _status;
};
////////////////////////////////////////////////
//
// TIMING
//
uint32_t DHT20::lastRead()
{
return _lastRead;
};
uint32_t DHT20::lastRequest()
{
return _lastRequest;
};
////////////////////////////////////////////////
//
// PRIVATE
//
uint8_t DHT20::_crc8(uint8_t *ptr, uint8_t len)
{
uint8_t crc = 0xFF;
while (len--)
{
crc ^= *ptr++;
for (uint8_t i = 0; i < 8; i++)
{
if (crc & 0x80)
{
crc <<= 1;
crc ^= 0x31;
}
else
{
crc <<= 1;
}
}
}
return crc;
}
// Code based on demo code sent by www.aosong.com
// no further documentation.
// 0x1B returned 18, 0, 4
// 0x1C returned 18, 65, 0
// 0x1E returned 18, 8, 0
// 18 seems to be status register
// other values unknown.
bool DHT20::_resetRegister(uint8_t reg)
{
uint8_t value[3];
value[0] = reg;
value[1] = 0x00;
value[2] = 0x00;
int rv = i2c_write_blocking(i2c_default, DHT20_ADDRESS, value, 3, false);
if (rv != 3)
{
return false;
}
sleep_ms(5);
int bytes = i2c_read_blocking(i2c_default, DHT20_ADDRESS, value, 3, false);
sleep_ms(10);
value[0] = 0xB0;
i2c_write_blocking(i2c_default, DHT20_ADDRESS, value, 3, false);
sleep_ms(5);
return true;
}