mirror of https://github.com/ArduPilot/ardupilot
795 lines
22 KiB
C++
795 lines
22 KiB
C++
#include "AP_Logger_config.h"
|
|
|
|
#if HAL_LOGGING_ENABLED
|
|
|
|
#include "AP_Logger_Backend.h"
|
|
|
|
#include "LoggerMessageWriter.h"
|
|
|
|
#include "AP_Common/AP_FWVersion.h"
|
|
#include <AP_InternalError/AP_InternalError.h>
|
|
#include <AP_Scheduler/AP_Scheduler.h>
|
|
#include <AP_Rally/AP_Rally.h>
|
|
#include <AP_Vehicle/AP_Vehicle_Type.h>
|
|
#include <Filter/Filter.h>
|
|
#include "AP_Logger.h"
|
|
|
|
#if HAL_LOGGER_FENCE_ENABLED
|
|
#include <AC_Fence/AC_Fence.h>
|
|
#endif
|
|
|
|
extern const AP_HAL::HAL& hal;
|
|
|
|
AP_Logger_Backend::AP_Logger_Backend(AP_Logger &front,
|
|
class LoggerMessageWriter_DFLogStart *writer) :
|
|
_front(front),
|
|
_startup_messagewriter(writer)
|
|
{
|
|
writer->set_logger_backend(this);
|
|
}
|
|
|
|
uint8_t AP_Logger_Backend::num_types() const
|
|
{
|
|
return _front._num_types;
|
|
}
|
|
|
|
const struct LogStructure *AP_Logger_Backend::structure(uint8_t num) const
|
|
{
|
|
return _front.structure(num);
|
|
}
|
|
|
|
uint8_t AP_Logger_Backend::num_units() const
|
|
{
|
|
return _front._num_units;
|
|
}
|
|
|
|
const struct UnitStructure *AP_Logger_Backend::unit(uint8_t num) const
|
|
{
|
|
return _front.unit(num);
|
|
}
|
|
|
|
uint8_t AP_Logger_Backend::num_multipliers() const
|
|
{
|
|
return _front._num_multipliers;
|
|
}
|
|
|
|
const struct MultiplierStructure *AP_Logger_Backend::multiplier(uint8_t num) const
|
|
{
|
|
return _front.multiplier(num);
|
|
}
|
|
|
|
AP_Logger_Backend::vehicle_startup_message_Writer AP_Logger_Backend::vehicle_message_writer() const {
|
|
return _front._vehicle_messages;
|
|
}
|
|
|
|
void AP_Logger_Backend::periodic_10Hz(const uint32_t now)
|
|
{
|
|
}
|
|
|
|
void AP_Logger_Backend::periodic_1Hz()
|
|
{
|
|
if (_rotate_pending && !logging_enabled()) {
|
|
_rotate_pending = false;
|
|
// handle log rotation once we stop logging
|
|
stop_logging_async();
|
|
}
|
|
df_stats_log();
|
|
}
|
|
|
|
void AP_Logger_Backend::periodic_fullrate()
|
|
{
|
|
push_log_blocks();
|
|
}
|
|
|
|
void AP_Logger_Backend::periodic_tasks()
|
|
{
|
|
uint32_t now = AP_HAL::millis();
|
|
if (now - _last_periodic_1Hz > 1000) {
|
|
periodic_1Hz();
|
|
_last_periodic_1Hz = now;
|
|
}
|
|
if (now - _last_periodic_10Hz > 100) {
|
|
periodic_10Hz(now);
|
|
_last_periodic_10Hz = now;
|
|
}
|
|
periodic_fullrate();
|
|
}
|
|
|
|
void AP_Logger_Backend::start_new_log_reset_variables()
|
|
{
|
|
_dropped = 0;
|
|
_startup_messagewriter->reset();
|
|
_front.backend_starting_new_log(this);
|
|
_formats_written.clearall();
|
|
}
|
|
|
|
// We may need to make sure data is loggable before starting the
|
|
// EKF; when allow_start_ekf we should be able to log that data
|
|
bool AP_Logger_Backend::allow_start_ekf() const
|
|
{
|
|
if (!_startup_messagewriter->fmt_done()) {
|
|
return false;
|
|
}
|
|
// we need to push all startup messages out, or the code in
|
|
// WriteBlockCheckStartupMessages bites us.
|
|
if (!_startup_messagewriter->finished()) {
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
// this method can be overridden to do extra things with your buffer.
|
|
// for example, in AP_Logger_MAVLink we may push messages into the UART.
|
|
void AP_Logger_Backend::push_log_blocks() {
|
|
WriteMoreStartupMessages();
|
|
}
|
|
|
|
// source more messages from the startup message writer:
|
|
void AP_Logger_Backend::WriteMoreStartupMessages()
|
|
{
|
|
#if APM_BUILD_TYPE(APM_BUILD_Replay)
|
|
return;
|
|
#endif
|
|
|
|
if (_startup_messagewriter->finished()) {
|
|
return;
|
|
}
|
|
|
|
_writing_startup_messages = true;
|
|
_startup_messagewriter->process();
|
|
_writing_startup_messages = false;
|
|
}
|
|
|
|
/*
|
|
* support for Write():
|
|
*/
|
|
|
|
|
|
// output a FMT message if not already done so
|
|
void AP_Logger_Backend::Safe_Write_Emit_FMT(uint8_t msg_type)
|
|
{
|
|
if (have_emitted_format_for_type(LogMessages(msg_type))) {
|
|
return;
|
|
}
|
|
Write_Emit_FMT(msg_type);
|
|
}
|
|
|
|
bool AP_Logger_Backend::Write_Emit_FMT(uint8_t msg_type)
|
|
{
|
|
#if APM_BUILD_TYPE(APM_BUILD_Replay)
|
|
if (msg_type < REPLAY_LOG_NEW_MSG_MIN || msg_type > REPLAY_LOG_NEW_MSG_MAX) {
|
|
// don't re-emit FMU msgs unless they are in the replay range
|
|
return true;
|
|
}
|
|
#endif
|
|
|
|
// get log structure from front end:
|
|
struct AP_Logger::log_write_fmt_strings ls = {};
|
|
struct LogStructure logstruct = {
|
|
// these will be overwritten, but need to keep the compiler happy:
|
|
0,
|
|
0,
|
|
ls.name,
|
|
ls.format,
|
|
ls.labels,
|
|
ls.units,
|
|
ls.multipliers
|
|
};
|
|
if (!_front.fill_logstructure(logstruct, msg_type)) {
|
|
// this is a bug; we've been asked to write out the FMT
|
|
// message for a msg_type, but the frontend can't supply the
|
|
// required information
|
|
INTERNAL_ERROR(AP_InternalError::error_t::logger_missing_logstructure);
|
|
return false;
|
|
}
|
|
|
|
if (!Write_Format(&logstruct)) {
|
|
return false;
|
|
}
|
|
if (!Write_Format_Units(&logstruct)) {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool AP_Logger_Backend::Write(const uint8_t msg_type, va_list arg_list, bool is_critical, bool is_streaming)
|
|
{
|
|
// stack-allocate a buffer so we can WriteBlock(); this could be
|
|
// 255 bytes! If we were willing to lose the WriteBlock
|
|
// abstraction we could do WriteBytes() here instead?
|
|
const char *fmt = nullptr;
|
|
uint8_t msg_len;
|
|
AP_Logger::log_write_fmt *f;
|
|
for (f = _front.log_write_fmts; f; f=f->next) {
|
|
if (f->msg_type == msg_type) {
|
|
fmt = f->fmt;
|
|
msg_len = f->msg_len;
|
|
break;
|
|
}
|
|
}
|
|
if (fmt == nullptr) {
|
|
INTERNAL_ERROR(AP_InternalError::error_t::logger_logwrite_missingfmt);
|
|
return false;
|
|
}
|
|
if (bufferspace_available() < msg_len) {
|
|
return false;
|
|
}
|
|
|
|
uint8_t buffer[msg_len];
|
|
uint8_t offset = 0;
|
|
buffer[offset++] = HEAD_BYTE1;
|
|
buffer[offset++] = HEAD_BYTE2;
|
|
buffer[offset++] = msg_type;
|
|
for (uint8_t i=0; i<strlen(fmt); i++) {
|
|
uint8_t charlen = 0;
|
|
switch(fmt[i]) {
|
|
case 'b': {
|
|
int8_t tmp = va_arg(arg_list, int);
|
|
memcpy(&buffer[offset], &tmp, sizeof(int8_t));
|
|
offset += sizeof(int8_t);
|
|
break;
|
|
}
|
|
case 'h':
|
|
case 'c': {
|
|
int16_t tmp = va_arg(arg_list, int);
|
|
memcpy(&buffer[offset], &tmp, sizeof(int16_t));
|
|
offset += sizeof(int16_t);
|
|
break;
|
|
}
|
|
case 'd': {
|
|
double tmp = va_arg(arg_list, double);
|
|
memcpy(&buffer[offset], &tmp, sizeof(double));
|
|
offset += sizeof(double);
|
|
break;
|
|
}
|
|
case 'i':
|
|
case 'L':
|
|
case 'e': {
|
|
int32_t tmp = va_arg(arg_list, int);
|
|
memcpy(&buffer[offset], &tmp, sizeof(int32_t));
|
|
offset += sizeof(int32_t);
|
|
break;
|
|
}
|
|
case 'f': {
|
|
float tmp = va_arg(arg_list, double);
|
|
memcpy(&buffer[offset], &tmp, sizeof(float));
|
|
offset += sizeof(float);
|
|
break;
|
|
}
|
|
case 'g': {
|
|
Float16_t tmp;
|
|
tmp.set(va_arg(arg_list, double));;
|
|
memcpy(&buffer[offset], &tmp, sizeof(tmp));
|
|
offset += sizeof(tmp);
|
|
break;
|
|
}
|
|
case 'n':
|
|
charlen = 4;
|
|
break;
|
|
case 'M':
|
|
case 'B': {
|
|
uint8_t tmp = va_arg(arg_list, int);
|
|
memcpy(&buffer[offset], &tmp, sizeof(uint8_t));
|
|
offset += sizeof(uint8_t);
|
|
break;
|
|
}
|
|
case 'H':
|
|
case 'C': {
|
|
uint16_t tmp = va_arg(arg_list, int);
|
|
memcpy(&buffer[offset], &tmp, sizeof(uint16_t));
|
|
offset += sizeof(uint16_t);
|
|
break;
|
|
}
|
|
case 'I':
|
|
case 'E': {
|
|
uint32_t tmp = va_arg(arg_list, uint32_t);
|
|
memcpy(&buffer[offset], &tmp, sizeof(uint32_t));
|
|
offset += sizeof(uint32_t);
|
|
break;
|
|
}
|
|
case 'N':
|
|
charlen = 16;
|
|
break;
|
|
case 'Z':
|
|
charlen = 64;
|
|
break;
|
|
case 'q': {
|
|
int64_t tmp = va_arg(arg_list, int64_t);
|
|
memcpy(&buffer[offset], &tmp, sizeof(int64_t));
|
|
offset += sizeof(int64_t);
|
|
break;
|
|
}
|
|
case 'Q': {
|
|
uint64_t tmp = va_arg(arg_list, uint64_t);
|
|
memcpy(&buffer[offset], &tmp, sizeof(uint64_t));
|
|
offset += sizeof(uint64_t);
|
|
break;
|
|
}
|
|
case 'a': {
|
|
int16_t *tmp = va_arg(arg_list, int16_t*);
|
|
const uint8_t bytes = 32*2;
|
|
memcpy(&buffer[offset], tmp, bytes);
|
|
offset += bytes;
|
|
break;
|
|
}
|
|
}
|
|
if (charlen != 0) {
|
|
char *tmp = va_arg(arg_list, char*);
|
|
uint8_t len = strnlen(tmp, charlen);
|
|
memcpy(&buffer[offset], tmp, len);
|
|
memset(&buffer[offset+len], 0, charlen-len);
|
|
offset += charlen;
|
|
}
|
|
}
|
|
|
|
return WritePrioritisedBlock(buffer, msg_len, is_critical, is_streaming);
|
|
}
|
|
|
|
bool AP_Logger_Backend::StartNewLogOK() const
|
|
{
|
|
if (logging_started()) {
|
|
return false;
|
|
}
|
|
if (_front._log_bitmask == 0) {
|
|
return false;
|
|
}
|
|
if (_front.in_log_download()) {
|
|
return false;
|
|
}
|
|
if (!hal.scheduler->in_main_thread()) {
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
// validate that pBuffer looks like a message, extract message type.
|
|
// Returns false if this doesn't look like a valid message.
|
|
bool AP_Logger_Backend::message_type_from_block(const void *pBuffer, uint16_t size, LogMessages &type) const
|
|
{
|
|
if (size < 3) {
|
|
return false;
|
|
}
|
|
if (((uint8_t*)pBuffer)[0] != HEAD_BYTE1 ||
|
|
((uint8_t*)pBuffer)[1] != HEAD_BYTE2) {
|
|
// Not passed a message
|
|
INTERNAL_ERROR(AP_InternalError::error_t::flow_of_control);
|
|
return false;
|
|
}
|
|
type = LogMessages(((uint8_t*)pBuffer)[2]);
|
|
return true;
|
|
}
|
|
|
|
#if CONFIG_HAL_BOARD == HAL_BOARD_SITL
|
|
void AP_Logger_Backend::validate_WritePrioritisedBlock(const void *pBuffer,
|
|
uint16_t size)
|
|
{
|
|
// just check the first few packets to avoid too much overhead
|
|
// (finding the structures is expensive)
|
|
static uint16_t count = 0;
|
|
if (count > 65534) {
|
|
return;
|
|
}
|
|
count++;
|
|
|
|
// we assume here that we ever WritePrioritisedBlock for a single
|
|
// message. If this assumption becomes false we can't do these
|
|
// checks.
|
|
if (size < 3) {
|
|
AP_HAL::panic("Short prioritised block");
|
|
}
|
|
LogMessages type;
|
|
if (!message_type_from_block(pBuffer, size, type)) {
|
|
AP_HAL::panic("Not passed a message");
|
|
}
|
|
uint8_t type_len;
|
|
const char *name_src;
|
|
const struct LogStructure *s = _front.structure_for_msg_type(type);
|
|
if (s == nullptr) {
|
|
const struct AP_Logger::log_write_fmt *t = _front.log_write_fmt_for_msg_type(type);
|
|
if (t == nullptr) {
|
|
AP_HAL::panic("No structure for msg_type=%u", type);
|
|
}
|
|
type_len = t->msg_len;
|
|
name_src = t->name;
|
|
} else {
|
|
type_len = s->msg_len;
|
|
name_src = s->name;
|
|
}
|
|
if (type_len != size) {
|
|
char name[5] = {}; // get a null-terminated string
|
|
if (name_src != nullptr) {
|
|
memcpy(name, name_src, 4);
|
|
} else {
|
|
strncpy(name, "?NM?", ARRAY_SIZE(name));
|
|
}
|
|
AP_HAL::panic("Size mismatch for %u (%s) (expected=%u got=%u)",
|
|
type, name, type_len, size);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
bool AP_Logger_Backend::ensure_format_emitted(const void *pBuffer, uint16_t size)
|
|
{
|
|
#if APM_BUILD_TYPE(APM_BUILD_Replay)
|
|
// we trust that Replay will correctly emit formats as required
|
|
return true;
|
|
#endif
|
|
|
|
// extract the ID:
|
|
LogMessages type;
|
|
if (!message_type_from_block(pBuffer, size, type)) {
|
|
return false;
|
|
}
|
|
if (have_emitted_format_for_type(type)) {
|
|
return true;
|
|
}
|
|
|
|
// make sure the FMT message has gone out!
|
|
if (type == LOG_FORMAT_MSG) {
|
|
// kind of? Our caller is just about to emit this....
|
|
return true;
|
|
}
|
|
if (!have_emitted_format_for_type(LOG_FORMAT_MSG) &&
|
|
!Write_Emit_FMT(LOG_FORMAT_MSG)) {
|
|
return false;
|
|
}
|
|
|
|
return Write_Emit_FMT(type);
|
|
}
|
|
|
|
bool AP_Logger_Backend::WritePrioritisedBlock(const void *pBuffer, uint16_t size, bool is_critical, bool writev_streaming)
|
|
{
|
|
#if CONFIG_HAL_BOARD == HAL_BOARD_SITL && !APM_BUILD_TYPE(APM_BUILD_Replay)
|
|
validate_WritePrioritisedBlock(pBuffer, size);
|
|
#endif
|
|
if (!ShouldLog(is_critical)) {
|
|
return false;
|
|
}
|
|
if (StartNewLogOK()) {
|
|
start_new_log();
|
|
}
|
|
if (!WritesOK()) {
|
|
return false;
|
|
}
|
|
|
|
if (!is_critical && rate_limiter != nullptr) {
|
|
const uint8_t *msgbuf = (const uint8_t *)pBuffer;
|
|
if (!rate_limiter->should_log(msgbuf[2], writev_streaming)) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (!ensure_format_emitted(pBuffer, size)) {
|
|
return false;
|
|
}
|
|
|
|
return _WritePrioritisedBlock(pBuffer, size, is_critical);
|
|
}
|
|
|
|
bool AP_Logger_Backend::ShouldLog(bool is_critical)
|
|
{
|
|
if (!_front.WritesEnabled()) {
|
|
return false;
|
|
}
|
|
if (!_initialised) {
|
|
return false;
|
|
}
|
|
|
|
if (!_startup_messagewriter->finished() &&
|
|
!hal.scheduler->in_main_thread()) {
|
|
// only the main thread may write startup messages out
|
|
return false;
|
|
}
|
|
|
|
if (_front.in_log_download() &&
|
|
_front._last_mavlink_log_transfer_message_handled_ms != 0) {
|
|
if (AP_HAL::millis() - _front._last_mavlink_log_transfer_message_handled_ms < 10000) {
|
|
if (!_front.vehicle_is_armed()) {
|
|
// user is transferring files via mavlink
|
|
return false;
|
|
}
|
|
} else {
|
|
_front._last_mavlink_log_transfer_message_handled_ms = 0;
|
|
}
|
|
}
|
|
|
|
if (is_critical && have_logged_armed && !_front._params.file_disarm_rot) {
|
|
// if we have previously logged while armed then we log all
|
|
// critical messages from then on. That fixes a problem where
|
|
// logs show the wrong flight mode if you disarm then arm again
|
|
return true;
|
|
}
|
|
|
|
if (!_front.vehicle_is_armed() && !_front.log_while_disarmed()) {
|
|
return false;
|
|
}
|
|
|
|
if (_front.vehicle_is_armed()) {
|
|
have_logged_armed = true;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void AP_Logger_Backend::PrepForArming()
|
|
{
|
|
if (_rotate_pending) {
|
|
_rotate_pending = false;
|
|
stop_logging();
|
|
}
|
|
if (logging_started()) {
|
|
return;
|
|
}
|
|
PrepForArming_start_logging();
|
|
}
|
|
|
|
bool AP_Logger_Backend::Write_MessageF(const char *fmt, ...)
|
|
{
|
|
char msg[65] {}; // sizeof(log_Message.msg) + null-termination
|
|
|
|
va_list ap;
|
|
va_start(ap, fmt);
|
|
hal.util->vsnprintf(msg, sizeof(msg), fmt, ap);
|
|
va_end(ap);
|
|
|
|
return Write_Message(msg);
|
|
}
|
|
|
|
#if HAL_RALLY_ENABLED
|
|
// Write rally points
|
|
bool AP_Logger_Backend::Write_RallyPoint(uint8_t total,
|
|
uint8_t sequence,
|
|
const RallyLocation &rally_point)
|
|
{
|
|
const struct log_Rally pkt_rally{
|
|
LOG_PACKET_HEADER_INIT(LOG_RALLY_MSG),
|
|
time_us : AP_HAL::micros64(),
|
|
total : total,
|
|
sequence : sequence,
|
|
latitude : rally_point.lat,
|
|
longitude : rally_point.lng,
|
|
altitude : rally_point.alt,
|
|
flags : rally_point.flags
|
|
};
|
|
return WriteBlock(&pkt_rally, sizeof(pkt_rally));
|
|
}
|
|
|
|
// Write rally points
|
|
bool AP_Logger_Backend::Write_Rally()
|
|
{
|
|
// kick off asynchronous write:
|
|
return _startup_messagewriter->writeallrallypoints();
|
|
}
|
|
#endif
|
|
|
|
#if HAL_LOGGER_FENCE_ENABLED
|
|
// Write a fence point
|
|
bool AP_Logger_Backend::Write_FencePoint(uint8_t total, uint8_t sequence, const AC_PolyFenceItem &fence_point)
|
|
{
|
|
const struct log_Fence pkt_fence{
|
|
LOG_PACKET_HEADER_INIT(LOG_FENCE_MSG),
|
|
time_us : AP_HAL::micros64(),
|
|
total : total,
|
|
sequence : sequence,
|
|
type : uint8_t(fence_point.type),
|
|
latitude : fence_point.loc.x,
|
|
longitude : fence_point.loc.y,
|
|
vertex_count : fence_point.vertex_count,
|
|
radius : fence_point.radius
|
|
};
|
|
return WriteBlock(&pkt_fence, sizeof(pkt_fence));
|
|
}
|
|
|
|
// Write all fence points
|
|
bool AP_Logger_Backend::Write_Fence()
|
|
{
|
|
// kick off asynchronous write:
|
|
return _startup_messagewriter->writeallfence();
|
|
}
|
|
#endif // HAL_LOGGER_FENCE_ENABLED
|
|
|
|
|
|
bool AP_Logger_Backend::Write_VER()
|
|
{
|
|
const AP_FWVersion &fwver = AP::fwversion();
|
|
|
|
log_VER pkt{
|
|
LOG_PACKET_HEADER_INIT(LOG_VER_MSG),
|
|
time_us : AP_HAL::micros64(),
|
|
board_type : fwver.board_type,
|
|
board_subtype: fwver.board_subtype,
|
|
major: fwver.major,
|
|
minor: fwver.minor,
|
|
patch: fwver.patch,
|
|
fw_type: fwver.fw_type,
|
|
git_hash: fwver.fw_hash,
|
|
};
|
|
strncpy(pkt.fw_string, fwver.fw_string, ARRAY_SIZE(pkt.fw_string)-1);
|
|
|
|
#ifdef APJ_BOARD_ID
|
|
pkt._APJ_BOARD_ID = APJ_BOARD_ID;
|
|
#endif
|
|
pkt.build_type = fwver.vehicle_type;
|
|
pkt.filter_version = AP_FILTER_VERSION;
|
|
|
|
return WriteCriticalBlock(&pkt, sizeof(pkt));
|
|
}
|
|
|
|
/*
|
|
convert a list entry number back into a log number (which can then
|
|
be converted into a filename). A "list entry number" is a sequence
|
|
where the oldest log has a number of 1, the second-from-oldest 2,
|
|
and so on. Thus the highest list entry number is equal to the
|
|
number of logs.
|
|
*/
|
|
uint16_t AP_Logger_Backend::log_num_from_list_entry(const uint16_t list_entry)
|
|
{
|
|
uint16_t oldest_log = find_oldest_log();
|
|
if (oldest_log == 0) {
|
|
return 0;
|
|
}
|
|
|
|
uint32_t log_num = oldest_log + list_entry - 1;
|
|
const auto max_logs_num = _front.get_max_num_logs();
|
|
if (log_num > (uint32_t)max_logs_num) {
|
|
log_num -= max_logs_num;
|
|
}
|
|
return (uint16_t)log_num;
|
|
}
|
|
|
|
// find_oldest_log - find oldest log
|
|
// returns 0 if no log was found
|
|
uint16_t AP_Logger_Backend::find_oldest_log()
|
|
{
|
|
if (_cached_oldest_log != 0) {
|
|
return _cached_oldest_log;
|
|
}
|
|
|
|
uint16_t last_log_num = find_last_log();
|
|
if (last_log_num == 0) {
|
|
return 0;
|
|
}
|
|
|
|
_cached_oldest_log = last_log_num - get_num_logs() + 1;
|
|
|
|
return _cached_oldest_log;
|
|
}
|
|
|
|
void AP_Logger_Backend::vehicle_was_disarmed()
|
|
{
|
|
if (_front._params.file_disarm_rot &&
|
|
!_front._params.log_replay) {
|
|
// rotate our log. Closing the current one and letting the
|
|
// logging restart naturally based on log_disarmed should do
|
|
// the trick:
|
|
_rotate_pending = true;
|
|
}
|
|
}
|
|
|
|
// this sensor is enabled if we should be logging at the moment
|
|
bool AP_Logger_Backend::logging_enabled() const
|
|
{
|
|
if (hal.util->get_soft_armed() ||
|
|
_front.log_while_disarmed()) {
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void AP_Logger_Backend::Write_AP_Logger_Stats_File(const struct df_stats &_stats)
|
|
{
|
|
const struct log_DSF pkt {
|
|
LOG_PACKET_HEADER_INIT(LOG_DF_FILE_STATS),
|
|
time_us : AP_HAL::micros64(),
|
|
dropped : _dropped,
|
|
blocks : _stats.blocks,
|
|
bytes : _stats.bytes,
|
|
buf_space_min : _stats.buf_space_min,
|
|
buf_space_max : _stats.buf_space_max,
|
|
buf_space_avg : (_stats.blocks) ? (_stats.buf_space_sigma / _stats.blocks) : 0,
|
|
};
|
|
WriteBlock(&pkt, sizeof(pkt));
|
|
}
|
|
|
|
void AP_Logger_Backend::df_stats_gather(const uint16_t bytes_written, uint32_t space_remaining)
|
|
{
|
|
if (space_remaining < stats.buf_space_min) {
|
|
stats.buf_space_min = space_remaining;
|
|
}
|
|
if (space_remaining > stats.buf_space_max) {
|
|
stats.buf_space_max = space_remaining;
|
|
}
|
|
stats.buf_space_sigma += space_remaining;
|
|
stats.bytes += bytes_written;
|
|
stats.blocks++;
|
|
}
|
|
|
|
void AP_Logger_Backend::df_stats_clear() {
|
|
memset(&stats, '\0', sizeof(stats));
|
|
stats.buf_space_min = -1;
|
|
}
|
|
|
|
void AP_Logger_Backend::df_stats_log() {
|
|
Write_AP_Logger_Stats_File(stats);
|
|
df_stats_clear();
|
|
}
|
|
|
|
|
|
// class to handle rate limiting of log messages
|
|
AP_Logger_RateLimiter::AP_Logger_RateLimiter(const AP_Logger &_front, const AP_Float &_limit_hz, const AP_Float &_disarm_limit_hz)
|
|
: front(_front),
|
|
rate_limit_hz(_limit_hz),
|
|
disarm_rate_limit_hz(_disarm_limit_hz)
|
|
{
|
|
}
|
|
|
|
/*
|
|
return false if a streaming message should not be sent yet
|
|
*/
|
|
bool AP_Logger_RateLimiter::should_log_streaming(uint8_t msgid, float rate_hz)
|
|
{
|
|
if (front._log_pause) {
|
|
return false;
|
|
}
|
|
const uint16_t now = AP_HAL::millis16();
|
|
uint16_t delta_ms = now - last_send_ms[msgid];
|
|
if (is_positive(rate_hz) && delta_ms < 1000.0 / rate_hz) {
|
|
// too soon
|
|
return false;
|
|
}
|
|
last_send_ms[msgid] = now;
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
return true if the message is not a streaming message or the gap
|
|
from the last message is more than the message rate
|
|
*/
|
|
bool AP_Logger_RateLimiter::should_log(uint8_t msgid, bool writev_streaming)
|
|
{
|
|
float rate_hz = rate_limit_hz;
|
|
if (!hal.util->get_soft_armed() &&
|
|
!AP::logger().in_log_persistance() &&
|
|
!is_zero(disarm_rate_limit_hz)) {
|
|
rate_hz = disarm_rate_limit_hz;
|
|
}
|
|
if (!is_positive(rate_hz) && !front._log_pause) {
|
|
// no rate limiting if not paused and rate is zero(user changed the parameter)
|
|
return true;
|
|
}
|
|
if (last_send_ms[msgid] == 0 && !writev_streaming) {
|
|
// might be non streaming. check the not_streaming bitmask
|
|
// cache
|
|
if (not_streaming.get(msgid)) {
|
|
return true;
|
|
}
|
|
const auto *mtype = front.structure_for_msg_type(msgid);
|
|
if (mtype == nullptr ||
|
|
mtype->streaming == false) {
|
|
not_streaming.set(msgid);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
#if !defined(HAL_BUILD_AP_PERIPH)
|
|
// if we've already decided on sending this msgid in this tick then use the
|
|
// same decision again
|
|
const uint16_t sched_ticks = AP::scheduler().ticks();
|
|
if (sched_ticks == last_sched_count[msgid]) {
|
|
return last_return.get(msgid);
|
|
}
|
|
last_sched_count[msgid] = sched_ticks;
|
|
#endif
|
|
|
|
bool ret = should_log_streaming(msgid, rate_hz);
|
|
if (ret) {
|
|
last_return.set(msgid);
|
|
} else {
|
|
last_return.clear(msgid);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
#endif // HAL_LOGGING_ENABLED
|