Merge pull request #159 from sheinz/fix/dht
DHT11/DHT22 sensor library fixed.
This commit is contained in:
		
						commit
						6915caf49c
					
				
					 3 changed files with 182 additions and 106 deletions
				
			
		| 
						 | 
					@ -2,6 +2,8 @@
 | 
				
			||||||
 *
 | 
					 *
 | 
				
			||||||
 * This sample code is in the public domain.
 | 
					 * This sample code is in the public domain.
 | 
				
			||||||
 */
 | 
					 */
 | 
				
			||||||
 | 
					#include <stdio.h>
 | 
				
			||||||
 | 
					#include <stdlib.h>
 | 
				
			||||||
#include "espressif/esp_common.h"
 | 
					#include "espressif/esp_common.h"
 | 
				
			||||||
#include "esp/uart.h"
 | 
					#include "esp/uart.h"
 | 
				
			||||||
#include "FreeRTOS.h"
 | 
					#include "FreeRTOS.h"
 | 
				
			||||||
| 
						 | 
					@ -13,12 +15,12 @@
 | 
				
			||||||
 * to read and print a new temperature and humidity measurement
 | 
					 * to read and print a new temperature and humidity measurement
 | 
				
			||||||
 * from a sensor attached to GPIO pin 4.
 | 
					 * from a sensor attached to GPIO pin 4.
 | 
				
			||||||
 */
 | 
					 */
 | 
				
			||||||
int const dht_gpio = 4;
 | 
					uint8_t const dht_gpio = 4;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void dhtMeasurementTask(void *pvParameters)
 | 
					void dhtMeasurementTask(void *pvParameters)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
  int8_t temperature = 0;
 | 
					    int16_t temperature = 0;
 | 
				
			||||||
  int8_t humidity = 0;
 | 
					    int16_t humidity = 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // DHT sensors that come mounted on a PCB generally have
 | 
					    // DHT sensors that come mounted on a PCB generally have
 | 
				
			||||||
    // pull-up resistors on the data pin.  It is recommended
 | 
					    // pull-up resistors on the data pin.  It is recommended
 | 
				
			||||||
