Ardupilot2/libraries/AP_NavEKF/EKF_Buffer.cpp
Andrew Tridgell a941e4cd41 AP_NavEKF: re-implemented EKF ring buffer
this fixes a bug where elemnts being pushed into the buffer more
slowly than we recall can be lost

for example, if you push a single element in then try a recall it will
fail
2022-06-24 20:25:39 +10:00

195 lines
4.4 KiB
C++

/*
common EKF Buffer class. This handles the storage buffers for EKF data to
bring it onto the fusion time horizon
*/
#include "EKF_Buffer.h"
#include <stdlib.h>
#include <string.h>
#include <AP_InternalError/AP_InternalError.h>
// constructor
ekf_ring_buffer::ekf_ring_buffer(uint8_t _elsize) :
elsize(_elsize),
buffer(nullptr)
{}
bool ekf_ring_buffer::init(uint8_t _size)
{
if (buffer) {
free(buffer);
}
buffer = calloc(_size, elsize);
if (buffer == nullptr) {
return false;
}
size = _size;
reset();
return true;
}
/*
get buffer offset for an index
*/
void *ekf_ring_buffer::get_offset(uint8_t idx) const
{
return (void*)(((uint8_t*)buffer)+idx*uint32_t(elsize));
}
/*
get a reference to the timestamp for an index
*/
uint32_t ekf_ring_buffer::time_ms(uint8_t idx) const
{
EKF_obs_element_t *el = (EKF_obs_element_t *)get_offset(idx);
return el->time_ms;
}
/*
Search through a ring buffer and return the newest data that is
older than the time specified by sample_time_ms
Returns false if no data can be found that is less than 100msec old
*/
bool ekf_ring_buffer::recall(void *element, const uint32_t sample_time_ms)
{
bool ret = false;
while (count > 0) {
const uint32_t toldest = time_ms(oldest);
const int32_t dt = sample_time_ms - toldest;
const bool matches = dt >= 0 && dt < 100;
if (matches) {
memcpy(element, get_offset(oldest), elsize);
ret = true;
}
if (dt < 0) {
// the oldest element is younger than we want, stop
// searching and don't consume this element
break;
}
// discard the sample
count--;
oldest = (oldest+1) % size;
}
return ret;
}
/*
* Writes data and timestamp to a Ring buffer and advances indices that
* define the location of the newest and oldest data
*/
void ekf_ring_buffer::push(const void *element)
{
if (buffer == nullptr) {
return;
}
// Advance head to next available index
const uint8_t head = (oldest+count) % size;
// New data is written at the head
memcpy(get_offset(head), element, elsize);
if (count < size) {
count++;
} else {
oldest = (oldest+1) % size;
}
}
// zeroes all data in the ring buffer
void ekf_ring_buffer::reset()
{
count = 0;
oldest = 0;
}
////////////////////////////////////////////////////
/*
IMU buffer operations implemented separately due to different
semantics
*/
// constructor
ekf_imu_buffer::ekf_imu_buffer(uint8_t _elsize) :
elsize(_elsize)
{}
/*
get buffer offset for an index
*/
void *ekf_imu_buffer::get_offset(uint8_t idx) const
{
return (void*)(((uint8_t*)buffer)+idx*uint32_t(elsize));
}
// initialise buffer, returns false when allocation has failed
bool ekf_imu_buffer::init(uint32_t size)
{
if (buffer != nullptr) {
// allow for init twice
free(buffer);
}
buffer = calloc(size, elsize);
if (buffer == nullptr) {
return false;
}
_size = size;
_youngest = 0;
_oldest = 0;
return true;
}
/*
Writes data to a Ring buffer and advances indices that
define the location of the newest and oldest data
*/
void ekf_imu_buffer::push_youngest_element(const void *element)
{
if (!buffer) {
INTERNAL_ERROR(AP_InternalError::error_t::flow_of_control);
return;
}
// push youngest to the buffer
_youngest = (_youngest+1) % _size;
memcpy(get_offset(_youngest), element, elsize);
// set oldest data index
_oldest = (_youngest+1) % _size;
if (_oldest == 0) {
_filled = true;
}
}
// retrieve the oldest data from the ring buffer tail
void ekf_imu_buffer::get_oldest_element(void *element)
{
if (buffer == nullptr) {
INTERNAL_ERROR(AP_InternalError::error_t::flow_of_control);
memset(element, 0, elsize);
} else {
memcpy(element, get_offset(_oldest), elsize);
}
}
// writes the same data to all elements in the ring buffer
void ekf_imu_buffer::reset_history(const void *element)
{
for (uint8_t index=0; index<_size; index++) {
memcpy(get_offset(index), element, elsize);
}
}
// zeroes all data in the ring buffer
void ekf_imu_buffer::reset()
{
_youngest = 0;
_oldest = 0;
memset(buffer, 0, _size*uint32_t(elsize));
}
// retrieves data from the ring buffer at a specified index
void *ekf_imu_buffer::get(uint8_t index) const
{
return get_offset(index);
}