1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/drivers/dht/dht.c

147 lines
3.7 KiB
C
Raw Normal View History

/*
* Copyright 2015 Ludwig Knüpfer
* 2015 Christian Mehlis
* 2016-2017 Freie Universität Berlin
*
* This file is subject to the terms and conditions of the GNU Lesser
* General Public License v2.1. See the file LICENSE in the top level
* directory for more details.
*/
/**
2016-01-21 22:33:02 +01:00
* @ingroup drivers_dht
* @{
*
* @file
* @brief Device driver implementation for the DHT 11 and 22
* temperature and humidity sensor
*
* @author Ludwig Knüpfer <ludwig.knuepfer@fu-berlin.de>
* @author Christian Mehlis <mehlis@inf.fu-berlin.de>
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
*
* @}
*/
#include <stdint.h>
#include <string.h>
#include "log.h"
#include "assert.h"
2015-07-30 15:58:43 +02:00
#include "xtimer.h"
#include "timex.h"
#include "periph/gpio.h"
#include "dht.h"
#include "dht_params.h"
2016-01-21 18:42:36 +01:00
#define ENABLE_DEBUG (0)
#include "debug.h"
#define PULSE_WIDTH_THRESHOLD (40U)
static uint16_t read(gpio_t pin, int bits)
{
uint16_t res = 0;
for (int i = 0; i < bits; i++) {
2016-08-21 06:44:40 +02:00
uint32_t start, end;
res <<= 1;
/* measure the length between the next rising and falling flanks (the
* time the pin is high - smoke up :-) */
while (!gpio_read(pin)) {}
start = xtimer_now_usec();
while (gpio_read(pin)) {}
end = xtimer_now_usec();
/* if the high phase was more than 40us, we got a 1 */
if ((end - start) > PULSE_WIDTH_THRESHOLD) {
res |= 0x0001;
}
}
return res;
}
2016-01-21 16:22:26 +01:00
int dht_init(dht_t *dev, const dht_params_t *params)
{
DEBUG("dht_init\n");
/* check parameters and configuration */
assert(dev && params &&
((dev->type == DHT11) || (dev->type == DHT22) || (dev->type == DHT21)));
*dev = *params;
2016-01-21 16:22:26 +01:00
gpio_init(dev->pin, GPIO_OUT);
2016-01-21 16:22:26 +01:00
gpio_set(dev->pin);
xtimer_usleep(2000 * US_PER_MS);
2016-01-21 16:22:26 +01:00
DEBUG("dht_init: success\n");
return DHT_OK;
2016-01-21 16:22:26 +01:00
}
2017-06-20 17:32:45 +02:00
int dht_read(const dht_t *dev, int16_t *temp, int16_t *hum)
2016-01-21 16:22:26 +01:00
{
uint8_t csum, sum;
2016-01-21 18:42:36 +01:00
uint16_t raw_hum, raw_temp;
2016-01-21 16:22:26 +01:00
assert(dev && temp && hum);
/* send init signal to device */
2016-01-21 16:22:26 +01:00
gpio_clear(dev->pin);
xtimer_usleep(20 * US_PER_MS);
2016-01-21 16:22:26 +01:00
gpio_set(dev->pin);
2015-07-30 15:58:43 +02:00
xtimer_usleep(40);
/* sync on device */
2016-03-17 14:55:01 +01:00
gpio_init(dev->pin, dev->in_mode);
while (!gpio_read(dev->pin)) {}
while (gpio_read(dev->pin)) {}
/*
* data is read in sequentially, highest bit first:
* 40 .. 24 23 .. 8 7 .. 0
* [humidity][temperature][checksum]
*/
/* read the humidity, temperature, and checksum bits */
2016-01-21 18:42:36 +01:00
raw_hum = read(dev->pin, 16);
raw_temp = read(dev->pin, 16);
csum = (uint8_t)read(dev->pin, 8);
/* set pin high again - so we can trigger the next reading by pulling it low
* again */
2016-02-20 17:10:02 +01:00
gpio_init(dev->pin, GPIO_OUT);
gpio_set(dev->pin);
2016-01-21 18:42:36 +01:00
/* validate the checksum */
sum = (raw_temp >> 8) + (raw_temp & 0xff) + (raw_hum >> 8) + (raw_hum & 0xff);
if ((sum != csum) || (csum == 0)) {
DEBUG("error: checksum invalid\n");
return DHT_NOCSUM;
}
2016-01-21 18:42:36 +01:00
/* parse the RAW values */
DEBUG("RAW values: temp: %7i hum: %7i\n", (int)raw_temp, (int)raw_hum);
switch (dev->type) {
case DHT11:
2016-01-21 18:42:36 +01:00
*temp = (int16_t)((raw_temp >> 8) * 10);
*hum = (int16_t)((raw_hum >> 8) * 10);
break;
case DHT22:
2016-01-21 18:42:36 +01:00
*hum = (int16_t)raw_hum;
/* if the high-bit is set, the value is negative */
if (raw_temp & 0x8000) {
*temp = (int16_t)((raw_temp & ~0x8000) * -1);
}
else {
2016-01-21 18:42:36 +01:00
*temp = (int16_t)raw_temp;
}
break;
default:
return DHT_NODEV; /* this should never be reached */
}
return DHT_OK;
}