2013-08-28 13:57:45 +02:00
|
|
|
//
|
|
|
|
// FILE: dht.cpp
|
|
|
|
// AUTHOR: Rob Tillaart
|
2023-10-25 17:25:03 +02:00
|
|
|
// VERSION: 0.1.36
|
2020-11-27 11:10:47 +01:00
|
|
|
// PURPOSE: DHT Temperature & Humidity Sensor library for Arduino, AVR optimized
|
|
|
|
// URL: https://github.com/RobTillaart/DHTlib
|
|
|
|
// http://arduino.cc/playground/Main/DHTLib
|
2013-08-28 13:57:45 +02:00
|
|
|
//
|
2023-10-25 17:25:03 +02:00
|
|
|
// inspired by DHT11 library
|
2013-08-28 13:57:45 +02:00
|
|
|
|
2021-12-16 16:44:08 +01:00
|
|
|
|
2013-08-28 13:57:45 +02:00
|
|
|
#include "dht.h"
|
|
|
|
|
2021-12-16 16:44:08 +01:00
|
|
|
|
|
|
|
#define DHTLIB_DHT11_WAKEUP 18
|
|
|
|
#define DHTLIB_DHT_WAKEUP 1
|
|
|
|
|
|
|
|
#define DHTLIB_DHT11_LEADING_ZEROS 1
|
|
|
|
#define DHTLIB_DHT_LEADING_ZEROS 6
|
|
|
|
|
|
|
|
|
2023-10-25 17:25:03 +02:00
|
|
|
// max timeout is 100 microseconds.
|
|
|
|
// For a 16 MHz processor 100 microseconds is 1600 clock cycles
|
|
|
|
// loops using DHTLIB_TIMEOUT use at least 4 clock cycles
|
|
|
|
// so 100 us takes max 400 loops
|
|
|
|
// so by dividing F_CPU by 40000 we "fail" as fast as possible
|
2021-12-16 16:44:08 +01:00
|
|
|
#ifndef F_CPU
|
2023-10-25 17:25:03 +02:00
|
|
|
#define DHTLIB_TIMEOUT 1000 // should be approx. clock/40000
|
2021-12-16 16:44:08 +01:00
|
|
|
#else
|
|
|
|
#define DHTLIB_TIMEOUT (F_CPU/40000)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2013-08-28 13:57:45 +02:00
|
|
|
/////////////////////////////////////////////////////
|
|
|
|
//
|
2023-10-25 17:25:03 +02:00
|
|
|
// PUBLIC
|
2013-08-28 13:57:45 +02:00
|
|
|
//
|
|
|
|
|
2015-03-01 11:53:47 +01:00
|
|
|
int8_t dht::read11(uint8_t pin)
|
2013-08-28 13:57:45 +02:00
|
|
|
{
|
2023-10-25 17:25:03 +02:00
|
|
|
// READ VALUES
|
2018-04-03 19:08:27 +02:00
|
|
|
if (_disableIRQ) noInterrupts();
|
2015-03-01 11:53:47 +01:00
|
|
|
int8_t result = _readSensor(pin, DHTLIB_DHT11_WAKEUP, DHTLIB_DHT11_LEADING_ZEROS);
|
2018-04-03 19:08:27 +02:00
|
|
|
if (_disableIRQ) interrupts();
|
2014-10-15 08:00:59 +02:00
|
|
|
|
2023-10-25 17:25:03 +02:00
|
|
|
// these bits are always zero, masking them reduces errors.
|
2015-01-21 21:46:04 +01:00
|
|
|
bits[0] &= 0x7F;
|
|
|
|
bits[2] &= 0x7F;
|
2013-08-28 13:57:45 +02:00
|
|
|
|
2023-10-25 17:25:03 +02:00
|
|
|
// CONVERT AND STORE
|
|
|
|
humidity = bits[0]; // bits[1] == 0;
|
|
|
|
temperature = bits[2]; // bits[3] == 0;
|
2013-08-28 13:57:45 +02:00
|
|
|
|
2023-10-25 17:25:03 +02:00
|
|
|
// TEST CHECKSUM
|
2017-09-20 14:47:53 +02:00
|
|
|
uint8_t sum = bits[0] + bits[1] + bits[2] + bits[3];
|
2014-10-08 16:37:30 +02:00
|
|
|
if (bits[4] != sum)
|
|
|
|
{
|
|
|
|
return DHTLIB_ERROR_CHECKSUM;
|
|
|
|
}
|
2014-10-15 08:00:59 +02:00
|
|
|
return result;
|
2013-08-28 13:57:45 +02:00
|
|
|
}
|
|
|
|
|
2021-12-16 16:44:08 +01:00
|
|
|
|
2017-12-12 20:58:58 +01:00
|
|
|
int8_t dht::read12(uint8_t pin)
|
|
|
|
{
|
2023-10-25 17:25:03 +02:00
|
|
|
// READ VALUES
|
2018-04-03 19:08:27 +02:00
|
|
|
if (_disableIRQ) noInterrupts();
|
2017-12-12 20:58:58 +01:00
|
|
|
int8_t result = _readSensor(pin, DHTLIB_DHT11_WAKEUP, DHTLIB_DHT11_LEADING_ZEROS);
|
2018-04-03 19:08:27 +02:00
|
|
|
if (_disableIRQ) interrupts();
|
2017-12-12 20:58:58 +01:00
|
|
|
|
2023-10-25 17:25:03 +02:00
|
|
|
// CONVERT AND STORE
|
2017-12-12 20:58:58 +01:00
|
|
|
humidity = bits[0] + bits[1] * 0.1;
|
2018-09-02 16:26:19 +02:00
|
|
|
temperature = bits[2] + (bits[3] & 0x7F) * 0.1;
|
2023-10-25 17:25:03 +02:00
|
|
|
if (bits[3] & 0x80) // negative temperature
|
2017-12-12 20:58:58 +01:00
|
|
|
{
|
|
|
|
temperature = -temperature;
|
|
|
|
}
|
|
|
|
|
2023-10-25 17:25:03 +02:00
|
|
|
// TEST CHECKSUM
|
2017-12-12 20:58:58 +01:00
|
|
|
uint8_t sum = bits[0] + bits[1] + bits[2] + bits[3];
|
|
|
|
if (bits[4] != sum)
|
|
|
|
{
|
|
|
|
return DHTLIB_ERROR_CHECKSUM;
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2021-12-16 16:44:08 +01:00
|
|
|
|
2015-03-01 11:53:47 +01:00
|
|
|
int8_t dht::read(uint8_t pin)
|
2013-08-28 13:57:45 +02:00
|
|
|
{
|
2023-10-25 17:25:03 +02:00
|
|
|
// READ VALUES
|
2018-04-03 19:08:27 +02:00
|
|
|
if (_disableIRQ) noInterrupts();
|
2015-03-01 11:53:47 +01:00
|
|
|
int8_t result = _readSensor(pin, DHTLIB_DHT_WAKEUP, DHTLIB_DHT_LEADING_ZEROS);
|
2018-04-03 19:08:27 +02:00
|
|
|
if (_disableIRQ) interrupts();
|
2014-10-15 08:00:59 +02:00
|
|
|
|
2023-10-25 17:25:03 +02:00
|
|
|
// these bits are always zero, masking them reduces errors.
|
2014-10-15 07:48:27 +02:00
|
|
|
bits[0] &= 0x03;
|
|
|
|
bits[2] &= 0x83;
|
2014-02-08 13:53:52 +01:00
|
|
|
|
2023-10-25 17:25:03 +02:00
|
|
|
// CONVERT AND STORE
|
2021-11-13 19:57:17 +01:00
|
|
|
humidity = (bits[0] * 256 + bits[1]) * 0.1;
|
|
|
|
int16_t t = ((bits[2] & 0x7F) * 256 + bits[3]);
|
|
|
|
if (t == 0)
|
|
|
|
{
|
2023-10-25 17:25:03 +02:00
|
|
|
temperature = 0.0; // prevent -0.0;
|
2021-11-13 19:57:17 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
temperature = t * 0.1;
|
|
|
|
if((bits[2] & 0x80) == 0x80 )
|
|
|
|
{
|
|
|
|
temperature = -temperature;
|
|
|
|
}
|
|
|
|
}
|
2014-02-08 13:53:52 +01:00
|
|
|
|
2023-10-25 17:25:03 +02:00
|
|
|
// HEXDUMP DEBUG
|
2021-01-29 12:31:58 +01:00
|
|
|
/*
|
|
|
|
Serial.println();
|
|
|
|
// CHECKSUM
|
|
|
|
if (_bits[4] < 0x10) Serial.print(0);
|
|
|
|
Serial.print(_bits[4], HEX);
|
|
|
|
Serial.print(" ");
|
|
|
|
// TEMPERATURE
|
|
|
|
if (_bits[2] < 0x10) Serial.print(0);
|
|
|
|
Serial.print(_bits[2], HEX);
|
|
|
|
if (_bits[3] < 0x10) Serial.print(0);
|
|
|
|
Serial.print(_bits[3], HEX);
|
|
|
|
Serial.print(" ");
|
|
|
|
Serial.print(temperature, 1);
|
|
|
|
Serial.print(" ");
|
|
|
|
// HUMIDITY
|
|
|
|
if (_bits[0] < 0x10) Serial.print(0);
|
|
|
|
Serial.print(_bits[0], HEX);
|
|
|
|
if (_bits[1] < 0x10) Serial.print(0);
|
|
|
|
Serial.print(_bits[1], HEX);
|
|
|
|
Serial.print(" ");
|
|
|
|
Serial.print(humidity, 1);
|
|
|
|
*/
|
|
|
|
|
2023-10-25 17:25:03 +02:00
|
|
|
// TEST CHECKSUM
|
2014-02-08 13:53:52 +01:00
|
|
|
uint8_t sum = bits[0] + bits[1] + bits[2] + bits[3];
|
2014-06-26 20:48:20 +02:00
|
|
|
if (bits[4] != sum)
|
|
|
|
{
|
|
|
|
return DHTLIB_ERROR_CHECKSUM;
|
|
|
|
}
|
2014-10-15 08:00:59 +02:00
|
|
|
return result;
|
2013-08-28 13:57:45 +02:00
|
|
|
}
|
|
|
|
|
2021-12-16 16:44:08 +01:00
|
|
|
|
2013-08-28 13:57:45 +02:00
|
|
|
/////////////////////////////////////////////////////
|
|
|
|
//
|
2023-10-25 17:25:03 +02:00
|
|
|
// PRIVATE
|
2013-08-28 13:57:45 +02:00
|
|
|
//
|
2015-03-01 11:53:47 +01:00
|
|
|
int8_t dht::_readSensor(uint8_t pin, uint8_t wakeupDelay, uint8_t leadingZeroBits)
|
2015-01-21 21:46:04 +01:00
|
|
|
{
|
2023-10-25 17:25:03 +02:00
|
|
|
// INIT BUFFERVAR TO RECEIVE DATA
|
2014-02-08 13:53:52 +01:00
|
|
|
uint8_t mask = 128;
|
|
|
|
uint8_t idx = 0;
|
|
|
|
|
2014-11-25 19:40:25 +01:00
|
|
|
uint8_t data = 0;
|
|
|
|
uint8_t state = LOW;
|
|
|
|
uint8_t pstate = LOW;
|
|
|
|
uint16_t zeroLoop = DHTLIB_TIMEOUT;
|
|
|
|
uint16_t delta = 0;
|
|
|
|
|
2023-10-25 17:25:03 +02:00
|
|
|
leadingZeroBits = 40 - leadingZeroBits; // reverse counting...
|
2014-11-25 19:40:25 +01:00
|
|
|
|
2023-10-25 17:25:03 +02:00
|
|
|
// replace digitalRead() with Direct Port Reads.
|
|
|
|
// reduces footprint ~100 bytes => portability issue?
|
|
|
|
// direct port read is about 3x faster
|
2014-10-08 16:37:30 +02:00
|
|
|
uint8_t bit = digitalPinToBitMask(pin);
|
|
|
|
uint8_t port = digitalPinToPort(pin);
|
2014-10-05 10:52:52 +02:00
|
|
|
volatile uint8_t *PIR = portInputRegister(port);
|
|
|
|
|
2023-10-25 17:25:03 +02:00
|
|
|
// REQUEST SAMPLE
|
2014-02-08 13:53:52 +01:00
|
|
|
pinMode(pin, OUTPUT);
|
2023-10-25 17:25:03 +02:00
|
|
|
digitalWrite(pin, LOW); // T-be
|
2017-03-06 21:10:50 +01:00
|
|
|
if (wakeupDelay > 8) delay(wakeupDelay);
|
|
|
|
else delayMicroseconds(wakeupDelay * 1000UL);
|
2023-10-25 17:25:03 +02:00
|
|
|
digitalWrite(pin, HIGH); // T-go
|
2014-02-08 13:53:52 +01:00
|
|
|
pinMode(pin, INPUT);
|
|
|
|
|
2023-10-25 17:25:03 +02:00
|
|
|
uint16_t loopCount = DHTLIB_TIMEOUT * 2; // 200 uSec max
|
|
|
|
// while(digitalRead(pin) == HIGH)
|
2014-10-08 16:37:30 +02:00
|
|
|
while ((*PIR & bit) != LOW )
|
|
|
|
{
|
2023-10-25 17:25:03 +02:00
|
|
|
if (--loopCount == 0)
|
2018-03-26 22:30:03 +02:00
|
|
|
{
|
|
|
|
return DHTLIB_ERROR_CONNECT;
|
|
|
|
}
|
2014-10-08 16:37:30 +02:00
|
|
|
}
|
|
|
|
|
2023-10-25 17:25:03 +02:00
|
|
|
// GET ACKNOWLEDGE or TIMEOUT
|
2014-10-08 16:37:30 +02:00
|
|
|
loopCount = DHTLIB_TIMEOUT;
|
2023-10-25 17:25:03 +02:00
|
|
|
// while(digitalRead(pin) == LOW)
|
|
|
|
while ((*PIR & bit) == LOW ) // T-rel
|
2014-06-01 09:48:06 +02:00
|
|
|
{
|
2023-10-25 17:25:03 +02:00
|
|
|
if (--loopCount == 0)
|
2018-03-26 22:30:03 +02:00
|
|
|
{
|
|
|
|
return DHTLIB_ERROR_ACK_L;
|
|
|
|
}
|
2014-06-01 09:48:06 +02:00
|
|
|
}
|
2014-02-08 13:53:52 +01:00
|
|
|
|
2014-10-08 16:37:30 +02:00
|
|
|
loopCount = DHTLIB_TIMEOUT;
|
2023-10-25 17:25:03 +02:00
|
|
|
// while(digitalRead(pin) == HIGH)
|
|
|
|
while ((*PIR & bit) != LOW ) // T-reh
|
2014-06-01 09:48:06 +02:00
|
|
|
{
|
2018-03-26 22:30:03 +02:00
|
|
|
if (--loopCount == 0)
|
|
|
|
{
|
|
|
|
return DHTLIB_ERROR_ACK_H;
|
|
|
|
}
|
2014-06-01 09:48:06 +02:00
|
|
|
}
|
2014-02-08 13:53:52 +01:00
|
|
|
|
2014-10-08 16:37:30 +02:00
|
|
|
loopCount = DHTLIB_TIMEOUT;
|
|
|
|
|
2023-10-25 17:25:03 +02:00
|
|
|
// READ THE OUTPUT - 40 BITS => 5 BYTES
|
2014-10-08 16:37:30 +02:00
|
|
|
for (uint8_t i = 40; i != 0; )
|
2014-02-08 13:53:52 +01:00
|
|
|
{
|
2023-10-25 17:25:03 +02:00
|
|
|
// WAIT FOR FALLING EDGE
|
2014-10-08 16:37:30 +02:00
|
|
|
state = (*PIR & bit);
|
|
|
|
if (state == LOW && pstate != LOW)
|
2014-02-08 13:53:52 +01:00
|
|
|
{
|
2023-10-25 17:25:03 +02:00
|
|
|
if (i > leadingZeroBits) // DHT22 first 6 bits are all zero !! DHT11 only 1
|
2014-10-15 08:00:59 +02:00
|
|
|
{
|
|
|
|
zeroLoop = min(zeroLoop, loopCount);
|
|
|
|
delta = (DHTLIB_TIMEOUT - zeroLoop)/4;
|
|
|
|
}
|
2023-10-25 17:25:03 +02:00
|
|
|
else if ( loopCount <= (zeroLoop - delta) ) // long -> one
|
2014-10-08 16:37:30 +02:00
|
|
|
{
|
2014-10-15 07:48:27 +02:00
|
|
|
data |= mask;
|
2014-10-08 16:37:30 +02:00
|
|
|
}
|
|
|
|
mask >>= 1;
|
2023-10-25 17:25:03 +02:00
|
|
|
if (mask == 0) // next byte
|
2014-10-08 16:37:30 +02:00
|
|
|
{
|
|
|
|
mask = 128;
|
2014-10-15 07:48:27 +02:00
|
|
|
bits[idx] = data;
|
2014-10-08 16:37:30 +02:00
|
|
|
idx++;
|
2014-10-15 07:48:27 +02:00
|
|
|
data = 0;
|
2014-10-08 16:37:30 +02:00
|
|
|
}
|
2023-10-25 17:25:03 +02:00
|
|
|
// next bit
|
2014-10-08 16:37:30 +02:00
|
|
|
--i;
|
2014-10-15 08:00:59 +02:00
|
|
|
|
2023-10-25 17:25:03 +02:00
|
|
|
// reset timeout flag
|
2014-10-08 16:37:30 +02:00
|
|
|
loopCount = DHTLIB_TIMEOUT;
|
2014-02-08 13:53:52 +01:00
|
|
|
}
|
2014-10-08 16:37:30 +02:00
|
|
|
pstate = state;
|
2023-10-25 17:25:03 +02:00
|
|
|
// Check timeout
|
2014-10-15 07:48:27 +02:00
|
|
|
if (--loopCount == 0)
|
|
|
|
{
|
2018-03-26 22:30:03 +02:00
|
|
|
return DHTLIB_ERROR_TIMEOUT;
|
2014-10-15 07:48:27 +02:00
|
|
|
}
|
2014-10-15 08:00:59 +02:00
|
|
|
|
2014-02-08 13:53:52 +01:00
|
|
|
}
|
2023-10-25 17:25:03 +02:00
|
|
|
// pinMode(pin, OUTPUT);
|
|
|
|
// digitalWrite(pin, HIGH);
|
2014-02-08 13:53:52 +01:00
|
|
|
return DHTLIB_OK;
|
2013-08-28 13:57:45 +02:00
|
|
|
}
|
2021-12-16 16:44:08 +01:00
|
|
|
|
|
|
|
|
2023-10-25 17:25:03 +02:00
|
|
|
// -- END OF FILE --
|
2021-12-16 16:44:08 +01:00
|
|
|
|