2013-08-29 02:34:34 -03:00
|
|
|
/*
|
|
|
|
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/>.
|
|
|
|
*/
|
|
|
|
|
2012-06-29 02:06:28 -03:00
|
|
|
/*
|
2015-05-04 22:49:54 -03:00
|
|
|
SITL.cpp - software in the loop state
|
2012-06-29 02:06:28 -03:00
|
|
|
*/
|
|
|
|
|
2015-10-22 10:58:33 -03:00
|
|
|
#include "SITL.h"
|
|
|
|
|
2015-08-11 03:28:46 -03:00
|
|
|
#include <AP_Common/AP_Common.h>
|
|
|
|
#include <AP_HAL/AP_HAL.h>
|
|
|
|
#include <GCS_MAVLink/GCS_MAVLink.h>
|
2015-11-16 00:10:29 -04:00
|
|
|
#include <DataFlash/DataFlash.h>
|
2012-06-29 02:06:28 -03:00
|
|
|
|
2014-01-03 01:01:18 -04:00
|
|
|
extern const AP_HAL::HAL& hal;
|
|
|
|
|
2015-10-22 10:04:42 -03:00
|
|
|
namespace SITL {
|
|
|
|
|
2012-06-29 02:06:28 -03:00
|
|
|
// table of user settable parameters
|
2015-10-25 14:03:46 -03:00
|
|
|
const AP_Param::GroupInfo SITL::var_info[] = {
|
2015-05-08 18:22:37 -03:00
|
|
|
AP_GROUPINFO("BARO_RND", 0, SITL, baro_noise, 0.2f),
|
|
|
|
AP_GROUPINFO("GYR_RND", 1, SITL, gyro_noise, 0),
|
|
|
|
AP_GROUPINFO("ACC_RND", 2, SITL, accel_noise, 0),
|
|
|
|
AP_GROUPINFO("MAG_RND", 3, SITL, mag_noise, 0),
|
2012-08-06 22:02:14 -03:00
|
|
|
AP_GROUPINFO("GPS_DISABLE",4, SITL, gps_disable, 0),
|
2015-05-08 18:22:37 -03:00
|
|
|
AP_GROUPINFO("DRIFT_SPEED",5, SITL, drift_speed, 0.05f),
|
2012-08-06 22:02:14 -03:00
|
|
|
AP_GROUPINFO("DRIFT_TIME", 6, SITL, drift_time, 5),
|
2015-04-13 20:57:17 -03:00
|
|
|
AP_GROUPINFO("GPS_DELAY", 7, SITL, gps_delay, 1),
|
2012-08-17 01:21:43 -03:00
|
|
|
AP_GROUPINFO("ENGINE_MUL", 8, SITL, engine_mul, 1),
|
2013-04-16 22:39:23 -03:00
|
|
|
AP_GROUPINFO("WIND_SPD", 9, SITL, wind_speed, 0),
|
2012-08-24 08:22:20 -03:00
|
|
|
AP_GROUPINFO("WIND_DIR", 10, SITL, wind_direction, 180),
|
2015-06-28 22:25:59 -03:00
|
|
|
AP_GROUPINFO("WIND_TURB", 11, SITL, wind_turbulance, 0),
|
2013-02-16 05:15:57 -04:00
|
|
|
AP_GROUPINFO("GPS_TYPE", 12, SITL, gps_type, SITL::GPS_TYPE_UBLOX),
|
2013-02-16 07:00:16 -04:00
|
|
|
AP_GROUPINFO("GPS_BYTELOSS", 13, SITL, gps_byteloss, 0),
|
2013-05-06 21:38:36 -03:00
|
|
|
AP_GROUPINFO("GPS_NUMSATS", 14, SITL, gps_numsats, 10),
|
2013-05-27 00:37:18 -03:00
|
|
|
AP_GROUPINFO("MAG_ERROR", 15, SITL, mag_error, 0),
|
2016-10-25 05:54:56 -03:00
|
|
|
AP_GROUPINFO("SERVO_SPEED", 16, SITL, servo_speed, 0.14),
|
2013-09-19 03:51:03 -03:00
|
|
|
AP_GROUPINFO("GPS_GLITCH", 17, SITL, gps_glitch, 0),
|
2013-10-01 03:39:16 -03:00
|
|
|
AP_GROUPINFO("GPS_HZ", 18, SITL, gps_hertz, 5),
|
2015-04-24 02:01:28 -03:00
|
|
|
AP_GROUPINFO("BATT_VOLTAGE", 19, SITL, batt_voltage, 12.6f),
|
2016-05-10 16:42:08 -03:00
|
|
|
AP_GROUPINFO("ARSPD_RND", 20, SITL, arspd_noise, 0.5f),
|
2013-11-06 21:49:08 -04:00
|
|
|
AP_GROUPINFO("ACCEL_FAIL", 21, SITL, accel_fail, 0),
|
2013-11-24 05:16:46 -04:00
|
|
|
AP_GROUPINFO("BARO_DRIFT", 22, SITL, baro_drift, 0),
|
2013-11-28 06:43:25 -04:00
|
|
|
AP_GROUPINFO("SONAR_GLITCH", 23, SITL, sonar_glitch, 0),
|
|
|
|
AP_GROUPINFO("SONAR_RND", 24, SITL, sonar_noise, 0),
|
2013-12-19 18:40:31 -04:00
|
|
|
AP_GROUPINFO("RC_FAIL", 25, SITL, rc_fail, 0),
|
2014-03-02 16:07:09 -04:00
|
|
|
AP_GROUPINFO("GPS2_ENABLE", 26, SITL, gps2_enable, 0),
|
2014-04-11 03:47:26 -03:00
|
|
|
AP_GROUPINFO("BARO_DISABLE", 27, SITL, baro_disable, 0),
|
2014-04-21 02:24:45 -03:00
|
|
|
AP_GROUPINFO("FLOAT_EXCEPT", 28, SITL, float_exception, 1),
|
2014-05-12 06:44:15 -03:00
|
|
|
AP_GROUPINFO("MAG_MOT", 29, SITL, mag_mot, 0),
|
2014-07-28 05:40:17 -03:00
|
|
|
AP_GROUPINFO("ACC_BIAS", 30, SITL, accel_bias, 0),
|
|
|
|
AP_GROUPINFO("BARO_GLITCH", 31, SITL, baro_glitch, 0),
|
2014-08-10 09:36:38 -03:00
|
|
|
AP_GROUPINFO("SONAR_SCALE", 32, SITL, sonar_scale, 12.1212f),
|
2015-01-02 06:49:56 -04:00
|
|
|
AP_GROUPINFO("FLOW_ENABLE", 33, SITL, flow_enable, 0),
|
2015-01-03 06:47:28 -04:00
|
|
|
AP_GROUPINFO("TERRAIN", 34, SITL, terrain_enable, 1),
|
2015-01-05 23:02:43 -04:00
|
|
|
AP_GROUPINFO("FLOW_RATE", 35, SITL, flow_rate, 10),
|
2015-01-05 23:18:44 -04:00
|
|
|
AP_GROUPINFO("FLOW_DELAY", 36, SITL, flow_delay, 0),
|
2015-02-20 17:32:49 -04:00
|
|
|
AP_GROUPINFO("GPS_DRIFTALT", 37, SITL, gps_drift_alt, 0),
|
2015-04-13 03:22:57 -03:00
|
|
|
AP_GROUPINFO("BARO_DELAY", 38, SITL, baro_delay, 0),
|
|
|
|
AP_GROUPINFO("MAG_DELAY", 39, SITL, mag_delay, 0),
|
|
|
|
AP_GROUPINFO("WIND_DELAY", 40, SITL, wind_delay, 0),
|
2015-04-20 19:29:27 -03:00
|
|
|
AP_GROUPINFO("MAG_OFS", 41, SITL, mag_ofs, 0),
|
2015-07-12 03:26:43 -03:00
|
|
|
AP_GROUPINFO("ACC2_RND", 42, SITL, accel2_noise, 0),
|
2016-05-10 16:42:08 -03:00
|
|
|
AP_GROUPINFO("ARSPD_FAIL", 43, SITL, arspd_fail, 0),
|
2016-01-19 00:28:53 -04:00
|
|
|
AP_GROUPINFO("GYR_SCALE", 44, SITL, gyro_scale, 0),
|
2016-06-15 21:20:01 -03:00
|
|
|
AP_GROUPINFO("ADSB_COUNT", 45, SITL, adsb_plane_count, -1),
|
2016-07-21 04:02:13 -03:00
|
|
|
AP_GROUPINFO("ADSB_RADIUS", 46, SITL, adsb_radius_m, 10000),
|
2016-06-15 00:47:19 -03:00
|
|
|
AP_GROUPINFO("ADSB_ALT", 47, SITL, adsb_altitude_m, 1000),
|
2016-06-17 00:46:12 -03:00
|
|
|
AP_GROUPINFO("MAG_ALY", 48, SITL, mag_anomaly_ned, 0),
|
|
|
|
AP_GROUPINFO("MAG_ALY_HGT", 49, SITL, mag_anomaly_hgt, 1.0f),
|
2016-07-21 21:57:45 -03:00
|
|
|
AP_GROUPINFO("PIN_MASK", 50, SITL, pin_mask, 0),
|
2016-08-16 18:16:03 -03:00
|
|
|
AP_GROUPINFO("ADSB_TX", 51, SITL, adsb_tx, 0),
|
2016-09-18 18:45:24 -03:00
|
|
|
AP_GROUPINFO("SPEEDUP", 52, SITL, speedup, -1),
|
2016-10-14 20:42:45 -03:00
|
|
|
AP_GROUPINFO("IMU_POS", 53, SITL, imu_pos_offset, 0),
|
|
|
|
AP_GROUPINFO("GPS_POS", 54, SITL, gps_pos_offset, 0),
|
|
|
|
AP_GROUPINFO("SONAR_POS", 55, SITL, rngfnd_pos_offset, 0),
|
|
|
|
AP_GROUPINFO("FLOW_POS", 56, SITL, optflow_pos_offset, 0),
|
2016-12-18 00:53:40 -04:00
|
|
|
AP_GROUPINFO("ACC2_BIAS", 57, SITL, accel2_bias, 0),
|
2017-01-30 15:04:10 -04:00
|
|
|
AP_GROUPINFO("GPS_NOISE", 58, SITL, gps_noise, 0),
|
2017-03-03 22:39:04 -04:00
|
|
|
AP_GROUPINFO("GP2_GLITCH", 59, SITL, gps2_glitch, 0),
|
2017-03-20 17:32:54 -03:00
|
|
|
AP_GROUPINFO("ENGINE_FAIL", 60, SITL, engine_fail, 0),
|
2012-06-29 02:06:28 -03:00
|
|
|
AP_GROUPEND
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/* report SITL state via MAVLink */
|
|
|
|
void SITL::simstate_send(mavlink_channel_t chan)
|
|
|
|
{
|
2015-05-04 22:49:54 -03:00
|
|
|
float yaw;
|
2012-06-29 02:06:28 -03:00
|
|
|
|
2015-05-04 22:49:54 -03:00
|
|
|
// convert to same conventions as DCM
|
|
|
|
yaw = state.yawDeg;
|
|
|
|
if (yaw > 180) {
|
|
|
|
yaw -= 360;
|
|
|
|
}
|
2012-06-29 02:06:28 -03:00
|
|
|
|
|
|
|
mavlink_msg_simstate_send(chan,
|
|
|
|
ToRad(state.rollDeg),
|
|
|
|
ToRad(state.pitchDeg),
|
|
|
|
ToRad(yaw),
|
|
|
|
state.xAccel,
|
|
|
|
state.yAccel,
|
|
|
|
state.zAccel,
|
2015-10-22 10:15:20 -03:00
|
|
|
radians(state.rollRate),
|
|
|
|
radians(state.pitchRate),
|
|
|
|
radians(state.yawRate),
|
2013-08-14 01:27:39 -03:00
|
|
|
state.latitude*1.0e7,
|
|
|
|
state.longitude*1.0e7);
|
2012-06-29 02:06:28 -03:00
|
|
|
}
|
|
|
|
|
2014-01-03 01:01:18 -04:00
|
|
|
/* report SITL state to DataFlash */
|
2015-11-16 00:10:29 -04:00
|
|
|
void SITL::Log_Write_SIMSTATE(DataFlash_Class *DataFlash)
|
2014-01-03 01:01:18 -04:00
|
|
|
{
|
2015-05-04 22:49:54 -03:00
|
|
|
float yaw;
|
2014-01-03 01:01:18 -04:00
|
|
|
|
2015-05-04 22:49:54 -03:00
|
|
|
// convert to same conventions as DCM
|
|
|
|
yaw = state.yawDeg;
|
|
|
|
if (yaw > 180) {
|
|
|
|
yaw -= 360;
|
|
|
|
}
|
2014-01-03 01:01:18 -04:00
|
|
|
|
|
|
|
struct log_AHRS pkt = {
|
|
|
|
LOG_PACKET_HEADER_INIT(LOG_SIMSTATE_MSG),
|
2015-11-19 23:15:52 -04:00
|
|
|
time_us : AP_HAL::micros64(),
|
2015-05-04 22:49:54 -03:00
|
|
|
roll : (int16_t)(state.rollDeg*100),
|
|
|
|
pitch : (int16_t)(state.pitchDeg*100),
|
|
|
|
yaw : (uint16_t)(wrap_360_cd(yaw*100)),
|
|
|
|
alt : (float)state.altitude,
|
|
|
|
lat : (int32_t)(state.latitude*1.0e7),
|
2017-04-15 08:20:28 -03:00
|
|
|
lng : (int32_t)(state.longitude*1.0e7),
|
|
|
|
q1 : state.quaternion.q1,
|
|
|
|
q2 : state.quaternion.q2,
|
|
|
|
q3 : state.quaternion.q3,
|
|
|
|
q4 : state.quaternion.q4,
|
2014-01-03 01:01:18 -04:00
|
|
|
};
|
2015-11-16 00:10:29 -04:00
|
|
|
DataFlash->WriteBlock(&pkt, sizeof(pkt));
|
2014-01-03 01:01:18 -04:00
|
|
|
}
|
|
|
|
|
2015-01-03 00:44:19 -04:00
|
|
|
/*
|
|
|
|
convert a set of roll rates from earth frame to body frame
|
|
|
|
output values are in radians/second
|
|
|
|
*/
|
2012-06-29 02:06:28 -03:00
|
|
|
void SITL::convert_body_frame(double rollDeg, double pitchDeg,
|
|
|
|
double rollRate, double pitchRate, double yawRate,
|
|
|
|
double *p, double *q, double *r)
|
|
|
|
{
|
2015-05-04 22:49:54 -03:00
|
|
|
double phi, theta, phiDot, thetaDot, psiDot;
|
2012-06-29 02:06:28 -03:00
|
|
|
|
2015-05-04 22:49:54 -03:00
|
|
|
phi = ToRad(rollDeg);
|
|
|
|
theta = ToRad(pitchDeg);
|
|
|
|
phiDot = ToRad(rollRate);
|
|
|
|
thetaDot = ToRad(pitchRate);
|
|
|
|
psiDot = ToRad(yawRate);
|
2012-06-29 02:06:28 -03:00
|
|
|
|
2015-05-30 09:51:38 -03:00
|
|
|
*p = phiDot - psiDot*sin(theta);
|
|
|
|
*q = cos(phi)*thetaDot + sin(phi)*psiDot*cos(theta);
|
|
|
|
*r = cos(phi)*psiDot*cos(theta) - sin(phi)*thetaDot;
|
2012-06-29 02:06:28 -03:00
|
|
|
}
|
|
|
|
|
2015-05-02 07:29:16 -03:00
|
|
|
|
|
|
|
/*
|
|
|
|
convert angular velocities from body frame to
|
|
|
|
earth frame.
|
2015-05-04 22:49:54 -03:00
|
|
|
|
2015-05-02 07:29:16 -03:00
|
|
|
all inputs and outputs are in radians/s
|
|
|
|
*/
|
|
|
|
Vector3f SITL::convert_earth_frame(const Matrix3f &dcm, const Vector3f &gyro)
|
|
|
|
{
|
|
|
|
float p = gyro.x;
|
|
|
|
float q = gyro.y;
|
|
|
|
float r = gyro.z;
|
|
|
|
|
|
|
|
float phi, theta, psi;
|
|
|
|
dcm.to_euler(&phi, &theta, &psi);
|
|
|
|
|
|
|
|
float phiDot = p + tanf(theta)*(q*sinf(phi) + r*cosf(phi));
|
|
|
|
float thetaDot = q*cosf(phi) - r*sinf(phi);
|
|
|
|
if (fabsf(cosf(theta)) < 1.0e-20f) {
|
|
|
|
theta += 1.0e-10f;
|
|
|
|
}
|
|
|
|
float psiDot = (q*sinf(phi) + r*cosf(phi))/cosf(theta);
|
|
|
|
return Vector3f(phiDot, thetaDot, psiDot);
|
|
|
|
}
|
|
|
|
|
2015-10-22 10:04:42 -03:00
|
|
|
} // namespace SITL
|