2013-04-19 04:49:16 -03:00
|
|
|
#include <stdlib.h>
|
2015-10-26 09:01:52 -03:00
|
|
|
|
2015-08-11 03:28:46 -03:00
|
|
|
#include <AP_AHRS/AP_AHRS.h>
|
2015-10-26 09:01:52 -03:00
|
|
|
#include <AP_Baro/AP_Baro.h>
|
2015-08-15 19:53:26 -03:00
|
|
|
#include <AP_BattMonitor/AP_BattMonitor.h>
|
2015-08-11 03:28:46 -03:00
|
|
|
#include <AP_Compass/AP_Compass.h>
|
2015-10-26 09:01:52 -03:00
|
|
|
#include <AP_HAL/AP_HAL.h>
|
|
|
|
#include <AP_Math/AP_Math.h>
|
|
|
|
#include <AP_Param/AP_Param.h>
|
2016-03-24 22:11:11 -03:00
|
|
|
#include <AP_Motors/AP_Motors.h>
|
|
|
|
#include <AC_AttitudeControl/AC_AttitudeControl.h>
|
|
|
|
#include <AC_AttitudeControl/AC_PosControl.h>
|
2019-04-05 19:48:11 -03:00
|
|
|
#include <AP_RSSI/AP_RSSI.h>
|
2019-06-13 23:58:15 -03:00
|
|
|
#include <AP_GPS/AP_GPS.h>
|
2013-04-19 04:49:16 -03:00
|
|
|
|
2019-01-18 00:23:42 -04:00
|
|
|
#include "AP_Logger.h"
|
|
|
|
#include "AP_Logger_File.h"
|
|
|
|
#include "AP_Logger_MAVLink.h"
|
|
|
|
#include "LoggerMessageWriter.h"
|
2015-08-06 09:18:28 -03:00
|
|
|
|
2013-04-19 04:49:16 -03:00
|
|
|
extern const AP_HAL::HAL& hal;
|
2013-01-14 23:02:08 -04:00
|
|
|
|
|
|
|
|
2013-04-19 10:19:16 -03:00
|
|
|
/*
|
2015-06-25 10:53:20 -03:00
|
|
|
write a structure format to the log - should be in frontend
|
2013-04-19 10:19:16 -03:00
|
|
|
*/
|
2019-01-18 00:24:08 -04:00
|
|
|
void AP_Logger_Backend::Fill_Format(const struct LogStructure *s, struct log_Format &pkt)
|
2013-04-19 10:19:16 -03:00
|
|
|
{
|
2013-12-20 22:56:54 -04:00
|
|
|
memset(&pkt, 0, sizeof(pkt));
|
2013-12-16 20:12:42 -04:00
|
|
|
pkt.head1 = HEAD_BYTE1;
|
|
|
|
pkt.head2 = HEAD_BYTE2;
|
|
|
|
pkt.msgid = LOG_FORMAT_MSG;
|
2015-12-23 12:06:58 -04:00
|
|
|
pkt.type = s->msg_type;
|
|
|
|
pkt.length = s->msg_len;
|
2015-10-25 13:23:25 -03:00
|
|
|
strncpy(pkt.name, s->name, sizeof(pkt.name));
|
|
|
|
strncpy(pkt.format, s->format, sizeof(pkt.format));
|
|
|
|
strncpy(pkt.labels, s->labels, sizeof(pkt.labels));
|
2013-12-16 20:12:42 -04:00
|
|
|
}
|
|
|
|
|
2015-12-07 20:51:46 -04:00
|
|
|
/*
|
|
|
|
Pack a LogStructure packet into a structure suitable to go to the logfile:
|
|
|
|
*/
|
2019-01-18 00:24:08 -04:00
|
|
|
void AP_Logger_Backend::Fill_Format_Units(const struct LogStructure *s, struct log_Format_Units &pkt)
|
2015-12-07 20:51:46 -04:00
|
|
|
{
|
|
|
|
memset(&pkt, 0, sizeof(pkt));
|
|
|
|
pkt.head1 = HEAD_BYTE1;
|
|
|
|
pkt.head2 = HEAD_BYTE2;
|
|
|
|
pkt.msgid = LOG_FORMAT_UNITS_MSG;
|
|
|
|
pkt.time_us = AP_HAL::micros64();
|
|
|
|
pkt.format_type = s->msg_type;
|
|
|
|
strncpy(pkt.units, s->units, sizeof(pkt.units));
|
|
|
|
strncpy(pkt.multipliers, s->multipliers, sizeof(pkt.multipliers));
|
|
|
|
}
|
|
|
|
|
2013-12-16 20:12:42 -04:00
|
|
|
/*
|
|
|
|
write a structure format to the log
|
|
|
|
*/
|
2019-01-18 00:24:08 -04:00
|
|
|
bool AP_Logger_Backend::Write_Format(const struct LogStructure *s)
|
2013-12-16 20:12:42 -04:00
|
|
|
{
|
|
|
|
struct log_Format pkt;
|
2019-01-18 00:24:08 -04:00
|
|
|
Fill_Format(s, pkt);
|
2015-08-06 09:18:28 -03:00
|
|
|
return WriteCriticalBlock(&pkt, sizeof(pkt));
|
2013-04-19 10:19:16 -03:00
|
|
|
}
|
|
|
|
|
2015-12-07 20:51:46 -04:00
|
|
|
/*
|
|
|
|
write a unit definition
|
|
|
|
*/
|
2019-01-18 00:24:08 -04:00
|
|
|
bool AP_Logger_Backend::Write_Unit(const struct UnitStructure *s)
|
2015-12-07 20:51:46 -04:00
|
|
|
{
|
|
|
|
struct log_Unit pkt = {
|
|
|
|
LOG_PACKET_HEADER_INIT(LOG_UNIT_MSG),
|
|
|
|
time_us : AP_HAL::micros64(),
|
|
|
|
type : s->ID,
|
|
|
|
unit : { }
|
|
|
|
};
|
|
|
|
strncpy(pkt.unit, s->unit, sizeof(pkt.unit));
|
|
|
|
|
|
|
|
return WriteCriticalBlock(&pkt, sizeof(pkt));
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
write a unit-multiplier definition
|
|
|
|
*/
|
2019-01-18 00:24:08 -04:00
|
|
|
bool AP_Logger_Backend::Write_Multiplier(const struct MultiplierStructure *s)
|
2015-12-07 20:51:46 -04:00
|
|
|
{
|
|
|
|
struct log_Format_Multiplier pkt = {
|
|
|
|
LOG_PACKET_HEADER_INIT(LOG_MULT_MSG),
|
|
|
|
time_us : AP_HAL::micros64(),
|
|
|
|
type : s->ID,
|
|
|
|
multiplier : s->multiplier,
|
|
|
|
};
|
|
|
|
|
|
|
|
return WriteCriticalBlock(&pkt, sizeof(pkt));
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
write the units for a format to the log
|
|
|
|
*/
|
2019-01-18 00:24:08 -04:00
|
|
|
bool AP_Logger_Backend::Write_Format_Units(const struct LogStructure *s)
|
2015-12-07 20:51:46 -04:00
|
|
|
{
|
|
|
|
struct log_Format_Units pkt;
|
2019-01-18 00:24:08 -04:00
|
|
|
Fill_Format_Units(s, pkt);
|
2015-12-07 20:51:46 -04:00
|
|
|
return WriteCriticalBlock(&pkt, sizeof(pkt));
|
|
|
|
}
|
|
|
|
|
2013-04-19 04:49:16 -03:00
|
|
|
/*
|
|
|
|
write a parameter to the log
|
|
|
|
*/
|
2019-01-18 00:24:08 -04:00
|
|
|
bool AP_Logger_Backend::Write_Parameter(const char *name, float value)
|
2013-04-19 04:49:16 -03:00
|
|
|
{
|
|
|
|
struct log_Parameter pkt = {
|
|
|
|
LOG_PACKET_HEADER_INIT(LOG_PARAMETER_MSG),
|
2015-11-19 23:15:08 -04:00
|
|
|
time_us : AP_HAL::micros64(),
|
2013-04-19 04:49:16 -03:00
|
|
|
name : {},
|
|
|
|
value : value
|
|
|
|
};
|
|
|
|
strncpy(pkt.name, name, sizeof(pkt.name));
|
2015-08-06 09:18:28 -03:00
|
|
|
return WriteCriticalBlock(&pkt, sizeof(pkt));
|
2013-04-19 04:49:16 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
write a parameter to the log
|
|
|
|
*/
|
2019-01-18 00:24:08 -04:00
|
|
|
bool AP_Logger_Backend::Write_Parameter(const AP_Param *ap,
|
2015-11-09 18:14:22 -04:00
|
|
|
const AP_Param::ParamToken &token,
|
|
|
|
enum ap_var_type type)
|
2013-04-19 04:49:16 -03:00
|
|
|
{
|
|
|
|
char name[16];
|
|
|
|
ap->copy_name_token(token, &name[0], sizeof(name), true);
|
2019-01-18 00:24:08 -04:00
|
|
|
return Write_Parameter(name, ap->cast_to_float(type));
|
2013-04-19 04:49:16 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Write an GPS packet
|
2019-01-18 00:24:08 -04:00
|
|
|
void AP_Logger::Write_GPS(uint8_t i, uint64_t time_us)
|
2013-04-19 04:49:16 -03:00
|
|
|
{
|
2018-04-11 09:17:46 -03:00
|
|
|
const AP_GPS &gps = AP::gps();
|
2016-05-05 00:37:16 -03:00
|
|
|
if (time_us == 0) {
|
|
|
|
time_us = AP_HAL::micros64();
|
|
|
|
}
|
2015-08-05 03:25:48 -03:00
|
|
|
const struct Location &loc = gps.location(i);
|
|
|
|
struct log_GPS pkt = {
|
|
|
|
LOG_PACKET_HEADER_INIT((uint8_t)(LOG_GPS_MSG+i)),
|
2016-05-05 00:37:16 -03:00
|
|
|
time_us : time_us,
|
2015-08-05 03:25:48 -03:00
|
|
|
status : (uint8_t)gps.status(i),
|
|
|
|
gps_week_ms : gps.time_week_ms(i),
|
|
|
|
gps_week : gps.time_week(i),
|
|
|
|
num_sats : gps.num_sats(i),
|
|
|
|
hdop : gps.get_hdop(i),
|
|
|
|
latitude : loc.lat,
|
|
|
|
longitude : loc.lng,
|
|
|
|
altitude : loc.alt,
|
2016-05-04 21:23:34 -03:00
|
|
|
ground_speed : gps.ground_speed(i),
|
2016-05-04 22:21:34 -03:00
|
|
|
ground_course : gps.ground_course(i),
|
2015-08-05 03:25:48 -03:00
|
|
|
vel_z : gps.velocity(i).z,
|
|
|
|
used : (uint8_t)(gps.primary_sensor() == i)
|
|
|
|
};
|
|
|
|
WriteBlock(&pkt, sizeof(pkt));
|
2015-09-09 01:52:52 -03:00
|
|
|
|
2016-05-12 13:50:13 -03:00
|
|
|
/* write auxiliary accuracy information as well */
|
2015-09-09 01:52:52 -03:00
|
|
|
float hacc = 0, vacc = 0, sacc = 0;
|
|
|
|
gps.horizontal_accuracy(i, hacc);
|
|
|
|
gps.vertical_accuracy(i, vacc);
|
|
|
|
gps.speed_accuracy(i, sacc);
|
|
|
|
struct log_GPA pkt2 = {
|
|
|
|
LOG_PACKET_HEADER_INIT((uint8_t)(LOG_GPA_MSG+i)),
|
2016-05-05 00:37:16 -03:00
|
|
|
time_us : time_us,
|
2015-09-09 01:52:52 -03:00
|
|
|
vdop : gps.get_vdop(i),
|
2017-04-16 19:56:55 -03:00
|
|
|
hacc : (uint16_t)MIN((hacc*100), UINT16_MAX),
|
|
|
|
vacc : (uint16_t)MIN((vacc*100), UINT16_MAX),
|
|
|
|
sacc : (uint16_t)MIN((sacc*100), UINT16_MAX),
|
2016-05-05 03:03:59 -03:00
|
|
|
have_vv : (uint8_t)gps.have_vertical_velocity(i),
|
2017-07-08 20:06:47 -03:00
|
|
|
sample_ms : gps.last_message_time_ms(i),
|
|
|
|
delta_ms : gps.last_message_delta_time_ms(i)
|
2015-09-09 01:52:52 -03:00
|
|
|
};
|
|
|
|
WriteBlock(&pkt2, sizeof(pkt2));
|
2013-04-19 04:49:16 -03:00
|
|
|
}
|
|
|
|
|
2015-09-10 07:27:47 -03:00
|
|
|
|
2013-11-25 17:44:00 -04:00
|
|
|
// Write an RCIN packet
|
2019-01-18 00:24:08 -04:00
|
|
|
void AP_Logger::Write_RCIN(void)
|
2013-11-25 17:44:00 -04:00
|
|
|
{
|
2018-08-08 21:34:15 -03:00
|
|
|
uint16_t values[14] = {};
|
|
|
|
rc().get_radio_in(values, ARRAY_SIZE(values));
|
2013-11-25 17:44:00 -04:00
|
|
|
struct log_RCIN pkt = {
|
|
|
|
LOG_PACKET_HEADER_INIT(LOG_RCIN_MSG),
|
2015-11-19 23:15:08 -04:00
|
|
|
time_us : AP_HAL::micros64(),
|
2018-04-26 09:00:30 -03:00
|
|
|
chan1 : values[0],
|
|
|
|
chan2 : values[1],
|
|
|
|
chan3 : values[2],
|
|
|
|
chan4 : values[3],
|
|
|
|
chan5 : values[4],
|
|
|
|
chan6 : values[5],
|
|
|
|
chan7 : values[6],
|
|
|
|
chan8 : values[7],
|
|
|
|
chan9 : values[8],
|
|
|
|
chan10 : values[9],
|
|
|
|
chan11 : values[10],
|
|
|
|
chan12 : values[11],
|
|
|
|
chan13 : values[12],
|
|
|
|
chan14 : values[13]
|
2013-11-25 17:44:00 -04:00
|
|
|
};
|
|
|
|
WriteBlock(&pkt, sizeof(pkt));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Write an SERVO packet
|
2019-01-18 00:24:08 -04:00
|
|
|
void AP_Logger::Write_RCOUT(void)
|
2013-11-25 17:44:00 -04:00
|
|
|
{
|
2013-11-27 07:14:17 -04:00
|
|
|
struct log_RCOUT pkt = {
|
|
|
|
LOG_PACKET_HEADER_INIT(LOG_RCOUT_MSG),
|
2015-11-19 23:15:08 -04:00
|
|
|
time_us : AP_HAL::micros64(),
|
2013-11-25 17:44:00 -04:00
|
|
|
chan1 : hal.rcout->read(0),
|
|
|
|
chan2 : hal.rcout->read(1),
|
|
|
|
chan3 : hal.rcout->read(2),
|
|
|
|
chan4 : hal.rcout->read(3),
|
|
|
|
chan5 : hal.rcout->read(4),
|
|
|
|
chan6 : hal.rcout->read(5),
|
|
|
|
chan7 : hal.rcout->read(6),
|
2014-08-12 12:08:17 -03:00
|
|
|
chan8 : hal.rcout->read(7),
|
|
|
|
chan9 : hal.rcout->read(8),
|
|
|
|
chan10 : hal.rcout->read(9),
|
|
|
|
chan11 : hal.rcout->read(10),
|
2016-06-06 00:42:03 -03:00
|
|
|
chan12 : hal.rcout->read(11),
|
|
|
|
chan13 : hal.rcout->read(12),
|
|
|
|
chan14 : hal.rcout->read(13)
|
2013-11-25 17:44:00 -04:00
|
|
|
};
|
|
|
|
WriteBlock(&pkt, sizeof(pkt));
|
|
|
|
}
|
|
|
|
|
2015-09-04 12:50:23 -03:00
|
|
|
// Write an RSSI packet
|
2019-04-05 19:48:11 -03:00
|
|
|
void AP_Logger::Write_RSSI()
|
2015-09-04 12:50:23 -03:00
|
|
|
{
|
2019-04-05 19:48:11 -03:00
|
|
|
AP_RSSI *rssi = AP::rssi();
|
|
|
|
if (rssi == nullptr) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-09-04 12:50:23 -03:00
|
|
|
struct log_RSSI pkt = {
|
|
|
|
LOG_PACKET_HEADER_INIT(LOG_RSSI_MSG),
|
2015-11-19 23:15:08 -04:00
|
|
|
time_us : AP_HAL::micros64(),
|
2019-04-05 19:48:11 -03:00
|
|
|
RXRSSI : rssi->read_receiver_rssi()
|
2015-09-04 12:50:23 -03:00
|
|
|
};
|
|
|
|
WriteBlock(&pkt, sizeof(pkt));
|
|
|
|
}
|
|
|
|
|
2019-01-18 00:24:08 -04:00
|
|
|
void AP_Logger::Write_Baro_instance(uint64_t time_us, uint8_t baro_instance, enum LogMessages type)
|
2014-01-27 19:35:18 -04:00
|
|
|
{
|
2018-03-05 16:36:23 -04:00
|
|
|
AP_Baro &baro = AP::baro();
|
2016-05-17 22:27:39 -03:00
|
|
|
float climbrate = baro.get_climb_rate();
|
2016-05-13 17:01:43 -03:00
|
|
|
float drift_offset = baro.get_baro_drift_offset();
|
2017-03-24 02:00:22 -03:00
|
|
|
float ground_temp = baro.get_ground_temperature();
|
2014-01-27 19:35:18 -04:00
|
|
|
struct log_BARO pkt = {
|
2018-04-09 03:04:17 -03:00
|
|
|
LOG_PACKET_HEADER_INIT(type),
|
2015-05-21 22:34:05 -03:00
|
|
|
time_us : time_us,
|
2017-09-27 21:19:35 -03:00
|
|
|
altitude : baro.get_altitude(baro_instance),
|
|
|
|
pressure : baro.get_pressure(baro_instance),
|
|
|
|
temperature : (int16_t)(baro.get_temperature(baro_instance) * 100 + 0.5f),
|
2016-05-17 22:27:39 -03:00
|
|
|
climbrate : climbrate,
|
2017-09-27 21:19:35 -03:00
|
|
|
sample_time_ms: baro.get_last_update(baro_instance),
|
2016-05-13 17:01:43 -03:00
|
|
|
drift_offset : drift_offset,
|
2017-03-24 02:00:22 -03:00
|
|
|
ground_temp : ground_temp,
|
2014-01-27 19:35:18 -04:00
|
|
|
};
|
|
|
|
WriteBlock(&pkt, sizeof(pkt));
|
2017-09-27 21:19:35 -03:00
|
|
|
}
|
2015-10-14 12:53:14 -03:00
|
|
|
|
2017-09-27 21:19:35 -03:00
|
|
|
// Write a BARO packet
|
2019-01-18 00:24:08 -04:00
|
|
|
void AP_Logger::Write_Baro(uint64_t time_us)
|
2017-09-27 21:19:35 -03:00
|
|
|
{
|
|
|
|
if (time_us == 0) {
|
|
|
|
time_us = AP_HAL::micros64();
|
|
|
|
}
|
2018-03-05 16:36:23 -04:00
|
|
|
const AP_Baro &baro = AP::baro();
|
2019-01-18 00:24:08 -04:00
|
|
|
Write_Baro_instance(time_us, 0, LOG_BARO_MSG);
|
2015-01-05 17:37:23 -04:00
|
|
|
if (baro.num_instances() > 1 && baro.healthy(1)) {
|
2019-01-18 00:24:08 -04:00
|
|
|
Write_Baro_instance(time_us, 1, LOG_BAR2_MSG);
|
2015-01-05 17:37:23 -04:00
|
|
|
}
|
2015-09-10 07:27:47 -03:00
|
|
|
if (baro.num_instances() > 2 && baro.healthy(2)) {
|
2019-01-18 00:24:08 -04:00
|
|
|
Write_Baro_instance(time_us, 2, LOG_BAR3_MSG);
|
2015-09-10 07:27:47 -03:00
|
|
|
}
|
2014-01-27 19:35:18 -04:00
|
|
|
}
|
2013-04-19 04:49:16 -03:00
|
|
|
|
2019-01-18 00:24:08 -04:00
|
|
|
void AP_Logger::Write_IMU_instance(const uint64_t time_us, const uint8_t imu_instance, const enum LogMessages type)
|
2013-04-19 04:49:16 -03:00
|
|
|
{
|
2018-03-10 05:36:33 -04:00
|
|
|
const AP_InertialSensor &ins = AP::ins();
|
2017-09-28 23:34:58 -03:00
|
|
|
const Vector3f &gyro = ins.get_gyro(imu_instance);
|
|
|
|
const Vector3f &accel = ins.get_accel(imu_instance);
|
2013-04-19 04:49:16 -03:00
|
|
|
struct log_IMU pkt = {
|
2017-09-28 23:34:58 -03:00
|
|
|
LOG_PACKET_HEADER_INIT(type),
|
2015-05-21 22:34:05 -03:00
|
|
|
time_us : time_us,
|
2013-04-19 04:49:16 -03:00
|
|
|
gyro_x : gyro.x,
|
|
|
|
gyro_y : gyro.y,
|
|
|
|
gyro_z : gyro.z,
|
|
|
|
accel_x : accel.x,
|
|
|
|
accel_y : accel.y,
|
2014-12-29 06:19:50 -04:00
|
|
|
accel_z : accel.z,
|
2017-09-28 23:34:58 -03:00
|
|
|
gyro_error : ins.get_gyro_error_count(imu_instance),
|
|
|
|
accel_error : ins.get_accel_error_count(imu_instance),
|
|
|
|
temperature : ins.get_temperature(imu_instance),
|
|
|
|
gyro_health : (uint8_t)ins.get_gyro_health(imu_instance),
|
|
|
|
accel_health : (uint8_t)ins.get_accel_health(imu_instance),
|
|
|
|
gyro_rate : ins.get_gyro_rate_hz(imu_instance),
|
|
|
|
accel_rate : ins.get_accel_rate_hz(imu_instance),
|
2013-04-19 04:49:16 -03:00
|
|
|
};
|
|
|
|
WriteBlock(&pkt, sizeof(pkt));
|
2017-09-28 23:34:58 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Write an raw accel/gyro data packet
|
2019-01-18 00:24:08 -04:00
|
|
|
void AP_Logger::Write_IMU()
|
2017-09-28 23:34:58 -03:00
|
|
|
{
|
|
|
|
uint64_t time_us = AP_HAL::micros64();
|
|
|
|
|
2018-03-10 05:36:33 -04:00
|
|
|
const AP_InertialSensor &ins = AP::ins();
|
|
|
|
|
2019-01-18 00:24:08 -04:00
|
|
|
Write_IMU_instance(time_us, 0, LOG_IMU_MSG);
|
2013-12-08 05:45:04 -04:00
|
|
|
if (ins.get_gyro_count() < 2 && ins.get_accel_count() < 2) {
|
|
|
|
return;
|
|
|
|
}
|
2015-10-14 12:48:19 -03:00
|
|
|
|
2019-01-18 00:24:08 -04:00
|
|
|
Write_IMU_instance(time_us, 1, LOG_IMU2_MSG);
|
2017-09-28 23:34:58 -03:00
|
|
|
|
2014-06-26 01:04:55 -03:00
|
|
|
if (ins.get_gyro_count() < 3 && ins.get_accel_count() < 3) {
|
|
|
|
return;
|
|
|
|
}
|
2017-09-28 23:34:58 -03:00
|
|
|
|
2019-01-18 00:24:08 -04:00
|
|
|
Write_IMU_instance(time_us, 2, LOG_IMU3_MSG);
|
2013-12-08 05:45:04 -04:00
|
|
|
}
|
|
|
|
|
2015-06-13 10:06:49 -03:00
|
|
|
// Write an accel/gyro delta time data packet
|
2019-01-18 00:24:08 -04:00
|
|
|
void AP_Logger::Write_IMUDT_instance(const uint64_t time_us, const uint8_t imu_instance, const enum LogMessages type)
|
2015-06-13 10:06:49 -03:00
|
|
|
{
|
2018-03-10 05:36:33 -04:00
|
|
|
const AP_InertialSensor &ins = AP::ins();
|
2015-06-13 10:06:49 -03:00
|
|
|
float delta_t = ins.get_delta_time();
|
2017-09-29 00:28:34 -03:00
|
|
|
float delta_vel_t = ins.get_delta_velocity_dt(imu_instance);
|
|
|
|
float delta_ang_t = ins.get_delta_angle_dt(imu_instance);
|
2015-06-13 10:06:49 -03:00
|
|
|
Vector3f delta_angle, delta_velocity;
|
2017-09-29 00:28:34 -03:00
|
|
|
ins.get_delta_angle(imu_instance, delta_angle);
|
|
|
|
ins.get_delta_velocity(imu_instance, delta_velocity);
|
2015-06-13 10:06:49 -03:00
|
|
|
|
|
|
|
struct log_IMUDT pkt = {
|
2017-09-29 00:28:34 -03:00
|
|
|
LOG_PACKET_HEADER_INIT(type),
|
2015-06-13 10:06:49 -03:00
|
|
|
time_us : time_us,
|
|
|
|
delta_time : delta_t,
|
|
|
|
delta_vel_dt : delta_vel_t,
|
2016-04-26 02:50:29 -03:00
|
|
|
delta_ang_dt : delta_ang_t,
|
2015-06-13 10:06:49 -03:00
|
|
|
delta_ang_x : delta_angle.x,
|
|
|
|
delta_ang_y : delta_angle.y,
|
|
|
|
delta_ang_z : delta_angle.z,
|
|
|
|
delta_vel_x : delta_velocity.x,
|
|
|
|
delta_vel_y : delta_velocity.y,
|
|
|
|
delta_vel_z : delta_velocity.z
|
|
|
|
};
|
2017-09-29 00:28:34 -03:00
|
|
|
WriteBlock(&pkt, sizeof(pkt));
|
|
|
|
}
|
|
|
|
|
2019-01-18 00:24:08 -04:00
|
|
|
void AP_Logger::Write_IMUDT(uint64_t time_us, uint8_t imu_mask)
|
2017-09-29 00:28:34 -03:00
|
|
|
{
|
2018-03-10 05:36:33 -04:00
|
|
|
const AP_InertialSensor &ins = AP::ins();
|
2016-05-08 23:26:30 -03:00
|
|
|
if (imu_mask & 1) {
|
2019-01-18 00:24:08 -04:00
|
|
|
Write_IMUDT_instance(time_us, 0, LOG_IMUDT_MSG);
|
2016-05-08 23:26:30 -03:00
|
|
|
}
|
2016-05-04 23:16:16 -03:00
|
|
|
if ((ins.get_gyro_count() < 2 && ins.get_accel_count() < 2) || !ins.use_gyro(1)) {
|
2015-06-13 10:06:49 -03:00
|
|
|
return;
|
|
|
|
}
|
2015-10-14 12:48:19 -03:00
|
|
|
|
2016-05-08 23:26:30 -03:00
|
|
|
if (imu_mask & 2) {
|
2019-01-18 00:24:08 -04:00
|
|
|
Write_IMUDT_instance(time_us, 1, LOG_IMUDT2_MSG);
|
2016-05-08 23:26:30 -03:00
|
|
|
}
|
2015-06-13 10:06:49 -03:00
|
|
|
|
2016-05-04 23:16:16 -03:00
|
|
|
if ((ins.get_gyro_count() < 3 && ins.get_accel_count() < 3) || !ins.use_gyro(2)) {
|
2015-06-13 10:06:49 -03:00
|
|
|
return;
|
|
|
|
}
|
2017-09-29 00:28:34 -03:00
|
|
|
|
2016-05-08 23:26:30 -03:00
|
|
|
if (imu_mask & 4) {
|
2019-01-18 00:24:08 -04:00
|
|
|
Write_IMUDT_instance(time_us, 2, LOG_IMUDT3_MSG);
|
2016-05-08 23:26:30 -03:00
|
|
|
}
|
2015-06-13 10:06:49 -03:00
|
|
|
}
|
|
|
|
|
2019-01-18 00:24:08 -04:00
|
|
|
void AP_Logger::Write_Vibration()
|
2015-06-11 10:26:45 -03:00
|
|
|
{
|
2015-11-19 23:15:08 -04:00
|
|
|
uint64_t time_us = AP_HAL::micros64();
|
2018-03-10 05:36:33 -04:00
|
|
|
const AP_InertialSensor &ins = AP::ins();
|
|
|
|
const Vector3f vibration = ins.get_vibration_levels();
|
2015-06-11 10:26:45 -03:00
|
|
|
struct log_Vibe pkt = {
|
|
|
|
LOG_PACKET_HEADER_INIT(LOG_VIBE_MSG),
|
|
|
|
time_us : time_us,
|
|
|
|
vibe_x : vibration.x,
|
|
|
|
vibe_y : vibration.y,
|
|
|
|
vibe_z : vibration.z,
|
|
|
|
clipping_0 : ins.get_accel_clip_count(0),
|
|
|
|
clipping_1 : ins.get_accel_clip_count(1),
|
|
|
|
clipping_2 : ins.get_accel_clip_count(2)
|
|
|
|
};
|
|
|
|
WriteBlock(&pkt, sizeof(pkt));
|
|
|
|
}
|
|
|
|
|
2019-01-18 00:24:08 -04:00
|
|
|
bool AP_Logger_Backend::Write_Mission_Cmd(const AP_Mission &mission,
|
2015-11-09 18:14:22 -04:00
|
|
|
const AP_Mission::Mission_Command &cmd)
|
2015-06-30 01:33:50 -03:00
|
|
|
{
|
2018-11-24 23:10:07 -04:00
|
|
|
mavlink_mission_item_int_t mav_cmd = {};
|
|
|
|
AP_Mission::mission_cmd_to_mavlink_int(cmd,mav_cmd);
|
|
|
|
struct log_Cmd pkt = {
|
|
|
|
LOG_PACKET_HEADER_INIT(LOG_CMD_MSG),
|
|
|
|
time_us : AP_HAL::micros64(),
|
|
|
|
command_total : mission.num_commands(),
|
|
|
|
sequence : mav_cmd.seq,
|
|
|
|
command : mav_cmd.command,
|
|
|
|
param1 : mav_cmd.param1,
|
|
|
|
param2 : mav_cmd.param2,
|
|
|
|
param3 : mav_cmd.param3,
|
|
|
|
param4 : mav_cmd.param4,
|
|
|
|
latitude : mav_cmd.x,
|
|
|
|
longitude : mav_cmd.y,
|
|
|
|
altitude : mav_cmd.z,
|
|
|
|
frame : mav_cmd.frame
|
|
|
|
};
|
|
|
|
return WriteBlock(&pkt, sizeof(pkt));
|
2015-06-30 01:33:50 -03:00
|
|
|
}
|
|
|
|
|
2019-01-29 21:43:24 -04:00
|
|
|
void AP_Logger_Backend::Write_EntireMission()
|
2015-06-30 01:33:50 -03:00
|
|
|
{
|
2019-01-18 00:23:42 -04:00
|
|
|
LoggerMessageWriter_WriteEntireMission writer;
|
2019-02-11 04:38:01 -04:00
|
|
|
writer.set_logger_backend(this);
|
2015-08-06 09:18:28 -03:00
|
|
|
writer.process();
|
2015-06-30 01:33:50 -03:00
|
|
|
}
|
|
|
|
|
2013-05-02 20:18:14 -03:00
|
|
|
// Write a text message to the log
|
2019-01-18 00:24:08 -04:00
|
|
|
bool AP_Logger_Backend::Write_Message(const char *message)
|
2013-05-02 20:18:14 -03:00
|
|
|
{
|
|
|
|
struct log_Message pkt = {
|
|
|
|
LOG_PACKET_HEADER_INIT(LOG_MESSAGE_MSG),
|
2015-11-19 23:15:08 -04:00
|
|
|
time_us : AP_HAL::micros64(),
|
2013-05-02 20:18:14 -03:00
|
|
|
msg : {}
|
|
|
|
};
|
|
|
|
strncpy(pkt.msg, message, sizeof(pkt.msg));
|
2015-08-27 23:47:16 -03:00
|
|
|
return WriteCriticalBlock(&pkt, sizeof(pkt));
|
2013-05-02 20:18:14 -03:00
|
|
|
}
|
|
|
|
|
2019-01-18 00:24:08 -04:00
|
|
|
void AP_Logger::Write_Power(void)
|
2014-02-13 07:07:32 -04:00
|
|
|
{
|
2019-01-15 21:15:24 -04:00
|
|
|
#if CONFIG_HAL_BOARD == HAL_BOARD_CHIBIOS
|
2018-04-12 20:49:48 -03:00
|
|
|
uint8_t safety_and_armed = uint8_t(hal.util->safety_switch_state());
|
|
|
|
if (hal.util->get_soft_armed()) {
|
|
|
|
// encode armed state in bit 3
|
|
|
|
safety_and_armed |= 1U<<2;
|
|
|
|
}
|
2014-02-13 07:07:32 -04:00
|
|
|
struct log_POWR pkt = {
|
|
|
|
LOG_PACKET_HEADER_INIT(LOG_POWR_MSG),
|
2015-11-19 23:15:08 -04:00
|
|
|
time_us : AP_HAL::micros64(),
|
2016-06-02 17:55:05 -03:00
|
|
|
Vcc : hal.analogin->board_voltage(),
|
|
|
|
Vservo : hal.analogin->servorail_voltage(),
|
2018-04-12 20:49:48 -03:00
|
|
|
flags : hal.analogin->power_status_flags(),
|
|
|
|
safety_and_arm : safety_and_armed
|
2014-02-13 07:07:32 -04:00
|
|
|
};
|
|
|
|
WriteBlock(&pkt, sizeof(pkt));
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2014-01-03 01:01:08 -04:00
|
|
|
// Write an AHRS2 packet
|
2019-01-18 00:24:08 -04:00
|
|
|
void AP_Logger::Write_AHRS2(AP_AHRS &ahrs)
|
2014-01-03 01:01:08 -04:00
|
|
|
{
|
|
|
|
Vector3f euler;
|
|
|
|
struct Location loc;
|
2017-04-15 08:19:36 -03:00
|
|
|
Quaternion quat;
|
2019-03-12 06:45:14 -03:00
|
|
|
if (!ahrs.get_secondary_attitude(euler) || !ahrs.get_secondary_position(loc) || !ahrs.get_secondary_quaternion(quat)) {
|
2014-01-03 01:01:08 -04:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
struct log_AHRS pkt = {
|
2014-01-30 18:33:12 -04:00
|
|
|
LOG_PACKET_HEADER_INIT(LOG_AHR2_MSG),
|
2015-11-19 23:15:08 -04:00
|
|
|
time_us : AP_HAL::micros64(),
|
2014-01-03 01:01:08 -04:00
|
|
|
roll : (int16_t)(degrees(euler.x)*100),
|
|
|
|
pitch : (int16_t)(degrees(euler.y)*100),
|
|
|
|
yaw : (uint16_t)(wrap_360_cd(degrees(euler.z)*100)),
|
|
|
|
alt : loc.alt*1.0e-2f,
|
|
|
|
lat : loc.lat,
|
2017-04-15 08:19:36 -03:00
|
|
|
lng : loc.lng,
|
|
|
|
q1 : quat.q1,
|
|
|
|
q2 : quat.q2,
|
|
|
|
q3 : quat.q3,
|
|
|
|
q4 : quat.q4,
|
2014-01-03 01:01:08 -04:00
|
|
|
};
|
|
|
|
WriteBlock(&pkt, sizeof(pkt));
|
|
|
|
}
|
|
|
|
|
2015-05-14 22:59:40 -03:00
|
|
|
// Write a POS packet
|
2019-01-18 00:24:08 -04:00
|
|
|
void AP_Logger::Write_POS(AP_AHRS &ahrs)
|
2015-05-14 22:59:40 -03:00
|
|
|
{
|
|
|
|
Location loc;
|
|
|
|
if (!ahrs.get_position(loc)) {
|
|
|
|
return;
|
|
|
|
}
|
2017-01-30 15:09:58 -04:00
|
|
|
float home, origin;
|
|
|
|
ahrs.get_relative_position_D_home(home);
|
2015-05-14 22:59:40 -03:00
|
|
|
struct log_POS pkt = {
|
|
|
|
LOG_PACKET_HEADER_INIT(LOG_POS_MSG),
|
2017-01-30 15:09:58 -04:00
|
|
|
time_us : AP_HAL::micros64(),
|
|
|
|
lat : loc.lat,
|
|
|
|
lng : loc.lng,
|
|
|
|
alt : loc.alt*1.0e-2f,
|
|
|
|
rel_home_alt : -home,
|
2017-07-21 23:27:45 -03:00
|
|
|
rel_origin_alt : ahrs.get_relative_position_D_origin(origin) ? -origin : quiet_nanf(),
|
2015-05-14 22:59:40 -03:00
|
|
|
};
|
|
|
|
WriteBlock(&pkt, sizeof(pkt));
|
|
|
|
}
|
|
|
|
|
2014-01-03 20:51:22 -04:00
|
|
|
#if AP_AHRS_NAVEKF_AVAILABLE
|
2016-07-14 02:08:43 -03:00
|
|
|
|
2017-04-27 22:34:02 -03:00
|
|
|
|
|
|
|
/*
|
|
|
|
write an EKF timing message
|
|
|
|
*/
|
2019-01-18 00:24:08 -04:00
|
|
|
void AP_Logger::Write_EKF_Timing(const char *name, uint64_t time_us, const struct ekf_timing &timing)
|
2017-04-27 22:34:02 -03:00
|
|
|
{
|
2019-01-18 00:24:08 -04:00
|
|
|
Write(name,
|
2018-05-23 05:56:37 -03:00
|
|
|
"TimeUS,Cnt,IMUMin,IMUMax,EKFMin,EKFMax,AngMin,AngMax,VMin,VMax",
|
|
|
|
"QIffffffff",
|
2017-04-27 22:34:02 -03:00
|
|
|
time_us,
|
|
|
|
timing.count,
|
|
|
|
(double)timing.dtIMUavg_min,
|
|
|
|
(double)timing.dtIMUavg_max,
|
2017-04-28 04:15:07 -03:00
|
|
|
(double)timing.dtEKFavg_min,
|
|
|
|
(double)timing.dtEKFavg_max,
|
2017-04-27 22:34:02 -03:00
|
|
|
(double)timing.delAngDT_min,
|
|
|
|
(double)timing.delAngDT_max,
|
|
|
|
(double)timing.delVelDT_min,
|
|
|
|
(double)timing.delVelDT_max);
|
|
|
|
}
|
|
|
|
|
2014-01-30 18:33:12 -04:00
|
|
|
#endif
|
2014-01-03 20:51:22 -04:00
|
|
|
|
2019-01-18 00:24:08 -04:00
|
|
|
void AP_Logger::Write_Radio(const mavlink_radio_t &packet)
|
2014-03-11 14:05:02 -03:00
|
|
|
{
|
|
|
|
struct log_Radio pkt = {
|
|
|
|
LOG_PACKET_HEADER_INIT(LOG_RADIO_MSG),
|
2015-11-19 23:15:08 -04:00
|
|
|
time_us : AP_HAL::micros64(),
|
2014-03-11 14:05:02 -03:00
|
|
|
rssi : packet.rssi,
|
|
|
|
remrssi : packet.remrssi,
|
|
|
|
txbuf : packet.txbuf,
|
|
|
|
noise : packet.noise,
|
|
|
|
remnoise : packet.remnoise,
|
|
|
|
rxerrors : packet.rxerrors,
|
|
|
|
fixed : packet.fixed
|
|
|
|
};
|
2016-11-22 11:34:53 -04:00
|
|
|
WriteBlock(&pkt, sizeof(pkt));
|
2014-03-11 14:05:02 -03:00
|
|
|
}
|
|
|
|
|
2014-06-10 23:55:04 -03:00
|
|
|
// Write a Camera packet
|
2019-06-27 04:24:04 -03:00
|
|
|
void AP_Logger::Write_CameraInfo(enum LogMessages msg, const Location ¤t_loc, uint64_t timestamp_us)
|
2014-06-10 23:55:04 -03:00
|
|
|
{
|
2019-06-27 04:24:04 -03:00
|
|
|
const AP_AHRS &ahrs = AP::ahrs();
|
|
|
|
|
2016-01-28 18:51:52 -04:00
|
|
|
int32_t altitude, altitude_rel, altitude_gps;
|
2019-01-02 00:46:33 -04:00
|
|
|
if (current_loc.relative_alt) {
|
2014-06-10 23:55:04 -03:00
|
|
|
altitude = current_loc.alt+ahrs.get_home().alt;
|
|
|
|
altitude_rel = current_loc.alt;
|
|
|
|
} else {
|
|
|
|
altitude = current_loc.alt;
|
|
|
|
altitude_rel = current_loc.alt - ahrs.get_home().alt;
|
|
|
|
}
|
2017-11-12 20:31:31 -04:00
|
|
|
const AP_GPS &gps = AP::gps();
|
2016-01-28 18:51:52 -04:00
|
|
|
if (gps.status() >= AP_GPS::GPS_OK_FIX_3D) {
|
|
|
|
altitude_gps = gps.location().alt;
|
|
|
|
} else {
|
|
|
|
altitude_gps = 0;
|
|
|
|
}
|
2014-06-10 23:55:04 -03:00
|
|
|
|
|
|
|
struct log_Camera pkt = {
|
2016-02-26 01:16:04 -04:00
|
|
|
LOG_PACKET_HEADER_INIT(static_cast<uint8_t>(msg)),
|
2018-05-14 22:04:09 -03:00
|
|
|
time_us : timestamp_us?timestamp_us:AP_HAL::micros64(),
|
2014-06-10 23:55:04 -03:00
|
|
|
gps_time : gps.time_week_ms(),
|
|
|
|
gps_week : gps.time_week(),
|
|
|
|
latitude : current_loc.lat,
|
|
|
|
longitude : current_loc.lng,
|
|
|
|
altitude : altitude,
|
|
|
|
altitude_rel: altitude_rel,
|
2016-01-28 18:51:52 -04:00
|
|
|
altitude_gps: altitude_gps,
|
2014-06-10 23:55:04 -03:00
|
|
|
roll : (int16_t)ahrs.roll_sensor,
|
|
|
|
pitch : (int16_t)ahrs.pitch_sensor,
|
|
|
|
yaw : (uint16_t)ahrs.yaw_sensor
|
|
|
|
};
|
2015-10-06 01:13:40 -03:00
|
|
|
WriteCriticalBlock(&pkt, sizeof(pkt));
|
2014-06-10 23:55:04 -03:00
|
|
|
}
|
2014-11-13 06:38:33 -04:00
|
|
|
|
2016-01-28 18:51:52 -04:00
|
|
|
// Write a Camera packet
|
2019-06-27 04:24:04 -03:00
|
|
|
void AP_Logger::Write_Camera(const Location ¤t_loc, uint64_t timestamp_us)
|
2016-01-28 18:51:52 -04:00
|
|
|
{
|
2019-06-27 04:24:04 -03:00
|
|
|
Write_CameraInfo(LOG_CAMERA_MSG, current_loc, timestamp_us);
|
2016-01-28 18:51:52 -04:00
|
|
|
}
|
|
|
|
|
2016-01-06 20:29:52 -04:00
|
|
|
// Write a Trigger packet
|
2019-06-27 04:24:04 -03:00
|
|
|
void AP_Logger::Write_Trigger(const Location ¤t_loc)
|
2016-01-06 20:29:52 -04:00
|
|
|
{
|
2019-06-27 04:24:04 -03:00
|
|
|
Write_CameraInfo(LOG_TRIGGER_MSG, current_loc, 0);
|
2016-01-06 20:29:52 -04:00
|
|
|
}
|
|
|
|
|
2014-12-18 12:55:11 -04:00
|
|
|
// Write an attitude packet
|
2019-01-18 00:24:08 -04:00
|
|
|
void AP_Logger::Write_Attitude(AP_AHRS &ahrs, const Vector3f &targets)
|
2017-03-30 20:37:35 -03:00
|
|
|
{
|
|
|
|
struct log_Attitude pkt = {
|
|
|
|
LOG_PACKET_HEADER_INIT(LOG_ATTITUDE_MSG),
|
|
|
|
time_us : AP_HAL::micros64(),
|
|
|
|
control_roll : (int16_t)targets.x,
|
|
|
|
roll : (int16_t)ahrs.roll_sensor,
|
|
|
|
control_pitch : (int16_t)targets.y,
|
|
|
|
pitch : (int16_t)ahrs.pitch_sensor,
|
2019-01-13 00:24:05 -04:00
|
|
|
control_yaw : (uint16_t)wrap_360_cd(targets.z),
|
|
|
|
yaw : (uint16_t)wrap_360_cd(ahrs.yaw_sensor),
|
2017-03-30 20:37:35 -03:00
|
|
|
error_rp : (uint16_t)(ahrs.get_error_rp() * 100),
|
|
|
|
error_yaw : (uint16_t)(ahrs.get_error_yaw() * 100)
|
|
|
|
};
|
|
|
|
WriteBlock(&pkt, sizeof(pkt));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Write an attitude packet
|
2019-01-18 00:24:08 -04:00
|
|
|
void AP_Logger::Write_AttitudeView(AP_AHRS_View &ahrs, const Vector3f &targets)
|
2014-12-18 12:55:11 -04:00
|
|
|
{
|
|
|
|
struct log_Attitude pkt = {
|
|
|
|
LOG_PACKET_HEADER_INIT(LOG_ATTITUDE_MSG),
|
2015-11-19 23:15:08 -04:00
|
|
|
time_us : AP_HAL::micros64(),
|
2014-12-18 12:55:11 -04:00
|
|
|
control_roll : (int16_t)targets.x,
|
|
|
|
roll : (int16_t)ahrs.roll_sensor,
|
|
|
|
control_pitch : (int16_t)targets.y,
|
|
|
|
pitch : (int16_t)ahrs.pitch_sensor,
|
2019-02-14 17:07:58 -04:00
|
|
|
control_yaw : (uint16_t)wrap_360_cd(targets.z),
|
|
|
|
yaw : (uint16_t)wrap_360_cd(ahrs.yaw_sensor),
|
2014-12-18 12:55:11 -04:00
|
|
|
error_rp : (uint16_t)(ahrs.get_error_rp() * 100),
|
|
|
|
error_yaw : (uint16_t)(ahrs.get_error_yaw() * 100)
|
|
|
|
};
|
|
|
|
WriteBlock(&pkt, sizeof(pkt));
|
|
|
|
}
|
|
|
|
|
2019-01-18 00:24:08 -04:00
|
|
|
void AP_Logger::Write_Current_instance(const uint64_t time_us,
|
2017-09-29 00:09:59 -03:00
|
|
|
const uint8_t battery_instance,
|
2017-09-29 00:15:20 -03:00
|
|
|
const enum LogMessages type,
|
|
|
|
const enum LogMessages celltype)
|
2014-12-18 17:34:29 -04:00
|
|
|
{
|
2018-01-16 15:09:47 -04:00
|
|
|
AP_BattMonitor &battery = AP::battery();
|
2017-09-29 00:09:59 -03:00
|
|
|
float temp;
|
|
|
|
bool has_temp = battery.get_temperature(temp, battery_instance);
|
|
|
|
struct log_Current pkt = {
|
|
|
|
LOG_PACKET_HEADER_INIT(type),
|
|
|
|
time_us : time_us,
|
|
|
|
voltage : battery.voltage(battery_instance),
|
|
|
|
voltage_resting : battery.voltage_resting_estimate(battery_instance),
|
|
|
|
current_amps : battery.current_amps(battery_instance),
|
2018-02-15 12:10:37 -04:00
|
|
|
current_total : battery.consumed_mah(battery_instance),
|
2017-12-05 05:54:11 -04:00
|
|
|
consumed_wh : battery.consumed_wh(battery_instance),
|
2017-09-29 00:09:59 -03:00
|
|
|
temperature : (int16_t)(has_temp ? (temp * 100) : 0),
|
|
|
|
resistance : battery.get_resistance(battery_instance)
|
|
|
|
};
|
|
|
|
WriteBlock(&pkt, sizeof(pkt));
|
|
|
|
|
|
|
|
// individual cell voltages
|
|
|
|
if (battery.has_cell_voltages(battery_instance)) {
|
|
|
|
const AP_BattMonitor::cells &cells = battery.get_cell_voltages(battery_instance);
|
|
|
|
struct log_Current_Cells cell_pkt = {
|
2017-09-29 00:15:20 -03:00
|
|
|
LOG_PACKET_HEADER_INIT(celltype),
|
2017-09-29 00:09:59 -03:00
|
|
|
time_us : time_us,
|
|
|
|
voltage : battery.voltage(battery_instance)
|
2016-06-02 18:00:57 -03:00
|
|
|
};
|
2017-09-29 00:09:59 -03:00
|
|
|
for (uint8_t i = 0; i < ARRAY_SIZE(cells.cells); i++) {
|
|
|
|
cell_pkt.cell_voltages[i] = cells.cells[i] + 1;
|
|
|
|
}
|
|
|
|
WriteBlock(&cell_pkt, sizeof(cell_pkt));
|
2017-04-08 16:04:12 -03:00
|
|
|
|
2017-09-29 00:09:59 -03:00
|
|
|
// check battery structure can hold all cells
|
|
|
|
static_assert(ARRAY_SIZE(cells.cells) == (sizeof(cell_pkt.cell_voltages) / sizeof(cell_pkt.cell_voltages[0])),
|
|
|
|
"Battery cell number doesn't match in library and log structure");
|
|
|
|
}
|
|
|
|
}
|
2017-04-08 16:04:12 -03:00
|
|
|
|
2017-09-29 00:09:59 -03:00
|
|
|
// Write an Current data packet
|
2019-01-18 00:24:08 -04:00
|
|
|
void AP_Logger::Write_Current()
|
2017-09-29 00:09:59 -03:00
|
|
|
{
|
2018-09-12 20:39:30 -03:00
|
|
|
// Big painful assert to ensure that logging won't produce suprising results when the
|
|
|
|
// number of battery monitors changes, does have the built in expectation that
|
|
|
|
// LOG_COMPASS_MSG follows the last LOG_CURRENT_CELLSx_MSG
|
|
|
|
static_assert(((LOG_CURRENT_MSG + AP_BATT_MONITOR_MAX_INSTANCES) == LOG_CURRENT_CELLS_MSG) &&
|
|
|
|
((LOG_CURRENT_CELLS_MSG + AP_BATT_MONITOR_MAX_INSTANCES) == LOG_COMPASS_MSG),
|
|
|
|
"The number of batt monitors has changed without updating the log "
|
|
|
|
"table entries. Please add new enums for LOG_CURRENT_MSG, LOG_CURRENT_CELLS_MSG "
|
|
|
|
"directly following the highest indexed fields. Don't forget to update the log "
|
|
|
|
"description table as well.");
|
|
|
|
|
2017-09-29 00:09:59 -03:00
|
|
|
const uint64_t time_us = AP_HAL::micros64();
|
2018-01-16 15:09:47 -04:00
|
|
|
const uint8_t num_instances = AP::battery().num_instances();
|
2018-09-12 20:39:30 -03:00
|
|
|
for (uint8_t i = 0; i < num_instances; i++) {
|
2019-01-18 00:24:08 -04:00
|
|
|
Write_Current_instance(time_us,
|
2018-09-12 20:39:30 -03:00
|
|
|
i,
|
|
|
|
(LogMessages)((uint8_t)LOG_CURRENT_MSG + i),
|
|
|
|
(LogMessages)((uint8_t)LOG_CURRENT_CELLS_MSG + i));
|
2016-06-02 18:00:57 -03:00
|
|
|
}
|
2014-12-18 17:34:29 -04:00
|
|
|
}
|
|
|
|
|
2019-01-18 00:24:08 -04:00
|
|
|
void AP_Logger::Write_Compass_instance(const uint64_t time_us, const uint8_t mag_instance, const enum LogMessages type)
|
2014-12-20 15:08:13 -04:00
|
|
|
{
|
2018-06-25 08:57:09 -03:00
|
|
|
const Compass &compass = AP::compass();
|
|
|
|
|
2017-09-28 23:53:31 -03:00
|
|
|
const Vector3f &mag_field = compass.get_field(mag_instance);
|
|
|
|
const Vector3f &mag_offsets = compass.get_offsets(mag_instance);
|
|
|
|
const Vector3f &mag_motor_offsets = compass.get_motor_offsets(mag_instance);
|
2014-12-20 15:08:13 -04:00
|
|
|
struct log_Compass pkt = {
|
2017-09-28 23:53:31 -03:00
|
|
|
LOG_PACKET_HEADER_INIT(type),
|
2016-05-05 00:37:16 -03:00
|
|
|
time_us : time_us,
|
2014-12-20 15:08:13 -04:00
|
|
|
mag_x : (int16_t)mag_field.x,
|
|
|
|
mag_y : (int16_t)mag_field.y,
|
|
|
|
mag_z : (int16_t)mag_field.z,
|
|
|
|
offset_x : (int16_t)mag_offsets.x,
|
|
|
|
offset_y : (int16_t)mag_offsets.y,
|
|
|
|
offset_z : (int16_t)mag_offsets.z,
|
|
|
|
motor_offset_x : (int16_t)mag_motor_offsets.x,
|
|
|
|
motor_offset_y : (int16_t)mag_motor_offsets.y,
|
2015-04-18 09:31:48 -03:00
|
|
|
motor_offset_z : (int16_t)mag_motor_offsets.z,
|
2017-09-28 23:53:31 -03:00
|
|
|
health : (uint8_t)compass.healthy(mag_instance),
|
|
|
|
SUS : compass.last_update_usec(mag_instance)
|
2014-12-20 15:08:13 -04:00
|
|
|
};
|
|
|
|
WriteBlock(&pkt, sizeof(pkt));
|
2017-09-28 23:53:31 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Write a Compass packet
|
2019-01-18 00:24:08 -04:00
|
|
|
void AP_Logger::Write_Compass(uint64_t time_us)
|
2017-09-28 23:53:31 -03:00
|
|
|
{
|
|
|
|
if (time_us == 0) {
|
|
|
|
time_us = AP_HAL::micros64();
|
|
|
|
}
|
2018-06-25 08:57:09 -03:00
|
|
|
const Compass &compass = AP::compass();
|
|
|
|
if (compass.get_count() > 0) {
|
2019-01-18 00:24:08 -04:00
|
|
|
Write_Compass_instance(time_us, 0, LOG_COMPASS_MSG);
|
2018-06-25 08:57:09 -03:00
|
|
|
}
|
2015-10-14 13:05:36 -03:00
|
|
|
|
2015-01-16 18:27:01 -04:00
|
|
|
if (compass.get_count() > 1) {
|
2019-01-18 00:24:08 -04:00
|
|
|
Write_Compass_instance(time_us, 1, LOG_COMPASS2_MSG);
|
2015-01-16 18:27:01 -04:00
|
|
|
}
|
2015-10-14 13:05:36 -03:00
|
|
|
|
2015-01-16 18:27:01 -04:00
|
|
|
if (compass.get_count() > 2) {
|
2019-01-18 00:24:08 -04:00
|
|
|
Write_Compass_instance(time_us, 2, LOG_COMPASS3_MSG);
|
2015-01-16 18:27:01 -04:00
|
|
|
}
|
2014-12-20 15:08:13 -04:00
|
|
|
}
|
|
|
|
|
2014-12-22 16:10:02 -04:00
|
|
|
// Write a mode packet.
|
2019-01-18 00:24:08 -04:00
|
|
|
bool AP_Logger_Backend::Write_Mode(uint8_t mode, uint8_t reason)
|
2014-12-22 16:10:02 -04:00
|
|
|
{
|
|
|
|
struct log_Mode pkt = {
|
|
|
|
LOG_PACKET_HEADER_INIT(LOG_MODE_MSG),
|
2015-11-19 23:15:08 -04:00
|
|
|
time_us : AP_HAL::micros64(),
|
2014-12-22 16:10:02 -04:00
|
|
|
mode : mode,
|
2016-01-25 19:45:27 -04:00
|
|
|
mode_num : mode,
|
|
|
|
mode_reason : reason
|
2014-12-22 16:10:02 -04:00
|
|
|
};
|
2015-08-27 23:47:16 -03:00
|
|
|
return WriteCriticalBlock(&pkt, sizeof(pkt));
|
2014-12-22 16:10:02 -04:00
|
|
|
}
|
|
|
|
|
2014-11-20 06:53:12 -04:00
|
|
|
// Write ESC status messages
|
2019-02-11 01:48:39 -04:00
|
|
|
// id starts from 0
|
|
|
|
// rpm is eRPM (rpm * 100)
|
|
|
|
// voltage is in centi-volts
|
|
|
|
// current is in centi-amps
|
|
|
|
// temperature is in centi-degrees Celsius
|
|
|
|
// current_tot is in centi-amp hours
|
|
|
|
void AP_Logger::Write_ESC(uint8_t id, uint64_t time_us, int32_t rpm, uint16_t voltage, uint16_t current, int16_t temperature, uint16_t current_tot)
|
2014-11-13 06:38:33 -04:00
|
|
|
{
|
2019-02-11 01:48:39 -04:00
|
|
|
// sanity check id
|
|
|
|
if (id >= 8) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
struct log_Esc pkt = {
|
|
|
|
LOG_PACKET_HEADER_INIT(uint8_t(LOG_ESC1_MSG+id)),
|
|
|
|
time_us : time_us,
|
|
|
|
rpm : rpm,
|
|
|
|
voltage : voltage,
|
|
|
|
current : current,
|
|
|
|
temperature : temperature,
|
|
|
|
current_tot : current_tot
|
|
|
|
};
|
|
|
|
WriteBlock(&pkt, sizeof(pkt));
|
2014-11-13 06:38:33 -04:00
|
|
|
}
|
2015-01-19 18:10:33 -04:00
|
|
|
|
2015-05-21 22:42:08 -03:00
|
|
|
// Write a Yaw PID packet
|
2019-01-18 00:24:08 -04:00
|
|
|
void AP_Logger::Write_PID(uint8_t msg_type, const PID_Info &info)
|
2015-05-21 22:42:08 -03:00
|
|
|
{
|
|
|
|
struct log_PID pkt = {
|
|
|
|
LOG_PACKET_HEADER_INIT(msg_type),
|
2015-11-19 23:15:08 -04:00
|
|
|
time_us : AP_HAL::micros64(),
|
2015-05-23 18:50:32 -03:00
|
|
|
desired : info.desired,
|
2018-08-08 01:11:43 -03:00
|
|
|
actual : info.actual,
|
2015-05-21 22:42:08 -03:00
|
|
|
P : info.P,
|
|
|
|
I : info.I,
|
|
|
|
D : info.D,
|
2018-08-08 01:11:43 -03:00
|
|
|
FF : info.FF
|
2015-05-21 22:42:08 -03:00
|
|
|
};
|
|
|
|
WriteBlock(&pkt, sizeof(pkt));
|
|
|
|
}
|
2015-07-03 08:49:45 -03:00
|
|
|
|
2019-01-18 00:24:08 -04:00
|
|
|
void AP_Logger::Write_Origin(uint8_t origin_type, const Location &loc)
|
2015-07-03 08:49:45 -03:00
|
|
|
{
|
2015-11-19 23:15:08 -04:00
|
|
|
uint64_t time_us = AP_HAL::micros64();
|
2015-07-03 08:49:45 -03:00
|
|
|
struct log_ORGN pkt = {
|
|
|
|
LOG_PACKET_HEADER_INIT(LOG_ORGN_MSG),
|
|
|
|
time_us : time_us,
|
|
|
|
origin_type : origin_type,
|
|
|
|
latitude : loc.lat,
|
|
|
|
longitude : loc.lng,
|
|
|
|
altitude : loc.alt
|
|
|
|
};
|
|
|
|
WriteBlock(&pkt, sizeof(pkt));
|
|
|
|
}
|
2015-08-07 07:34:14 -03:00
|
|
|
|
2019-01-18 00:24:08 -04:00
|
|
|
void AP_Logger::Write_RPM(const AP_RPM &rpm_sensor)
|
2015-08-07 07:34:14 -03:00
|
|
|
{
|
|
|
|
struct log_RPM pkt = {
|
|
|
|
LOG_PACKET_HEADER_INIT(LOG_RPM_MSG),
|
2015-11-19 23:15:08 -04:00
|
|
|
time_us : AP_HAL::micros64(),
|
2015-08-07 07:34:14 -03:00
|
|
|
rpm1 : rpm_sensor.get_rpm(0),
|
|
|
|
rpm2 : rpm_sensor.get_rpm(1)
|
|
|
|
};
|
|
|
|
WriteBlock(&pkt, sizeof(pkt));
|
2016-01-28 18:51:52 -04:00
|
|
|
}
|
2016-03-24 22:11:11 -03:00
|
|
|
|
2017-04-18 11:43:03 -03:00
|
|
|
// Write a rate packet
|
2019-01-18 00:24:08 -04:00
|
|
|
void AP_Logger::Write_Rate(const AP_AHRS_View *ahrs,
|
2016-03-24 22:11:11 -03:00
|
|
|
const AP_Motors &motors,
|
|
|
|
const AC_AttitudeControl &attitude_control,
|
|
|
|
const AC_PosControl &pos_control)
|
|
|
|
{
|
|
|
|
const Vector3f &rate_targets = attitude_control.rate_bf_targets();
|
|
|
|
const Vector3f &accel_target = pos_control.get_accel_target();
|
|
|
|
struct log_Rate pkt_rate = {
|
|
|
|
LOG_PACKET_HEADER_INIT(LOG_RATE_MSG),
|
|
|
|
time_us : AP_HAL::micros64(),
|
2016-06-18 07:47:45 -03:00
|
|
|
control_roll : degrees(rate_targets.x),
|
2018-12-21 02:15:40 -04:00
|
|
|
roll : degrees(ahrs->get_gyro().x),
|
2016-03-24 22:11:11 -03:00
|
|
|
roll_out : motors.get_roll(),
|
2016-06-18 07:47:45 -03:00
|
|
|
control_pitch : degrees(rate_targets.y),
|
2018-12-21 02:15:40 -04:00
|
|
|
pitch : degrees(ahrs->get_gyro().y),
|
2016-03-24 22:11:11 -03:00
|
|
|
pitch_out : motors.get_pitch(),
|
2016-06-18 07:47:45 -03:00
|
|
|
control_yaw : degrees(rate_targets.z),
|
2018-12-21 02:15:40 -04:00
|
|
|
yaw : degrees(ahrs->get_gyro().z),
|
2016-03-24 22:11:11 -03:00
|
|
|
yaw_out : motors.get_yaw(),
|
|
|
|
control_accel : (float)accel_target.z,
|
2018-12-21 02:15:40 -04:00
|
|
|
accel : (float)(-(ahrs->get_accel_ef_blended().z + GRAVITY_MSS) * 100.0f),
|
2016-03-24 22:11:11 -03:00
|
|
|
accel_out : motors.get_throttle()
|
|
|
|
};
|
|
|
|
WriteBlock(&pkt_rate, sizeof(pkt_rate));
|
|
|
|
}
|
2016-07-03 23:14:26 -03:00
|
|
|
|
2017-04-03 23:47:04 -03:00
|
|
|
// Write visual odometry sensor data
|
2019-01-18 00:24:08 -04:00
|
|
|
void AP_Logger::Write_VisualOdom(float time_delta, const Vector3f &angle_delta, const Vector3f &position_delta, float confidence)
|
2017-04-03 23:47:04 -03:00
|
|
|
{
|
|
|
|
struct log_VisualOdom pkt_visualodom = {
|
|
|
|
LOG_PACKET_HEADER_INIT(LOG_VISUALODOM_MSG),
|
|
|
|
time_us : AP_HAL::micros64(),
|
|
|
|
time_delta : time_delta,
|
|
|
|
angle_delta_x : angle_delta.x,
|
|
|
|
angle_delta_y : angle_delta.y,
|
|
|
|
angle_delta_z : angle_delta.z,
|
|
|
|
position_delta_x : position_delta.x,
|
|
|
|
position_delta_y : position_delta.y,
|
|
|
|
position_delta_z : position_delta.z,
|
|
|
|
confidence : confidence
|
|
|
|
};
|
|
|
|
WriteBlock(&pkt_visualodom, sizeof(log_VisualOdom));
|
|
|
|
}
|
2017-04-09 08:17:17 -03:00
|
|
|
|
|
|
|
// Write AOA and SSA
|
2019-01-18 00:24:08 -04:00
|
|
|
void AP_Logger::Write_AOA_SSA(AP_AHRS &ahrs)
|
2017-04-09 08:17:17 -03:00
|
|
|
{
|
|
|
|
struct log_AOA_SSA aoa_ssa = {
|
|
|
|
LOG_PACKET_HEADER_INIT(LOG_AOA_SSA_MSG),
|
|
|
|
time_us : AP_HAL::micros64(),
|
|
|
|
AOA : ahrs.getAOA(),
|
|
|
|
SSA : ahrs.getSSA()
|
|
|
|
};
|
|
|
|
|
|
|
|
WriteBlock(&aoa_ssa, sizeof(aoa_ssa));
|
|
|
|
}
|
2017-04-18 11:43:03 -03:00
|
|
|
|
|
|
|
// Write beacon sensor (position) data
|
2019-01-18 00:24:08 -04:00
|
|
|
void AP_Logger::Write_Beacon(AP_Beacon &beacon)
|
2017-04-18 11:43:03 -03:00
|
|
|
{
|
2018-02-22 00:53:55 -04:00
|
|
|
if (!beacon.enabled()) {
|
|
|
|
return;
|
|
|
|
}
|
2017-04-18 11:43:03 -03:00
|
|
|
// position
|
|
|
|
Vector3f pos;
|
|
|
|
float accuracy = 0.0f;
|
|
|
|
beacon.get_vehicle_position_ned(pos, accuracy);
|
|
|
|
|
|
|
|
struct log_Beacon pkt_beacon = {
|
|
|
|
LOG_PACKET_HEADER_INIT(LOG_BEACON_MSG),
|
|
|
|
time_us : AP_HAL::micros64(),
|
|
|
|
health : (uint8_t)beacon.healthy(),
|
|
|
|
count : (uint8_t)beacon.count(),
|
|
|
|
dist0 : beacon.beacon_distance(0),
|
|
|
|
dist1 : beacon.beacon_distance(1),
|
|
|
|
dist2 : beacon.beacon_distance(2),
|
|
|
|
dist3 : beacon.beacon_distance(3),
|
|
|
|
posx : pos.x,
|
|
|
|
posy : pos.y,
|
|
|
|
posz : pos.z
|
|
|
|
};
|
|
|
|
WriteBlock(&pkt_beacon, sizeof(pkt_beacon));
|
|
|
|
}
|
2017-07-14 14:00:13 -03:00
|
|
|
|
|
|
|
// Write proximity sensor distances
|
2019-01-18 00:24:08 -04:00
|
|
|
void AP_Logger::Write_Proximity(AP_Proximity &proximity)
|
2017-07-14 14:00:13 -03:00
|
|
|
{
|
|
|
|
// exit immediately if not enabled
|
|
|
|
if (proximity.get_status() == AP_Proximity::Proximity_NotConnected) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-07-14 14:00:21 -03:00
|
|
|
AP_Proximity::Proximity_Distance_Array dist_array {};
|
|
|
|
proximity.get_horizontal_distances(dist_array);
|
2017-07-14 14:00:13 -03:00
|
|
|
|
|
|
|
float dist_up;
|
|
|
|
if (!proximity.get_upward_distance(dist_up)) {
|
|
|
|
dist_up = 0.0f;
|
|
|
|
}
|
|
|
|
|
|
|
|
float close_ang = 0.0f, close_dist = 0.0f;
|
|
|
|
proximity.get_closest_object(close_ang, close_dist);
|
|
|
|
|
|
|
|
struct log_Proximity pkt_proximity = {
|
|
|
|
LOG_PACKET_HEADER_INIT(LOG_PROXIMITY_MSG),
|
|
|
|
time_us : AP_HAL::micros64(),
|
|
|
|
health : (uint8_t)proximity.get_status(),
|
2017-07-14 14:00:21 -03:00
|
|
|
dist0 : dist_array.distance[0],
|
|
|
|
dist45 : dist_array.distance[1],
|
|
|
|
dist90 : dist_array.distance[2],
|
|
|
|
dist135 : dist_array.distance[3],
|
|
|
|
dist180 : dist_array.distance[4],
|
|
|
|
dist225 : dist_array.distance[5],
|
|
|
|
dist270 : dist_array.distance[6],
|
|
|
|
dist315 : dist_array.distance[7],
|
2017-07-14 14:00:13 -03:00
|
|
|
distup : dist_up,
|
|
|
|
closest_angle : close_ang,
|
|
|
|
closest_dist : close_dist
|
|
|
|
};
|
|
|
|
WriteBlock(&pkt_proximity, sizeof(pkt_proximity));
|
|
|
|
}
|
2017-07-26 14:09:33 -03:00
|
|
|
|
2019-01-18 00:24:08 -04:00
|
|
|
void AP_Logger::Write_SRTL(bool active, uint16_t num_points, uint16_t max_points, uint8_t action, const Vector3f& breadcrumb)
|
2017-07-26 14:09:33 -03:00
|
|
|
{
|
|
|
|
struct log_SRTL pkt_srtl = {
|
|
|
|
LOG_PACKET_HEADER_INIT(LOG_SRTL_MSG),
|
|
|
|
time_us : AP_HAL::micros64(),
|
|
|
|
active : active,
|
|
|
|
num_points : num_points,
|
|
|
|
max_points : max_points,
|
|
|
|
action : action,
|
|
|
|
N : breadcrumb.x,
|
|
|
|
E : breadcrumb.y,
|
|
|
|
D : breadcrumb.z
|
|
|
|
};
|
|
|
|
WriteBlock(&pkt_srtl, sizeof(pkt_srtl));
|
|
|
|
}
|
2019-06-08 01:59:04 -03:00
|
|
|
|
|
|
|
void AP_Logger::Write_OA(uint8_t algorithm, const Location& final_dest, const Location& oa_dest)
|
|
|
|
{
|
|
|
|
struct log_OA pkt = {
|
|
|
|
LOG_PACKET_HEADER_INIT(LOG_OA_MSG),
|
|
|
|
time_us : AP_HAL::micros64(),
|
|
|
|
algorithm : algorithm,
|
|
|
|
final_lat : final_dest.lat,
|
|
|
|
final_lng : final_dest.lng,
|
|
|
|
oa_lat : oa_dest.lat,
|
|
|
|
oa_lng : oa_dest.lng,
|
|
|
|
};
|
|
|
|
WriteBlock(&pkt, sizeof(pkt));
|
|
|
|
}
|