2013-12-29 07:56:30 -04:00
|
|
|
/// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*-
|
|
|
|
/*
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
2015-08-11 03:28:41 -03:00
|
|
|
#include <AP_Param/AP_Param.h>
|
2015-06-29 02:59:53 -03:00
|
|
|
#include "Parameters.h"
|
|
|
|
#include "VehicleType.h"
|
|
|
|
#include "MsgHandler.h"
|
2013-12-29 07:56:30 -04:00
|
|
|
|
2014-05-13 06:10:24 -03:00
|
|
|
#ifndef INT16_MIN
|
2014-05-01 04:54:31 -03:00
|
|
|
#define INT16_MIN -32768
|
|
|
|
#define INT16_MAX 32767
|
2014-05-13 06:10:24 -03:00
|
|
|
#endif
|
2014-05-01 04:54:31 -03:00
|
|
|
|
2013-12-29 07:56:30 -04:00
|
|
|
#include "LogReader.h"
|
2015-06-12 21:23:12 -03:00
|
|
|
#include "DataFlashFileReader.h"
|
2016-05-25 07:46:18 -03:00
|
|
|
#include "Replay.h"
|
2013-12-29 07:56:30 -04:00
|
|
|
|
2015-10-22 12:36:14 -03:00
|
|
|
#if CONFIG_HAL_BOARD == HAL_BOARD_SITL
|
|
|
|
#include <SITL/SITL.h>
|
|
|
|
#endif
|
|
|
|
|
2015-05-09 19:20:32 -03:00
|
|
|
#define streq(x, y) (!strcmp(x, y))
|
|
|
|
|
2015-10-16 17:22:11 -03:00
|
|
|
const AP_HAL::HAL& hal = AP_HAL::get_HAL();
|
2013-12-29 07:56:30 -04:00
|
|
|
|
2015-06-20 09:30:00 -03:00
|
|
|
ReplayVehicle replayvehicle;
|
2013-12-29 18:47:50 -04:00
|
|
|
|
2016-05-16 04:21:52 -03:00
|
|
|
struct globals globals;
|
|
|
|
|
2015-06-20 09:30:00 -03:00
|
|
|
#define GSCALAR(v, name, def) { replayvehicle.g.v.vtype, name, Parameters::k_param_ ## v, &replayvehicle.g.v, {def_value : def} }
|
|
|
|
#define GOBJECT(v, name, class) { AP_PARAM_GROUP, name, Parameters::k_param_ ## v, &replayvehicle.v, {group_info : class::var_info} }
|
|
|
|
#define GOBJECTN(v, pname, name, class) { AP_PARAM_GROUP, name, Parameters::k_param_ ## pname, &replayvehicle.v, {group_info : class::var_info} }
|
2014-01-04 20:39:43 -04:00
|
|
|
|
2015-10-25 14:03:46 -03:00
|
|
|
const AP_Param::Info ReplayVehicle::var_info[] = {
|
2015-06-01 03:14:11 -03:00
|
|
|
GSCALAR(dummy, "_DUMMY", 0),
|
2014-03-01 00:15:46 -04:00
|
|
|
|
2015-06-01 03:14:11 -03:00
|
|
|
// barometer ground calibration. The GND_ prefix is chosen for
|
|
|
|
// compatibility with previous releases of ArduPlane
|
|
|
|
// @Group: GND_
|
|
|
|
// @Path: ../libraries/AP_Baro/AP_Baro.cpp
|
|
|
|
GOBJECT(barometer, "GND_", AP_Baro),
|
|
|
|
|
|
|
|
// @Group: INS_
|
|
|
|
// @Path: ../libraries/AP_InertialSensor/AP_InertialSensor.cpp
|
|
|
|
GOBJECT(ins, "INS_", AP_InertialSensor),
|
|
|
|
|
|
|
|
// @Group: AHRS_
|
|
|
|
// @Path: ../libraries/AP_AHRS/AP_AHRS.cpp
|
|
|
|
GOBJECT(ahrs, "AHRS_", AP_AHRS),
|
|
|
|
|
|
|
|
// @Group: ARSPD_
|
|
|
|
// @Path: ../libraries/AP_Airspeed/AP_Airspeed.cpp
|
|
|
|
GOBJECT(airspeed, "ARSPD_", AP_Airspeed),
|
|
|
|
|
|
|
|
// @Group: EKF_
|
|
|
|
// @Path: ../libraries/AP_NavEKF/AP_NavEKF.cpp
|
|
|
|
GOBJECTN(EKF, NavEKF, "EKF_", NavEKF),
|
|
|
|
|
2015-09-22 22:55:26 -03:00
|
|
|
// @Group: EK2_
|
|
|
|
// @Path: ../libraries/AP_NavEKF2/AP_NavEKF2.cpp
|
|
|
|
GOBJECTN(EKF2, NavEKF2, "EK2_", NavEKF2),
|
|
|
|
|
2015-06-01 03:14:11 -03:00
|
|
|
// @Group: COMPASS_
|
|
|
|
// @Path: ../libraries/AP_Compass/AP_Compass.cpp
|
|
|
|
GOBJECT(compass, "COMPASS_", Compass),
|
|
|
|
|
2016-05-08 23:48:30 -03:00
|
|
|
// @Group: LOG
|
|
|
|
// @Path: ../libraries/DataFlash/DataFlash.cpp
|
|
|
|
GOBJECT(dataflash, "LOG", DataFlash_Class),
|
|
|
|
|
2015-06-01 03:14:11 -03:00
|
|
|
AP_VAREND
|
2014-12-07 20:25:22 -04:00
|
|
|
};
|
|
|
|
|
2015-06-20 09:30:00 -03:00
|
|
|
|
|
|
|
void ReplayVehicle::load_parameters(void)
|
2015-06-01 03:14:11 -03:00
|
|
|
{
|
2016-05-03 00:21:42 -03:00
|
|
|
unlink("Replay.stg");
|
2015-06-01 03:14:11 -03:00
|
|
|
if (!AP_Param::check_var_info()) {
|
2015-11-19 23:05:11 -04:00
|
|
|
AP_HAL::panic("Bad parameter table");
|
2015-06-01 03:14:11 -03:00
|
|
|
}
|
2016-05-03 00:21:42 -03:00
|
|
|
AP_Param::set_default_by_name("EKF_ENABLE", 1);
|
|
|
|
AP_Param::set_default_by_name("EK2_ENABLE", 1);
|
2016-05-08 23:48:30 -03:00
|
|
|
AP_Param::set_default_by_name("LOG_REPLAY", 1);
|
2016-05-03 00:21:42 -03:00
|
|
|
AP_Param::set_default_by_name("AHRS_EKF_TYPE", 2);
|
2016-05-09 00:58:37 -03:00
|
|
|
AP_Param::set_default_by_name("LOG_FILE_BUFSIZE", 60);
|
2015-06-01 03:14:11 -03:00
|
|
|
}
|
2014-03-01 00:15:46 -04:00
|
|
|
|
2016-05-25 21:10:27 -03:00
|
|
|
static const struct LogStructure min_log_structure[] = {
|
|
|
|
{ LOG_FORMAT_MSG, sizeof(log_Format),
|
|
|
|
"FMT", "BBnNZ", "Type,Length,Name,Format,Columns" },
|
|
|
|
{ LOG_PARAMETER_MSG, sizeof(log_Parameter),
|
|
|
|
"PARM", "QNf", "TimeUS,Name,Value" },
|
|
|
|
{ LOG_MESSAGE_MSG, sizeof(log_Message),
|
|
|
|
"MSG", "QZ", "TimeUS,Message"},
|
|
|
|
};
|
|
|
|
|
2015-07-04 09:43:54 -03:00
|
|
|
void ReplayVehicle::setup(void)
|
|
|
|
{
|
2015-09-23 05:56:15 -03:00
|
|
|
load_parameters();
|
|
|
|
|
2016-05-25 21:10:27 -03:00
|
|
|
// we pass a minimal log structure, as we will be outputting the
|
|
|
|
// log structures we need manually, to prevent FMT duplicates
|
|
|
|
dataflash.Init(min_log_structure, ARRAY_SIZE(min_log_structure));
|
2015-06-20 09:30:00 -03:00
|
|
|
dataflash.StartNewLog();
|
|
|
|
|
|
|
|
ahrs.set_compass(&compass);
|
|
|
|
ahrs.set_fly_forward(true);
|
|
|
|
ahrs.set_wind_estimation(true);
|
|
|
|
ahrs.set_correct_centrifugal(true);
|
|
|
|
ahrs.set_ekf_use(true);
|
2016-05-03 19:26:02 -03:00
|
|
|
|
2015-09-23 05:56:15 -03:00
|
|
|
EKF2.set_enable(true);
|
|
|
|
|
2015-06-20 09:30:00 -03:00
|
|
|
printf("Starting disarmed\n");
|
|
|
|
hal.util->set_soft_armed(false);
|
|
|
|
|
|
|
|
barometer.init();
|
|
|
|
barometer.setHIL(0);
|
|
|
|
barometer.update();
|
|
|
|
compass.init();
|
|
|
|
ins.set_hil_mode();
|
|
|
|
}
|
|
|
|
|
|
|
|
Replay replay(replayvehicle);
|
|
|
|
|
2015-06-01 03:14:11 -03:00
|
|
|
void Replay::usage(void)
|
2014-03-01 00:15:46 -04:00
|
|
|
{
|
|
|
|
::printf("Options:\n");
|
2015-06-09 21:09:56 -03:00
|
|
|
::printf("\t--parm NAME=VALUE set parameter NAME to VALUE\n");
|
|
|
|
::printf("\t--accel-mask MASK set accel mask (1=accel1 only, 2=accel2 only, 3=both)\n");
|
|
|
|
::printf("\t--gyro-mask MASK set gyro mask (1=gyro1 only, 2=gyro2 only, 3=both)\n");
|
|
|
|
::printf("\t--arm-time time arm at time (milliseconds)\n");
|
2015-06-16 01:26:07 -03:00
|
|
|
::printf("\t--no-imt don't use IMT data\n");
|
2015-06-25 01:17:00 -03:00
|
|
|
::printf("\t--check-generate generate CHEK messages in output\n");
|
2015-06-26 02:12:17 -03:00
|
|
|
::printf("\t--check check solution against CHEK messages\n");
|
|
|
|
::printf("\t--tolerance-euler tolerance for euler angles in degrees\n");
|
|
|
|
::printf("\t--tolerance-pos tolerance for position in meters\n");
|
|
|
|
::printf("\t--tolerance-vel tolerance for velocity in meters/second\n");
|
2015-07-07 02:13:51 -03:00
|
|
|
::printf("\t--nottypes list of msg types not to output, comma separated\n");
|
2015-07-07 02:55:41 -03:00
|
|
|
::printf("\t--downsample downsampling rate for output\n");
|
2016-05-05 01:08:52 -03:00
|
|
|
::printf("\t--logmatch match logging rate to source\n");
|
2016-05-16 04:21:52 -03:00
|
|
|
::printf("\t--no-params don't use parameters from the log\n");
|
2016-03-10 18:54:00 -04:00
|
|
|
::printf("\t--no-fpe do not generate floating point exceptions\n");
|
2014-03-01 00:15:46 -04:00
|
|
|
}
|
|
|
|
|
2015-06-25 01:17:00 -03:00
|
|
|
|
|
|
|
enum {
|
2015-06-26 02:12:17 -03:00
|
|
|
OPT_CHECK = 128,
|
|
|
|
OPT_CHECK_GENERATE,
|
|
|
|
OPT_TOLERANCE_EULER,
|
|
|
|
OPT_TOLERANCE_POS,
|
2015-07-07 02:13:51 -03:00
|
|
|
OPT_TOLERANCE_VEL,
|
2015-07-07 02:55:41 -03:00
|
|
|
OPT_NOTTYPES,
|
2016-05-05 01:08:52 -03:00
|
|
|
OPT_DOWNSAMPLE,
|
2016-05-16 04:21:52 -03:00
|
|
|
OPT_LOGMATCH,
|
|
|
|
OPT_NOPARAMS,
|
2016-03-10 18:54:00 -04:00
|
|
|
OPT_PARAM_FILE,
|
|
|
|
OPT_NO_FPE,
|
2015-06-25 01:17:00 -03:00
|
|
|
};
|
|
|
|
|
2015-06-18 22:57:19 -03:00
|
|
|
void Replay::flush_dataflash(void) {
|
|
|
|
_vehicle.dataflash.flush();
|
|
|
|
}
|
|
|
|
|
2015-07-07 02:13:51 -03:00
|
|
|
/*
|
|
|
|
create a list from a comma separated string
|
|
|
|
*/
|
|
|
|
const char **Replay::parse_list_from_string(const char *str_in)
|
|
|
|
{
|
|
|
|
uint16_t comma_count=0;
|
|
|
|
const char *p;
|
|
|
|
for (p=str_in; *p; p++) {
|
|
|
|
if (*p == ',') comma_count++;
|
|
|
|
}
|
|
|
|
|
|
|
|
char *str = strdup(str_in);
|
|
|
|
if (str == NULL) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
const char **ret = (const char **)calloc(comma_count+2, sizeof(char *));
|
|
|
|
if (ret == NULL) {
|
|
|
|
free(str);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
char *saveptr = NULL;
|
|
|
|
uint16_t idx = 0;
|
|
|
|
for (p=strtok_r(str, ",", &saveptr); p; p=strtok_r(NULL, ",", &saveptr)) {
|
|
|
|
ret[idx++] = p;
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2015-06-09 21:09:56 -03:00
|
|
|
void Replay::_parse_command_line(uint8_t argc, char * const argv[])
|
2013-12-29 07:56:30 -04:00
|
|
|
{
|
2015-06-09 21:09:56 -03:00
|
|
|
const struct GetOptLong::option options[] = {
|
2016-03-10 18:54:00 -04:00
|
|
|
// name has_arg flag val
|
2015-06-09 21:09:56 -03:00
|
|
|
{"parm", true, 0, 'p'},
|
|
|
|
{"param", true, 0, 'p'},
|
2016-05-25 07:46:18 -03:00
|
|
|
{"param-file", true, 0, OPT_PARAM_FILE},
|
2015-06-09 21:09:56 -03:00
|
|
|
{"help", false, 0, 'h'},
|
|
|
|
{"accel-mask", true, 0, 'a'},
|
|
|
|
{"gyro-mask", true, 0, 'g'},
|
|
|
|
{"arm-time", true, 0, 'A'},
|
2015-06-16 01:26:07 -03:00
|
|
|
{"no-imt", false, 0, 'n'},
|
2015-06-26 02:12:17 -03:00
|
|
|
{"check-generate", false, 0, OPT_CHECK_GENERATE},
|
|
|
|
{"check", false, 0, OPT_CHECK},
|
|
|
|
{"tolerance-euler", true, 0, OPT_TOLERANCE_EULER},
|
|
|
|
{"tolerance-pos", true, 0, OPT_TOLERANCE_POS},
|
|
|
|
{"tolerance-vel", true, 0, OPT_TOLERANCE_VEL},
|
2015-07-07 02:13:51 -03:00
|
|
|
{"nottypes", true, 0, OPT_NOTTYPES},
|
2015-07-07 02:55:41 -03:00
|
|
|
{"downsample", true, 0, OPT_DOWNSAMPLE},
|
2016-05-05 01:08:52 -03:00
|
|
|
{"logmatch", false, 0, OPT_LOGMATCH},
|
2016-05-16 04:21:52 -03:00
|
|
|
{"no-params", false, 0, OPT_NOPARAMS},
|
2016-03-10 18:54:00 -04:00
|
|
|
{"no-fpe", false, 0, OPT_NO_FPE},
|
2015-06-09 21:09:56 -03:00
|
|
|
{0, false, 0, 0}
|
|
|
|
};
|
|
|
|
|
|
|
|
GetOptLong gopt(argc, argv, "r:p:ha:g:A:", options);
|
2014-01-04 20:39:43 -04:00
|
|
|
|
2014-02-26 04:34:01 -04:00
|
|
|
int opt;
|
2015-06-09 21:09:56 -03:00
|
|
|
while ((opt = gopt.getoption()) != -1) {
|
2014-02-26 04:34:01 -04:00
|
|
|
switch (opt) {
|
2014-03-01 17:00:13 -04:00
|
|
|
case 'g':
|
2015-06-09 21:09:56 -03:00
|
|
|
logreader.set_gyro_mask(strtol(gopt.optarg, NULL, 0));
|
2014-03-01 17:00:13 -04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 'a':
|
2015-06-09 21:09:56 -03:00
|
|
|
logreader.set_accel_mask(strtol(gopt.optarg, NULL, 0));
|
2014-03-01 17:00:13 -04:00
|
|
|
break;
|
|
|
|
|
2014-04-21 00:00:59 -03:00
|
|
|
case 'A':
|
2015-06-09 21:09:56 -03:00
|
|
|
arm_time_ms = strtol(gopt.optarg, NULL, 0);
|
2014-04-21 00:00:59 -03:00
|
|
|
break;
|
|
|
|
|
2015-06-16 01:26:07 -03:00
|
|
|
case 'n':
|
|
|
|
use_imt = false;
|
|
|
|
logreader.set_use_imt(use_imt);
|
|
|
|
break;
|
|
|
|
|
2015-06-25 01:17:00 -03:00
|
|
|
case 'p': {
|
2015-06-09 21:09:56 -03:00
|
|
|
const char *eq = strchr(gopt.optarg, '=');
|
2014-03-01 00:15:46 -04:00
|
|
|
if (eq == NULL) {
|
|
|
|
::printf("Usage: -p NAME=VALUE\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
2016-05-25 07:46:18 -03:00
|
|
|
struct user_parameter *u = new user_parameter;
|
|
|
|
strncpy(u->name, gopt.optarg, eq-gopt.optarg);
|
|
|
|
u->value = atof(eq+1);
|
|
|
|
u->next = user_parameters;
|
|
|
|
user_parameters = u;
|
2014-03-01 00:15:46 -04:00
|
|
|
break;
|
2014-02-26 04:34:01 -04:00
|
|
|
}
|
2015-06-25 01:17:00 -03:00
|
|
|
|
2015-06-26 02:12:17 -03:00
|
|
|
case OPT_CHECK_GENERATE:
|
|
|
|
check_generate = true;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case OPT_CHECK:
|
|
|
|
check_solution = true;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case OPT_TOLERANCE_EULER:
|
|
|
|
tolerance_euler = atof(gopt.optarg);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case OPT_TOLERANCE_POS:
|
|
|
|
tolerance_pos = atof(gopt.optarg);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case OPT_TOLERANCE_VEL:
|
|
|
|
tolerance_vel = atof(gopt.optarg);
|
2015-06-25 01:17:00 -03:00
|
|
|
break;
|
|
|
|
|
2015-07-07 02:13:51 -03:00
|
|
|
case OPT_NOTTYPES:
|
|
|
|
nottypes = parse_list_from_string(gopt.optarg);
|
|
|
|
break;
|
|
|
|
|
2015-07-07 02:55:41 -03:00
|
|
|
case OPT_DOWNSAMPLE:
|
|
|
|
downsample = atoi(gopt.optarg);
|
|
|
|
break;
|
|
|
|
|
2016-05-05 01:08:52 -03:00
|
|
|
case OPT_LOGMATCH:
|
|
|
|
logmatch = true;
|
|
|
|
break;
|
2016-05-16 04:21:52 -03:00
|
|
|
|
|
|
|
case OPT_NOPARAMS:
|
|
|
|
globals.no_params = true;
|
|
|
|
break;
|
2016-05-25 07:46:18 -03:00
|
|
|
|
|
|
|
case OPT_PARAM_FILE:
|
|
|
|
load_param_file(gopt.optarg);
|
|
|
|
break;
|
2016-05-05 01:08:52 -03:00
|
|
|
|
2016-03-10 18:54:00 -04:00
|
|
|
case OPT_NO_FPE:
|
|
|
|
generate_fpe = false;
|
|
|
|
break;
|
|
|
|
|
2015-06-25 01:17:00 -03:00
|
|
|
case 'h':
|
|
|
|
default:
|
|
|
|
usage();
|
|
|
|
exit(0);
|
|
|
|
}
|
2014-02-26 04:34:01 -04:00
|
|
|
}
|
|
|
|
|
2015-06-09 21:09:56 -03:00
|
|
|
argv += gopt.optind;
|
|
|
|
argc -= gopt.optind;
|
2014-02-26 04:34:01 -04:00
|
|
|
|
|
|
|
if (argc > 0) {
|
|
|
|
filename = argv[0];
|
2014-02-22 17:17:01 -04:00
|
|
|
}
|
2015-06-09 21:09:56 -03:00
|
|
|
}
|
|
|
|
|
2015-07-04 02:14:14 -03:00
|
|
|
class IMUCounter : public DataFlashFileReader {
|
2015-06-12 21:23:12 -03:00
|
|
|
public:
|
2015-07-04 02:14:14 -03:00
|
|
|
IMUCounter() {}
|
2015-06-12 21:23:12 -03:00
|
|
|
bool handle_log_format_msg(const struct log_Format &f);
|
|
|
|
bool handle_msg(const struct log_Format &f, uint8_t *msg);
|
|
|
|
|
2016-05-24 21:34:41 -03:00
|
|
|
uint64_t last_clock_timestamp = 0;
|
|
|
|
float last_parm_value = 0;
|
|
|
|
char last_parm_name[17] {};
|
2015-06-12 21:23:12 -03:00
|
|
|
private:
|
2016-05-24 21:34:41 -03:00
|
|
|
MsgHandler *handler = nullptr;
|
|
|
|
MsgHandler *parm_handler = nullptr;
|
2015-06-12 21:23:12 -03:00
|
|
|
};
|
2015-07-04 02:14:14 -03:00
|
|
|
|
|
|
|
bool IMUCounter::handle_log_format_msg(const struct log_Format &f) {
|
2015-09-09 22:26:12 -03:00
|
|
|
if (!strncmp(f.name,"IMU",4) ||
|
|
|
|
!strncmp(f.name,"IMT",4)) {
|
2016-05-24 21:34:41 -03:00
|
|
|
// an IMU or IMT message message format
|
2015-06-12 21:23:12 -03:00
|
|
|
handler = new MsgHandler(f);
|
|
|
|
}
|
2016-05-24 21:34:41 -03:00
|
|
|
if (strncmp(f.name,"PARM",4) == 0) {
|
|
|
|
// PARM message message format
|
|
|
|
parm_handler = new MsgHandler(f);
|
|
|
|
}
|
2015-06-12 21:23:12 -03:00
|
|
|
|
|
|
|
return true;
|
|
|
|
};
|
2015-07-04 02:14:14 -03:00
|
|
|
|
|
|
|
bool IMUCounter::handle_msg(const struct log_Format &f, uint8_t *msg) {
|
2016-05-24 21:34:41 -03:00
|
|
|
if (strncmp(f.name,"PARM",4) == 0) {
|
|
|
|
// gather parameter values to check for SCHED_LOOP_RATE
|
|
|
|
parm_handler->field_value(msg, "Name", last_parm_name, sizeof(last_parm_name));
|
|
|
|
parm_handler->field_value(msg, "Value", last_parm_value);
|
|
|
|
return true;
|
|
|
|
}
|
2015-09-09 22:26:12 -03:00
|
|
|
if (strncmp(f.name,"IMU",4) &&
|
|
|
|
strncmp(f.name,"IMT",4)) {
|
2015-07-04 02:14:14 -03:00
|
|
|
// not an IMU message
|
2015-06-12 21:23:12 -03:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-09-09 22:26:12 -03:00
|
|
|
if (handler->field_value(msg, "TimeUS", last_clock_timestamp)) {
|
|
|
|
} else if (handler->field_value(msg, "TimeMS", last_clock_timestamp)) {
|
|
|
|
last_clock_timestamp *= 1000;
|
2015-06-12 21:23:12 -03:00
|
|
|
} else {
|
2015-09-09 22:26:12 -03:00
|
|
|
::printf("Unable to find timestamp in message");
|
2015-06-12 21:23:12 -03:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-07-04 02:14:14 -03:00
|
|
|
/*
|
|
|
|
find information about the log
|
|
|
|
*/
|
|
|
|
bool Replay::find_log_info(struct log_information &info)
|
|
|
|
{
|
|
|
|
IMUCounter reader;
|
2015-06-12 21:23:12 -03:00
|
|
|
if (!reader.open_log(filename)) {
|
|
|
|
perror(filename);
|
|
|
|
exit(1);
|
|
|
|
}
|
2015-09-09 22:26:12 -03:00
|
|
|
char clock_source[5] = { };
|
2015-06-12 21:23:12 -03:00
|
|
|
int samplecount = 0;
|
|
|
|
uint64_t prev = 0;
|
2015-07-04 02:14:14 -03:00
|
|
|
uint64_t smallest_delta = 0;
|
2016-05-10 02:40:30 -03:00
|
|
|
uint64_t total_delta = 0;
|
2015-06-12 21:23:12 -03:00
|
|
|
prev = 0;
|
2015-09-09 22:26:12 -03:00
|
|
|
const uint16_t samples_required = 1000;
|
|
|
|
while (samplecount < samples_required) {
|
2015-06-12 21:23:12 -03:00
|
|
|
char type[5];
|
|
|
|
if (!reader.update(type)) {
|
|
|
|
break;
|
|
|
|
}
|
2015-09-09 22:26:12 -03:00
|
|
|
|
2016-05-24 21:34:41 -03:00
|
|
|
if (streq(type, "PARM") && streq(reader.last_parm_name, "SCHED_LOOP_RATE")) {
|
|
|
|
// get rate directly from parameters
|
|
|
|
info.update_rate = reader.last_parm_value;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-09-09 22:26:12 -03:00
|
|
|
if (strlen(clock_source) == 0) {
|
2015-11-30 19:49:43 -04:00
|
|
|
// If you want to add a clock source, also add it to
|
|
|
|
// handle_msg and handle_log_format_msg, above. Note that
|
|
|
|
// ordering is important here. For example, when we log
|
|
|
|
// IMT we may reduce the logging speed of IMU, so then
|
|
|
|
// using IMU as your clock source will lead to incorrect
|
|
|
|
// behaviour.
|
|
|
|
if (streq(type, "IMT")) {
|
2016-05-03 00:21:42 -03:00
|
|
|
strcpy(clock_source, "IMT");
|
2015-11-30 19:49:43 -04:00
|
|
|
} else if (streq(type, "IMU")) {
|
2016-05-03 00:21:42 -03:00
|
|
|
strcpy(clock_source, "IMU");
|
2015-09-09 22:26:12 -03:00
|
|
|
} else {
|
|
|
|
continue;
|
|
|
|
}
|
2016-05-03 00:21:42 -03:00
|
|
|
hal.console->printf("Using clock source %s\n", clock_source);
|
|
|
|
}
|
|
|
|
// IMT if available always overrides
|
|
|
|
if (streq(type, "IMT") && strcmp(clock_source, "IMT") != 0) {
|
|
|
|
strcpy(clock_source, "IMT");
|
|
|
|
hal.console->printf("Changing clock source to %s\n", clock_source);
|
2016-05-04 02:25:14 -03:00
|
|
|
samplecount = 0;
|
|
|
|
prev = 0;
|
|
|
|
smallest_delta = 0;
|
2016-05-10 02:40:30 -03:00
|
|
|
total_delta = 0;
|
2015-09-09 22:26:12 -03:00
|
|
|
}
|
|
|
|
if (streq(type, clock_source)) {
|
2015-06-12 21:23:12 -03:00
|
|
|
if (prev == 0) {
|
2015-09-09 22:26:12 -03:00
|
|
|
prev = reader.last_clock_timestamp;
|
2015-06-12 21:23:12 -03:00
|
|
|
} else {
|
2015-09-09 22:26:12 -03:00
|
|
|
uint64_t delta = reader.last_clock_timestamp - prev;
|
2016-05-10 02:40:30 -03:00
|
|
|
if (delta < 40000 && delta > 1000) {
|
|
|
|
if (smallest_delta == 0 || delta < smallest_delta) {
|
|
|
|
smallest_delta = delta;
|
|
|
|
}
|
|
|
|
samplecount++;
|
|
|
|
total_delta += delta;
|
2015-07-04 02:14:14 -03:00
|
|
|
}
|
2015-06-12 21:23:12 -03:00
|
|
|
}
|
2016-05-10 02:40:30 -03:00
|
|
|
prev = reader.last_clock_timestamp;
|
2015-06-12 21:23:12 -03:00
|
|
|
}
|
2015-09-09 22:26:12 -03:00
|
|
|
|
2016-05-03 19:26:02 -03:00
|
|
|
if (streq(type, "IMU2")) {
|
2015-07-04 02:14:14 -03:00
|
|
|
info.have_imu2 = true;
|
|
|
|
}
|
2016-05-03 19:26:02 -03:00
|
|
|
if (streq(type, "IMT")) {
|
|
|
|
info.have_imt = true;
|
|
|
|
}
|
|
|
|
if (streq(type, "IMT2")) {
|
|
|
|
info.have_imt2 = true;
|
|
|
|
}
|
2015-06-12 21:23:12 -03:00
|
|
|
}
|
2015-07-04 02:14:14 -03:00
|
|
|
if (smallest_delta == 0) {
|
2015-09-09 22:26:12 -03:00
|
|
|
::printf("Unable to determine log rate - insufficient IMU/IMT messages? (need=%d got=%d)", samples_required, samplecount);
|
2015-07-04 02:14:14 -03:00
|
|
|
return false;
|
2015-06-12 21:23:12 -03:00
|
|
|
}
|
|
|
|
|
2016-05-10 02:40:30 -03:00
|
|
|
float average_delta = total_delta / samplecount;
|
|
|
|
float rate = 1.0e6f/average_delta;
|
|
|
|
printf("average_delta=%.2f smallest_delta=%lu samplecount=%lu\n",
|
|
|
|
average_delta, (unsigned long)smallest_delta, (unsigned long)samplecount);
|
2015-07-04 02:14:14 -03:00
|
|
|
if (rate < 100) {
|
|
|
|
info.update_rate = 50;
|
|
|
|
} else {
|
|
|
|
info.update_rate = 400;
|
2015-06-12 21:23:12 -03:00
|
|
|
}
|
2015-07-04 02:14:14 -03:00
|
|
|
return true;
|
2015-06-12 21:23:12 -03:00
|
|
|
}
|
|
|
|
|
2015-06-18 22:57:19 -03:00
|
|
|
// catch floating point exceptions
|
|
|
|
static void _replay_sig_fpe(int signum)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "ERROR: Floating point exception - flushing dataflash...\n");
|
|
|
|
replay.flush_dataflash();
|
|
|
|
fprintf(stderr, "ERROR: ... and aborting.\n");
|
2015-07-02 01:27:10 -03:00
|
|
|
if (replay.check_solution) {
|
|
|
|
FILE *f = fopen("replay_results.txt","a");
|
|
|
|
fprintf(f, "%s\tFPE\tFPE\tFPE\tFPE\tFPE\n",
|
|
|
|
replay.log_filename);
|
|
|
|
fclose(f);
|
|
|
|
}
|
2015-06-18 22:57:19 -03:00
|
|
|
abort();
|
|
|
|
}
|
|
|
|
|
2015-06-09 21:09:56 -03:00
|
|
|
void Replay::setup()
|
|
|
|
{
|
|
|
|
::printf("Starting\n");
|
|
|
|
|
|
|
|
uint8_t argc;
|
|
|
|
char * const *argv;
|
|
|
|
|
|
|
|
hal.util->commandline_arguments(argc, argv);
|
|
|
|
|
|
|
|
_parse_command_line(argc, argv);
|
2014-02-22 17:17:01 -04:00
|
|
|
|
2015-07-08 21:20:45 -03:00
|
|
|
if (!check_generate) {
|
|
|
|
logreader.set_save_chek_messages(true);
|
|
|
|
}
|
|
|
|
|
2016-03-10 18:54:00 -04:00
|
|
|
if (generate_fpe) {
|
|
|
|
// SITL_State::_parse_command_line sets up an FPE handler. We
|
|
|
|
// can do better:
|
|
|
|
feenableexcept(FE_INVALID | FE_OVERFLOW);
|
|
|
|
signal(SIGFPE, _replay_sig_fpe);
|
|
|
|
} else {
|
|
|
|
// disable floating point exception generation:
|
|
|
|
int exceptions = FE_OVERFLOW | FE_DIVBYZERO;
|
|
|
|
#ifndef __i386__
|
|
|
|
// i386 with gcc doesn't work with FE_INVALID
|
|
|
|
exceptions |= FE_INVALID;
|
|
|
|
#endif
|
|
|
|
if (feclearexcept(exceptions)) {
|
|
|
|
::fprintf(stderr, "Failed to disable floating point exceptions: %s", strerror(errno));
|
|
|
|
}
|
|
|
|
signal(SIGFPE, SIG_IGN);
|
|
|
|
}
|
2015-06-18 22:57:19 -03:00
|
|
|
|
2014-02-22 17:17:01 -04:00
|
|
|
hal.console->printf("Processing log %s\n", filename);
|
2015-06-12 21:23:12 -03:00
|
|
|
|
2015-07-01 23:23:48 -03:00
|
|
|
// remember filename for reporting
|
|
|
|
log_filename = filename;
|
|
|
|
|
2015-07-04 02:14:14 -03:00
|
|
|
if (!find_log_info(log_info)) {
|
|
|
|
printf("Update to get log information\n");
|
|
|
|
exit(1);
|
2015-06-12 21:23:12 -03:00
|
|
|
}
|
|
|
|
|
2015-07-04 02:14:14 -03:00
|
|
|
hal.console->printf("Using an update rate of %u Hz\n", log_info.update_rate);
|
2014-02-28 23:24:51 -04:00
|
|
|
|
2015-06-01 03:14:11 -03:00
|
|
|
if (!logreader.open_log(filename)) {
|
2014-02-28 23:24:51 -04:00
|
|
|
perror(filename);
|
|
|
|
exit(1);
|
|
|
|
}
|
2013-12-29 07:56:30 -04:00
|
|
|
|
2015-06-20 09:30:00 -03:00
|
|
|
_vehicle.setup();
|
2015-09-23 05:56:15 -03:00
|
|
|
|
2015-10-22 03:40:21 -03:00
|
|
|
inhibit_gyro_cal();
|
2016-05-03 19:26:02 -03:00
|
|
|
|
|
|
|
if (log_info.update_rate == 400) {
|
|
|
|
// assume copter for 400Hz
|
|
|
|
_vehicle.ahrs.set_vehicle_class(AHRS_VEHICLE_COPTER);
|
|
|
|
_vehicle.ahrs.set_fly_forward(false);
|
|
|
|
} else if (log_info.update_rate == 50) {
|
|
|
|
// assume copter for 400Hz
|
|
|
|
_vehicle.ahrs.set_vehicle_class(AHRS_VEHICLE_FIXED_WING);
|
|
|
|
_vehicle.ahrs.set_fly_forward(true);
|
|
|
|
}
|
|
|
|
|
2015-07-04 02:14:14 -03:00
|
|
|
set_ins_update_rate(log_info.update_rate);
|
2013-12-29 07:56:30 -04:00
|
|
|
|
2014-01-04 20:39:43 -04:00
|
|
|
plotf = fopen("plot.dat", "w");
|
|
|
|
plotf2 = fopen("plot2.dat", "w");
|
2014-01-30 18:27:56 -04:00
|
|
|
ekf1f = fopen("EKF1.dat", "w");
|
|
|
|
ekf2f = fopen("EKF2.dat", "w");
|
|
|
|
ekf3f = fopen("EKF3.dat", "w");
|
|
|
|
ekf4f = fopen("EKF4.dat", "w");
|
|
|
|
|
2016-05-03 19:26:02 -03:00
|
|
|
fprintf(plotf, "time SIM.Roll SIM.Pitch SIM.Yaw BAR.Alt FLIGHT.Roll FLIGHT.Pitch FLIGHT.Yaw FLIGHT.dN FLIGHT.dE AHR2.Roll AHR2.Pitch AHR2.Yaw DCM.Roll DCM.Pitch DCM.Yaw EKF.Roll EKF.Pitch EKF.Yaw INAV.dN INAV.dE INAV.Alt EKF.dN EKF.dE EKF.Alt\n");
|
2014-01-30 18:27:56 -04:00
|
|
|
fprintf(plotf2, "time E1 E2 E3 VN VE VD PN PE PD GX GY GZ WN WE MN ME MD MX MY MZ E1ref E2ref E3ref\n");
|
|
|
|
fprintf(ekf1f, "timestamp TimeMS Roll Pitch Yaw VN VE VD PN PE PD GX GY GZ\n");
|
|
|
|
fprintf(ekf2f, "timestamp TimeMS AX AY AZ VWN VWE MN ME MD MX MY MZ\n");
|
|
|
|
fprintf(ekf3f, "timestamp TimeMS IVN IVE IVD IPN IPE IPD IMX IMY IMZ IVT\n");
|
2014-05-14 05:39:18 -03:00
|
|
|
fprintf(ekf4f, "timestamp TimeMS SV SP SH SMX SMY SMZ SVT OFN EFE FS DS\n");
|
2014-01-04 20:39:43 -04:00
|
|
|
}
|
|
|
|
|
2015-06-25 01:17:00 -03:00
|
|
|
void Replay::set_ins_update_rate(uint16_t _update_rate) {
|
2015-12-26 00:44:44 -04:00
|
|
|
_vehicle.ins.init(_update_rate);
|
2015-06-20 09:30:00 -03:00
|
|
|
}
|
|
|
|
|
2015-10-22 03:40:21 -03:00
|
|
|
void Replay::inhibit_gyro_cal() {
|
|
|
|
// swiped from LR_MsgHandler.cpp; until we see PARM messages, we
|
|
|
|
// don't have a PARM handler available to set parameters.
|
|
|
|
enum ap_var_type var_type;
|
|
|
|
AP_Param *vp = AP_Param::find("INS_GYR_CAL", &var_type);
|
|
|
|
if (vp == NULL) {
|
|
|
|
::fprintf(stderr, "No GYR_CAL parameter found\n");
|
|
|
|
abort();
|
|
|
|
}
|
|
|
|
((AP_Float *)vp)->set(AP_InertialSensor::GYRO_CAL_NEVER);
|
|
|
|
|
|
|
|
// logreader.set_parameter("GYR_CAL", AP_InertialSensor::GYRO_CAL_NEVER);
|
|
|
|
}
|
2014-03-01 00:15:46 -04:00
|
|
|
|
|
|
|
/*
|
|
|
|
setup user -p parameters
|
|
|
|
*/
|
2015-06-01 03:14:11 -03:00
|
|
|
void Replay::set_user_parameters(void)
|
2014-03-01 00:15:46 -04:00
|
|
|
{
|
2016-05-25 07:46:18 -03:00
|
|
|
for (struct user_parameter *u=user_parameters; u; u=u->next) {
|
|
|
|
if (!logreader.set_parameter(u->name, u->value)) {
|
|
|
|
::printf("Failed to set parameter %s to %f\n", u->name, u->value);
|
2014-03-01 00:15:46 -04:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-05 01:08:52 -03:00
|
|
|
/*
|
|
|
|
write out EKF log messages
|
|
|
|
*/
|
|
|
|
void Replay::write_ekf_logs(void)
|
|
|
|
{
|
|
|
|
if (!LogReader::in_list("EKF", nottypes)) {
|
|
|
|
_vehicle.dataflash.Log_Write_EKF(_vehicle.ahrs,false);
|
|
|
|
}
|
|
|
|
if (!LogReader::in_list("AHRS2", nottypes)) {
|
|
|
|
_vehicle.dataflash.Log_Write_AHRS2(_vehicle.ahrs);
|
|
|
|
}
|
|
|
|
if (!LogReader::in_list("POS", nottypes)) {
|
|
|
|
_vehicle.dataflash.Log_Write_POS(_vehicle.ahrs);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-01 03:14:11 -03:00
|
|
|
void Replay::read_sensors(const char *type)
|
2014-01-04 20:39:43 -04:00
|
|
|
{
|
2015-05-09 19:20:32 -03:00
|
|
|
if (!done_parameters && !streq(type,"FMT") && !streq(type,"PARM")) {
|
2014-03-01 00:15:46 -04:00
|
|
|
done_parameters = true;
|
|
|
|
set_user_parameters();
|
|
|
|
}
|
2015-04-20 02:07:13 -03:00
|
|
|
|
2016-05-04 04:53:20 -03:00
|
|
|
if (done_parameters && streq(type, "PARM")) {
|
|
|
|
set_user_parameters();
|
|
|
|
}
|
|
|
|
|
2015-07-04 02:14:14 -03:00
|
|
|
if (!done_home_init) {
|
|
|
|
if (streq(type, "GPS") &&
|
|
|
|
(_vehicle.gps.status() >= AP_GPS::GPS_OK_FIX_3D) && done_baro_init) {
|
|
|
|
const Location &loc = _vehicle.gps.location();
|
|
|
|
::printf("GPS Lock at %.7f %.7f %.2fm time=%.1f seconds\n",
|
|
|
|
loc.lat * 1.0e-7f,
|
|
|
|
loc.lng * 1.0e-7f,
|
|
|
|
loc.alt * 0.01f,
|
2015-11-19 23:05:11 -04:00
|
|
|
AP_HAL::millis()*0.001f);
|
2015-07-04 02:14:14 -03:00
|
|
|
_vehicle.ahrs.set_home(loc);
|
|
|
|
_vehicle.compass.set_initial_location(loc.lat, loc.lng);
|
|
|
|
done_home_init = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-05-09 19:20:32 -03:00
|
|
|
if (streq(type,"GPS")) {
|
2015-06-20 09:30:00 -03:00
|
|
|
_vehicle.gps.update();
|
|
|
|
if (_vehicle.gps.status() >= AP_GPS::GPS_OK_FIX_3D) {
|
|
|
|
_vehicle.ahrs.estimate_wind();
|
2014-02-17 18:11:46 -04:00
|
|
|
}
|
2015-05-09 19:20:32 -03:00
|
|
|
} else if (streq(type,"MAG")) {
|
2015-06-20 09:30:00 -03:00
|
|
|
_vehicle.compass.read();
|
2015-05-09 19:20:32 -03:00
|
|
|
} else if (streq(type,"ARSP")) {
|
2015-06-20 09:30:00 -03:00
|
|
|
_vehicle.ahrs.set_airspeed(&_vehicle.airspeed);
|
2015-05-09 19:20:32 -03:00
|
|
|
} else if (streq(type,"BARO")) {
|
2015-06-20 09:30:00 -03:00
|
|
|
_vehicle.barometer.update();
|
2015-04-20 02:07:13 -03:00
|
|
|
if (!done_baro_init) {
|
|
|
|
done_baro_init = true;
|
|
|
|
::printf("Barometer initialised\n");
|
2015-06-20 09:30:00 -03:00
|
|
|
_vehicle.barometer.update_calibration();
|
2015-04-20 02:07:13 -03:00
|
|
|
}
|
2015-05-27 08:20:38 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
bool run_ahrs = false;
|
2016-05-03 19:26:02 -03:00
|
|
|
if (log_info.have_imt2) {
|
|
|
|
run_ahrs = streq(type, "IMT2");
|
|
|
|
_vehicle.ahrs.force_ekf_start();
|
|
|
|
} else if (log_info.have_imt) {
|
|
|
|
run_ahrs = streq(type, "IMT");
|
|
|
|
_vehicle.ahrs.force_ekf_start();
|
|
|
|
} else if (log_info.have_imu2) {
|
|
|
|
run_ahrs = streq(type, "IMU2");
|
|
|
|
} else {
|
|
|
|
run_ahrs = streq(type, "IMU");
|
2015-05-27 08:20:38 -03:00
|
|
|
}
|
2015-06-26 02:12:17 -03:00
|
|
|
|
|
|
|
/*
|
|
|
|
always run AHRS on CHECK messages when checking the solution
|
|
|
|
*/
|
|
|
|
if (check_solution) {
|
|
|
|
run_ahrs = streq(type, "CHEK");
|
|
|
|
}
|
|
|
|
|
2015-05-27 08:20:38 -03:00
|
|
|
if (run_ahrs) {
|
2015-06-20 09:30:00 -03:00
|
|
|
_vehicle.ahrs.update();
|
|
|
|
if (_vehicle.ahrs.get_home().lat != 0) {
|
|
|
|
_vehicle.inertial_nav.update(_vehicle.ins.get_delta_time());
|
2015-04-20 02:07:13 -03:00
|
|
|
}
|
2016-05-05 01:08:52 -03:00
|
|
|
if ((downsample == 0 || ++output_counter % downsample == 0) && !logmatch) {
|
|
|
|
write_ekf_logs();
|
2015-07-07 02:13:51 -03:00
|
|
|
}
|
2015-06-20 09:30:00 -03:00
|
|
|
if (_vehicle.ahrs.healthy() != ahrs_healthy) {
|
|
|
|
ahrs_healthy = _vehicle.ahrs.healthy();
|
2015-05-19 02:21:12 -03:00
|
|
|
printf("AHRS health: %u at %lu\n",
|
|
|
|
(unsigned)ahrs_healthy,
|
2015-11-19 23:05:11 -04:00
|
|
|
(unsigned long)AP_HAL::millis());
|
2014-01-05 01:37:47 -04:00
|
|
|
}
|
2015-06-26 02:12:17 -03:00
|
|
|
if (check_generate) {
|
|
|
|
log_check_generate();
|
|
|
|
} else if (check_solution) {
|
|
|
|
log_check_solution();
|
2015-06-25 01:17:00 -03:00
|
|
|
}
|
2013-12-29 08:21:09 -04:00
|
|
|
}
|
2016-05-05 01:08:52 -03:00
|
|
|
|
|
|
|
if (logmatch && streq(type, "NKF1")) {
|
|
|
|
write_ekf_logs();
|
|
|
|
}
|
2013-12-29 07:56:30 -04:00
|
|
|
}
|
|
|
|
|
2015-06-25 01:17:00 -03:00
|
|
|
|
|
|
|
/*
|
|
|
|
copy current data to CHEK message
|
|
|
|
*/
|
2015-06-26 02:12:17 -03:00
|
|
|
void Replay::log_check_generate(void)
|
2015-06-25 01:17:00 -03:00
|
|
|
{
|
|
|
|
Vector3f euler;
|
|
|
|
Vector3f velocity;
|
|
|
|
Location loc {};
|
|
|
|
|
|
|
|
_vehicle.EKF.getEulerAngles(euler);
|
|
|
|
_vehicle.EKF.getVelNED(velocity);
|
|
|
|
_vehicle.EKF.getLLH(loc);
|
|
|
|
|
|
|
|
struct log_Chek packet = {
|
|
|
|
LOG_PACKET_HEADER_INIT(LOG_CHEK_MSG),
|
2015-11-19 23:05:11 -04:00
|
|
|
time_us : AP_HAL::micros64(),
|
2015-06-25 01:17:00 -03:00
|
|
|
roll : (int16_t)(100*degrees(euler.x)), // roll angle (centi-deg, displayed as deg due to format string)
|
|
|
|
pitch : (int16_t)(100*degrees(euler.y)), // pitch angle (centi-deg, displayed as deg due to format string)
|
|
|
|
yaw : (uint16_t)wrap_360_cd(100*degrees(euler.z)), // yaw angle (centi-deg, displayed as deg due to format string)
|
|
|
|
lat : loc.lat,
|
|
|
|
lng : loc.lng,
|
|
|
|
alt : loc.alt*0.01f,
|
|
|
|
vnorth : velocity.x,
|
|
|
|
veast : velocity.y,
|
|
|
|
vdown : velocity.z
|
|
|
|
};
|
|
|
|
|
|
|
|
_vehicle.dataflash.WriteBlock(&packet, sizeof(packet));
|
|
|
|
}
|
|
|
|
|
2015-06-26 02:12:17 -03:00
|
|
|
|
|
|
|
/*
|
|
|
|
check current solution against CHEK message
|
|
|
|
*/
|
|
|
|
void Replay::log_check_solution(void)
|
|
|
|
{
|
|
|
|
const LR_MsgHandler::CheckState &check_state = logreader.get_check_state();
|
|
|
|
Vector3f euler;
|
|
|
|
Vector3f velocity;
|
|
|
|
Location loc {};
|
|
|
|
|
|
|
|
_vehicle.EKF.getEulerAngles(euler);
|
|
|
|
_vehicle.EKF.getVelNED(velocity);
|
|
|
|
_vehicle.EKF.getLLH(loc);
|
|
|
|
|
|
|
|
float roll_error = degrees(fabsf(euler.x - check_state.euler.x));
|
|
|
|
float pitch_error = degrees(fabsf(euler.y - check_state.euler.y));
|
2016-04-27 06:35:18 -03:00
|
|
|
float yaw_error = wrap_180_cd(100*degrees(fabsf(euler.z - check_state.euler.z)))*0.01f;
|
2015-06-26 02:12:17 -03:00
|
|
|
float vel_error = (velocity - check_state.velocity).length();
|
|
|
|
float pos_error = get_distance(check_state.pos, loc);
|
|
|
|
|
2015-11-27 13:11:58 -04:00
|
|
|
check_result.max_roll_error = MAX(check_result.max_roll_error, roll_error);
|
|
|
|
check_result.max_pitch_error = MAX(check_result.max_pitch_error, pitch_error);
|
|
|
|
check_result.max_yaw_error = MAX(check_result.max_yaw_error, yaw_error);
|
|
|
|
check_result.max_vel_error = MAX(check_result.max_vel_error, vel_error);
|
|
|
|
check_result.max_pos_error = MAX(check_result.max_pos_error, pos_error);
|
2015-06-26 02:12:17 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-06-01 03:14:11 -03:00
|
|
|
void Replay::loop()
|
2013-12-29 07:56:30 -04:00
|
|
|
{
|
|
|
|
while (true) {
|
2015-05-10 22:16:05 -03:00
|
|
|
char type[5];
|
2014-04-21 00:00:59 -03:00
|
|
|
|
2015-11-19 23:05:11 -04:00
|
|
|
if (arm_time_ms >= 0 && AP_HAL::millis() > (uint32_t)arm_time_ms) {
|
2015-01-28 19:45:14 -04:00
|
|
|
if (!hal.util->get_soft_armed()) {
|
|
|
|
hal.util->set_soft_armed(true);
|
2015-11-19 23:05:11 -04:00
|
|
|
::printf("Arming at %u ms\n", (unsigned)AP_HAL::millis());
|
2014-04-21 00:00:59 -03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-01 03:14:11 -03:00
|
|
|
if (!logreader.update(type)) {
|
2015-11-19 23:05:11 -04:00
|
|
|
::printf("End of log at %.1f seconds\n", AP_HAL::millis()*0.001f);
|
2013-12-29 18:47:50 -04:00
|
|
|
fclose(plotf);
|
2015-06-26 02:12:17 -03:00
|
|
|
break;
|
2013-12-29 07:56:30 -04:00
|
|
|
}
|
2014-01-04 20:39:43 -04:00
|
|
|
read_sensors(type);
|
2013-12-29 07:56:30 -04:00
|
|
|
|
2015-05-09 19:20:32 -03:00
|
|
|
if (streq(type,"ATT")) {
|
2013-12-29 08:21:09 -04:00
|
|
|
Vector3f ekf_euler;
|
2013-12-31 17:54:56 -04:00
|
|
|
Vector3f velNED;
|
|
|
|
Vector3f posNED;
|
|
|
|
Vector3f gyroBias;
|
2014-10-30 20:47:57 -03:00
|
|
|
float accelWeighting;
|
|
|
|
float accelZBias1;
|
|
|
|
float accelZBias2;
|
2014-01-30 18:27:56 -04:00
|
|
|
Vector3f windVel;
|
2013-12-31 17:54:56 -04:00
|
|
|
Vector3f magNED;
|
|
|
|
Vector3f magXYZ;
|
2014-01-04 20:39:43 -04:00
|
|
|
Vector3f DCM_attitude;
|
2014-01-05 01:37:47 -04:00
|
|
|
Vector3f ekf_relpos;
|
2014-01-30 18:27:56 -04:00
|
|
|
Vector3f velInnov;
|
|
|
|
Vector3f posInnov;
|
|
|
|
Vector3f magInnov;
|
|
|
|
float tasInnov;
|
2014-04-01 01:33:32 -03:00
|
|
|
float velVar;
|
|
|
|
float posVar;
|
|
|
|
float hgtVar;
|
2014-01-30 18:27:56 -04:00
|
|
|
Vector3f magVar;
|
2014-04-01 01:33:32 -03:00
|
|
|
float tasVar;
|
|
|
|
Vector2f offset;
|
2016-05-10 03:09:08 -03:00
|
|
|
uint16_t faultStatus;
|
2014-01-30 18:27:56 -04:00
|
|
|
|
2015-12-10 18:06:48 -04:00
|
|
|
const Matrix3f &dcm_matrix = _vehicle.ahrs.AP_AHRS_DCM::get_rotation_body_to_ned();
|
2014-03-01 17:00:13 -04:00
|
|
|
dcm_matrix.to_euler(&DCM_attitude.x, &DCM_attitude.y, &DCM_attitude.z);
|
2015-06-20 09:30:00 -03:00
|
|
|
_vehicle.EKF.getEulerAngles(ekf_euler);
|
|
|
|
_vehicle.EKF.getVelNED(velNED);
|
|
|
|
_vehicle.EKF.getPosNED(posNED);
|
|
|
|
_vehicle.EKF.getGyroBias(gyroBias);
|
|
|
|
_vehicle.EKF.getIMU1Weighting(accelWeighting);
|
|
|
|
_vehicle.EKF.getAccelZBias(accelZBias1, accelZBias2);
|
|
|
|
_vehicle.EKF.getWind(windVel);
|
|
|
|
_vehicle.EKF.getMagNED(magNED);
|
|
|
|
_vehicle.EKF.getMagXYZ(magXYZ);
|
|
|
|
_vehicle.EKF.getInnovations(velInnov, posInnov, magInnov, tasInnov);
|
|
|
|
_vehicle.EKF.getVariances(velVar, posVar, hgtVar, magVar, tasVar, offset);
|
|
|
|
_vehicle.EKF.getFilterFaults(faultStatus);
|
|
|
|
_vehicle.EKF.getPosNED(ekf_relpos);
|
|
|
|
Vector3f inav_pos = _vehicle.inertial_nav.get_position() * 0.01f;
|
2013-12-31 17:54:56 -04:00
|
|
|
float temp = degrees(ekf_euler.z);
|
2014-01-30 18:27:56 -04:00
|
|
|
|
2013-12-31 17:54:56 -04:00
|
|
|
if (temp < 0.0f) temp = temp + 360.0f;
|
2016-05-03 19:26:02 -03:00
|
|
|
fprintf(plotf, "%.3f %.1f %.1f %.1f %.2f %.1f %.1f %.1f %.2f %.2f %.1f %.1f %.1f %.1f %.1f %.1f %.1f %.1f %.1f %.2f %.2f %.2f %.2f %.2f %.2f\n",
|
2015-11-19 23:05:11 -04:00
|
|
|
AP_HAL::millis() * 0.001f,
|
2015-06-01 03:14:11 -03:00
|
|
|
logreader.get_sim_attitude().x,
|
|
|
|
logreader.get_sim_attitude().y,
|
|
|
|
logreader.get_sim_attitude().z,
|
2015-06-20 09:30:00 -03:00
|
|
|
_vehicle.barometer.get_altitude(),
|
2015-06-01 03:14:11 -03:00
|
|
|
logreader.get_attitude().x,
|
|
|
|
logreader.get_attitude().y,
|
|
|
|
wrap_180_cd(logreader.get_attitude().z*100)*0.01f,
|
|
|
|
logreader.get_inavpos().x,
|
|
|
|
logreader.get_inavpos().y,
|
|
|
|
logreader.get_ahr2_attitude().x,
|
|
|
|
logreader.get_ahr2_attitude().y,
|
|
|
|
wrap_180_cd(logreader.get_ahr2_attitude().z*100)*0.01f,
|
2014-01-04 20:39:43 -04:00
|
|
|
degrees(DCM_attitude.x),
|
|
|
|
degrees(DCM_attitude.y),
|
|
|
|
degrees(DCM_attitude.z),
|
2014-01-03 00:38:08 -04:00
|
|
|
degrees(ekf_euler.x),
|
|
|
|
degrees(ekf_euler.y),
|
2014-01-05 01:37:47 -04:00
|
|
|
degrees(ekf_euler.z),
|
|
|
|
inav_pos.x,
|
|
|
|
inav_pos.y,
|
|
|
|
inav_pos.z,
|
|
|
|
ekf_relpos.x,
|
|
|
|
ekf_relpos.y,
|
|
|
|
-ekf_relpos.z);
|
2014-01-30 18:27:56 -04:00
|
|
|
fprintf(plotf2, "%.3f %.1f %.1f %.1f %.1f %.1f %.1f %.1f %.1f %.1f %.1f %.1f %.1f %.1f %.1f %.1f %.1f %.1f %.1f %.1f %.1f %.1f %.1f %.1f\n",
|
2015-11-19 23:05:11 -04:00
|
|
|
AP_HAL::millis() * 0.001f,
|
2013-12-29 18:47:50 -04:00
|
|
|
degrees(ekf_euler.x),
|
|
|
|
degrees(ekf_euler.y),
|
2013-12-31 17:54:56 -04:00
|
|
|
temp,
|
|
|
|
velNED.x,
|
|
|
|
velNED.y,
|
|
|
|
velNED.z,
|
|
|
|
posNED.x,
|
|
|
|
posNED.y,
|
|
|
|
posNED.z,
|
2014-01-30 18:27:56 -04:00
|
|
|
60*degrees(gyroBias.x),
|
|
|
|
60*degrees(gyroBias.y),
|
|
|
|
60*degrees(gyroBias.z),
|
|
|
|
windVel.x,
|
|
|
|
windVel.y,
|
2013-12-31 17:54:56 -04:00
|
|
|
magNED.x,
|
|
|
|
magNED.y,
|
|
|
|
magNED.z,
|
|
|
|
magXYZ.x,
|
|
|
|
magXYZ.y,
|
|
|
|
magXYZ.z,
|
2015-06-01 03:14:11 -03:00
|
|
|
logreader.get_attitude().x,
|
|
|
|
logreader.get_attitude().y,
|
|
|
|
logreader.get_attitude().z);
|
2014-01-30 18:27:56 -04:00
|
|
|
|
|
|
|
// define messages for EKF1 data packet
|
|
|
|
int16_t roll = (int16_t)(100*degrees(ekf_euler.x)); // roll angle (centi-deg)
|
|
|
|
int16_t pitch = (int16_t)(100*degrees(ekf_euler.y)); // pitch angle (centi-deg)
|
|
|
|
uint16_t yaw = (uint16_t)wrap_360_cd(100*degrees(ekf_euler.z)); // yaw angle (centi-deg)
|
|
|
|
float velN = (float)(velNED.x); // velocity North (m/s)
|
|
|
|
float velE = (float)(velNED.y); // velocity East (m/s)
|
|
|
|
float velD = (float)(velNED.z); // velocity Down (m/s)
|
|
|
|
float posN = (float)(posNED.x); // metres North
|
|
|
|
float posE = (float)(posNED.y); // metres East
|
|
|
|
float posD = (float)(posNED.z); // metres Down
|
2014-12-08 17:19:46 -04:00
|
|
|
float gyrX = (float)(6000*degrees(gyroBias.x)); // centi-deg/min
|
|
|
|
float gyrY = (float)(6000*degrees(gyroBias.y)); // centi-deg/min
|
|
|
|
float gyrZ = (float)(6000*degrees(gyroBias.z)); // centi-deg/min
|
2014-01-30 18:27:56 -04:00
|
|
|
|
|
|
|
// print EKF1 data packet
|
2014-12-08 17:19:46 -04:00
|
|
|
fprintf(ekf1f, "%.3f %u %d %d %u %.2f %.2f %.2f %.2f %.2f %.2f %.0f %.0f %.0f\n",
|
2015-11-19 23:05:11 -04:00
|
|
|
AP_HAL::millis() * 0.001f,
|
|
|
|
AP_HAL::millis(),
|
2014-01-30 18:27:56 -04:00
|
|
|
roll,
|
|
|
|
pitch,
|
|
|
|
yaw,
|
|
|
|
velN,
|
|
|
|
velE,
|
|
|
|
velD,
|
|
|
|
posN,
|
|
|
|
posE,
|
|
|
|
posD,
|
|
|
|
gyrX,
|
|
|
|
gyrY,
|
|
|
|
gyrZ);
|
|
|
|
|
|
|
|
// define messages for EKF2 data packet
|
2014-10-30 20:47:57 -03:00
|
|
|
int8_t accWeight = (int8_t)(100*accelWeighting);
|
|
|
|
int8_t acc1 = (int8_t)(100*accelZBias1);
|
|
|
|
int8_t acc2 = (int8_t)(100*accelZBias2);
|
2014-01-30 18:27:56 -04:00
|
|
|
int16_t windN = (int16_t)(100*windVel.x);
|
|
|
|
int16_t windE = (int16_t)(100*windVel.y);
|
|
|
|
int16_t magN = (int16_t)(magNED.x);
|
|
|
|
int16_t magE = (int16_t)(magNED.y);
|
|
|
|
int16_t magD = (int16_t)(magNED.z);
|
|
|
|
int16_t magX = (int16_t)(magXYZ.x);
|
|
|
|
int16_t magY = (int16_t)(magXYZ.y);
|
|
|
|
int16_t magZ = (int16_t)(magXYZ.z);
|
|
|
|
|
|
|
|
// print EKF2 data packet
|
|
|
|
fprintf(ekf2f, "%.3f %d %d %d %d %d %d %d %d %d %d %d %d\n",
|
2015-11-19 23:05:11 -04:00
|
|
|
AP_HAL::millis() * 0.001f,
|
|
|
|
AP_HAL::millis(),
|
2014-10-30 20:47:57 -03:00
|
|
|
accWeight,
|
|
|
|
acc1,
|
|
|
|
acc2,
|
2014-01-30 18:27:56 -04:00
|
|
|
windN,
|
|
|
|
windE,
|
|
|
|
magN,
|
|
|
|
magE,
|
|
|
|
magD,
|
|
|
|
magX,
|
|
|
|
magY,
|
|
|
|
magZ);
|
|
|
|
|
|
|
|
// define messages for EKF3 data packet
|
|
|
|
int16_t innovVN = (int16_t)(100*velInnov.x);
|
|
|
|
int16_t innovVE = (int16_t)(100*velInnov.y);
|
|
|
|
int16_t innovVD = (int16_t)(100*velInnov.z);
|
|
|
|
int16_t innovPN = (int16_t)(100*posInnov.x);
|
|
|
|
int16_t innovPE = (int16_t)(100*posInnov.y);
|
|
|
|
int16_t innovPD = (int16_t)(100*posInnov.z);
|
|
|
|
int16_t innovMX = (int16_t)(magInnov.x);
|
|
|
|
int16_t innovMY = (int16_t)(magInnov.y);
|
|
|
|
int16_t innovMZ = (int16_t)(magInnov.z);
|
|
|
|
int16_t innovVT = (int16_t)(100*tasInnov);
|
|
|
|
|
|
|
|
// print EKF3 data packet
|
|
|
|
fprintf(ekf3f, "%.3f %d %d %d %d %d %d %d %d %d %d %d\n",
|
2015-11-19 23:05:11 -04:00
|
|
|
AP_HAL::millis() * 0.001f,
|
|
|
|
AP_HAL::millis(),
|
2014-01-30 18:27:56 -04:00
|
|
|
innovVN,
|
|
|
|
innovVE,
|
|
|
|
innovVD,
|
|
|
|
innovPN,
|
|
|
|
innovPE,
|
|
|
|
innovPD,
|
|
|
|
innovMX,
|
|
|
|
innovMY,
|
|
|
|
innovMZ,
|
|
|
|
innovVT);
|
|
|
|
|
|
|
|
// define messages for EKF4 data packet
|
2014-05-01 04:54:31 -03:00
|
|
|
int16_t sqrtvarV = (int16_t)(constrain_float(100*velVar,INT16_MIN,INT16_MAX));
|
|
|
|
int16_t sqrtvarP = (int16_t)(constrain_float(100*posVar,INT16_MIN,INT16_MAX));
|
|
|
|
int16_t sqrtvarH = (int16_t)(constrain_float(100*hgtVar,INT16_MIN,INT16_MAX));
|
|
|
|
int16_t sqrtvarMX = (int16_t)(constrain_float(100*magVar.x,INT16_MIN,INT16_MAX));
|
|
|
|
int16_t sqrtvarMY = (int16_t)(constrain_float(100*magVar.y,INT16_MIN,INT16_MAX));
|
|
|
|
int16_t sqrtvarMZ = (int16_t)(constrain_float(100*magVar.z,INT16_MIN,INT16_MAX));
|
|
|
|
int16_t sqrtvarVT = (int16_t)(constrain_float(100*tasVar,INT16_MIN,INT16_MAX));
|
|
|
|
int16_t offsetNorth = (int8_t)(constrain_float(offset.x,INT16_MIN,INT16_MAX));
|
|
|
|
int16_t offsetEast = (int8_t)(constrain_float(offset.y,INT16_MIN,INT16_MAX));
|
2014-01-30 18:27:56 -04:00
|
|
|
|
|
|
|
// print EKF4 data packet
|
2015-01-02 20:53:10 -04:00
|
|
|
fprintf(ekf4f, "%.3f %u %d %d %d %d %d %d %d %d %d %d\n",
|
2015-11-19 23:05:11 -04:00
|
|
|
AP_HAL::millis() * 0.001f,
|
|
|
|
(unsigned)AP_HAL::millis(),
|
2015-01-02 20:53:10 -04:00
|
|
|
(int)sqrtvarV,
|
|
|
|
(int)sqrtvarP,
|
|
|
|
(int)sqrtvarH,
|
|
|
|
(int)sqrtvarMX,
|
|
|
|
(int)sqrtvarMY,
|
|
|
|
(int)sqrtvarMZ,
|
|
|
|
(int)sqrtvarVT,
|
|
|
|
(int)offsetNorth,
|
|
|
|
(int)offsetEast,
|
|
|
|
(int)faultStatus);
|
2013-12-29 07:56:30 -04:00
|
|
|
}
|
|
|
|
}
|
2015-06-26 02:12:17 -03:00
|
|
|
|
2015-07-04 02:14:14 -03:00
|
|
|
flush_dataflash();
|
|
|
|
|
2015-06-26 02:12:17 -03:00
|
|
|
if (check_solution) {
|
|
|
|
report_checks();
|
|
|
|
}
|
|
|
|
exit(0);
|
|
|
|
}
|
|
|
|
|
2015-07-04 02:14:14 -03:00
|
|
|
|
|
|
|
bool Replay::show_error(const char *text, float max_error, float tolerance)
|
|
|
|
{
|
|
|
|
bool failed = max_error > tolerance;
|
|
|
|
printf("%s:\t%.2f %c %.2f\n",
|
|
|
|
text,
|
|
|
|
max_error,
|
|
|
|
failed?'>':'<',
|
|
|
|
tolerance);
|
|
|
|
return failed;
|
|
|
|
}
|
|
|
|
|
2015-06-26 02:12:17 -03:00
|
|
|
/*
|
|
|
|
report results of --check
|
|
|
|
*/
|
|
|
|
void Replay::report_checks(void)
|
|
|
|
{
|
|
|
|
bool failed = false;
|
|
|
|
if (tolerance_euler < 0.01f) {
|
|
|
|
tolerance_euler = 0.01f;
|
|
|
|
}
|
2015-07-01 23:23:48 -03:00
|
|
|
FILE *f = fopen("replay_results.txt","a");
|
|
|
|
if (f != NULL) {
|
2015-07-02 01:27:10 -03:00
|
|
|
fprintf(f, "%s\t%.3f\t%.3f\t%.3f\t%.3f\t%.3f\n",
|
2015-07-01 23:23:48 -03:00
|
|
|
log_filename,
|
|
|
|
check_result.max_roll_error,
|
|
|
|
check_result.max_pitch_error,
|
|
|
|
check_result.max_yaw_error,
|
|
|
|
check_result.max_pos_error,
|
|
|
|
check_result.max_vel_error);
|
|
|
|
fclose(f);
|
|
|
|
}
|
2015-07-04 02:14:14 -03:00
|
|
|
failed |= show_error("Roll error", check_result.max_roll_error, tolerance_euler);
|
|
|
|
failed |= show_error("Pitch error", check_result.max_pitch_error, tolerance_euler);
|
|
|
|
failed |= show_error("Yaw error", check_result.max_yaw_error, tolerance_euler);
|
|
|
|
failed |= show_error("Position error", check_result.max_pos_error, tolerance_pos);
|
|
|
|
failed |= show_error("Velocity error", check_result.max_vel_error, tolerance_vel);
|
2015-06-26 02:12:17 -03:00
|
|
|
if (failed) {
|
|
|
|
printf("Checks failed\n");
|
|
|
|
exit(1);
|
|
|
|
} else {
|
|
|
|
printf("Checks passed\n");
|
|
|
|
}
|
2013-12-29 07:56:30 -04:00
|
|
|
}
|
|
|
|
|
2016-05-25 07:46:18 -03:00
|
|
|
/*
|
|
|
|
parse a parameter file line
|
|
|
|
*/
|
|
|
|
bool Replay::parse_param_line(char *line, char **vname, float &value)
|
|
|
|
{
|
|
|
|
if (line[0] == '#') {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
char *saveptr = NULL;
|
|
|
|
char *pname = strtok_r(line, ", =\t", &saveptr);
|
|
|
|
if (pname == NULL) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (strlen(pname) > AP_MAX_NAME_SIZE) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
const char *value_s = strtok_r(NULL, ", =\t", &saveptr);
|
|
|
|
if (value_s == NULL) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
value = atof(value_s);
|
|
|
|
*vname = pname;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
load a default set of parameters from a file
|
|
|
|
*/
|
|
|
|
void Replay::load_param_file(const char *pfilename)
|
|
|
|
{
|
|
|
|
FILE *f = fopen(pfilename, "r");
|
|
|
|
if (f == NULL) {
|
|
|
|
printf("Failed to open parameter file: %s\n", pfilename);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
char line[100];
|
|
|
|
|
|
|
|
while (fgets(line, sizeof(line)-1, f)) {
|
|
|
|
char *pname;
|
|
|
|
float value;
|
|
|
|
if (!parse_param_line(line, &pname, value)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
struct user_parameter *u = new user_parameter;
|
|
|
|
strncpy(u->name, pname, sizeof(u->name));
|
|
|
|
u->value = value;
|
|
|
|
u->next = user_parameters;
|
|
|
|
user_parameters = u;
|
|
|
|
}
|
|
|
|
fclose(f);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
see if a user parameter is set
|
|
|
|
*/
|
|
|
|
bool Replay::check_user_param(const char *name)
|
|
|
|
{
|
|
|
|
for (struct user_parameter *u=user_parameters; u; u=u->next) {
|
|
|
|
if (strcmp(name, u->name) == 0) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-10-19 16:59:30 -03:00
|
|
|
AP_HAL_MAIN_CALLBACKS(&replay);
|