177 lines
4.9 KiB
C
177 lines
4.9 KiB
C
|
/****************************************************************************
|
||
|
* apps/examples/bme680/bme680_main.c
|
||
|
*
|
||
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||
|
* contributor license agreements. See the NOTICE file distributed with
|
||
|
* this work for additional information regarding copyright ownership. The
|
||
|
* ASF licenses this file to you under the Apache License, Version 2.0 (the
|
||
|
* "License"); you may not use this file except in compliance with the
|
||
|
* License. You may obtain a copy of the License at
|
||
|
*
|
||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||
|
*
|
||
|
* Unless required by applicable law or agreed to in writing, software
|
||
|
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||
|
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||
|
* License for the specific language governing permissions and limitations
|
||
|
* under the License.
|
||
|
*
|
||
|
****************************************************************************/
|
||
|
|
||
|
/****************************************************************************
|
||
|
* Included Files
|
||
|
****************************************************************************/
|
||
|
|
||
|
#include <nuttx/config.h>
|
||
|
#include <nuttx/sensors/sensor.h>
|
||
|
#include <nuttx/sensors/bme680.h>
|
||
|
#include <stdio.h>
|
||
|
#include <fcntl.h>
|
||
|
#include <poll.h>
|
||
|
#include <unistd.h>
|
||
|
#include <sys/ioctl.h>
|
||
|
#include <time.h>
|
||
|
|
||
|
#define NB_LOWERHALFS 3
|
||
|
|
||
|
/* Structure used when polling the sub-sensors */
|
||
|
|
||
|
struct data
|
||
|
{
|
||
|
void *data_struct;
|
||
|
uint16_t data_size;
|
||
|
};
|
||
|
|
||
|
/****************************************************************************
|
||
|
* Public Functions
|
||
|
****************************************************************************/
|
||
|
|
||
|
/****************************************************************************
|
||
|
* bme680_main
|
||
|
****************************************************************************/
|
||
|
|
||
|
int main(int argc, FAR char *argv[])
|
||
|
{
|
||
|
int baro_fd;
|
||
|
int hum_fd;
|
||
|
int gas_fd;
|
||
|
uint16_t seconds;
|
||
|
int ret;
|
||
|
|
||
|
/* This example works when all of the sub-sensors of
|
||
|
* the BME680 are enabled.
|
||
|
*/
|
||
|
|
||
|
struct sensor_baro baro_data;
|
||
|
struct sensor_humi humi_data;
|
||
|
struct sensor_gas gas_data;
|
||
|
|
||
|
/* Open each lowerhalf file to be able to read the data.
|
||
|
* When the pressure measurement is deactivated, sensor_temp0 should
|
||
|
* be opened instead (to get the temperature measurement).
|
||
|
*/
|
||
|
|
||
|
baro_fd = open("/dev/uorb/sensor_baro0", O_RDONLY | O_NONBLOCK);
|
||
|
if (baro_fd < 0)
|
||
|
{
|
||
|
printf("Failed to open barometer lowerhalf.\n");
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
hum_fd = open("/dev/uorb/sensor_humi0", O_RDONLY | O_NONBLOCK);
|
||
|
if (hum_fd < 0)
|
||
|
{
|
||
|
printf("Failed to open humidity sensor lowerhalf.\n");
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
gas_fd = open("/dev/uorb/sensor_gas0", O_RDONLY | O_NONBLOCK);
|
||
|
if (gas_fd < 0)
|
||
|
{
|
||
|
printf("Failed to open gas lowerhalf.\n");
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
/* Configure the sensor */
|
||
|
|
||
|
struct bme680_config_s config;
|
||
|
|
||
|
/* Set oversampling */
|
||
|
|
||
|
config.temp_os = BME680_OS_2X;
|
||
|
config.press_os = BME680_OS_16X;
|
||
|
config.filter_coef = BME680_FILTER_COEF3;
|
||
|
config.hum_os = BME680_OS_1X;
|
||
|
|
||
|
/* Set heater parameters */
|
||
|
|
||
|
config.target_temp = 300; /* degrees Celsius */
|
||
|
config.amb_temp = 30; /* degrees Celsius */
|
||
|
config.heater_duration = 100; /* milliseconds */
|
||
|
|
||
|
config.nb_conv = 0;
|
||
|
|
||
|
ret = ioctl(baro_fd, SNIOC_CALIBRATE, &config);
|
||
|
|
||
|
struct pollfd pfds[] = {
|
||
|
{.fd = baro_fd, .events = POLLIN},
|
||
|
{.fd = hum_fd, .events = POLLIN},
|
||
|
{.fd = gas_fd, .events = POLLIN}
|
||
|
};
|
||
|
|
||
|
struct data sensor_data[] = {
|
||
|
{.data_struct = &baro_data, .data_size = sizeof(struct sensor_baro)},
|
||
|
{.data_struct = &humi_data, .data_size = sizeof(struct sensor_humi)},
|
||
|
{.data_struct = &gas_data, .data_size = sizeof(struct sensor_gas)}
|
||
|
};
|
||
|
|
||
|
seconds = 5 * 60;
|
||
|
|
||
|
/* Wait ~5 minutes for the sensor to accomodate to the surroundings.
|
||
|
* The first measurements are not accurate. The longer the wait, the more
|
||
|
* accurate the results are.
|
||
|
*/
|
||
|
|
||
|
while (seconds > 0)
|
||
|
{
|
||
|
ret = poll(pfds, NB_LOWERHALFS, -1);
|
||
|
if (ret < 0)
|
||
|
{
|
||
|
perror("Could not poll sensor.");
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
/* Go through lowerhalfs and read the data */
|
||
|
|
||
|
for (int i = 0; i < NB_LOWERHALFS; i++)
|
||
|
{
|
||
|
if (pfds[i].revents & POLLIN)
|
||
|
{
|
||
|
ret = read(pfds[i].fd, sensor_data[i].data_struct,
|
||
|
sensor_data[i].data_size);
|
||
|
|
||
|
if (ret != sensor_data[i].data_size)
|
||
|
{
|
||
|
perror("Could not read from sub-sensor.");
|
||
|
return ret;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
seconds -= 3;
|
||
|
}
|
||
|
|
||
|
printf("Temperature [C] = %f\n"
|
||
|
"Pressure [hPa] = %f\n"
|
||
|
"Humidity [rH] = %f\n"
|
||
|
"Gas resistance [kOhm] = %f\n",
|
||
|
baro_data.temperature, baro_data.pressure,
|
||
|
humi_data.humidity, gas_data.gas_resistance);
|
||
|
|
||
|
close(baro_fd);
|
||
|
close(hum_fd);
|
||
|
close(gas_fd);
|
||
|
|
||
|
return 0;
|
||
|
}
|