2018-11-21 08:48:20 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2018 Gunar Schorcht
|
|
|
|
*
|
|
|
|
* 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_sht3x
|
|
|
|
* @brief SAUL adaption for Sensirion SHT30/SHT31/SHT35 devices
|
|
|
|
* @author Gunar Schorcht <gunar@schorcht.net>
|
|
|
|
* @file
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <errno.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "phydat.h"
|
|
|
|
#include "saul.h"
|
|
|
|
#include "sht3x.h"
|
|
|
|
#include "sht3x_params.h"
|
|
|
|
|
2019-07-18 15:17:31 +02:00
|
|
|
#define SHT3X_NUM ARRAY_SIZE(sht3x_params)
|
2018-11-21 08:48:20 +01:00
|
|
|
extern sht3x_dev_t sht3x_devs[SHT3X_NUM];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Humidity and temperature sensor values are fetched by separate saul
|
|
|
|
* functions. To avoid double waiting for the sensor and readout of the
|
|
|
|
* sensor values, we read both sensor values once, if necessary, and
|
|
|
|
* store them in local variables to provide them in the separate saul
|
|
|
|
* read functions.
|
|
|
|
*/
|
|
|
|
static bool _temp_valid[SHT3X_NUM] = { false };
|
|
|
|
static bool _hum_valid[SHT3X_NUM] = { false };
|
|
|
|
static int16_t _temp[SHT3X_NUM];
|
|
|
|
static int16_t _hum[SHT3X_NUM];
|
|
|
|
|
|
|
|
static unsigned _dev2index (const sht3x_dev_t *dev)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* returns the index of the device in sht3x_devs[] or SHT3_NUM
|
|
|
|
* if not found
|
|
|
|
*/
|
|
|
|
for (unsigned i = 0; i < SHT3X_NUM; i++) {
|
|
|
|
if (dev == &sht3x_devs[i]) {
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return SHT3X_NUM;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int read(int dev)
|
|
|
|
{
|
|
|
|
/* read both sensor values */
|
|
|
|
unsigned res = sht3x_read(&sht3x_devs[dev], &_temp[dev], &_hum[dev]);
|
|
|
|
if (res != SHT3X_OK) {
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
/* mark both sensor values as valid */
|
|
|
|
_temp_valid[dev] = true;
|
|
|
|
_hum_valid[dev] = true;
|
|
|
|
return SHT3X_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int read_temp(const void *dev, phydat_t *data)
|
|
|
|
{
|
|
|
|
/* find the device index */
|
|
|
|
unsigned dev_index = _dev2index((const sht3x_dev_t *)dev);
|
|
|
|
if (dev_index == SHT3X_NUM) {
|
|
|
|
/* return with error if device index could not be found */
|
|
|
|
return -ECANCELED;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* either local variable is valid or fetching it was successful */
|
|
|
|
if (_temp_valid[dev_index] || read(dev_index) == SHT3X_OK) {
|
|
|
|
/* mark local variable as invalid */
|
|
|
|
_temp_valid[dev_index] = false;
|
|
|
|
|
|
|
|
data->val[0] = _temp[dev_index];
|
|
|
|
data->unit = UNIT_TEMP_C;
|
|
|
|
data->scale = -2;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
return -ECANCELED;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int read_hum(const void *dev, phydat_t *data)
|
|
|
|
{
|
|
|
|
/* find the device index */
|
|
|
|
unsigned dev_index = _dev2index((const sht3x_dev_t *)dev);
|
|
|
|
if (dev_index == SHT3X_NUM) {
|
|
|
|
/* return with error if device index could not be found */
|
|
|
|
return -ECANCELED;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* either local variable is valid or fetching it was successful */
|
|
|
|
if (_hum_valid[dev_index] || read(dev_index) == SHT3X_OK) {
|
|
|
|
/* mark local variable as invalid */
|
|
|
|
_hum_valid[dev_index] = false;
|
|
|
|
|
|
|
|
data->val[0] = _hum[dev_index];
|
|
|
|
data->unit = UNIT_PERCENT;
|
|
|
|
data->scale = -2;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
return -ECANCELED;
|
|
|
|
}
|
|
|
|
|
|
|
|
const saul_driver_t sht3x_saul_driver_temperature = {
|
|
|
|
.read = read_temp,
|
|
|
|
.write = saul_notsup,
|
|
|
|
.type = SAUL_SENSE_TEMP
|
|
|
|
};
|
|
|
|
|
|
|
|
const saul_driver_t sht3x_saul_driver_humidity = {
|
|
|
|
.read = read_hum,
|
|
|
|
.write = saul_notsup,
|
|
|
|
.type = SAUL_SENSE_HUM
|
|
|
|
};
|