zephyr/drivers/sensor/broadcom/afbr_s50/afbr_s50_decoder.c
Luis Ubieda 820975a409 sensor: afbr_s50: Add private channel to obtain pixel matrix
Each time this sensor gets a reading, it contains a matrix of 4 x 32
pixels containing distance readings, from which the 1-D result is
calculated. The private channel would expose this array through
Sensor APIs.

Signed-off-by: Luis Ubieda <luisf@croxel.com>
2025-05-28 21:29:13 +02:00

181 lines
4.2 KiB
C

/*
* Copyright (c) 2025 Croxel Inc.
* Copyright (c) 2025 CogniPilot Foundation
*
* SPDX-License-Identifier: Apache-2.0
*/
#define DT_DRV_COMPAT brcm_afbr_s50
#include <zephyr/drivers/sensor_clock.h>
#include <api/argus_res.h>
#include <zephyr/drivers/sensor/afbr_s50.h>
#include "afbr_s50_decoder.h"
#include <zephyr/logging/log.h>
LOG_MODULE_REGISTER(AFBR_S50_DECODER, CONFIG_SENSOR_LOG_LEVEL);
uint8_t afbr_s50_encode_channel(uint16_t chan)
{
switch (chan) {
case SENSOR_CHAN_DISTANCE:
return BIT(0);
case SENSOR_CHAN_AFBR_S50_PIXELS:
return BIT(1);
default:
return 0;
}
}
uint8_t afbr_s50_encode_event(enum sensor_trigger_type trigger)
{
if (trigger == SENSOR_TRIG_DATA_READY) {
return BIT(0);
}
return 0;
}
static int afbr_s50_decoder_get_frame_count(const uint8_t *buffer,
struct sensor_chan_spec chan_spec,
uint16_t *frame_count)
{
const struct afbr_s50_edata *edata = (const struct afbr_s50_edata *)buffer;
if (chan_spec.chan_idx != 0) {
return -ENOTSUP;
}
switch (chan_spec.chan_type) {
case SENSOR_CHAN_DISTANCE:
case SENSOR_CHAN_AFBR_S50_PIXELS:
if (edata->header.channels & afbr_s50_encode_channel(chan_spec.chan_type)) {
*frame_count = 1;
return 0;
}
break;
default:
break;
}
*frame_count = 0;
return 0;
}
static int afbr_s50_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_DISTANCE:
*base_size = sizeof(struct sensor_q31_data);
*frame_size = sizeof(struct sensor_q31_sample_data);
return 0;
case SENSOR_CHAN_AFBR_S50_PIXELS:
*base_size = sizeof(struct sensor_q31_data) +
31 * sizeof(struct sensor_q31_sample_data);
*frame_size = 32 * sizeof(struct sensor_q31_sample_data);
return 0;
default:
return -ENOTSUP;
}
}
static int afbr_s50_decoder_decode(const uint8_t *buffer,
struct sensor_chan_spec chan_spec,
uint32_t *fit,
uint16_t max_count,
void *data_out)
{
const struct afbr_s50_edata *edata = (const struct afbr_s50_edata *)buffer;
if (*fit != 0) {
return 0;
}
if (max_count == 0 || chan_spec.chan_idx != 0) {
return -EINVAL;
}
switch (chan_spec.chan_type) {
case SENSOR_CHAN_DISTANCE: {
struct sensor_q31_data *out = data_out;
if ((edata->header.channels & afbr_s50_encode_channel(SENSOR_CHAN_DISTANCE)) == 0) {
return -ENODATA;
}
out->header.base_timestamp_ns = edata->header.timestamp;
out->header.reading_count = 1;
/* Result comes encoded in Q9.22 format */
out->shift = 9;
out->readings[0].value = edata->payload.Bin.Range;
*fit = 1;
return 1;
}
case SENSOR_CHAN_AFBR_S50_PIXELS: {
struct sensor_q31_data *out = data_out;
if ((edata->header.channels &
afbr_s50_encode_channel(SENSOR_CHAN_AFBR_S50_PIXELS)) == 0) {
return -ENODATA;
}
out->header.base_timestamp_ns = edata->header.timestamp;
out->header.reading_count = 32;
/* Result comes encoded in Q9.22 format */
out->shift = 9;
for (size_t i = 0 ; i < 32 ; i++) {
if (edata->payload.Pixels[i].Amplitude == 0xFFFF ||
edata->payload.Pixels[i].Status != PIXEL_OK) {
LOG_DBG("Invalid pixel: %zu, Amplitude: %u, Status: %u", i,
edata->payload.Pixels[i].Amplitude,
edata->payload.Pixels[i].Status);
out->readings[i].value = AFBR_PIXEL_INVALID_VALUE;
} else {
out->readings[i].value = edata->payload.Pixels[i].Range;
}
out->readings[i].timestamp_delta = 0;
}
*fit = 1;
return 1;
}
default:
return -EINVAL;
}
}
static bool afbr_s50_decoder_has_trigger(const uint8_t *buffer,
enum sensor_trigger_type trigger)
{
const struct afbr_s50_edata *edata = (const struct afbr_s50_edata *)buffer;
if (trigger == SENSOR_TRIG_DATA_READY) {
return edata->header.events & afbr_s50_encode_event(SENSOR_TRIG_DATA_READY);
} else {
return false;
}
}
SENSOR_DECODER_API_DT_DEFINE() = {
.get_frame_count = afbr_s50_decoder_get_frame_count,
.get_size_info = afbr_s50_decoder_get_size_info,
.decode = afbr_s50_decoder_decode,
.has_trigger = afbr_s50_decoder_has_trigger,
};
int afbr_s50_get_decoder(const struct device *dev,
const struct sensor_decoder_api **decoder)
{
ARG_UNUSED(dev);
*decoder = &SENSOR_DECODER_NAME();
return 0;
}