2021-03-23 12:09:27 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2021 Inria
|
|
|
|
*
|
|
|
|
* 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_sgp30
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Device driver implementation for the sensors
|
|
|
|
*
|
|
|
|
* @author Francisco Molina <francois-xavier.molina@inria.fr>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
#include <string.h>
|
|
|
|
|
2022-09-26 18:53:05 +02:00
|
|
|
#include "checksum/crc8.h"
|
|
|
|
#include "irq.h"
|
2021-03-23 12:09:27 +01:00
|
|
|
#include "sgp30.h"
|
|
|
|
#include "sgp30_constants.h"
|
|
|
|
#include "sgp30_params.h"
|
|
|
|
#include "ztimer.h"
|
|
|
|
|
|
|
|
#define ENABLE_DEBUG 0
|
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @name Address pointer values for all SGP30 I2C commands
|
|
|
|
* @{
|
|
|
|
*/
|
|
|
|
typedef enum {
|
|
|
|
SGP30_CMD_INIT_AIR_QUALITY = 0x2003, /**< Start measurement mode */
|
|
|
|
SGP30_CMD_MEASURE_AIR_QUALITY = 0x2008, /**< Measure air quality values */
|
|
|
|
SGP30_CMD_GET_BASELINE = 0x2015, /**< Get baseline values */
|
|
|
|
SGP30_CMD_SET_BASELINE = 0x201e, /**< Set baseline values CMD */
|
|
|
|
SGP30_CMD_SET_HUMIDITY = 0x2061, /**< Set absolute humidity value
|
|
|
|
for compensation */
|
|
|
|
SGP30_CMD_MEASURE_TEST = 0x2032, /**< Perform on-chip self test */
|
|
|
|
SGP30_CMD_GET_FEATURE_SET_VERSION = 0x202f, /**< Get feature set version */
|
|
|
|
SGP30_CMD_MEASURE_RAW_SIGNALS = 0x2050, /**< Read raw H2 and Ethanol signals */
|
|
|
|
SGP30_CMD_READ_SERIAL = 0x3682, /**< Read serial number */
|
|
|
|
SGP30_CMD_SOFT_RESET = 0x0006, /**< Perform General Call reset */
|
|
|
|
} sgp30_cmd_t;
|
|
|
|
/** @} */
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @name Max delays for SGP30 I2C commands completion
|
|
|
|
* @{
|
|
|
|
*/
|
|
|
|
#define SGP30_DELAY_INIT_AIR_QUALITY (10 * US_PER_MS)
|
|
|
|
#define SGP30_DELAY_MEASURE_AIR_QUALITY (12 * US_PER_MS)
|
|
|
|
#define SGP30_DELAY_GET_BASELINE (10 * US_PER_MS)
|
|
|
|
#define SGP30_DELAY_SET_BASELINE (10 * US_PER_MS)
|
|
|
|
#define SGP30_DELAY_SET_HUMIDITY (10 * US_PER_MS)
|
|
|
|
#define SGP30_DELAY_MEASURE_TEST (220 * US_PER_MS)
|
|
|
|
#define SGP30_DELAY_GET_FEATURE_SET_VERSION (2 * US_PER_MS)
|
|
|
|
#define SGP30_DELAY_MEASURE_RAW_SIGNALS (25 * US_PER_MS)
|
|
|
|
#define SGP30_DELAY_READ_SERIAL (500)
|
|
|
|
#define SGP30_DELAY_SOFT_RESET (60 * US_PER_MS)
|
|
|
|
/** @} */
|
|
|
|
|
|
|
|
/* Device power up time */
|
|
|
|
#define SGP30_POWER_UP_TIME (60 * US_PER_MS)
|
|
|
|
/* Polynomial used in crc check */
|
|
|
|
#define SGP30_CRC8_POLYNOMIAL 0x31
|
|
|
|
/* Seed used in crc check */
|
|
|
|
#define SGP30_CRC8_SEED 0xFF
|
|
|
|
|
|
|
|
static inline uint8_t _crc8(const void *buf, size_t len)
|
|
|
|
{
|
|
|
|
return crc8(buf, len, SGP30_CRC8_POLYNOMIAL, SGP30_CRC8_SEED);
|
|
|
|
}
|
|
|
|
|
|
|
|
void _set_uint16_and_crc(uint8_t *buf, uint16_t *val)
|
|
|
|
{
|
|
|
|
buf[0] = *val >> 8;
|
|
|
|
buf[1] = *val & 0xFF;
|
|
|
|
buf[2] = _crc8(buf, sizeof(uint16_t));
|
|
|
|
}
|
|
|
|
|
|
|
|
int _get_uint16_and_check_crc(uint8_t *buf, uint16_t *val)
|
|
|
|
{
|
|
|
|
if (_crc8(buf, sizeof(uint16_t)) == buf[2]) {
|
|
|
|
*val = (buf[0] << 8) + buf[1];
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return -EBADMSG;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _rx_tx_data(sgp30_t *dev, uint16_t cmd, uint8_t *data,
|
|
|
|
size_t len, uint32_t delay, bool do_read)
|
|
|
|
{
|
|
|
|
int res = 0;
|
|
|
|
|
2021-11-25 13:43:06 +01:00
|
|
|
i2c_acquire(dev->params.i2c_dev);
|
2021-03-23 12:09:27 +01:00
|
|
|
|
|
|
|
uint8_t frame_cmd[sizeof(cmd) + len];
|
|
|
|
frame_cmd[0] = cmd >> 8;
|
|
|
|
frame_cmd[1] = cmd & 0xFF;
|
|
|
|
|
|
|
|
if (len == 0 || do_read) {
|
|
|
|
res = i2c_write_bytes(dev->params.i2c_dev, SGP30_I2C_ADDRESS,
|
|
|
|
&frame_cmd[0], sizeof(cmd), 0);
|
|
|
|
}
|
|
|
|
if (res == 0 && do_read) {
|
|
|
|
/* delay for command completion */
|
|
|
|
if (!irq_is_in()) {
|
|
|
|
ztimer_sleep(ZTIMER_USEC, delay);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
ztimer_spin(ZTIMER_USEC, delay);
|
|
|
|
}
|
|
|
|
res = i2c_read_bytes(dev->params.i2c_dev, SGP30_I2C_ADDRESS,
|
|
|
|
data, len, 0);
|
|
|
|
|
|
|
|
}
|
|
|
|
else if (res == 0 && !do_read) {
|
|
|
|
memcpy(&frame_cmd[2], data, len);
|
|
|
|
res = i2c_write_bytes(dev->params.i2c_dev, SGP30_I2C_ADDRESS,
|
|
|
|
frame_cmd, sizeof(cmd) + len, 0);
|
|
|
|
if (res) {
|
|
|
|
DEBUG_PUTS("[sgp30]: failed to write data");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
i2c_release(dev->params.i2c_dev);
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
int _read_measurements(sgp30_t *dev, sgp30_data_t *data)
|
|
|
|
{
|
|
|
|
uint8_t frame[6];
|
|
|
|
|
|
|
|
if (_rx_tx_data(dev, SGP30_CMD_MEASURE_AIR_QUALITY, frame, sizeof(frame),
|
|
|
|
SGP30_DELAY_MEASURE_AIR_QUALITY, true)) {
|
|
|
|
return -EPROTO;
|
|
|
|
}
|
|
|
|
if (_get_uint16_and_check_crc(&frame[0], &data->eco2) ||
|
|
|
|
_get_uint16_and_check_crc(&frame[3], &data->tvoc)) {
|
|
|
|
return -EBADMSG;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef MODULE_SGP30_STRICT
|
|
|
|
static void _read_cb(void *arg)
|
|
|
|
{
|
|
|
|
sgp30_t *dev = (sgp30_t *)arg;
|
|
|
|
|
|
|
|
if (!dev->ready) {
|
|
|
|
dev->ready = true;
|
|
|
|
}
|
|
|
|
_read_measurements(dev, &dev->_data);
|
|
|
|
dev->_data.timestamp = ztimer_now(ZTIMER_USEC);
|
|
|
|
ztimer_set(ZTIMER_USEC, &dev->_timer, SGP30_RECOMMENDED_SAMPLING_PERIOD);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
int sgp30_start_air_quality(sgp30_t *dev)
|
|
|
|
{
|
|
|
|
int ret = _rx_tx_data(dev, SGP30_CMD_INIT_AIR_QUALITY, NULL, 0,
|
|
|
|
SGP30_DELAY_INIT_AIR_QUALITY, false);
|
|
|
|
|
|
|
|
#ifdef MODULE_SGP30_STRICT
|
|
|
|
if (ret == 0) {
|
|
|
|
ztimer_set(ZTIMER_USEC, &dev->_timer, SGP30_AIR_QUALITY_INIT_DELAY_US);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
return ret ? -EPROTO : 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int sgp30_init(sgp30_t *dev, const sgp30_params_t *params)
|
|
|
|
{
|
|
|
|
assert(dev && params);
|
|
|
|
dev->params = *params;
|
|
|
|
#ifdef MODULE_SGP30_STRICT
|
|
|
|
dev->ready = false;
|
|
|
|
dev->_timer.callback = _read_cb;
|
|
|
|
dev->_timer.arg = dev;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* delay while powering up */
|
|
|
|
ztimer_sleep(ZTIMER_USEC, SGP30_POWER_UP_TIME);
|
|
|
|
|
|
|
|
/* read future set */
|
|
|
|
uint16_t version;
|
|
|
|
sgp30_read_future_set(dev, &version);
|
|
|
|
if (version < SGP30_REQUIRED_FEATURE_SET) {
|
|
|
|
DEBUG("[sgp30]: wrong version number, %04x instead of %04x\n",
|
|
|
|
version, SGP30_REQUIRED_FEATURE_SET);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* read serial id */
|
|
|
|
uint8_t serial[SGP30_SERIAL_ID_LEN];
|
|
|
|
if (sgp30_read_serial_number(dev, serial, SGP30_SERIAL_ID_LEN)) {
|
|
|
|
DEBUG_PUTS("[sgp30]: could not read serial number");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
if (IS_ACTIVE(ENABLE_DEBUG)) {
|
|
|
|
DEBUG_PUTS("[sgp30]: serial");
|
|
|
|
for (uint8_t i = 0; i < SGP30_SERIAL_ID_LEN; i++) {
|
|
|
|
DEBUG("%02x ", serial[i]);
|
|
|
|
}
|
|
|
|
DEBUG_PUTS("\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
/* start air quality measurement */
|
|
|
|
if (sgp30_start_air_quality(dev)) {
|
|
|
|
DEBUG_PUTS("[sgp30]: could not start air quality measurements ");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int sgp30_reset(sgp30_t *dev)
|
|
|
|
{
|
|
|
|
assert(dev);
|
|
|
|
int ret = _rx_tx_data(dev, SGP30_CMD_SOFT_RESET, NULL, 0, SGP30_DELAY_SOFT_RESET, false);
|
|
|
|
#ifdef MODULE_SGP30_STRICT
|
|
|
|
if (ret == 0) {
|
|
|
|
ztimer_remove(ZTIMER_USEC, &dev->_timer);
|
|
|
|
dev->ready = false;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
return ret ? -EPROTO : 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int sgp30_read_serial_number(sgp30_t *dev, uint8_t *buf, size_t len)
|
|
|
|
{
|
2021-04-09 18:19:33 +02:00
|
|
|
(void) len;
|
2021-03-23 12:09:27 +01:00
|
|
|
assert(dev && buf && (len == SGP30_SERIAL_ID_LEN));
|
|
|
|
uint8_t frame[9];
|
|
|
|
if (_rx_tx_data(dev, SGP30_CMD_READ_SERIAL, (uint8_t *)frame, sizeof(frame),
|
|
|
|
SGP30_DELAY_READ_SERIAL, true)) {
|
|
|
|
DEBUG_PUTS("[sgp30]: fail read");
|
|
|
|
return -EPROTO;
|
|
|
|
}
|
|
|
|
/* the serial id is in big endian format */
|
|
|
|
uint16_t tmp[SGP30_SERIAL_ID_LEN/2];
|
|
|
|
if (_get_uint16_and_check_crc(&frame[0], &tmp[2]) ||
|
|
|
|
_get_uint16_and_check_crc(&frame[3], &tmp[1]) ||
|
|
|
|
_get_uint16_and_check_crc(&frame[6], &tmp[0])) {
|
|
|
|
DEBUG_PUTS("[sgp30]: wrong crc");
|
|
|
|
return -EBADMSG;
|
|
|
|
}
|
|
|
|
memcpy(buf, tmp, SGP30_SERIAL_ID_LEN);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int sgp30_read_future_set(sgp30_t *dev, uint16_t *version)
|
|
|
|
{
|
|
|
|
uint8_t frame[3];
|
|
|
|
|
|
|
|
if (_rx_tx_data(dev, SGP30_CMD_GET_FEATURE_SET_VERSION, frame, sizeof(frame),
|
|
|
|
SGP30_DELAY_GET_FEATURE_SET_VERSION, true)) {
|
|
|
|
return -EPROTO;
|
|
|
|
}
|
|
|
|
if (_get_uint16_and_check_crc(&frame[0], version)) {
|
|
|
|
DEBUG_PUTS("[sgp30]: wrong crc");
|
|
|
|
return -EBADMSG;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int sgp30_set_absolute_humidity(sgp30_t *dev, uint32_t a_humidity)
|
|
|
|
{
|
|
|
|
/* max value is (255g/m3+255/256g/m3), or 255999 mg/m3*/
|
|
|
|
if (a_humidity > 256000LU) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* scale down to g/m^3 */
|
|
|
|
uint16_t humidity_scaled =
|
|
|
|
(uint16_t)(((uint64_t)a_humidity * 256 * 16777) >> 24);
|
|
|
|
|
|
|
|
uint8_t frame[3];
|
|
|
|
_set_uint16_and_crc(&frame[0], &humidity_scaled);
|
|
|
|
int ret = _rx_tx_data(dev, SGP30_CMD_SET_HUMIDITY, frame, sizeof(frame),
|
|
|
|
SGP30_DELAY_SET_HUMIDITY, false);
|
|
|
|
return ret ? -EPROTO : 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int sgp30_set_baseline(sgp30_t *dev, sgp30_data_t *data)
|
|
|
|
{
|
|
|
|
uint8_t frame[6];
|
|
|
|
|
|
|
|
_set_uint16_and_crc(&frame[0], &data->eco2);
|
|
|
|
_set_uint16_and_crc(&frame[3], &data->tvoc);
|
|
|
|
|
|
|
|
int ret = _rx_tx_data(dev, SGP30_CMD_SET_BASELINE, frame, sizeof(frame),
|
|
|
|
SGP30_DELAY_SET_BASELINE, false);
|
|
|
|
return ret ? -EPROTO : 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int sgp30_get_baseline(sgp30_t *dev, sgp30_data_t *data)
|
|
|
|
{
|
|
|
|
uint8_t frame[6];
|
|
|
|
|
|
|
|
if (_rx_tx_data(dev, SGP30_CMD_GET_BASELINE, frame, sizeof(frame),
|
|
|
|
SGP30_DELAY_GET_BASELINE, true)) {
|
|
|
|
return -EPROTO;
|
|
|
|
}
|
|
|
|
if (_get_uint16_and_check_crc(&frame[0], &data->eco2) ||
|
|
|
|
_get_uint16_and_check_crc(&frame[3], &data->tvoc)) {
|
|
|
|
return -EBADMSG;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int sgp30_read_measurements(sgp30_t *dev, sgp30_data_t *data)
|
|
|
|
{
|
|
|
|
#ifdef MODULE_SGP30_STRICT
|
|
|
|
if (dev->ready) {
|
|
|
|
unsigned state = irq_disable();
|
|
|
|
memcpy(data, &dev->_data, sizeof(sgp30_data_t));
|
|
|
|
irq_restore(state);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return -EAGAIN;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
#else
|
|
|
|
return _read_measurements(dev, data);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
int sgp30_read_raw_measurements(sgp30_t *dev, sgp30_raw_data_t *data)
|
|
|
|
{
|
|
|
|
uint8_t frame[6];
|
|
|
|
|
|
|
|
if (_rx_tx_data(dev, SGP30_CMD_MEASURE_RAW_SIGNALS, frame, sizeof(frame),
|
|
|
|
SGP30_DELAY_MEASURE_RAW_SIGNALS, true)) {
|
|
|
|
return -EPROTO;
|
|
|
|
}
|
|
|
|
if (_get_uint16_and_check_crc(&frame[0], &data->raw_ethanol) ||
|
|
|
|
_get_uint16_and_check_crc(&frame[3], &data->raw_h2)) {
|
|
|
|
DEBUG_PUTS("[sgp30]: wrong crc");
|
|
|
|
return -EBADMSG;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef MODULE_SGP30_STRICT
|
|
|
|
bool sgp30_ready(sgp30_t *dev)
|
|
|
|
{
|
|
|
|
return dev->ready;
|
|
|
|
}
|
|
|
|
#endif
|