mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
312 lines
9.4 KiB
C
312 lines
9.4 KiB
C
/*
|
|
* Copyright (C) 2016 Kees Bakker, SODAQ
|
|
* 2017 Inria
|
|
* 2018 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.
|
|
*/
|
|
|
|
/**
|
|
* @defgroup drivers_bmx280 BMP280/BME280 temperature, pressure and humidity sensor
|
|
* @ingroup drivers_sensors
|
|
* @brief Device driver interface for the Bosch BMP280 and BME280 sensors
|
|
*
|
|
* BMP280 and BME280 measure temperature in centi °C and pressure in Pa. BME280
|
|
* can also measure relative humidity in %.
|
|
*
|
|
* For more information, see the datasheets:
|
|
* * [BMP280](https://ae-bst.resource.bosch.com/media/_tech/media/datasheets/BST-BMP280-DS001.pdf)
|
|
* * [BME280](https://ae-bst.resource.bosch.com/media/_tech/media/datasheets/BST-BME280-DS002.pdf)
|
|
*
|
|
* This driver provides @ref drivers_saul capabilities.
|
|
*
|
|
* ## Usage
|
|
*
|
|
* To include this driver to your application, simply add one of the following
|
|
* to the application's Makefile:
|
|
*
|
|
* ```make
|
|
* # BME280 connected via SPI
|
|
* USEMODULE += bme280_spi
|
|
* # BME280 connected via I2C
|
|
* USEMODULE += bme280_i2c
|
|
* # BMP280 via SPI
|
|
* USEMODULE += bmp280_spi
|
|
* # BMP280 via I2C
|
|
* USEMODULE += bmp280_i2c
|
|
*
|
|
* # When using I2C, specify the default I2C device to use,
|
|
* # and the BME280's address (see datasheet).
|
|
* # The values below are the defaults:
|
|
* CFLAGS += -DBMX280_PARAM_I2C_DEV=I2C_DEV\(0\)
|
|
* CFLAGS += -DBMX280_PARAM_I2C_ADDR=0x77
|
|
*
|
|
* # Using SPI, you can override the default configuration by specifying the
|
|
* # used SPI bus and the ship select pin:
|
|
* CFLAGS += -DBMX280_PARAM_SPI_DEV=SPI_DEV\(x\)
|
|
* CFLAGS += -DBMX280_PARAM_CS=GPIO_PIN\(y,z\)
|
|
* ```
|
|
*
|
|
* This way the default parameters in `drivers/bmx280/include/bmx280_params.h`
|
|
* are replaced by these new values.
|
|
*
|
|
* @{
|
|
* @file
|
|
* @brief Device driver interface for the BMP280 and BME280 sensors
|
|
*
|
|
* @details There are three sensor values that can be read: temperature,
|
|
* pressure and humidity. The BME280 device usually measures them
|
|
* all at once. It is possible to skip measuring either of the
|
|
* values by changing the oversampling settings. The driver is
|
|
* written in such a way that a measurement is only started from
|
|
* the function that reads the temperature. In other words, you
|
|
* always have to call bmx280_read_temperature, and then optionally
|
|
* you can call the other two.
|
|
*
|
|
* @author Kees Bakker <kees@sodaq.com>
|
|
* @author Alexandre Abadie <alexandre.abadie@inria.fr>
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
|
*/
|
|
|
|
#ifndef BMX280_H
|
|
#define BMX280_H
|
|
|
|
#include <stdint.h>
|
|
#include "saul.h"
|
|
|
|
#if defined(MODULE_BME280_SPI) || defined(MODULE_BMP280_SPI)
|
|
#define BMX280_USE_SPI
|
|
#include "periph/spi.h"
|
|
#else
|
|
#include "periph/i2c.h"
|
|
#endif
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
|
|
/**
|
|
* @brief Select the number or raw data bytes depending on the device type
|
|
*/
|
|
#if defined(MODULE_BME280_SPI) || defined(MODULE_BME280_I2C)
|
|
#define BMX280_RAW_LEN (8U)
|
|
#else
|
|
#define BMX280_RAW_LEN (6U)
|
|
#endif
|
|
|
|
/**
|
|
* @brief Calibration struct for the BMX280 sensor
|
|
*
|
|
* This must be read from the device at startup.
|
|
*/
|
|
typedef struct {
|
|
uint16_t dig_T1; /**< T1 coefficient */
|
|
int16_t dig_T2; /**< T2 coefficient */
|
|
int16_t dig_T3; /**< T3 coefficient */
|
|
|
|
uint16_t dig_P1; /**< P1 coefficient */
|
|
int16_t dig_P2; /**< P2 coefficient */
|
|
int16_t dig_P3; /**< P3 coefficient */
|
|
int16_t dig_P4; /**< P4 coefficient */
|
|
int16_t dig_P5; /**< P5 coefficient */
|
|
int16_t dig_P6; /**< P6 coefficient */
|
|
int16_t dig_P7; /**< P7 coefficient */
|
|
int16_t dig_P8; /**< P8 coefficient */
|
|
int16_t dig_P9; /**< P9 coefficient */
|
|
|
|
uint8_t dig_H1; /**< H1 coefficient */
|
|
int16_t dig_H2; /**< H2 coefficient */
|
|
uint8_t dig_H3; /**< H3 coefficient */
|
|
int16_t dig_H4; /**< H4 coefficient */
|
|
int16_t dig_H5; /**< H5 coefficient */
|
|
int8_t dig_H6; /**< H6 coefficient */
|
|
} bmx280_calibration_t;
|
|
|
|
/**
|
|
* @brief Values for t_sb field of the BMX280 config register
|
|
*/
|
|
typedef enum {
|
|
BMX280_SB_0_5 = 0x00,
|
|
BMX280_SB_62_5 = 0x20,
|
|
BMX280_SB_125 = 0x40,
|
|
BMX280_SB_250 = 0x60,
|
|
BMX280_SB_500 = 0x80,
|
|
BMX280_SB_1000 = 0xa0,
|
|
BMX280_SB_10 = 0xc0,
|
|
BMX280_SB_20 = 0xe0,
|
|
} bmx280_t_sb_t;
|
|
|
|
/**
|
|
* @brief Values for filter field of the BMX280 config register
|
|
*/
|
|
typedef enum {
|
|
BMX280_FILTER_OFF = 0x00,
|
|
BMX280_FILTER_2 = 0x04,
|
|
BMX280_FILTER_4 = 0x08,
|
|
BMX280_FILTER_8 = 0x0c,
|
|
BMX280_FILTER_16 = 0x10,
|
|
} bmx280_filter_t;
|
|
|
|
/**
|
|
* @brief Values for mode field of the BMX280 ctrl_meas register
|
|
*/
|
|
typedef enum {
|
|
BMX280_MODE_SLEEP = 0x00,
|
|
BMX280_MODE_FORCED = 0x01,
|
|
BMX280_MODE_NORMAL = 0x03,
|
|
} bmx280_mode_t;
|
|
|
|
/**
|
|
* @brief Values for oversampling settings
|
|
*
|
|
* These values are used for:
|
|
* - osrs_h field of the BME280 ctrl_hum register
|
|
* - osrs_t field of the BMX280 ctrl_meas register
|
|
* - osrs_p field of the BMX280 ctrl_meas register
|
|
*/
|
|
typedef enum {
|
|
BMX280_OSRS_SKIPPED = 0x00,
|
|
BMX280_OSRS_X1 = 0x01,
|
|
BMX280_OSRS_X2 = 0x02,
|
|
BMX280_OSRS_X4 = 0x03,
|
|
BMX280_OSRS_X8 = 0x04,
|
|
BMX280_OSRS_X16 = 0x05,
|
|
} bmx280_osrs_t;
|
|
|
|
/**
|
|
* @brief Parameters for the BMX280 sensor
|
|
*
|
|
* These parameters are needed to configure the device at startup.
|
|
*/
|
|
typedef struct {
|
|
#ifdef BMX280_USE_SPI
|
|
/* SPI configuration */
|
|
spi_t spi; /**< SPI bus */
|
|
spi_clk_t clk; /**< clock speed for the SPI bus */
|
|
gpio_t cs; /**< chip select pin */
|
|
#else
|
|
/* I2C details */
|
|
i2c_t i2c_dev; /**< I2C device which is used */
|
|
uint8_t i2c_addr; /**< I2C address */
|
|
#endif
|
|
|
|
/* Config Register */
|
|
bmx280_t_sb_t t_sb; /**< standby */
|
|
bmx280_filter_t filter; /**< filter coefficient */
|
|
|
|
/* ctrl_meas */
|
|
bmx280_mode_t run_mode; /**< ctrl_meas mode */
|
|
bmx280_osrs_t temp_oversample; /**< ctrl_meas osrs_t */
|
|
bmx280_osrs_t press_oversample; /**< ctrl_meas osrs_p */
|
|
|
|
/* ctrl_hum */
|
|
bmx280_osrs_t humid_oversample; /**< ctrl_hum osrs_h */
|
|
} bmx280_params_t;
|
|
|
|
/**
|
|
* @brief Device descriptor for the BMX280 sensor
|
|
*/
|
|
typedef struct {
|
|
bmx280_params_t params; /**< Device Parameters */
|
|
bmx280_calibration_t calibration; /**< Calibration Data */
|
|
int32_t t_fine; /**< temperature compensation value */
|
|
uint8_t last_reading[BMX280_RAW_LEN]; /**< cache the RAW data */
|
|
} bmx280_t;
|
|
|
|
/**
|
|
* @brief Status and error return codes
|
|
*/
|
|
enum {
|
|
BMX280_OK = 0, /**< everything was fine */
|
|
BMX280_ERR_BUS = -1, /**< bus error */
|
|
BMX280_ERR_NODEV = -2, /**< did not detect BME280 or BMP280 */
|
|
};
|
|
|
|
/**
|
|
* @brief Export of SAUL interface for temperature sensor
|
|
*/
|
|
extern const saul_driver_t bmx280_temperature_saul_driver;
|
|
|
|
/**
|
|
* @brief Export of SAUL interface for pressure sensor
|
|
*/
|
|
extern const saul_driver_t bmx280_pressure_saul_driver;
|
|
|
|
#if defined(MODULE_BME280_SPI) || defined(MODULE_BME280_I2C)
|
|
/**
|
|
* @brief Export of SAUL interface for humidity sensor
|
|
*/
|
|
extern const saul_driver_t bme280_relative_humidity_saul_driver;
|
|
#endif
|
|
|
|
/**
|
|
* @brief Initialize the given BMX280 device
|
|
*
|
|
* @param[out] dev device descriptor of the given BMX280 device
|
|
* @param[in] params static configuration parameters
|
|
*
|
|
* @return BMX280_OK on success
|
|
* @return BMX280_ERR_BUS on bus error
|
|
* @return BMX280_ERR_NODEV if no corresponding device was found on the bus
|
|
*/
|
|
int bmx280_init(bmx280_t* dev, const bmx280_params_t* params);
|
|
|
|
/**
|
|
* @brief Read temperature value from the given BMX280 device
|
|
*
|
|
* The measured temperature is returned in centi °C (x.xx°C).
|
|
*
|
|
* @param[in] dev device to read from
|
|
*
|
|
* @return measured temperature in centi Celsius
|
|
* @return INT16_MIN on error
|
|
*/
|
|
int16_t bmx280_read_temperature(bmx280_t* dev);
|
|
|
|
/**
|
|
* @brief Read air pressure value from the given BMX280 device
|
|
*
|
|
* The air pressure is returned in PA (Pascal).
|
|
*
|
|
* This function returns the pressure data that was measured when
|
|
* bmx280_read_temperature() has been called last. So bmx280_read_temperature()
|
|
* has to be called before.
|
|
*
|
|
* If bmx280_read_temperatue() did not succeed in acquiring a new set of sensor
|
|
* data, the result of this function is undefined.
|
|
*
|
|
* @param[in] dev device to read from
|
|
*
|
|
* @return air pressure in PA
|
|
*/
|
|
uint32_t bmx280_read_pressure(bmx280_t *dev);
|
|
|
|
#if defined(MODULE_BME280_SPI) || defined(MODULE_BME280_I2C) || defined(DOXYGEN)
|
|
/**
|
|
* @brief Read humidity value from the given BME280 device
|
|
*
|
|
* The humidity is returned in centi %RH (x.xx% relative humidity).
|
|
*
|
|
* This function returns the pressure data that was measured when
|
|
* bmx280_read_temperature() has been called last. So bmx280_read_temperature()
|
|
* has to be called before.
|
|
*
|
|
* If bmx280_read_temperatue() did not succeed in acquiring a new set of sensor
|
|
* data, the result of this function is undefined.
|
|
*
|
|
* @param[in] dev device to read from
|
|
*
|
|
* @return humidity in centi %RH (i.e. the percentage times 100)
|
|
*/
|
|
uint16_t bme280_read_humidity(bmx280_t *dev);
|
|
#endif
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|
|
|
|
#endif /* BMX280_H */
|
|
/** @} */
|