272 lines
8.7 KiB
Markdown
Raw Normal View History

2022-01-11 20:48:39 +01:00
[![Arduino CI](https://github.com/RobTillaart/DHT20/workflows/Arduino%20CI/badge.svg)](https://github.com/marketplace/actions/arduino_ci)
[![Arduino-lint](https://github.com/RobTillaart/DHT20/actions/workflows/arduino-lint.yml/badge.svg)](https://github.com/RobTillaart/DHT20/actions/workflows/arduino-lint.yml)
[![JSON check](https://github.com/RobTillaart/DHT20/actions/workflows/jsoncheck.yml/badge.svg)](https://github.com/RobTillaart/DHT20/actions/workflows/jsoncheck.yml)
[![License: MIT](https://img.shields.io/badge/license-MIT-green.svg)](https://github.com/RobTillaart/DHT20/blob/master/LICENSE)
[![GitHub release](https://img.shields.io/github/release/RobTillaart/DHT20.svg?maxAge=3600)](https://github.com/RobTillaart/DHT20/releases)
# DHT20
Arduino library for I2C DHT20 temperature and humidity sensor.
## Description
2023-02-10 16:22:51 +01:00
The DHT20 is a humidity and temperature sensor.
2022-11-01 21:05:41 +01:00
The sensor has a fixed address of **0x38**.
It is not known if the address can be changed.
2022-01-11 20:48:39 +01:00
The library must be initiated by calling the **begin()** function,
or **begin(dataPin, clockPin)** for **ESP32** and similar platforms.
Thereafter one has to call the **read()** function to do the actual reading,
2022-11-01 21:05:41 +01:00
and call **getTemperature()** and **getHumidity()** to get the measured values.
Calling these latter again will return the same values until a new **read()** is done.
2022-01-11 20:48:39 +01:00
The **read()** call of this sensor is blocking for 80+ milliseconds (datasheet 7.4)
so the library also has a asynchronous interface. See below.
2022-09-18 19:13:39 +02:00
Since 0.1.3 and 0.1.4 the performance of **read()** has been optimized,
still blocking but less long for about 45 milliseconds.
2022-09-17 13:50:11 +02:00
2022-01-11 20:48:39 +01:00
2022-11-01 21:05:41 +01:00
### 0.2.0
In #8 a bug is described that the sensor "freezes".
Cause is not well understood.
Two solutions / workarounds are found:
- call **resetSensor()** before EVERY **read()**.
This is the preferred solution.
- use **Wire.setClock(200000)** 100 K and lower speeds freezes the sensor.
With clock set to 200 K and above the sensor seems to work for longer periods.
Tested several speeds on UNO, no pull ups, < 10 cm wire.
Note: setting the I2C clock possibly interferes with other devices on the I2C bus,
so it is not a solution in the end.
The 0.2.0 version embeds the **resetSensor()** into **requestData()** to
reset the sensor if needed in both synchronous and asynchronous calls.
This keeps the API simple. The reads are 1-2 ms slower than 0.1.4. (< 50 ms).
Still far below the 80 ms mentioned in the datasheet.
2023-02-10 16:22:51 +01:00
#### Tested
Verified to work with Arduino UNO and ESP32 and ESP8266 (see #8)
Please let me know if other platforms work (or not).
## I2C
#### Address
The sensor has a fixed address of **0x38**.
It is not known if the address can be changed.
#### Connection
2022-09-11 11:44:04 +02:00
2022-11-01 21:05:41 +01:00
Always check datasheet!
2022-09-11 11:44:04 +02:00
Front view
```
+--------------+
VDD ----| 1 |
SDA ----| 2 DHT20 |
GND ----| 3 |
SCL ----| 4 |
+--------------+
```
2023-02-10 16:22:51 +01:00
#### Performance
2022-09-11 11:44:04 +02:00
2023-02-10 16:22:51 +01:00
The datasheet states 400 KHz as the maximum speed.
Below the results of a small test that works well up to 800 KHz.
- Arduino UNO + 10 cm wires + no pull up + DHT20_I2C_speed.ino
Speed in KHz, Time in microseconds.
**read()**
| Speed | Time | notes |
|:-------:|:-------:|:--------|
| 100 | 44588 | default I2C speed |
| 200 | 43988 |
| 400 | 44040 | datasheet maximum |
| 600 | 43224 |
| 800 | 43988 |
**ASYNC: requestData()**
| Speed | Time | notes |
|:-------:|:-------:|:--------|
| 100 | 1676 | default I2C speed |
| 200 | 1384 |
| 400 | 1240 | datasheet maximum |
| 600 | 1188 |
| 800 | 1168 |
**ASYNC: readData()**
| Speed | Time | notes |
|:-------:|:-------:|:--------|
| 100 | 832 | default I2C speed |
| 200 | 464 |
| 400 | 284 | datasheet maximum |
| 600 | 212 |
| 800 | 188 |
The numbers indicate that the conversion takes > 40 milliseconds.
Requesting the measurement and fetching the data < 2.5 milliseconds.
Using the asynchronous interface frees up a lot of clock cycles.
Going beyond 400 KHz (datasheet max) does not save much extra time,
and should only be used if you are in a need for speed.
2022-09-11 11:44:04 +02:00
2022-01-11 20:48:39 +01:00
## Interface
2023-02-10 16:22:51 +01:00
```cpp
#include "DHT20.h"
```
2022-01-11 20:48:39 +01:00
2023-02-10 16:22:51 +01:00
#### Constructor
2022-01-11 20:48:39 +01:00
- **DHT20(TwoWire \*wire = &Wire)** constructor, using a specific Wire (I2C bus).
- **bool begin(uint8_t dataPin, uint8_t clockPin)** begin for ESP32 et al, to set I2C bus pins.
- **bool begin()** initializer for non ESP32. Returns true if connected.
- **bool isConnected()** returns true if the address of the DHT20 can be seen on the I2C bus.
2022-12-21 21:48:21 +01:00
- **uint8_t getAddress()** returns the (fixed) address - convenience.
2022-01-11 20:48:39 +01:00
2023-02-10 16:22:51 +01:00
#### Core
2022-01-11 20:48:39 +01:00
- **int8_t read()** read the sensor and store the values internally.
2022-11-01 21:05:41 +01:00
Returns the status of the read which should be 0 == **DHT20_OK**.
2022-01-11 20:48:39 +01:00
- **float getHumidity()** returns last Humidity read.
2022-11-01 21:05:41 +01:00
Multiple calls will return same value until a new **read()** is made.
2022-01-11 20:48:39 +01:00
- **float getTemperature()** returns last Temperature read.
2022-11-01 21:05:41 +01:00
Multiple calls will return same value until a new **read()** is made.
2022-01-11 20:48:39 +01:00
2023-02-10 16:22:51 +01:00
#### Offset
2022-01-11 20:48:39 +01:00
2023-02-10 16:22:51 +01:00
- **void setHumOffset(float offset = 0)** set an offset to calibrate the sensor (1st order).
Default offset is 0.
2022-11-01 21:05:41 +01:00
- **float getHumOffset()** return current humidity offset, default 0.
2023-02-10 16:22:51 +01:00
- **void setTempOffset(float offset = 0)** set an offset to calibrate the sensor (1st order).
Default offset is 0.
2022-11-01 21:05:41 +01:00
- **float getTempOffset()** return current temperature offset, default 0.
2022-01-11 20:48:39 +01:00
2023-02-10 16:22:51 +01:00
#### Asynchronous interface
2022-01-11 20:48:39 +01:00
2022-11-01 21:05:41 +01:00
There are two timings that need to be considered (from datasheet):
- time between requests = 1000 ms.
- time between request and data ready = 80 ms.
2022-09-11 11:44:04 +02:00
2022-09-18 19:13:39 +02:00
The async interface allows one to continue processing after a **requestData()** has been made.
2022-11-01 21:05:41 +01:00
Note there should be at least **1000 milliseconds** between subsequent requests.
2022-09-11 11:44:04 +02:00
2022-09-18 19:13:39 +02:00
With **bool isMeasuring()** one can check if a new measurement is ready.
2022-11-01 21:05:41 +01:00
Alternative is to delay for up to 80 ms.
2022-09-18 19:13:39 +02:00
If so the sensor can be read with **readData()**.
2022-09-11 11:44:04 +02:00
To interpret the read bits to temperature, humidity and status one needs to call **convert()** as last step.
2022-01-11 20:48:39 +01:00
- **int requestData()** signals the sensor to make a new measurement.
2022-09-11 11:44:04 +02:00
Note there must be at least 1000 milliseconds between requests!
2022-01-11 20:48:39 +01:00
- **int readData()** does the actual reading of the data.
2022-09-11 11:44:04 +02:00
- **int convert()** converts the read bits to temperature and humidity.
See the example **DHT20_async.ino**
2022-01-11 20:48:39 +01:00
2023-02-10 16:22:51 +01:00
#### Status
2022-01-11 20:48:39 +01:00
2022-09-16 19:27:01 +02:00
- **uint8_t readStatus()** forced read of the status only.
2022-09-18 19:13:39 +02:00
This function blocks a few milliseconds to optimize communication.
2022-09-17 13:50:11 +02:00
- **bool isCalibrated()** idem, wrapper around **readStatus()**
- **bool isMeasuring()** idem, wrapper around **readStatus()**
- **bool isIdle()** idem, wrapper around **readStatus()**
2022-09-18 19:13:39 +02:00
- **int internalStatus()** returns the internal status of the sensor. (for debug).
2022-09-16 19:27:01 +02:00
| status bit | meaning |
|:------------:|:---------------------------|
2022-09-18 19:13:39 +02:00
| 7 | 1 = measurement, 0 = idle |
2022-09-16 19:27:01 +02:00
| 6 - 4 | unknown |
2022-09-18 19:13:39 +02:00
| 3 | 1 = calibrated, 0 = not |
2022-09-16 19:27:01 +02:00
| 2 - 0 | unknown |
2022-01-11 20:48:39 +01:00
2022-09-18 19:13:39 +02:00
#### Experimental 0.1.4 resetSensor
2022-11-01 21:05:41 +01:00
Use with care!
2022-09-18 19:13:39 +02:00
- **uint8_t resetSensor()** if at startup the sensor does not return a status of 0x18,
three registers 0x1B, 0x1C and 0x1E need to be reset.
See datasheet 7.4 Sensor Reading Process, point 1.
There is no documentation about the meaning of these registers.
The code is based upon example code for the AHT20 (from manufacturer).
2022-11-01 21:05:41 +01:00
The call is needed to get the **read()** working well so it has been embedded into
the read calls. (0.2.0)
2022-09-18 19:13:39 +02:00
2023-02-10 16:22:51 +01:00
#### Timing
2022-01-11 20:48:39 +01:00
2022-09-17 13:50:11 +02:00
- **uint32_t lastRead()** last time the sensor is read in milliseconds since start.
- **uint32_t lastRequest()** last time a request is made to make a measurement.
2023-02-10 16:22:51 +01:00
#### Return codes
2022-01-11 20:48:39 +01:00
2023-02-10 16:22:51 +01:00
| name | value | notes |
|:-----------------------------|:-------:|:--------|
| DHT20_OK | 00 | OK
| DHT20_ERROR_CHECKSUM | -10 | values might be OK if they are like recent previous ones.
| DHT20_ERROR_CONNECT | -11 | check connection
| DHT20_MISSING_BYTES | -12 | check connection
| DHT20_ERROR_BYTES_ALL_ZERO | -13 | check connection
| DHT20_ERROR_READ_TIMEOUT | -14 |
| DHT20_ERROR_LASTREAD | -15 | wait 1 second between reads
2022-01-11 20:48:39 +01:00
## Future
2023-02-10 16:22:51 +01:00
#### Must
2022-11-01 21:05:41 +01:00
- improve documentation.
- investigate the bug from #8 further
2022-12-21 21:48:21 +01:00
(is done in 0.2.1 see issue #8)
2022-09-11 11:44:04 +02:00
2023-02-10 16:22:51 +01:00
#### Should
#### Could
2022-09-11 11:44:04 +02:00
2022-01-11 20:48:39 +01:00
- improve unit tests.
- investigate
- sensor calibration (website aosong?)
2023-02-10 16:22:51 +01:00
- can sensor address be changed?
2022-09-18 19:13:39 +02:00
- investigate optimizing timing in readStatus()
- delay(1) ==> microSeconds(???).
- connected flag?
2023-02-10 16:22:51 +01:00
- keep in sync DHT12 ?
2022-12-21 21:48:21 +01:00
2022-01-11 20:48:39 +01:00
2023-02-10 16:22:51 +01:00
#### Wont
2022-01-11 20:48:39 +01:00
- **void setIgnoreChecksum(bool = false)** ignore checksum flag speeds up communication a bit
2022-09-18 19:13:39 +02:00
- **bool getIgnoreChecksum()** get checksum flag. for completeness.
2022-01-11 20:48:39 +01:00