2020-05-22 09:20:39 +02:00

192 lines
4.5 KiB
C++

//
// FILE: dhtnew.cpp
// AUTHOR: Rob.Tillaart@gmail.com
// VERSION: 0.2.0
// PURPOSE: DHT Temperature & Humidity Sensor library for Arduino
// URL: https://github.com/RobTillaart/DHTNEW
//
// HISTORY:
// 0.1.0 2017-07-24 initial version based upon DHTStable
// 0.1.1 2017-07-29 add begin() to determine type once and for all instead of every call + refactor
// 0.1.2 2018-01-08 improved begin() + refactor()
// 0.1.3 2018-01-08 removed begin() + moved detection to read() function
// 0.1.4 2018-04-03 add get-/setDisableIRQ(bool b)
// 0.1.5 2019-01-20 fix negative temperature DHT22 - issue #120
// 0.1.6 2020-04-09 #pragma once, readme.md, own repo
// 0.1.7 2020-05-01 prevent premature read; add waitForReading flag (Kudo's to Mr-HaleYa),
// 0.2.0 2020-05-02 made temperature and humidity private (Kudo's to Mr-HaleYa),
#include "dhtnew.h"
// these defines are implementation only, not for user
#define DHTLIB_DHT11_WAKEUP 18
#define DHTLIB_DHT_WAKEUP 1
// datasheet state 1000 and 2000,// experiments [Mr-HaleYa] indicate a bit larger value to be robust.
#define DHTLIB_DHT11_READ_DELAY 1250
#define DHTLIB_DHT22_READ_DELAY 2250
// max timeout is 100usec.
// For a 16Mhz proc that is max 1600 clock cycles
// loops using TIMEOUT use at least 4 clock cycli
// so 100 us takes max 400 loops
// so by dividing F_CPU by 40000 we "fail" as fast as possible
#define DHTLIB_TIMEOUT (F_CPU/40000)
/////////////////////////////////////////////////////
//
// PUBLIC
//
DHTNEW::DHTNEW(uint8_t pin) { _pin = pin; };
// return values:
// DHTLIB_OK
// DHTLIB_ERROR_CHECKSUM
// DHTLIB_ERROR_TIMEOUT
int DHTNEW::read()
{
if (_type != 0)
{
uint16_t readDelay = DHTLIB_DHT22_READ_DELAY; // assume DHT22 compatible
if (_type == 11) readDelay = DHTLIB_DHT11_READ_DELAY;
while (millis() - _lastRead < readDelay)
{
if (!_waitForRead) return DHTLIB_OK;
yield();
}
return _read();
}
_type = 22;
_wakeupDelay = DHTLIB_DHT_WAKEUP;
int rv = _read();
if (rv == DHTLIB_OK) return rv;
_type = 11;
_wakeupDelay = DHTLIB_DHT11_WAKEUP;
rv = _read();
if (rv == DHTLIB_OK) return rv;
_type = 0; // retry next time
return rv;
}
// return values:
// DHTLIB_OK
// DHTLIB_ERROR_CHECKSUM
// DHTLIB_ERROR_TIMEOUT
int DHTNEW::_read()
{
// READ VALUES
if (_disableIRQ) noInterrupts();
int rv = _readSensor();
if (_disableIRQ) interrupts();
_lastRead = millis();
if (rv != DHTLIB_OK)
{
humidity = DHTLIB_INVALID_VALUE;
temperature = DHTLIB_INVALID_VALUE;
return rv; // propagate error value
}
if (_type == 22) // DHT22, DHT33, DHT44, compatible
{
humidity = (_bits[0] * 256 + _bits[1]) * 0.1;
temperature = ((_bits[2] & 0x7F) * 256 + _bits[3]) * 0.1;
}
else // if (_type == 11) // DHT11, DH12, compatible
{
humidity = _bits[0] + _bits[1] * 0.1;
temperature = _bits[2] + _bits[3] * 0.1;
}
if (_bits[2] & 0x80) // negative temperature
{
temperature = -temperature;
}
humidity = constrain(humidity + _humOffset, 0, 100);
temperature += _tempOffset;
// TEST CHECKSUM
uint8_t sum = _bits[0] + _bits[1] + _bits[2] + _bits[3];
if (_bits[4] != sum)
{
return DHTLIB_ERROR_CHECKSUM;
}
return DHTLIB_OK;
}
/////////////////////////////////////////////////////
//
// PRIVATE
//
// return values:
// DHTLIB_OK
// DHTLIB_ERROR_TIMEOUT
int DHTNEW::_readSensor()
{
// INIT BUFFERVAR TO RECEIVE DATA
uint8_t mask = 128;
uint8_t idx = 0;
// EMPTY BUFFER
for (uint8_t i = 0; i < 5; i++) _bits[i] = 0;
// REQUEST SAMPLE
pinMode(_pin, OUTPUT);
digitalWrite(_pin, LOW);
delay(_wakeupDelay);
pinMode(_pin, INPUT);
delayMicroseconds(40);
// GET ACKNOWLEDGE or TIMEOUT
uint16_t loopCnt = DHTLIB_TIMEOUT;
while(digitalRead(_pin) == LOW)
{
if (--loopCnt == 0) return DHTLIB_ERROR_TIMEOUT;
}
loopCnt = DHTLIB_TIMEOUT;
while(digitalRead(_pin) == HIGH)
{
if (--loopCnt == 0) return DHTLIB_ERROR_TIMEOUT;
}
// READ THE OUTPUT - 40 BITS => 5 BYTES
for (uint8_t i = 40; i != 0; i--)
{
loopCnt = DHTLIB_TIMEOUT;
while(digitalRead(_pin) == LOW)
{
if (--loopCnt == 0) return DHTLIB_ERROR_TIMEOUT;
}
uint32_t t = micros();
loopCnt = DHTLIB_TIMEOUT;
while(digitalRead(_pin) == HIGH)
{
if (--loopCnt == 0) return DHTLIB_ERROR_TIMEOUT;
}
if ((micros() - t) > 40)
{
_bits[idx] |= mask;
}
mask >>= 1;
if (mask == 0) // next byte?
{
mask = 0x80;
idx++;
}
}
return DHTLIB_OK;
}
// -- END OF FILE --