Implement the RTIO/Sensors V2/Async API for the AKM09918C. Add a decoder API implementation as well. Signed-off-by: Tristan Honscheid <honscheid@google.com>
96 lines
2.5 KiB
C
96 lines
2.5 KiB
C
/*
|
|
* Copyright (c) 2023 Google LLC
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
#include "akm09918c.h"
|
|
|
|
static int akm09918c_decoder_get_frame_count(const uint8_t *buffer, enum sensor_channel channel,
|
|
size_t channel_idx, uint16_t *frame_count)
|
|
{
|
|
ARG_UNUSED(buffer);
|
|
ARG_UNUSED(channel);
|
|
ARG_UNUSED(channel_idx);
|
|
|
|
/* This sensor lacks a FIFO; there will always only be one frame at a time. */
|
|
*frame_count = 1;
|
|
return 0;
|
|
}
|
|
|
|
static int akm09918c_decoder_get_size_info(enum sensor_channel channel, size_t *base_size,
|
|
size_t *frame_size)
|
|
{
|
|
switch (channel) {
|
|
case SENSOR_CHAN_MAGN_X:
|
|
case SENSOR_CHAN_MAGN_Y:
|
|
case SENSOR_CHAN_MAGN_Z:
|
|
case SENSOR_CHAN_MAGN_XYZ:
|
|
*base_size = sizeof(struct sensor_three_axis_data);
|
|
*frame_size = sizeof(struct sensor_three_axis_sample_data);
|
|
return 0;
|
|
default:
|
|
return -ENOTSUP;
|
|
}
|
|
}
|
|
|
|
/** Fixed shift value to use. All channels (MAGN_X, _Y, and _Z) have the same fixed range of
|
|
* +/- 49.12 Gauss.
|
|
*/
|
|
#define AKM09918C_SHIFT (6)
|
|
|
|
static int akm09918c_convert_raw_to_q31(int16_t reading, q31_t *out)
|
|
{
|
|
int64_t intermediate = ((int64_t)reading * AKM09918C_MICRO_GAUSS_PER_BIT) *
|
|
((int64_t)INT32_MAX + 1) /
|
|
((1 << AKM09918C_SHIFT) * INT64_C(1000000));
|
|
|
|
*out = CLAMP(intermediate, INT32_MIN, INT32_MAX);
|
|
return 0;
|
|
}
|
|
|
|
static int akm09918c_decoder_decode(const uint8_t *buffer, enum sensor_channel channel,
|
|
size_t channel_idx, uint32_t *fit,
|
|
uint16_t max_count, void *data_out)
|
|
{
|
|
const struct akm09918c_encoded_data *edata = (const struct akm09918c_encoded_data *)buffer;
|
|
|
|
if (*fit != 0) {
|
|
return 0;
|
|
}
|
|
|
|
switch (channel) {
|
|
case SENSOR_CHAN_MAGN_X:
|
|
case SENSOR_CHAN_MAGN_Y:
|
|
case SENSOR_CHAN_MAGN_Z:
|
|
case SENSOR_CHAN_MAGN_XYZ: {
|
|
struct sensor_three_axis_data *out = data_out;
|
|
|
|
out->header.base_timestamp_ns = edata->header.timestamp;
|
|
out->header.reading_count = 1;
|
|
out->shift = AKM09918C_SHIFT;
|
|
|
|
akm09918c_convert_raw_to_q31(edata->readings[0], &out->readings[0].x);
|
|
akm09918c_convert_raw_to_q31(edata->readings[1], &out->readings[0].y);
|
|
akm09918c_convert_raw_to_q31(edata->readings[2], &out->readings[0].z);
|
|
*fit = 1;
|
|
|
|
return 1;
|
|
}
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
}
|
|
|
|
SENSOR_DECODER_API_DT_DEFINE() = {
|
|
.get_frame_count = akm09918c_decoder_get_frame_count,
|
|
.get_size_info = akm09918c_decoder_get_size_info,
|
|
.decode = akm09918c_decoder_decode,
|
|
};
|
|
|
|
int akm09918c_get_decoder(const struct device *dev, const struct sensor_decoder_api **decoder)
|
|
{
|
|
ARG_UNUSED(dev);
|
|
*decoder = &SENSOR_DECODER_NAME();
|
|
|
|
return 0;
|
|
}
|