164 lines
3.9 KiB
C
164 lines
3.9 KiB
C
/*
|
|
* Copyright (c) 2024 Intel Corporation
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
#include "bme280.h"
|
|
#include <math.h>
|
|
|
|
static int bme280_decoder_get_frame_count(const uint8_t *buffer,
|
|
struct sensor_chan_spec chan_spec,
|
|
uint16_t *frame_count)
|
|
{
|
|
const struct bme280_encoded_data *edata = (const struct bme280_encoded_data *)buffer;
|
|
int32_t ret = -ENOTSUP;
|
|
|
|
if (chan_spec.chan_idx != 0) {
|
|
return ret;
|
|
}
|
|
|
|
/* This sensor lacks a FIFO; there will always only be one frame at a time. */
|
|
switch (chan_spec.chan_type) {
|
|
case SENSOR_CHAN_AMBIENT_TEMP:
|
|
*frame_count = edata->has_temp ? 1 : 0;
|
|
break;
|
|
case SENSOR_CHAN_PRESS:
|
|
*frame_count = edata->has_press ? 1 : 0;
|
|
break;
|
|
case SENSOR_CHAN_HUMIDITY:
|
|
*frame_count = edata->has_humidity ? 1 : 0;
|
|
break;
|
|
default:
|
|
return ret;
|
|
}
|
|
|
|
if (*frame_count > 0) {
|
|
ret = 0;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int bme280_decoder_get_size_info(struct sensor_chan_spec chan_spec, size_t *base_size,
|
|
size_t *frame_size)
|
|
{
|
|
switch (chan_spec.chan_type) {
|
|
case SENSOR_CHAN_AMBIENT_TEMP:
|
|
case SENSOR_CHAN_HUMIDITY:
|
|
case SENSOR_CHAN_PRESS:
|
|
*base_size = sizeof(struct sensor_q31_sample_data);
|
|
*frame_size = sizeof(struct sensor_q31_sample_data);
|
|
return 0;
|
|
default:
|
|
return -ENOTSUP;
|
|
}
|
|
}
|
|
|
|
#define BME280_HUM_SHIFT (22)
|
|
#define BME280_PRESS_SHIFT (24)
|
|
#define BME280_TEMP_SHIFT (24)
|
|
|
|
static void bme280_convert_double_to_q31(double reading, int32_t shift, q31_t *out)
|
|
{
|
|
reading = reading * pow(2, 31 - shift);
|
|
|
|
int64_t reading_round = (reading < 0) ? (reading - 0.5) : (reading + 0.5);
|
|
int32_t reading_q31 = CLAMP(reading_round, INT32_MIN, INT32_MAX);
|
|
|
|
if (reading_q31 < 0) {
|
|
reading_q31 = abs(reading_q31);
|
|
reading_q31 = ~reading_q31;
|
|
reading_q31++;
|
|
}
|
|
|
|
*out = reading_q31;
|
|
}
|
|
|
|
/* Refer to bme280.c bme280_channel_get() */
|
|
static void bme280_convert_signed_temp_raw_to_q31(int32_t reading, q31_t *out)
|
|
{
|
|
double temp_double = reading / 100.0;
|
|
|
|
bme280_convert_double_to_q31(temp_double, BME280_TEMP_SHIFT, out);
|
|
}
|
|
|
|
static void bme280_convert_unsigned_pressure_raw_to_q31(uint32_t reading, q31_t *out)
|
|
{
|
|
double press_double = (reading / 256.0) / 1000.0; /* Pa -> hPa */
|
|
|
|
bme280_convert_double_to_q31(press_double, BME280_PRESS_SHIFT, out);
|
|
}
|
|
|
|
static void bme280_convert_unsigned_humidity_raw_to_q31(uint32_t reading, q31_t *out)
|
|
{
|
|
double hum_double = (reading / 1024.0);
|
|
|
|
bme280_convert_double_to_q31(hum_double, BME280_HUM_SHIFT, out);
|
|
}
|
|
|
|
static int bme280_decoder_decode(const uint8_t *buffer, struct sensor_chan_spec chan_spec,
|
|
uint32_t *fit, uint16_t max_count, void *data_out)
|
|
{
|
|
const struct bme280_encoded_data *edata = (const struct bme280_encoded_data *)buffer;
|
|
|
|
if (*fit != 0) {
|
|
return 0;
|
|
}
|
|
|
|
struct sensor_q31_data *out = data_out;
|
|
|
|
out->header.base_timestamp_ns = edata->header.timestamp;
|
|
out->header.reading_count = 1;
|
|
|
|
switch (chan_spec.chan_type) {
|
|
case SENSOR_CHAN_AMBIENT_TEMP:
|
|
if (edata->has_temp) {
|
|
bme280_convert_signed_temp_raw_to_q31(edata->reading.comp_temp,
|
|
&out->readings[0].temperature);
|
|
out->shift = BME280_TEMP_SHIFT;
|
|
} else {
|
|
return -ENODATA;
|
|
}
|
|
break;
|
|
case SENSOR_CHAN_PRESS:
|
|
if (edata->has_press) {
|
|
bme280_convert_unsigned_pressure_raw_to_q31(edata->reading.comp_press,
|
|
&out->readings[0].pressure);
|
|
out->shift = BME280_PRESS_SHIFT;
|
|
} else {
|
|
return -ENODATA;
|
|
}
|
|
break;
|
|
case SENSOR_CHAN_HUMIDITY:
|
|
if (edata->has_humidity) {
|
|
bme280_convert_unsigned_humidity_raw_to_q31(edata->reading.comp_humidity,
|
|
&out->readings[0].humidity);
|
|
out->shift = BME280_HUM_SHIFT;
|
|
} else {
|
|
return -ENODATA;
|
|
}
|
|
break;
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
|
|
*fit = 1;
|
|
|
|
return 1;
|
|
}
|
|
|
|
|
|
SENSOR_DECODER_API_DT_DEFINE() = {
|
|
.get_frame_count = bme280_decoder_get_frame_count,
|
|
.get_size_info = bme280_decoder_get_size_info,
|
|
.decode = bme280_decoder_decode,
|
|
};
|
|
|
|
int bme280_get_decoder(const struct device *dev, const struct sensor_decoder_api **decoder)
|
|
{
|
|
ARG_UNUSED(dev);
|
|
*decoder = &SENSOR_DECODER_NAME();
|
|
|
|
return 0;
|
|
}
|