| 
						 | 
					@ -26,11 +28,12 @@ void dhtMeasurementTask(void *pvParameters)
 | 
				
			||||||
    gpio_set_pullup(dht_gpio, false, false);
 | 
					    gpio_set_pullup(dht_gpio, false, false);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    while(1) {
 | 
					    while(1) {
 | 
				
			||||||
 | 
					        if (dht_read_data(dht_gpio, &humidity, &temperature)) {
 | 
				
			||||||
    if (dht_fetch_data(dht_gpio, &humidity, &temperature)) {
 | 
					            printf("Humidity: %d%% Temp: %dC\n", 
 | 
				
			||||||
      printf("Humidity: %i%% Temp: %iC\n", humidity, temperature);
 | 
					                    humidity / 10, 
 | 
				
			||||||
 | 
					                    temperature / 10);
 | 
				
			||||||
        } else {
 | 
					        } else {
 | 
				
			||||||
      printf("Could not read data from sensor...");
 | 
					            printf("Could not read data from sensor\n");
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        // Three second delay...
 | 
					        // Three second delay...
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
							
								
								
									
										214
									
								
								extras/dht/dht.c
									
										
									
									
									
								
							
							
						
						
									
										214
									
								
								extras/dht/dht.c
									
										
									
									
									
								
							| 
						 | 
					@ -6,17 +6,20 @@
 | 
				
			||||||
 */
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#include "dht.h"
 | 
					#include "dht.h"
 | 
				
			||||||
 | 
					#include "FreeRTOS.h"
 | 
				
			||||||
#include "string.h"
 | 
					#include "string.h"
 | 
				
			||||||
#include "task.h"
 | 
					#include "task.h"
 | 
				
			||||||
#include "esp/gpio.h"
 | 
					#include "esp/gpio.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#include <espressif/esp_misc.h> // sdk_os_delay_us
 | 
					#include <espressif/esp_misc.h> // sdk_os_delay_us
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#ifndef DEBUG_DHT
 | 
					// DHT timer precision in microseconds
 | 
				
			||||||
#define DEBUG_DHT 0
 | 
					#define DHT_TIMER_INTERVAL   2
 | 
				
			||||||
#endif
 | 
					#define DHT_DATA_BITS  40
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#if DEBUG_DHT
 | 
					// #define DEBUG_DHT
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#ifdef DEBUG_DHT
 | 
				
			||||||
#define debug(fmt, ...) printf("%s" fmt "\n", "dht: ", ## __VA_ARGS__);
 | 
					#define debug(fmt, ...) printf("%s" fmt "\n", "dht: ", ## __VA_ARGS__);
 | 
				
			||||||
#else
 | 
					#else
 | 
				
			||||||
#define debug(fmt, ...) /* (do nothing) */
 | 
					#define debug(fmt, ...) /* (do nothing) */
 | 
				
			||||||
| 
						 | 
					@ -42,7 +45,7 @@
 | 
				
			||||||
 *  and then letting it float back high for a period of time that depends on the data bit.
 | 
					 *  and then letting it float back high for a period of time that depends on the data bit.
 | 
				
			||||||
 *  duration_data_high is shorter than 50us for a logic '0' and longer than 50us for logic '1'.
 | 
					 *  duration_data_high is shorter than 50us for a logic '0' and longer than 50us for logic '1'.
 | 
				
			||||||
 *
 | 
					 *
 | 
				
			||||||
 *  There are a total of 40 data bits trasnmitted sequentially. These bits are read into a byte array
 | 
					 *  There are a total of 40 data bits transmitted sequentially. These bits are read into a byte array
 | 
				
			||||||
 *  of length 5.  The first and third bytes are humidity (%) and temperature (C), respectively.  Bytes 2 and 4
 | 
					 *  of length 5.  The first and third bytes are humidity (%) and temperature (C), respectively.  Bytes 2 and 4
 | 
				
			||||||
 *  are zero-filled and the fifth is a checksum such that:
 | 
					 *  are zero-filled and the fifth is a checksum such that:
 | 
				
			||||||
 *
 | 
					 *
 | 
				
			||||||
| 
						 | 
					@ -50,95 +53,144 @@
 | 
				
			||||||
 *
 | 
					 *
 | 
				
			||||||
*/
 | 
					*/
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/*  
 | 
					
 | 
				
			||||||
 *  @pin                    the selected GPIO pin
 | 
					/**
 | 
				
			||||||
 *  @interval               how frequently the pin state is checked in microseconds
 | 
					 * Wait specified time for pin to go to a specified state.
 | 
				
			||||||
 *  @timeout                maximum length of time to wait for the expected pin state
 | 
					 * If timeout is reached and pin doesn't go to a requested state
 | 
				
			||||||
 *  @expected_pin_state     high (true) or low (false) pin state
 | 
					 * false is returned.
 | 
				
			||||||
 *  @counter                pointer to external uint8_t for tallying the duration waited for the pin state
 | 
					 * The elapsed time is returned in pointer 'duration' if it is not NULL.
 | 
				
			||||||
 */
 | 
					 */
 | 
				
			||||||
 | 
					static bool dht_await_pin_state(uint8_t pin, uint32_t timeout,
 | 
				
			||||||
bool dht_await_pin_state(uint8_t pin, uint8_t interval, uint8_t timeout, bool expected_pin_state, uint8_t * counter) {
 | 
					        bool expected_pin_state, uint32_t *duration)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
    for (*counter = 0; *counter < timeout; *counter+=interval) {
 | 
					    for (uint32_t i = 0; i < timeout; i += DHT_TIMER_INTERVAL) {
 | 
				
			||||||
        if (gpio_read(pin) == expected_pin_state) return true;
 | 
					        // need to wait at least a single interval to prevent reading a jitter
 | 
				
			||||||
        sdk_os_delay_us(interval);
 | 
					        sdk_os_delay_us(DHT_TIMER_INTERVAL);
 | 
				
			||||||
 | 
					        if (gpio_read(pin) == expected_pin_state) {
 | 
				
			||||||
 | 
					            if (duration) {
 | 
				
			||||||
 | 
					                *duration = i;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            return true;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    return false;
 | 
					    return false;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/*  
 | 
					/**
 | 
				
			||||||
 *  
 | 
					 * Request data from DHT and read raw bit stream.
 | 
				
			||||||
 *  
 | 
					 * The function call should be protected from task switching.
 | 
				
			||||||
 *  @pin                    the selected GPIO pin
 | 
					 * Return false if error occurred.
 | 
				
			||||||
 *  @humidity               pointer to external int8_t to store resulting humidity value
 | 
					 | 
				
			||||||
 *  @temperature            pointer to external int8_t to store resulting temperature value
 | 
					 | 
				
			||||||
 */
 | 
					 */
 | 
				
			||||||
 | 
					static inline bool dht_fetch_data(uint8_t pin, bool bits[DHT_DATA_BITS])
 | 
				
			||||||
bool dht_fetch_data(int8_t pin, int8_t * humidity, int8_t * temperature) {
 | 
					{
 | 
				
			||||||
    int8_t data[40] = {0};
 | 
					    uint32_t low_duration;
 | 
				
			||||||
    int8_t result[5] = {0};
 | 
					    uint32_t high_duration;
 | 
				
			||||||
    uint8_t i = 0;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    uint8_t init_phase_duration = 0;
 | 
					 | 
				
			||||||
    uint8_t duration_data_low = 0;
 | 
					 | 
				
			||||||
    uint8_t duration_data_high = 0;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    gpio_enable(pin, GPIO_OUT_OPEN_DRAIN);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    taskENTER_CRITICAL();
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // Phase 'A' pulling signal low to initiate read sequence
 | 
					    // Phase 'A' pulling signal low to initiate read sequence
 | 
				
			||||||
    gpio_write(pin, 0);
 | 
					    gpio_write(pin, 0);
 | 
				
			||||||
    sdk_os_delay_us(20000);
 | 
					    sdk_os_delay_us(20000);
 | 
				
			||||||
    gpio_write(pin, 1);
 | 
					    gpio_write(pin, 1);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // Step through Phase 'B' at 2us intervals, 40us max
 | 
					    // Step through Phase 'B', 40us
 | 
				
			||||||
    if (dht_await_pin_state(pin, 2, 40, false, &init_phase_duration)) {
 | 
					    if (!dht_await_pin_state(pin, 40, false, NULL)) {
 | 
				
			||||||
        // Step through Phase 'C ' at 2us intervals, 88us max
 | 
					        debug("Initialization error, problem in phase 'B'\n");
 | 
				
			||||||
        if (dht_await_pin_state(pin, 2, 88, true, &init_phase_duration)) {
 | 
					 | 
				
			||||||
            // Step through Phase 'D' at 2us intervals, 88us max
 | 
					 | 
				
			||||||
            if (dht_await_pin_state(pin, 2, 88, false, &init_phase_duration)) {
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
                // Read in each of the 40 bits of data...
 | 
					 | 
				
			||||||
                for (i = 0; i < 40; i++) {
 | 
					 | 
				
			||||||
                    if (dht_await_pin_state(pin, 2, 60, true, &duration_data_low)) {
 | 
					 | 
				
			||||||
                        if (dht_await_pin_state(pin, 2, 75, false, &duration_data_high)) {
 | 
					 | 
				
			||||||
                            data[i] = duration_data_high > duration_data_low;
 | 
					 | 
				
			||||||
                        }
 | 
					 | 
				
			||||||
                    }
 | 
					 | 
				
			||||||
                }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
                taskEXIT_CRITICAL();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
                for (i = 0; i < 40; i++) {
 | 
					 | 
				
			||||||
                    // Read each bit into 'result' byte array...
 | 
					 | 
				
			||||||
                    result[i/8] <<= 1;
 | 
					 | 
				
			||||||
                    result[i/8] |= data[i];
 | 
					 | 
				
			||||||
                }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
                if (result[4] == ((result[0] + result[1] + result[2] + result[3]) & 0xFF)) {
 | 
					 | 
				
			||||||
                    // Data valid, checksum succeeded...
 | 
					 | 
				
			||||||
                    *humidity = result[0];
 | 
					 | 
				
			||||||
                    *temperature = result[2];
 | 
					 | 
				
			||||||
                    debug("Successfully retrieved sensor data...");
 | 
					 | 
				
			||||||
                    return true;
 | 
					 | 
				
			||||||
                } else {
 | 
					 | 
				
			||||||
                    debug("Checksum failed, invalid data received from sensor...");
 | 
					 | 
				
			||||||
                }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
            } else {
 | 
					 | 
				
			||||||
                debug("Initialization error, problem in phase 'D'...");
 | 
					 | 
				
			||||||
            }
 | 
					 | 
				
			||||||
        } else {
 | 
					 | 
				
			||||||
            debug("Initialization error, problem in phase 'C'...");
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
    } else {
 | 
					 | 
				
			||||||
        debug("Initialization error, problem in phase 'B'...");
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
    
 | 
					 | 
				
			||||||
    taskEXIT_CRITICAL();
 | 
					 | 
				
			||||||
        return false;
 | 
					        return false;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // Step through Phase 'C', 88us
 | 
				
			||||||
 | 
					    if (!dht_await_pin_state(pin, 88, true, NULL)) {
 | 
				
			||||||
 | 
					        debug("Initialization error, problem in phase 'C'\n");
 | 
				
			||||||
 | 
					        return false;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // Step through Phase 'D', 88us
 | 
				
			||||||
 | 
					    if (!dht_await_pin_state(pin, 88, false, NULL)) {
 | 
				
			||||||
 | 
					        debug("Initialization error, problem in phase 'D'\n");
 | 
				
			||||||
 | 
					        return false;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // Read in each of the 40 bits of data...
 | 
				
			||||||
 | 
					    for (int i = 0; i < DHT_DATA_BITS; i++) {
 | 
				
			||||||
 | 
					        if (!dht_await_pin_state(pin, 65, true, &low_duration)) {
 | 
				
			||||||
 | 
					            debug("LOW bit timeout\n");
 | 
				
			||||||
 | 
					            return false;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        if (!dht_await_pin_state(pin, 75, false, &high_duration)){
 | 
				
			||||||
 | 
					            debug("HIGHT bit timeout\n");
 | 
				
			||||||
 | 
					            return false;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        bits[i] = high_duration > low_duration;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    return true;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/**
 | 
				
			||||||
 | 
					 * Pack two data bytes into single value and take into account sign bit.
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					static inline int16_t dht_convert_data(uint8_t msb, uint8_t lsb)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					    int16_t data;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#if DHT_TYPE == DHT22
 | 
				
			||||||
 | 
					    data = msb & 0x7F;
 | 
				
			||||||
 | 
					    data <<= 8;
 | 
				
			||||||
 | 
					    data |= lsb;
 | 
				
			||||||
 | 
					    if (msb & BIT(7)) {
 | 
				
			||||||
 | 
					        data = 0 - data;       // convert it to negative
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					#elif DHT_TYPE == DHT11
 | 
				
			||||||
 | 
					    data = msb * 10;
 | 
				
			||||||
 | 
					#else
 | 
				
			||||||
 | 
					#error "Unsupported DHT type"
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return data;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					bool dht_read_data(uint8_t pin, int16_t *humidity, int16_t *temperature)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					    bool bits[DHT_DATA_BITS];
 | 
				
			||||||
 | 
					    uint8_t data[DHT_DATA_BITS/8] = {0};
 | 
				
			||||||
 | 
					    bool result;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    gpio_enable(pin, GPIO_OUT_OPEN_DRAIN);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    taskENTER_CRITICAL();
 | 
				
			||||||
 | 
					    result = dht_fetch_data(pin, bits);
 | 
				
			||||||
 | 
					    taskEXIT_CRITICAL();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if (!result) {
 | 
				
			||||||
 | 
					        return false;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    for (uint8_t i = 0; i < DHT_DATA_BITS; i++) {
 | 
				
			||||||
 | 
					        // Read each bit into 'result' byte array...
 | 
				
			||||||
 | 
					        data[i/8] <<= 1;
 | 
				
			||||||
 | 
					        data[i/8] |= bits[i];
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if (data[4] != ((data[0] + data[1] + data[2] + data[3]) & 0xFF)) {
 | 
				
			||||||
 | 
					        debug("Checksum failed, invalid data received from sensor\n");
 | 
				
			||||||
 | 
					        return false;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    *humidity = dht_convert_data(data[0], data[1]);
 | 
				
			||||||
 | 
					    *temperature = dht_convert_data(data[2], data[3]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    debug("Sensor data: humidity=%d, temp=%d\n", *humidity, *temperature);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return true;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					bool dht_read_float_data(uint8_t pin, float *humidity, float *temperature)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					    int16_t i_humidity, i_temp;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if (dht_read_data(pin, &i_humidity, &i_temp)) {
 | 
				
			||||||
 | 
					        *humidity = (float)i_humidity / 10;
 | 
				
			||||||
 | 
					        *temperature = (float)i_temp / 10;
 | 
				
			||||||
 | 
					        return true;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    return false;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -5,13 +5,34 @@
 | 
				
			||||||
 *
 | 
					 *
 | 
				
			||||||
 */
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#ifndef __DTH_H__
 | 
					#ifndef __DHT_H__
 | 
				
			||||||
#define __DHT_H__
 | 
					#define __DHT_H__
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#include "FreeRTOS.h"
 | 
					#include <stdint.h>
 | 
				
			||||||
 | 
					#include <stdbool.h>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
bool dht_wait_for_pin_state(uint8_t pin, uint8_t interval, uint8_t timeout, bool expected_pin_sate, uint8_t * counter);
 | 
					#define DHT11       11
 | 
				
			||||||
bool dht_fetch_data(int8_t pin, int8_t * humidity, int8_t * temperature);
 | 
					#define DHT22       22
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#endif
 | 
					// Type of sensor to use
 | 
				
			||||||
 | 
					#define DHT_TYPE    DHT22
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/**
 | 
				
			||||||
 | 
					 * Read data from sensor on specified pin.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * Humidity and temperature is returned as integers.
 | 
				
			||||||
 | 
					 * For example: humidity=625 is 62.5 %
 | 
				
			||||||
 | 
					 *              temperature=24.4 is 24.4 degrees Celsius
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					bool dht_read_data(uint8_t pin, int16_t *humidity, int16_t *temperature);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/**
 | 
				
			||||||
 | 
					 * Float version of dht_read_data.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * Return values as floating point values.
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					bool dht_read_float_data(uint8_t pin, float *humidity, float *temperature);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#endif  // __DHT_H__
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue