mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-22 00:28:30 -04:00
desktop: fixed calculation of angular velocities
thanks to James Goppert for the frame conversion maths
This commit is contained in:
parent
65699a2f2f
commit
706c6c38fa
@ -2,6 +2,7 @@
|
|||||||
#define _DESKTOP_H
|
#define _DESKTOP_H
|
||||||
|
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
|
#include <sys/time.h>
|
||||||
|
|
||||||
struct desktop_info {
|
struct desktop_info {
|
||||||
bool slider; // slider switch state, True means CLI mode
|
bool slider; // slider switch state, True means CLI mode
|
||||||
@ -21,7 +22,10 @@ ssize_t sitl_gps_read(int fd, void *buf, size_t count);
|
|||||||
void sitl_update_compass(float heading, float roll, float pitch, float yaw);
|
void sitl_update_compass(float heading, float roll, float pitch, float yaw);
|
||||||
void sitl_update_gps(float latitude, float longitude, float altitude,
|
void sitl_update_gps(float latitude, float longitude, float altitude,
|
||||||
float speedN, float speedE, bool have_lock);
|
float speedN, float speedE, bool have_lock);
|
||||||
void sitl_update_adc(float roll, float pitch, float yaw, float airspeed);
|
void sitl_update_adc(float roll, float pitch, float yaw,
|
||||||
|
float rollRate, float pitchRate, float yawRate,
|
||||||
|
float xAccel, float yAccel, float zAccel,
|
||||||
|
float airspeed);
|
||||||
void sitl_setup_adc(void);
|
void sitl_setup_adc(void);
|
||||||
void sitl_update_barometer(float altitude);
|
void sitl_update_barometer(float altitude);
|
||||||
|
|
||||||
|
@ -48,6 +48,8 @@ static volatile struct {
|
|||||||
double heading; // degrees
|
double heading; // degrees
|
||||||
double speedN, speedE; // m/s
|
double speedN, speedE; // m/s
|
||||||
double roll, pitch, yaw; // degrees
|
double roll, pitch, yaw; // degrees
|
||||||
|
double rollRate, pitchRate, yawRate; // degrees per second
|
||||||
|
double xAccel, yAccel, zAccel; // meters/s/s
|
||||||
double airspeed; // m/s
|
double airspeed; // m/s
|
||||||
uint32_t update_count;
|
uint32_t update_count;
|
||||||
} sim_state;
|
} sim_state;
|
||||||
@ -140,6 +142,12 @@ static void sitl_fgear_input(void)
|
|||||||
sim_state.roll = d.fg_pkt.rollDeg;
|
sim_state.roll = d.fg_pkt.rollDeg;
|
||||||
sim_state.pitch = d.fg_pkt.pitchDeg;
|
sim_state.pitch = d.fg_pkt.pitchDeg;
|
||||||
sim_state.yaw = d.fg_pkt.yawDeg;
|
sim_state.yaw = d.fg_pkt.yawDeg;
|
||||||
|
sim_state.rollRate = d.fg_pkt.rollRate;
|
||||||
|
sim_state.pitchRate = d.fg_pkt.pitchRate;
|
||||||
|
sim_state.yawRate = d.fg_pkt.yawRate;
|
||||||
|
sim_state.xAccel = ft2m(d.fg_pkt.xAccel);
|
||||||
|
sim_state.yAccel = ft2m(d.fg_pkt.yAccel);
|
||||||
|
sim_state.zAccel = ft2m(d.fg_pkt.zAccel);
|
||||||
sim_state.heading = d.fg_pkt.heading;
|
sim_state.heading = d.fg_pkt.heading;
|
||||||
sim_state.airspeed = kt2mps(d.fg_pkt.airspeed);
|
sim_state.airspeed = kt2mps(d.fg_pkt.airspeed);
|
||||||
sim_state.update_count++;
|
sim_state.update_count++;
|
||||||
@ -287,7 +295,10 @@ static void timer_handler(int signum)
|
|||||||
sitl_update_gps(sim_state.latitude, sim_state.longitude,
|
sitl_update_gps(sim_state.latitude, sim_state.longitude,
|
||||||
sim_state.altitude,
|
sim_state.altitude,
|
||||||
sim_state.speedN, sim_state.speedE, true);
|
sim_state.speedN, sim_state.speedE, true);
|
||||||
sitl_update_adc(sim_state.roll, sim_state.pitch, sim_state.heading, sim_state.airspeed);
|
sitl_update_adc(sim_state.roll, sim_state.pitch, sim_state.yaw,
|
||||||
|
sim_state.rollRate, sim_state.pitchRate, sim_state.yawRate,
|
||||||
|
sim_state.xAccel, sim_state.yAccel, sim_state.zAccel,
|
||||||
|
sim_state.airspeed);
|
||||||
sitl_update_barometer(sim_state.altitude);
|
sitl_update_barometer(sim_state.altitude);
|
||||||
sitl_update_compass(sim_state.heading, sim_state.roll, sim_state.pitch, sim_state.heading);
|
sitl_update_compass(sim_state.heading, sim_state.roll, sim_state.pitch, sim_state.heading);
|
||||||
}
|
}
|
||||||
@ -333,7 +344,7 @@ void sitl_setup(void)
|
|||||||
|
|
||||||
// setup some initial values
|
// setup some initial values
|
||||||
sitl_update_barometer(desktop_state.initial_height);
|
sitl_update_barometer(desktop_state.initial_height);
|
||||||
sitl_update_adc(0, 0, 0, 0);
|
sitl_update_adc(0, 0, 0, 0, 0, 0, 0, 0, -9.8, 0);
|
||||||
sitl_update_compass(0, 0, 0, 0);
|
sitl_update_compass(0, 0, 0, 0);
|
||||||
sitl_update_gps(0, 0, 0, 0, 0, false);
|
sitl_update_gps(0, 0, 0, 0, 0, false);
|
||||||
}
|
}
|
||||||
|
@ -41,64 +41,41 @@ static uint16_t airspeed_sensor(float airspeed)
|
|||||||
|
|
||||||
inputs are in degrees
|
inputs are in degrees
|
||||||
*/
|
*/
|
||||||
void sitl_update_adc(float roll, float pitch, float yaw, float airspeed)
|
void sitl_update_adc(float roll, float pitch, float yaw,
|
||||||
|
float rollRate, float pitchRate, float yawRate,
|
||||||
|
float xAccel, float yAccel, float zAccel,
|
||||||
|
float airspeed)
|
||||||
{
|
{
|
||||||
static const uint8_t sensor_map[6] = { 1, 2, 0, 4, 5, 6 };
|
static const uint8_t sensor_map[6] = { 1, 2, 0, 4, 5, 6 };
|
||||||
static const float _sensor_signs[6] = { 1, -1, -1, 1, -1, -1 };
|
static const float _sensor_signs[6] = { 1, -1, -1, 1, -1, -1 };
|
||||||
const float accel_offset = 2041;
|
const float accel_offset = 2041;
|
||||||
const float gyro_offset = 1658;
|
const float gyro_offset = 1658;
|
||||||
#define ToRad(x) (x*0.01745329252) // *pi/180
|
#define ToRad(x) (x*0.01745329252) // *pi/180
|
||||||
const float _gyro_gain_x = 0.4;
|
const float _gyro_gain_x = ToRad(0.4);
|
||||||
const float _gyro_gain_y = 0.41;
|
const float _gyro_gain_y = ToRad(0.41);
|
||||||
const float _gyro_gain_z = 0.41;
|
const float _gyro_gain_z = ToRad(0.41);
|
||||||
const float _accel_scale = 9.80665 / 423.8;
|
const float _accel_scale = 9.80665 / 423.8;
|
||||||
float adc[7];
|
float adc[7];
|
||||||
float xAccel, yAccel, zAccel, scale;
|
float phi, theta, phiDot, thetaDot, psiDot;
|
||||||
float rollRate, pitchRate, yawRate;
|
float p, q, r;
|
||||||
static uint32_t last_update;
|
|
||||||
static float last_roll, last_pitch, last_yaw;
|
|
||||||
unsigned long delta_t;
|
|
||||||
|
|
||||||
// 200Hz max
|
/* convert the angular velocities from earth frame to
|
||||||
if (micros() - last_update < 5000) {
|
body frame. Thanks to James Goppert for the formula
|
||||||
return;
|
*/
|
||||||
}
|
phi = ToRad(roll);
|
||||||
|
theta = ToRad(pitch);
|
||||||
|
phiDot = ToRad(rollRate);
|
||||||
|
thetaDot = ToRad(pitchRate);
|
||||||
|
psiDot = ToRad(yawRate);
|
||||||
|
|
||||||
/* map roll/pitch/yaw to values the accelerometer would see */
|
p = phiDot - psiDot*sin(theta);
|
||||||
xAccel = sin(ToRad(pitch)) * cos(ToRad(roll));
|
q = cos(phi)*thetaDot + sin(phi)*psiDot*cos(theta);
|
||||||
yAccel = -sin(ToRad(roll)) * cos(ToRad(pitch));
|
r = cos(phi)*psiDot*cos(theta) - sin(phi)*thetaDot;
|
||||||
zAccel = -cos(ToRad(roll)) * cos(ToRad(pitch));
|
|
||||||
scale = 9.81 / sqrt((xAccel*xAccel)+(yAccel*yAccel)+(zAccel*zAccel));
|
|
||||||
xAccel *= scale;
|
|
||||||
yAccel *= scale;
|
|
||||||
zAccel *= scale;
|
|
||||||
|
|
||||||
/* map roll/pitch/yaw to values the gyro would see */
|
|
||||||
if (last_update == 0) {
|
|
||||||
rollRate = 0;
|
|
||||||
pitchRate = 0;
|
|
||||||
yawRate = 0;
|
|
||||||
delta_t = micros();
|
|
||||||
} else {
|
|
||||||
delta_t = micros() - last_update;
|
|
||||||
float rollChange, pitchChange, yawChange;
|
|
||||||
rollChange = normalise180(roll - last_roll);
|
|
||||||
pitchChange = normalise180(pitch - last_pitch);
|
|
||||||
yawChange = normalise180(yaw - last_yaw);
|
|
||||||
rollRate = 1.0e6 * rollChange / delta_t;
|
|
||||||
pitchRate = 1.0e6 * pitchChange / delta_t;
|
|
||||||
yawRate = 1.0e6 * yawChange / delta_t;
|
|
||||||
}
|
|
||||||
last_update += delta_t;
|
|
||||||
|
|
||||||
last_roll = roll;
|
|
||||||
last_pitch = pitch;
|
|
||||||
last_yaw = yaw;
|
|
||||||
|
|
||||||
/* work out the ADC channel values */
|
/* work out the ADC channel values */
|
||||||
adc[0] = (rollRate / (_gyro_gain_x * _sensor_signs[0])) + gyro_offset;
|
adc[0] = (p / (_gyro_gain_x * _sensor_signs[0])) + gyro_offset;
|
||||||
adc[1] = (pitchRate / (_gyro_gain_y * _sensor_signs[1])) + gyro_offset;
|
adc[1] = (q / (_gyro_gain_y * _sensor_signs[1])) + gyro_offset;
|
||||||
adc[2] = (yawRate / (_gyro_gain_z * _sensor_signs[2])) + gyro_offset;
|
adc[2] = (r / (_gyro_gain_z * _sensor_signs[2])) + gyro_offset;
|
||||||
|
|
||||||
adc[3] = (xAccel / (_accel_scale * _sensor_signs[3])) + accel_offset;
|
adc[3] = (xAccel / (_accel_scale * _sensor_signs[3])) + accel_offset;
|
||||||
adc[4] = (yAccel / (_accel_scale * _sensor_signs[4])) + accel_offset;
|
adc[4] = (yAccel / (_accel_scale * _sensor_signs[4])) + accel_offset;
|
||||||
@ -123,14 +100,22 @@ void sitl_update_adc(float roll, float pitch, float yaw, float airspeed)
|
|||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#if 0
|
static uint32_t last_report;
|
||||||
|
uint32_t tnow = millis();
|
||||||
extern AP_DCM dcm;
|
extern AP_DCM dcm;
|
||||||
Vector3f omega = dcm.get_gyro();
|
Vector3f omega = dcm.get_gyro();
|
||||||
printf("dt=%5u adc[2]=%6.1f roll=%6.1f / %6.1f yaw=%6.1f / %6.1f yawRate=%6.3f / %6.3f\n",
|
// report roll/pitch discrepancies
|
||||||
(unsigned)delta_t,
|
if (tnow - last_report > 5000 ||
|
||||||
adc[2],
|
(tnow - last_report > 1000 &&
|
||||||
roll, dcm.roll_sensor/100.0, yaw, dcm.yaw_sensor/100.0, yawRate, ToDeg(omega.z));
|
(fabs(roll - dcm.roll_sensor/100.0) > 5.0 ||
|
||||||
#endif
|
fabs(pitch - dcm.pitch_sensor/100.0) > 5.0))) {
|
||||||
|
last_report = tnow;
|
||||||
|
printf("roll=%5.1f / %5.1f pitch=%5.1f / %5.1f rr=%5.2f / %5.2f pr=%5.2f / %5.2f\n",
|
||||||
|
roll, dcm.roll_sensor/100.0,
|
||||||
|
pitch, dcm.pitch_sensor/100.0,
|
||||||
|
rollRate, ToDeg(omega.x),
|
||||||
|
pitchRate, ToDeg(omega.y));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user