ardupilot/ArduPlane/Log.cpp

610 lines
18 KiB
C++
Raw Normal View History

// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*-
2015-05-13 03:09:36 -03:00
#include "Plane.h"
#include "version.h"
2015-05-13 03:09:36 -03:00
#if LOGGING_ENABLED == ENABLED
#if CLI_ENABLED == ENABLED
// Code to Write and Read packets from DataFlash.log memory
// Code to interact with the user to dump or erase logs
// Creates a constant array of structs representing menu options
// and stores them in Flash memory, not RAM.
// User enters the string in the console to call the functions on the right.
// See class Menu in AP_Coommon for implementation details
static const struct Menu::command log_menu_commands[] = {
2015-05-13 03:09:36 -03:00
{"dump", MENU_FUNC(dump_log)},
{"erase", MENU_FUNC(erase_logs)},
{"enable", MENU_FUNC(select_logs)},
{"disable", MENU_FUNC(select_logs)}
};
// A Macro to create the Menu
MENU2(log_menu, "Log", log_menu_commands, FUNCTOR_BIND(&plane, &Plane::print_log_menu, bool));
2015-05-13 03:09:36 -03:00
bool Plane::print_log_menu(void)
{
cliSerial->println("logs enabled: ");
2012-08-21 23:19:50 -03:00
if (0 == g.log_bitmask) {
cliSerial->println("none");
2012-08-21 23:19:50 -03:00
}else{
// Macro to make the following code a bit easier on the eye.
// Pass it the capitalised name of the log option, as defined
// in defines.h but without the LOG_ prefix. It will check for
// the bit being set and print the name of the log option to suit.
#define PLOG(_s) if (g.log_bitmask & MASK_LOG_ ## _s) cliSerial->printf(" %s", # _s)
2012-08-21 23:19:50 -03:00
PLOG(ATTITUDE_FAST);
PLOG(ATTITUDE_MED);
PLOG(GPS);
PLOG(PM);
PLOG(CTUN);
PLOG(NTUN);
PLOG(MODE);
PLOG(IMU);
2012-08-21 23:19:50 -03:00
PLOG(CMD);
PLOG(CURRENT);
2013-04-19 10:54:40 -03:00
PLOG(COMPASS);
PLOG(TECS);
PLOG(CAMERA);
2013-11-25 17:44:17 -04:00
PLOG(RC);
PLOG(SONAR);
2012-08-21 23:19:50 -03:00
#undef PLOG
}
cliSerial->println();
2012-08-21 23:19:50 -03:00
2013-04-17 08:34:56 -03:00
DataFlash.ListAvailableLogs(cliSerial);
2012-08-21 23:19:50 -03:00
return(true);
}
2015-05-13 03:09:36 -03:00
int8_t Plane::dump_log(uint8_t argc, const Menu::arg *argv)
{
2015-05-13 03:09:36 -03:00
int16_t dump_log_num;
2013-02-22 19:15:15 -04:00
uint16_t dump_log_start;
uint16_t dump_log_end;
2012-08-21 23:19:50 -03:00
// check that the requested log number can be read
2015-05-13 03:09:36 -03:00
dump_log_num = argv[1].i;
2012-08-21 23:19:50 -03:00
2015-05-13 03:09:36 -03:00
if (dump_log_num == -2) {
2013-02-22 19:15:15 -04:00
DataFlash.DumpPageInfo(cliSerial);
2012-08-21 23:19:50 -03:00
return(-1);
2015-05-13 03:09:36 -03:00
} else if (dump_log_num <= 0) {
cliSerial->printf("dumping all\n");
2013-02-28 16:16:32 -04:00
Log_Read(0, 1, 0);
2012-08-21 23:19:50 -03:00
return(-1);
} else if ((argc != 2) || ((uint16_t)dump_log_num > DataFlash.get_num_logs())) {
cliSerial->printf("bad log number\n");
2012-08-21 23:19:50 -03:00
return(-1);
}
2015-05-13 03:09:36 -03:00
DataFlash.get_log_boundaries(dump_log_num, dump_log_start, dump_log_end);
Log_Read((uint16_t)dump_log_num, dump_log_start, dump_log_end);
return 0;
}
2015-05-13 03:09:36 -03:00
int8_t Plane::erase_logs(uint8_t argc, const Menu::arg *argv)
{
2011-12-28 00:53:14 -04:00
in_mavlink_delay = true;
do_erase_logs();
in_mavlink_delay = false;
return 0;
}
2015-05-13 03:09:36 -03:00
int8_t Plane::select_logs(uint8_t argc, const Menu::arg *argv)
{
2014-01-13 21:52:22 -04:00
uint32_t bits;
2012-08-21 23:19:50 -03:00
if (argc != 2) {
cliSerial->printf("missing log type\n");
2012-08-21 23:19:50 -03:00
return(-1);
}
bits = 0;
// Macro to make the following code a bit easier on the eye.
// Pass it the capitalised name of the log option, as defined
// in defines.h but without the LOG_ prefix. It will check for
// that name as the argument to the command, and set the bit in
// bits accordingly.
//
if (!strcasecmp(argv[1].str, "all")) {
2014-01-13 21:52:22 -04:00
bits = 0xFFFFFFFFUL;
2012-08-21 23:19:50 -03:00
} else {
#define TARG(_s) if (!strcasecmp(argv[1].str, # _s)) bits |= MASK_LOG_ ## _s
2012-08-21 23:19:50 -03:00
TARG(ATTITUDE_FAST);
TARG(ATTITUDE_MED);
TARG(GPS);
TARG(PM);
TARG(CTUN);
TARG(NTUN);
TARG(MODE);
TARG(IMU);
2012-08-21 23:19:50 -03:00
TARG(CMD);
TARG(CURRENT);
2013-04-19 10:54:40 -03:00
TARG(COMPASS);
TARG(TECS);
TARG(CAMERA);
2013-11-25 17:44:17 -04:00
TARG(RC);
TARG(SONAR);
2012-08-21 23:19:50 -03:00
#undef TARG
}
if (!strcasecmp(argv[0].str, "enable")) {
2012-08-21 23:19:50 -03:00
g.log_bitmask.set_and_save(g.log_bitmask | bits);
}else{
g.log_bitmask.set_and_save(g.log_bitmask & ~bits);
}
return(0);
}
2015-05-13 03:09:36 -03:00
int8_t Plane::process_logs(uint8_t argc, const Menu::arg *argv)
{
2012-08-21 23:19:50 -03:00
log_menu.run();
return 0;
}
#endif // CLI_ENABLED == ENABLED
2015-05-13 03:09:36 -03:00
void Plane::do_erase_logs(void)
{
2015-11-18 15:17:50 -04:00
gcs_send_text(MAV_SEVERITY_INFO, "Erasing logs");
DataFlash.EraseAll();
2015-11-18 15:17:50 -04:00
gcs_send_text(MAV_SEVERITY_INFO, "Log erase complete");
}
// Write an attitude packet
2015-05-13 03:09:36 -03:00
void Plane::Log_Write_Attitude(void)
{
if (!should_log(MASK_LOG_ATTITUDE_FAST)) {
return;
}
Vector3f targets; // Package up the targets into a vector for commonality with Copter usage of Log_Wrote_Attitude
targets.x = nav_roll_cd;
targets.y = nav_pitch_cd;
targets.z = 0; //Plane does not have the concept of navyaw. This is a placeholder.
DataFlash.Log_Write_Attitude(ahrs, targets);
if (quadplane.in_vtol_mode()) {
DataFlash.Log_Write_PID(LOG_PIDR_MSG, quadplane.attitude_control->get_rate_roll_pid().get_pid_info());
DataFlash.Log_Write_PID(LOG_PIDP_MSG, quadplane.attitude_control->get_rate_pitch_pid().get_pid_info());
DataFlash.Log_Write_PID(LOG_PIDY_MSG, quadplane.attitude_control->get_rate_yaw_pid().get_pid_info());
DataFlash.Log_Write_PID(LOG_PIDA_MSG, quadplane.pid_accel_z.get_pid_info() );
} else {
DataFlash.Log_Write_PID(LOG_PIDR_MSG, rollController.get_pid_info());
DataFlash.Log_Write_PID(LOG_PIDP_MSG, pitchController.get_pid_info());
DataFlash.Log_Write_PID(LOG_PIDY_MSG, yawController.get_pid_info());
DataFlash.Log_Write_PID(LOG_PIDS_MSG, steerController.get_pid_info());
}
#if AP_AHRS_NAVEKF_AVAILABLE
#if OPTFLOW == ENABLED
DataFlash.Log_Write_EKF(ahrs,optflow.enabled());
#else
DataFlash.Log_Write_EKF(ahrs,false);
#endif
2014-01-03 20:51:36 -04:00
DataFlash.Log_Write_AHRS2(ahrs);
#endif
2015-05-04 03:18:55 -03:00
#if CONFIG_HAL_BOARD == HAL_BOARD_SITL
2015-11-16 00:10:42 -04:00
sitl.Log_Write_SIMSTATE(&DataFlash);
#endif
2015-05-15 01:24:46 -03:00
DataFlash.Log_Write_POS(ahrs);
}
2013-04-19 10:37:21 -03:00
struct PACKED log_Performance {
LOG_PACKET_HEADER;
uint64_t time_us;
uint16_t num_long;
uint16_t main_loop_count;
uint32_t g_dt_max;
uint32_t g_dt_min;
uint32_t log_dropped;
};
// Write a performance monitoring packet. Total length : 19 bytes
2015-05-13 03:09:36 -03:00
void Plane::Log_Write_Performance()
{
struct log_Performance pkt = {
LOG_PACKET_HEADER_INIT(LOG_PERFORMANCE_MSG),
time_us : AP_HAL::micros64(),
num_long : perf.num_long,
main_loop_count : perf.mainLoop_count,
g_dt_max : perf.G_Dt_max,
g_dt_min : perf.G_Dt_min,
log_dropped : DataFlash.num_dropped() - perf.last_log_dropped
};
DataFlash.WriteCriticalBlock(&pkt, sizeof(pkt));
2012-08-21 23:19:50 -03:00
}
2013-04-19 10:37:21 -03:00
struct PACKED log_Startup {
LOG_PACKET_HEADER;
uint64_t time_us;
uint8_t startup_type;
uint16_t command_total;
};
void Plane::Log_Write_Startup(uint8_t type)
{
struct log_Startup pkt = {
LOG_PACKET_HEADER_INIT(LOG_STARTUP_MSG),
time_us : AP_HAL::micros64(),
startup_type : type,
command_total : mission.num_commands()
};
DataFlash.WriteCriticalBlock(&pkt, sizeof(pkt));
}
2013-04-19 10:37:21 -03:00
struct PACKED log_Control_Tuning {
LOG_PACKET_HEADER;
uint64_t time_us;
int16_t nav_roll_cd;
int16_t roll;
int16_t nav_pitch_cd;
int16_t pitch;
int16_t throttle_out;
int16_t rudder_out;
int16_t throttle_dem;
};
// Write a control tuning packet. Total length : 22 bytes
2015-05-13 03:09:36 -03:00
void Plane::Log_Write_Control_Tuning()
{
struct log_Control_Tuning pkt = {
2013-04-19 10:37:21 -03:00
LOG_PACKET_HEADER_INIT(LOG_CTUN_MSG),
time_us : AP_HAL::micros64(),
nav_roll_cd : (int16_t)nav_roll_cd,
roll : (int16_t)ahrs.roll_sensor,
nav_pitch_cd : (int16_t)nav_pitch_cd,
pitch : (int16_t)ahrs.pitch_sensor,
ArduPlane: Fix up after refactoring RC_Channel class Further to refactor of RC_Channel class which included adding get_xx set_xx methods, change reads and writes to the public members to calls to get and set functionsss old public member(int16_t) get function -> int16_t set function (int16_t) (expression where c is an object of type RC_Channel) c.radio_in c.get_radio_in() c.set_radio_in(v) c.control_in c.get_control_in() c.set_control_in(v) c.servo_out c.get_servo_out() c.set_servo_out(v) c.pwm_out c.get_pwm_out() // use existing c.radio_out c.get_radio_out() c.set_radio_out(v) c.radio_max c.get_radio_max() c.set_radio_max(v) c.radio_min c.get_radio_min() c.set_radio_min(v) c.radio_trim c.get_radio_trim() c.set_radio_trim(v); c.min_max_configured() // return true if min and max are configured Because data members of RC_Channels are now private and so cannot be written directly some overloads are provided in the Plane classes to provide the old functionality new overload Plane::stick_mix_channel(RC_Channel *channel) which forwards to the previously existing void stick_mix_channel(RC_Channel *channel, int16_t &servo_out); new overload Plane::channel_output_mixer(Rc_Channel* , RC_Channel*)const which forwards to (uint8_t mixing_type, int16_t & chan1, int16_t & chan2)const; Rename functions RC_Channel_aux::set_radio_trim(Aux_servo_function_t function) to RC_Channel_aux::set_trim_to_radio_in_for(Aux_servo_function_t function) RC_Channel_aux::set_servo_out(Aux_servo_function_t function, int16_t value) to RC_Channel_aux::set_servo_out_for(Aux_servo_function_t function, int16_t value) Rationale: RC_Channel is a complicated class, which combines several functionalities dealing with stick inputs in pwm and logical units, logical and actual actuator outputs, unit conversion etc, etc The intent of this PR is to clarify existing use of the class. At the basic level it should now be possible to grep all places where private variable is set by searching for the set_xx function. (The wider purpose is to provide a more generic and logically simpler method of output mixing. This is a small step)
2016-05-08 05:33:02 -03:00
throttle_out : (int16_t)channel_throttle->get_servo_out(),
rudder_out : (int16_t)channel_rudder->get_servo_out(),
throttle_dem : (int16_t)SpdHgt_Controller->get_throttle_demand()
};
DataFlash.WriteBlock(&pkt, sizeof(pkt));
}
// Write a TECS tuning packet
2015-05-13 03:09:36 -03:00
void Plane::Log_Write_TECS_Tuning(void)
{
SpdHgt_Controller->log_data(DataFlash, LOG_TECS_MSG);
}
2013-04-19 10:37:21 -03:00
struct PACKED log_Nav_Tuning {
LOG_PACKET_HEADER;
uint64_t time_us;
float wp_distance;
int16_t target_bearing_cd;
int16_t nav_bearing_cd;
int16_t altitude_error_cm;
int16_t airspeed_cm;
float altitude;
uint32_t groundspeed_cm;
2016-01-05 19:33:49 -04:00
float xtrack_error;
float xtrack_error_i;
};
2016-01-05 19:33:49 -04:00
// Write a navigation tuning packet
2015-05-13 03:09:36 -03:00
void Plane::Log_Write_Nav_Tuning()
{
struct log_Nav_Tuning pkt = {
2013-04-19 10:37:21 -03:00
LOG_PACKET_HEADER_INIT(LOG_NTUN_MSG),
time_us : AP_HAL::micros64(),
wp_distance : auto_state.wp_distance,
target_bearing_cd : (int16_t)nav_controller->target_bearing_cd(),
nav_bearing_cd : (int16_t)nav_controller->nav_bearing_cd(),
altitude_error_cm : (int16_t)altitude_error_cm,
airspeed_cm : (int16_t)airspeed.get_airspeed_cm(),
altitude : barometer.get_altitude(),
2016-01-05 19:33:49 -04:00
groundspeed_cm : (uint32_t)(gps.ground_speed()*100),
xtrack_error : nav_controller->crosstrack_error(),
xtrack_error_i : nav_controller->crosstrack_error_integrator()
};
DataFlash.WriteBlock(&pkt, sizeof(pkt));
}
struct PACKED log_Status {
LOG_PACKET_HEADER;
uint64_t time_us;
uint8_t is_flying;
float is_flying_probability;
uint8_t armed;
uint8_t safety;
bool is_crashed;
bool is_still;
uint8_t stage;
bool impact;
};
2015-05-13 03:09:36 -03:00
void Plane::Log_Write_Status()
{
struct log_Status pkt = {
LOG_PACKET_HEADER_INIT(LOG_STATUS_MSG)
,time_us : AP_HAL::micros64()
,is_flying : is_flying()
,is_flying_probability : isFlyingProbability
,armed : hal.util->get_soft_armed()
,safety : static_cast<uint8_t>(hal.util->safety_switch_state())
,is_crashed : crash_state.is_crashed
,is_still : plane.ins.is_still()
,stage : static_cast<uint8_t>(flight_stage)
,impact : crash_state.impact_detected
};
DataFlash.WriteBlock(&pkt, sizeof(pkt));
}
struct PACKED log_Sonar {
LOG_PACKET_HEADER;
uint64_t time_us;
uint16_t distance;
float voltage;
float baro_alt;
float groundspeed;
int8_t throttle;
uint8_t count;
float correction;
};
// Write a sonar packet
2015-05-13 03:09:36 -03:00
void Plane::Log_Write_Sonar()
{
#if RANGEFINDER_ENABLED == ENABLED
uint16_t distance = 0;
if (rangefinder.status() == RangeFinder::RangeFinder_Good) {
distance = rangefinder.distance_cm();
}
struct log_Sonar pkt = {
LOG_PACKET_HEADER_INIT(LOG_SONAR_MSG),
time_us : AP_HAL::micros64(),
distance : distance,
voltage : rangefinder.voltage_mv()*0.001f,
baro_alt : barometer.get_altitude(),
2014-03-28 16:52:48 -03:00
groundspeed : gps.ground_speed(),
throttle : (int8_t)(100 * channel_throttle->norm_output()),
count : rangefinder_state.in_range_count,
correction : rangefinder_state.correction
};
DataFlash.WriteBlock(&pkt, sizeof(pkt));
DataFlash.Log_Write_RFND(rangefinder);
#endif
}
struct PACKED log_Optflow {
LOG_PACKET_HEADER;
uint64_t time_us;
uint8_t surface_quality;
float flow_x;
float flow_y;
float body_x;
float body_y;
};
#if OPTFLOW == ENABLED
// Write an optical flow packet
2015-05-13 03:09:36 -03:00
void Plane::Log_Write_Optflow()
{
// exit immediately if not enabled
if (!optflow.enabled()) {
return;
}
const Vector2f &flowRate = optflow.flowRate();
const Vector2f &bodyRate = optflow.bodyRate();
struct log_Optflow pkt = {
LOG_PACKET_HEADER_INIT(LOG_OPTFLOW_MSG),
time_us : AP_HAL::micros64(),
surface_quality : optflow.quality(),
flow_x : flowRate.x,
flow_y : flowRate.y,
body_x : bodyRate.x,
body_y : bodyRate.y
};
DataFlash.WriteBlock(&pkt, sizeof(pkt));
}
#endif
2013-11-27 22:19:34 -04:00
struct PACKED log_Arm_Disarm {
LOG_PACKET_HEADER;
uint64_t time_us;
2013-11-27 22:19:34 -04:00
uint8_t arm_state;
uint16_t arm_checks;
};
2015-05-13 03:09:36 -03:00
void Plane::Log_Write_Current()
{
ArduPlane: Fix up after refactoring RC_Channel class Further to refactor of RC_Channel class which included adding get_xx set_xx methods, change reads and writes to the public members to calls to get and set functionsss old public member(int16_t) get function -> int16_t set function (int16_t) (expression where c is an object of type RC_Channel) c.radio_in c.get_radio_in() c.set_radio_in(v) c.control_in c.get_control_in() c.set_control_in(v) c.servo_out c.get_servo_out() c.set_servo_out(v) c.pwm_out c.get_pwm_out() // use existing c.radio_out c.get_radio_out() c.set_radio_out(v) c.radio_max c.get_radio_max() c.set_radio_max(v) c.radio_min c.get_radio_min() c.set_radio_min(v) c.radio_trim c.get_radio_trim() c.set_radio_trim(v); c.min_max_configured() // return true if min and max are configured Because data members of RC_Channels are now private and so cannot be written directly some overloads are provided in the Plane classes to provide the old functionality new overload Plane::stick_mix_channel(RC_Channel *channel) which forwards to the previously existing void stick_mix_channel(RC_Channel *channel, int16_t &servo_out); new overload Plane::channel_output_mixer(Rc_Channel* , RC_Channel*)const which forwards to (uint8_t mixing_type, int16_t & chan1, int16_t & chan2)const; Rename functions RC_Channel_aux::set_radio_trim(Aux_servo_function_t function) to RC_Channel_aux::set_trim_to_radio_in_for(Aux_servo_function_t function) RC_Channel_aux::set_servo_out(Aux_servo_function_t function, int16_t value) to RC_Channel_aux::set_servo_out_for(Aux_servo_function_t function, int16_t value) Rationale: RC_Channel is a complicated class, which combines several functionalities dealing with stick inputs in pwm and logical units, logical and actual actuator outputs, unit conversion etc, etc The intent of this PR is to clarify existing use of the class. At the basic level it should now be possible to grep all places where private variable is set by searching for the set_xx function. (The wider purpose is to provide a more generic and logically simpler method of output mixing. This is a small step)
2016-05-08 05:33:02 -03:00
DataFlash.Log_Write_Current(battery, channel_throttle->get_control_in());
2014-02-13 07:08:25 -04:00
// also write power status
DataFlash.Log_Write_Power();
}
2015-05-13 03:09:36 -03:00
void Plane::Log_Arm_Disarm() {
2013-11-27 22:19:34 -04:00
struct log_Arm_Disarm pkt = {
LOG_PACKET_HEADER_INIT(LOG_ARM_DISARM_MSG),
time_us : AP_HAL::micros64(),
2013-11-27 22:19:34 -04:00
arm_state : arming.is_armed(),
arm_checks : arming.get_enabled_checks()
};
DataFlash.WriteCriticalBlock(&pkt, sizeof(pkt));
2013-11-27 22:19:34 -04:00
}
2013-04-19 10:54:40 -03:00
2015-05-13 03:09:36 -03:00
void Plane::Log_Write_GPS(uint8_t instance)
{
2016-05-04 05:32:02 -03:00
if (!ahrs.have_ekf_logging()) {
DataFlash.Log_Write_GPS(gps, instance);
}
}
2015-05-13 03:09:36 -03:00
void Plane::Log_Write_IMU()
{
2013-11-03 23:37:32 -04:00
DataFlash.Log_Write_IMU(ins);
}
2015-05-13 03:09:36 -03:00
void Plane::Log_Write_RC(void)
2013-11-25 17:44:17 -04:00
{
DataFlash.Log_Write_RCIN();
DataFlash.Log_Write_RCOUT();
2015-09-04 12:51:03 -03:00
if (rssi.enabled()) {
DataFlash.Log_Write_RSSI(rssi);
}
2013-11-25 17:44:17 -04:00
}
2015-05-13 03:09:36 -03:00
void Plane::Log_Write_Baro(void)
{
2016-05-04 05:32:02 -03:00
if (!ahrs.have_ekf_logging()) {
DataFlash.Log_Write_Baro(barometer);
}
}
// Write a AIRSPEED packet
2015-05-13 03:09:36 -03:00
void Plane::Log_Write_Airspeed(void)
{
DataFlash.Log_Write_Airspeed(airspeed);
}
2015-07-05 23:01:17 -03:00
// log ahrs home and EKF origin to dataflash
void Plane::Log_Write_Home_And_Origin()
{
#if AP_AHRS_NAVEKF_AVAILABLE
// log ekf origin if set
Location ekf_orig;
if (ahrs.get_origin(ekf_orig)) {
2015-07-05 23:01:17 -03:00
DataFlash.Log_Write_Origin(LogOriginType::ekf_origin, ekf_orig);
}
#endif
// log ahrs home if set
if (home_is_set != HOME_UNSET) {
DataFlash.Log_Write_Origin(LogOriginType::ahrs_home, ahrs.get_home());
}
}
const struct LogStructure Plane::log_structure[] = {
2013-04-19 10:37:21 -03:00
LOG_COMMON_STRUCTURES,
{ LOG_PERFORMANCE_MSG, sizeof(log_Performance),
"PM", "QHHIII", "TimeUS,NLon,NLoop,MaxT,MinT,LogDrop" },
2013-04-19 10:37:21 -03:00
{ LOG_STARTUP_MSG, sizeof(log_Startup),
"STRT", "QBH", "TimeUS,SType,CTot" },
2013-04-19 10:37:21 -03:00
{ LOG_CTUN_MSG, sizeof(log_Control_Tuning),
"CTUN", "Qcccchhh", "TimeUS,NavRoll,Roll,NavPitch,Pitch,ThrOut,RdrOut,ThrDem" },
2013-04-19 10:37:21 -03:00
{ LOG_NTUN_MSG, sizeof(log_Nav_Tuning),
"NTUN", "QfccccfIff", "TimeUS,WpDist,TargBrg,NavBrg,AltErr,Arspd,Alt,GSpdCM,XT,XTi" },
{ LOG_SONAR_MSG, sizeof(log_Sonar),
"SONR", "QHfffbBf", "TimeUS,DistCM,Volt,BaroAlt,GSpd,Thr,Cnt,Corr" },
2013-11-27 22:19:34 -04:00
{ LOG_ARM_DISARM_MSG, sizeof(log_Arm_Disarm),
2015-07-19 20:46:19 -03:00
"ARM", "QBH", "TimeUS,ArmState,ArmChecks" },
{ LOG_ATRP_MSG, sizeof(AP_AutoTune::log_ATRP),
"ATRP", "QBBcfff", "TimeUS,Type,State,Servo,Demanded,Achieved,P" },
{ LOG_STATUS_MSG, sizeof(log_Status),
"STAT", "QBfBBBBBB", "TimeUS,isFlying,isFlyProb,Armed,Safety,Crash,Still,Stage,Hit" },
{ LOG_QTUN_MSG, sizeof(QuadPlane::log_QControl_Tuning),
"QTUN", "Qffffehhffff", "TimeUS,AngBst,ThrOut,DAlt,Alt,BarAlt,DCRt,CRt,DVx,DVy,DAx,DAy" },
#if OPTFLOW == ENABLED
{ LOG_OPTFLOW_MSG, sizeof(log_Optflow),
"OF", "QBffff", "TimeUS,Qual,flowX,flowY,bodyX,bodyY" },
#endif
2014-01-30 06:45:41 -04:00
TECS_LOG_FORMAT(LOG_TECS_MSG)
2013-04-19 10:37:21 -03:00
};
#if CLI_ENABLED == ENABLED
// Read the DataFlash.log memory : Packet Parser
void Plane::Log_Read(uint16_t list_entry, int16_t start_page, int16_t end_page)
{
cliSerial->printf("\n" FIRMWARE_STRING
"\nFree RAM: %u\n",
(unsigned)hal.util->available_memory());
2011-11-19 18:02:45 -04:00
cliSerial->println(HAL_BOARD_NAME);
2013-04-18 21:24:20 -03:00
DataFlash.LogReadProcess(list_entry, start_page, end_page,
FUNCTOR_BIND_MEMBER(&Plane::print_flight_mode, void, AP_HAL::BetterStream *, uint8_t),
cliSerial);
}
#endif // CLI_ENABLED
void Plane::Log_Write_Vehicle_Startup_Messages()
{
// only 200(?) bytes are guaranteed by DataFlash
Log_Write_Startup(TYPE_GROUNDSTART_MSG);
DataFlash.Log_Write_Mode(control_mode);
}
2013-04-19 10:37:21 -03:00
// start a new log
2015-05-13 03:09:36 -03:00
void Plane::start_logging()
{
DataFlash.set_mission(&mission);
DataFlash.setVehicle_Startup_Log_Writer(
FUNCTOR_BIND(&plane, &Plane::Log_Write_Vehicle_Startup_Messages, void)
);
DataFlash.StartNewLog();
}
2013-01-14 23:03:08 -04:00
2015-05-13 03:09:36 -03:00
/*
initialise logging subsystem
*/
void Plane::log_init(void)
{
2015-07-06 12:30:40 -03:00
DataFlash.Init(log_structure, ARRAY_SIZE(log_structure));
2015-05-13 03:09:36 -03:00
if (!DataFlash.CardInserted()) {
gcs_send_text(MAV_SEVERITY_WARNING, "No dataflash card inserted");
2015-05-13 03:09:36 -03:00
g.log_bitmask.set(0);
} else if (DataFlash.NeedPrep()) {
2015-11-18 15:17:50 -04:00
gcs_send_text(MAV_SEVERITY_INFO, "Preparing log system");
DataFlash.Prep();
2015-11-18 15:17:50 -04:00
gcs_send_text(MAV_SEVERITY_INFO, "Prepared log system");
2015-05-13 03:09:36 -03:00
for (uint8_t i=0; i<num_gcs; i++) {
gcs[i].reset_cli_timeout();
}
}
2015-05-13 03:09:36 -03:00
arming.set_logging_available(DataFlash.CardInserted());
}
2015-05-13 03:09:36 -03:00
#else // LOGGING_ENABLED
2015-07-31 18:26:14 -03:00
#if CLI_ENABLED == ENABLED
bool Plane::print_log_menu(void) { return true; }
int8_t Plane::dump_log(uint8_t argc, const Menu::arg *argv) { return 0; }
int8_t Plane::erase_logs(uint8_t argc, const Menu::arg *argv) { return 0; }
int8_t Plane::select_logs(uint8_t argc, const Menu::arg *argv) { return 0; }
int8_t Plane::process_logs(uint8_t argc, const Menu::arg *argv) { return 0; }
#endif // CLI_ENABLED == ENABLED
void Plane::do_erase_logs(void) {}
void Plane::Log_Write_Attitude(void) {}
void Plane::Log_Write_Performance() {}
void Plane::Log_Write_Startup(uint8_t type) {}
void Plane::Log_Write_Control_Tuning() {}
void Plane::Log_Write_TECS_Tuning(void) {}
void Plane::Log_Write_Nav_Tuning() {}
void Plane::Log_Write_Status() {}
void Plane::Log_Write_Sonar() {}
#if OPTFLOW == ENABLED
void Plane::Log_Write_Optflow() {}
#endif
2015-07-31 18:26:14 -03:00
void Plane::Log_Write_Current() {}
void Plane::Log_Arm_Disarm() {}
void Plane::Log_Write_GPS(uint8_t instance) {}
void Plane::Log_Write_IMU() {}
void Plane::Log_Write_RC(void) {}
void Plane::Log_Write_Baro(void) {}
void Plane::Log_Write_Airspeed(void) {}
void Plane::Log_Write_Home_And_Origin() {}
#if CLI_ENABLED == ENABLED
void Plane::Log_Read(uint16_t log_num, int16_t start_page, int16_t end_page) {}
#endif // CLI_ENABLED
void Plane::start_logging() {}
void Plane::log_init(void) {}
#endif // LOGGING_ENABLED