mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
103 lines
2.6 KiB
C
103 lines
2.6 KiB
C
/*
|
|
* Copyright (C) 2018 Koen Zandberg <koen@bergzand.net>
|
|
* Copyright (C) 2018 Beduino Master Projekt - University of Bremen
|
|
*
|
|
* 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.
|
|
*/
|
|
|
|
/**
|
|
* @ingroup drivers_mhz19
|
|
* @{
|
|
*
|
|
* @file
|
|
* @brief Device driver implementation for MH-Z19 CO2 sensor.
|
|
*
|
|
* @author Koen Zandberg <koen@bergzand.net>
|
|
* @author Christian Manal <manal@uni-bremen.de>
|
|
*
|
|
* @}
|
|
*/
|
|
|
|
#include "mhz19.h"
|
|
#include "mhz19_params.h"
|
|
#include "xtimer.h"
|
|
#include "mutex.h"
|
|
|
|
#define ENABLE_DEBUG 0
|
|
#include "debug.h"
|
|
|
|
#ifdef MODULE_MHZ19_PWM
|
|
int mhz19_init(mhz19_t *dev, const mhz19_params_t *params)
|
|
{
|
|
int16_t res;
|
|
|
|
dev->pin = params->pin;
|
|
|
|
if (gpio_init(dev->pin, GPIO_IN_PD) != 0) {
|
|
return MHZ19_ERR_INIT;
|
|
}
|
|
|
|
/* Take one measurement to make sure there's an mhz19 on the pin */
|
|
if (mhz19_get_ppm(dev, &res) == MHZ19_ERR_TIMEOUT) {
|
|
return MHZ19_ERR_TIMEOUT;
|
|
}
|
|
|
|
return MHZ19_OK;
|
|
}
|
|
|
|
int mhz19_get_ppm(mhz19_t *dev, int16_t *ppm)
|
|
{
|
|
uint32_t start, middle, end, th, tl;
|
|
/*
|
|
* Per the docs, one sample should take 1004ms +-5%. Worst case is
|
|
* that we come in right after the rising edge of the current cycle,
|
|
* so we want to wait two cycles plus some wiggle room at most for
|
|
* a measurement.
|
|
*/
|
|
int16_t timeout = 2200;
|
|
|
|
DEBUG("%s: Waiting for high level to end\n", __func__);
|
|
while (gpio_read(dev->pin) && timeout) {
|
|
timeout--;
|
|
xtimer_usleep(US_PER_MS);
|
|
}
|
|
|
|
DEBUG("%s: Waiting for initial rising edge\n", __func__);
|
|
while (!gpio_read(dev->pin) && timeout) {
|
|
timeout--;
|
|
xtimer_usleep(US_PER_MS);
|
|
}
|
|
|
|
start = xtimer_now_usec() / US_PER_MS;
|
|
DEBUG("%s: Waiting for falling edge\n", __func__);
|
|
while (gpio_read(dev->pin) && timeout) {
|
|
timeout--;
|
|
xtimer_usleep(US_PER_MS);
|
|
}
|
|
middle = xtimer_now_usec() / US_PER_MS;
|
|
DEBUG("%s: Waiting for rising edge\n", __func__);
|
|
while (!gpio_read(dev->pin) && timeout) {
|
|
timeout--;
|
|
xtimer_usleep(US_PER_MS);
|
|
}
|
|
|
|
/* If we waited too long for flanks, something went wrong */
|
|
if (!timeout) {
|
|
DEBUG("%s: Measurement timed out\n", __func__);
|
|
return MHZ19_ERR_TIMEOUT;
|
|
}
|
|
end = xtimer_now_usec() / US_PER_MS;
|
|
|
|
th = (middle - start);
|
|
tl = (end - middle);
|
|
|
|
*ppm = (int16_t)(2000 * (th - 2) / (th + tl - 4));
|
|
|
|
return MHZ19_OK;
|
|
}
|
|
#else
|
|
typedef int dont_be_pedantic;
|
|
#endif /* MODULE_MHZ19_PWM */
|