2018-09-28 23:48:47 +02: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_ccs811
|
|
|
|
* @brief Device Driver for AMS CCS811 digital gas sensor
|
|
|
|
* @author Gunar Schorcht <gunar@schorcht.net>
|
|
|
|
* @file
|
|
|
|
* @{
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef CCS811_H
|
|
|
|
#define CCS811_H
|
|
|
|
|
|
|
|
#include <stdint.h>
|
2018-12-13 11:43:20 +01:00
|
|
|
#include "periph/gpio.h"
|
2018-09-28 23:48:47 +02:00
|
|
|
#include "periph/i2c.h"
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C"
|
|
|
|
{
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @name CCS811 I2C addresses
|
|
|
|
* @{
|
|
|
|
*/
|
|
|
|
#define CCS811_I2C_ADDRESS_1 (0x5A) /**< default */
|
|
|
|
#define CCS811_I2C_ADDRESS_2 (0x5B)
|
|
|
|
/** @} */
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @name CCS811 IAQ value ranges
|
|
|
|
* @{
|
|
|
|
*/
|
|
|
|
#define CCS811_ECO2_RANGE_MIN (400) /**< eCO2 min in ppm */
|
|
|
|
#define CCS811_ECO2_RANGE_MAX (8192) /**< eCO2 max in ppm */
|
|
|
|
#define CCS811_TVOC_RANGE_MIN (0) /**< TVOC min in ppb */
|
|
|
|
#define CCS811_TVOC_RANGE_MAX (1187) /**< TVOC min in ppb */
|
|
|
|
/** @} */
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Driver error codes (returned as negative values)
|
|
|
|
*/
|
|
|
|
typedef enum {
|
|
|
|
CCS811_OK, /**< no error */
|
|
|
|
CCS811_ERROR_I2C, /**< I2C communication failure */
|
|
|
|
CCS811_ERROR_NO_DEV, /**< device not available */
|
|
|
|
CCS811_ERROR_NO_APP, /**< could not start application */
|
|
|
|
CCS811_ERROR_NO_NEW_DATA, /**< no new data (last valid data returned) */
|
|
|
|
CCS811_ERROR_NO_IAQ_DATA, /**< IAQ data not available in this mode */
|
|
|
|
CCS811_ERROR_WRITE_REG_INV, /**< invalid register address on write */
|
|
|
|
CCS811_ERROR_READ_REG_INV, /**< invalid register address on read */
|
|
|
|
CCS811_ERROR_MEASMODE_INV, /**< invalid measurement mode */
|
|
|
|
CCS811_ERROR_THRESH_INV, /**< invalid threshold parameters */
|
|
|
|
CCS811_ERROR_MAX_RESISTANCE, /**< maximum sensor resistance exceeded */
|
|
|
|
CCS811_ERROR_HEATER_FAULT, /**< heater current not in range */
|
|
|
|
CCS811_ERROR_HEATER_SUPPLY, /**< heater voltage not applied correctly */
|
|
|
|
CCS811_ERROR_NO_INT_PIN, /**< nINT signal pin not configured */
|
|
|
|
CCS811_ERROR_NO_WAKE_PIN, /**< nWAKE signal pin not configured */
|
|
|
|
CCS811_ERROR_NO_RESET_PIN, /**< nRESET signal pin not configured */
|
|
|
|
CCS811_ERROR_NOT_SUPPORTED, /**< function is not supported */
|
|
|
|
} ccs811_error_codes_t;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief CCS811 operation modes
|
|
|
|
*/
|
|
|
|
typedef enum {
|
|
|
|
CCS811_MODE_IDLE = 0, /**< Idle, low current mode */
|
|
|
|
CCS811_MODE_1S = 1, /**< Constant Power mode, IAQ values every 1 s */
|
|
|
|
CCS811_MODE_10S = 2, /**< Pulse Heating mode, IAQ values every 10 s */
|
|
|
|
CCS811_MODE_60S = 3, /**< Low Power Pulse Heating, IAQ values every 60 s */
|
|
|
|
CCS811_MODE_250MS = 4 /**< Constant Power mode, only RAW data every 250 ms */
|
|
|
|
} ccs811_mode_t;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief CCS811 interrupt mode
|
|
|
|
*/
|
|
|
|
typedef enum {
|
|
|
|
CCS811_INT_NONE = 0, /**< interrupt generation is disabled (default) */
|
|
|
|
CCS811_INT_DATA_READY, /**< nINT signal when new data are reade to read */
|
|
|
|
CCS811_INT_THRESHOLD, /**< nINT signal when new data reach thresholds */
|
|
|
|
} ccs811_int_mode_t;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief CCS811 device initialization parameters
|
|
|
|
*/
|
|
|
|
typedef struct {
|
|
|
|
|
|
|
|
i2c_t i2c_dev; /**< I2C device, clock stretching required (default I2C_DEV(0)) */
|
|
|
|
uint8_t i2c_addr; /**< I2C address (default CCS811_I2C_ADDRESS_1) */
|
2018-12-13 11:46:45 +01:00
|
|
|
ccs811_mode_t mode; /**< measurement mode used (default #CCS811_MODE_IDLE) */
|
|
|
|
#if MODULE_CCS811_FULL || DOXYGEN
|
2018-09-28 23:48:47 +02:00
|
|
|
gpio_t int_pin; /**< nINT signal pin (default GPIO_PIN(0, 0) */
|
2018-12-13 11:46:45 +01:00
|
|
|
ccs811_int_mode_t int_mode; /**< interrupt mode used (default #CCS811_INT_NONE) */
|
|
|
|
#endif
|
2018-09-28 23:48:47 +02:00
|
|
|
gpio_t wake_pin; /**< nWAKE signal pin (default GPIO_UNDEF) */
|
|
|
|
gpio_t reset_pin; /**< nRESET signal pin (default GPIO_UNDEF) */
|
|
|
|
} ccs811_params_t;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief CCS811 sensor device data structure
|
|
|
|
*/
|
|
|
|
typedef struct {
|
|
|
|
ccs811_params_t params; /**< device initialization parameters */
|
|
|
|
} ccs811_t;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Initialize a CCS811 sensor device
|
|
|
|
*
|
|
|
|
* The function resets the CCS811 sensor, checks its availability and
|
|
|
|
* initializes it according to the given configuration parameters.
|
|
|
|
*
|
|
|
|
* If #ccs811_params_t::reset_pin is configured
|
|
|
|
* - the pin is used for the hardware reset of the sensor, otherwise
|
|
|
|
* only a software reset is tried, and
|
|
|
|
* - the #ccs811_init function can be used at any time to reset the sensor.
|
|
|
|
*
|
|
|
|
* If #ccs811_params_t::wake_pin is configured, it use to switch the sensor
|
|
|
|
* into the sleep mode while the I2C interface is not used.
|
|
|
|
*
|
|
|
|
* @param[in] dev Device descriptor of CCS811 device to be initialized
|
|
|
|
* @param[in] params Configuration parameters used by initialization
|
|
|
|
*
|
|
|
|
* @note The I2C implementation of the MCU has to support clock stretching
|
|
|
|
* to get CCS811 working.
|
|
|
|
*
|
|
|
|
* @retval CCS811_OK on success
|
|
|
|
* @retval CCS811_ERROR_* on error, see #ccs811_error_codes_t
|
|
|
|
*/
|
|
|
|
int ccs811_init (ccs811_t *dev, const ccs811_params_t *params);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Read IAQ sensor values and/or RAW sensor data
|
|
|
|
*
|
|
|
|
* The function reads the IAQ sensor values (TVOC and eCO2) and/or the raw
|
|
|
|
* sensor data. For either \p iaq_tvoc2, \p iaq_eco2, \p raw_i, or \p raw_v
|
|
|
|
* also ```NULL``` can be passed, if their value are not of interest.
|
|
|
|
*
|
|
|
|
* @note
|
|
|
|
* - If the function is called and no new data are available, the function
|
|
|
|
* returns the results of the last measurement and the error code
|
|
|
|
* #CCS811_ERROR_NO_NEW_DATA.
|
|
|
|
* - The data-ready status function #ccs811_data_ready or the data-ready
|
|
|
|
* interrupt (#CCS811_INT_DATA_READY) can be used to determine whether
|
|
|
|
* new data are available.
|
|
|
|
* - In #CCS811_MODE_250MS, only RAW data are available. In
|
|
|
|
* that case, the function fails with error_code #CCS811_ERROR_NO_IAQ_DATA
|
|
|
|
* if \p iaq_tvoc and \p iaq_eco2 parameters are not ```NULL```.
|
|
|
|
*
|
|
|
|
* @param[in] dev Device descriptor of CCS811 device to read from
|
|
|
|
* @param[out] iaq_tvoc TVOC total volatile organic compound (0..1187 ppb)
|
|
|
|
* @param[out] iaq_eco2 eCO2 equivalent CO2 (400 - 8192 ppm)
|
|
|
|
* @param[out] raw_i Current through the sensor used for measuring (0..63 uA)
|
|
|
|
* @param[out] raw_v Voltage across the sensor measured (0..1023 = 1.65 V)
|
|
|
|
*
|
|
|
|
* @retval CCS811_OK on success and new data are returned
|
|
|
|
* @retval CCS811_ERROR_NO_NEW_DATA when no new data are available and last
|
|
|
|
* measurement results are returned.
|
|
|
|
* @retval CCS811_ERROR_* otherwise, see #ccs811_error_codes_t.
|
|
|
|
*/
|
|
|
|
int ccs811_read_iaq (const ccs811_t *dev,
|
|
|
|
uint16_t *iaq_tvoc, uint16_t *iaq_eco2,
|
|
|
|
uint16_t *raw_i, uint16_t *raw_v);
|
|
|
|
|
|
|
|
#if MODULE_CCS811_FULL || DOXYGEN
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Read the resistance of connected NTC thermistor
|
|
|
|
*
|
|
|
|
* CCS811 supports an external interface for connecting a negative thermal
|
|
|
|
* coefficient thermistor (R_NTC) to provide a cost effective and power
|
|
|
|
* efficient means of calculating the local ambient temperature. The sensor
|
|
|
|
* measures the voltage V_NTC across the R_NTC as well as the voltage V_REF
|
|
|
|
* across a connected reference resistor (R_REF).
|
|
|
|
*
|
|
|
|
* The function returns the current resistance of R_NTC using the equation
|
|
|
|
*
|
|
|
|
* R_NTC = R_REF / V_REF * V_NTC
|
|
|
|
*
|
|
|
|
* Using the data sheet of the NTC, the ambient temperature can be calculated.
|
|
|
|
*
|
|
|
|
* @param[in] dev Device descriptor of CCS811 device to read from
|
|
|
|
* @param[in] r_ref Resistance of R_REF in Ohm
|
|
|
|
* @param[out] r_ntc Resistance of R_NTC in Ohm
|
|
|
|
*
|
|
|
|
* @retval CCS811_OK on success
|
|
|
|
* @retval CCS811_ERROR_* on error, see #ccs811_error_codes_t
|
|
|
|
*/
|
|
|
|
int ccs811_read_ntc (const ccs811_t *dev, uint32_t r_ref, uint32_t *r_ntc);
|
|
|
|
|
|
|
|
#endif /* MODULE_CCS811_FULL || DOXYGEN */
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Data-ready status function
|
|
|
|
*
|
|
|
|
* The function reads the status register and returns CSS811_OK when new
|
|
|
|
* data are available. The function is usefull for polling the sensor.
|
|
|
|
*
|
|
|
|
* @param[in] dev Device descriptor of CCS811 device to read from
|
|
|
|
*
|
|
|
|
* @retval CCS811_OK when new data are available
|
|
|
|
* @retval CCS811_ERROR_NO_NEW_DATA when no new data are available
|
|
|
|
* @retval CCS811_ERROR_* otherwise, see #ccs811_error_codes_t.
|
|
|
|
*/
|
|
|
|
int ccs811_data_ready (const ccs811_t *dev);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Power down the sensor
|
|
|
|
*
|
|
|
|
* The feature disables sensor measurements by entering idle mode
|
|
|
|
* (#CCS811_MODE_IDLE). In addition, the function sets the low active
|
|
|
|
* signal ** nWAKE ** to high to completely deactivate the sensor.
|
|
|
|
* The sensor is then no longer accessible via I2C. The last sensor
|
|
|
|
* measurement mode is saved.
|
|
|
|
*
|
|
|
|
* The low active **nWAKE** signal pin has to be configured
|
|
|
|
* (#ccs811_params_t::wake_pin) accordingly. Otherwise, the function fails
|
|
|
|
* and returns with #CCS811_ERROR_NO_WAKE_PIN.
|
|
|
|
*
|
|
|
|
* @param[in] dev Device descriptor of CCS811 device to read from
|
|
|
|
*
|
|
|
|
* @retval CCS811_OK on success
|
|
|
|
* @retval CCS811_ERROR_NO_WAKE_PIN #ccs811_params_t::wake_pin not configured
|
|
|
|
* @retval CCS811_ERROR_* on error, see #ccs811_error_codes_t
|
|
|
|
*/
|
|
|
|
int ccs811_power_down (ccs811_t *dev);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Power up the sensor
|
|
|
|
*
|
|
|
|
* The function sets the low active signal ** nWAKE ** to low to activate the
|
|
|
|
* sensor and switches back from the idle mode (#CCS811_MODE_IDLE) to the
|
|
|
|
* last measurement mode.
|
|
|
|
*
|
|
|
|
* The low active **nWAKE** signal pin has to be configured
|
|
|
|
* (#ccs811_params_t::wake_pin) accordingly. Otherwise, the function fails
|
|
|
|
* and returns with #CCS811_ERROR_NO_WAKE_PIN.
|
|
|
|
*
|
|
|
|
* @note It may take several minutes before accurate readings are generated
|
|
|
|
* when the sensor switches back from idle mode to the previous
|
|
|
|
* measurement mode.
|
|
|
|
*
|
|
|
|
* @param[in] dev Device descriptor of CCS811 device to read from
|
|
|
|
*
|
|
|
|
* @retval CCS811_OK on success
|
|
|
|
* @retval CCS811_ERROR_NO_WAKE_PIN #ccs811_params_t::wake_pin not configured
|
|
|
|
* @retval CCS811_ERROR_* on error, see #ccs811_error_codes_t
|
|
|
|
*/
|
|
|
|
int ccs811_power_up (ccs811_t *dev);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Set the operation mode of the sensor
|
|
|
|
*
|
|
|
|
* The function sets the operating mode of the sensor.
|
|
|
|
*
|
|
|
|
* The sensor starts periodic measurements with the specified period if the
|
|
|
|
* \p mode parameter is either
|
|
|
|
*
|
|
|
|
* - #CCS811_MODE_1S,
|
|
|
|
* - #CCS811_MODE_10S,
|
|
|
|
* - #CCS811_MODE_60S, or
|
|
|
|
* - #CCS811_MODE_250MS.
|
|
|
|
*
|
|
|
|
* The #ccs811_read_iaq function can then be used to get sensor data at the
|
|
|
|
* same rate to get the results.
|
|
|
|
*
|
|
|
|
* In case, the \p mode parameter is #CCS811_MODE_IDLE, the sensor does not
|
|
|
|
* perform any measurements.
|
|
|
|
*
|
|
|
|
* @note
|
|
|
|
* - In #CCS811_MODE_250MS, only raw data are available. IAQ values would
|
|
|
|
* have to be calculated by the host in this mode.
|
|
|
|
* - Mode timings (the period) are subject to typical 2% tolerance due to
|
|
|
|
* accuracy of internal sensor clock.
|
|
|
|
* - After setting the sensor mode, the sensor needs up to 20 minutes, before
|
|
|
|
* accurate readings are generated.
|
|
|
|
* - When the sensor operating mode is changed to a new mode with
|
|
|
|
* a lower sample rate, e.g., from #CCS811_MODE_60S to #CCS811_MODE_1S, it
|
|
|
|
* should be placed in #CCS811_MODE_IDLE for at least 10 minutes before
|
|
|
|
* enabling the new mode.
|
|
|
|
*
|
|
|
|
* @param[in] dev Device descriptor of CCS811 device
|
|
|
|
* @param[in] mode CCS811 operation mode
|
|
|
|
*
|
|
|
|
* @retval CCS811_OK on success
|
|
|
|
* @retval CCS811_ERROR_* on error, see #ccs811_error_codes_t
|
|
|
|
*/
|
|
|
|
int ccs811_set_mode (ccs811_t *dev, ccs811_mode_t mode);
|
|
|
|
|
|
|
|
#if MODULE_CCS811_FULL || DOXYGEN
|
|
|
|
/**
|
|
|
|
* @brief Enable/disable data ready or threshold interrupt signal **nINT**
|
|
|
|
*
|
|
|
|
* CCS811 can trigger either
|
|
|
|
*
|
|
|
|
* - a data-ready interrupt (#CCS811_INT_DATA_READY) when new data become
|
|
|
|
* available or
|
|
|
|
* - a threshold interrupt (#CCS811_INT_THRESHOLD) if the new eCO2 data
|
|
|
|
* exceed defined thresholds (see #ccs811_set_eco2_thresholds).
|
|
|
|
*
|
|
|
|
* As soon as an interrupt condition occurs, the signal **nINT** is driven
|
|
|
|
* low. It stops being driven low when the sensor data is read with the
|
|
|
|
* #ccs811_read_iaq function. #ccs811_params_t::int_pin parameter has
|
|
|
|
* to be configured.
|
|
|
|
*
|
|
|
|
* With #CCS811_INT_NONE (the default), interrupt generation is disabled.
|
|
|
|
* The interrupt generation is disabled by default.
|
|
|
|
*
|
|
|
|
* @param[in] dev Device descriptor of CCS811 device
|
|
|
|
* @param[in] mode Enable the interrupt if true, otherwise disable it
|
|
|
|
*
|
|
|
|
* @retval CCS811_OK on success
|
|
|
|
* @retval CCS811_ERROR_NO_INT_PIN #ccs811_params_t::int_pin not configured
|
|
|
|
* @retval CCS811_ERROR_* on error, see #ccs811_error_codes_t
|
|
|
|
*/
|
|
|
|
int ccs811_set_int_mode (ccs811_t *dev, ccs811_int_mode_t mode);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Set environmental data
|
|
|
|
*
|
|
|
|
* If information about the environment are available from another sensor,
|
|
|
|
* they can be used by CCS811 to compensate gas readings due to
|
|
|
|
* temperature and humidity changes.
|
|
|
|
*
|
|
|
|
* @param[in] dev Device descriptor of CCS811 device
|
|
|
|
* @param[in] temp Temperature [hundredths of a degree Celsius]
|
|
|
|
* @param[in] hum Relative Humidity [hundredths of a percent]
|
|
|
|
*
|
|
|
|
* @retval CCS811_OK on success
|
|
|
|
* @retval CCS811_ERROR_* on error, see #ccs811_error_codes_t
|
|
|
|
*/
|
|
|
|
int ccs811_set_environmental_data (const ccs811_t *dev,
|
|
|
|
int16_t temp, int16_t hum);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Set eCO2 thresholds for threshold interrupts
|
|
|
|
*
|
|
|
|
* Threshold interrupts, if enabled (#ccs811_int_mode_t), are generated when
|
|
|
|
* new eCO2 value moves from the current range (LOW, MEDIUM or HIGH) to
|
|
|
|
* another range by more than one hysteresis value. The hysteresis is used
|
|
|
|
* to prevent multiple interrupts near a threshold.
|
|
|
|
*
|
|
|
|
* Ranges are defined as following:
|
|
|
|
*
|
|
|
|
* Name | Range | Value | Default
|
|
|
|
* :------|:------------------------------------------|:-------|:-------
|
|
|
|
* LOW | below the \p low parameter | > 400 | 1500
|
|
|
|
* MEDIUM | between the \p low and \p high parameters | | |
|
|
|
|
* HIGH | above the value of the \p high parameter | < 8192 | 2500
|
|
|
|
*
|
|
|
|
* @param[in] dev Device descriptor of CCS811 device
|
|
|
|
* @param[in] low Threshold LOW to MEDIUM
|
|
|
|
* @param[in] high Threshold MEDIUM to HIGH
|
|
|
|
* @param[in] hyst Hysteresis value (default 50)
|
|
|
|
*
|
|
|
|
* @retval CCS811_OK on success
|
|
|
|
* @retval CCS811_ERROR_* on error, see #ccs811_error_codes_t
|
|
|
|
*/
|
|
|
|
int ccs811_set_eco2_thresholds (const ccs811_t *dev,
|
|
|
|
uint16_t low, uint16_t high, uint8_t hyst);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Get the current baseline value from sensor
|
|
|
|
*
|
|
|
|
* The sensor supports automatic baseline correction over a minimum time of
|
|
|
|
* 24 hours. Using this function, the current baseline value can be saved
|
|
|
|
* before the sensor is powered down. This baseline can then be restored using
|
|
|
|
* the #ccs811_set_baseline function after sensor is powered up again to
|
|
|
|
* continue the automatic baseline process.
|
|
|
|
*
|
|
|
|
* @param[in] dev Device descriptor of CCS811 device
|
|
|
|
* @param[in] baseline Current baseline value
|
|
|
|
*
|
|
|
|
* @retval CCS811_OK on success
|
|
|
|
* @retval CCS811_ERROR_* on error, see #ccs811_error_codes_t
|
|
|
|
*/
|
|
|
|
int ccs811_get_baseline (const ccs811_t *dev, uint16_t *baseline);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Write a previously stored baseline value to the sensor
|
|
|
|
*
|
|
|
|
* The sensor supports automatic baseline correction over a minimum time of
|
|
|
|
* 24 hours. Using this function, a previously saved baseline
|
|
|
|
* (#ccs811_get_baseline) value can be restored after the sensor is powered
|
|
|
|
* up to continue the automatic baseline process.
|
|
|
|
*
|
|
|
|
* @note The baseline must be written after the conditioning period
|
|
|
|
* of 20 min after power up.
|
|
|
|
*
|
|
|
|
* @param[in] dev Device descriptor of CCS811 device
|
|
|
|
* @param[in] baseline Stored baseline value
|
|
|
|
*
|
|
|
|
* @retval CCS811_OK on success
|
|
|
|
* @retval CCS811_ERROR_* on error, see #ccs811_error_codes_t
|
|
|
|
*/
|
|
|
|
int ccs811_set_baseline (const ccs811_t *dev, uint16_t baseline);
|
|
|
|
#endif /* MODULE_CCS811_FULL || DOXYGEN */
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#endif /* CCS811_H */
|
|
|
|
/** @} */
|