2012-04-30 04:17:14 -03:00
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*-
2013-10-04 00:33:47 -03:00
// default sensors are present and healthy: gyro, accelerometer, rate_control, attitude_stabilization, yaw_position, altitude control, x/y position control, motor_control
#define MAVLINK_SENSOR_PRESENT_DEFAULT (MAV_SYS_STATUS_SENSOR_3D_GYRO | MAV_SYS_STATUS_SENSOR_3D_ACCEL | MAV_SYS_STATUS_SENSOR_ANGULAR_RATE_CONTROL | MAV_SYS_STATUS_SENSOR_ATTITUDE_STABILIZATION | MAV_SYS_STATUS_SENSOR_YAW_POSITION | MAV_SYS_STATUS_SENSOR_XY_POSITION_CONTROL | MAV_SYS_STATUS_SENSOR_MOTOR_OUTPUTS)
2012-04-30 04:17:14 -03:00
// use this to prevent recursion during sensor init
static bool in_mavlink_delay;
// true when we have received at least 1 MAVLink packet
static bool mavlink_active;
2013-10-27 20:33:52 -03:00
// true if we are out of time in our event timeslice
static bool gcs_out_of_time;
2012-04-30 04:17:14 -03:00
// check if a message will fit in the payload space available
#define CHECK_PAYLOAD_SIZE(id) if (payload_space < MAVLINK_MSG_ID_## id ##_LEN) return false
2012-12-18 15:30:42 -04:00
/*
* !!NOTE!!
*
* the use of NOINLINE separate functions for each message type avoids
* a compiler bug in gcc that would cause it to use far more stack
* space than is needed. Without the NOINLINE we use the sum of the
* stack needed for each message type. Please be careful to follow the
* pattern below when adding any new messages
2012-04-30 04:17:14 -03:00
*/
static NOINLINE void send_heartbeat(mavlink_channel_t chan)
{
uint8_t base_mode = MAV_MODE_FLAG_CUSTOM_MODE_ENABLED;
uint8_t system_status = MAV_STATE_ACTIVE;
uint32_t custom_mode = control_mode;
2013-02-08 22:11:43 -04:00
2013-03-28 20:25:53 -03:00
if (failsafe.triggered != 0) {
2013-02-08 22:11:43 -04:00
system_status = MAV_STATE_CRITICAL;
}
2012-04-30 04:17:14 -03:00
// work out the base_mode. This value is not very useful
// for APM, but we calculate it as best we can so a generic
// MAVLink enabled ground station can work out something about
// what the MAV is up to. The actual bit values are highly
// ambiguous for most of the APM flight modes. In practice, you
// only get useful information from the custom_mode, which maps to
// the APM flight mode and has a well defined meaning in the
// ArduPlane documentation
switch (control_mode) {
case MANUAL:
2012-05-14 16:21:29 -03:00
case LEARNING:
2013-03-01 07:32:57 -04:00
case STEERING:
2012-06-10 06:34:11 -03:00
base_mode = MAV_MODE_FLAG_MANUAL_INPUT_ENABLED;
2012-04-30 04:17:14 -03:00
break;
case AUTO:
case RTL:
case GUIDED:
2012-06-10 06:34:11 -03:00
base_mode = MAV_MODE_FLAG_GUIDED_ENABLED;
2012-04-30 04:17:14 -03:00
// note that MAV_MODE_FLAG_AUTO_ENABLED does not match what
// APM does in any mode, as that is defined as "system finds its own goal
// positions", which APM does not currently do
break;
case INITIALISING:
system_status = MAV_STATE_CALIBRATING;
break;
2013-03-28 18:53:20 -03:00
case HOLD:
system_status = 0;
break;
2012-04-30 04:17:14 -03:00
}
#if ENABLE_STICK_MIXING==ENABLED
if (control_mode != INITIALISING) {
// all modes except INITIALISING have some form of manual
// override if stick mixing is enabled
base_mode |= MAV_MODE_FLAG_MANUAL_INPUT_ENABLED;
}
#endif
#if HIL_MODE != HIL_MODE_DISABLED
base_mode |= MAV_MODE_FLAG_HIL_ENABLED;
#endif
// we are armed if we are not initialising
2014-02-18 19:54:04 -04:00
if (control_mode != INITIALISING && ahrs.get_armed()) {
2012-04-30 04:17:14 -03:00
base_mode |= MAV_MODE_FLAG_SAFETY_ARMED;
}
// indicate we have set a custom mode
base_mode |= MAV_MODE_FLAG_CUSTOM_MODE_ENABLED;
mavlink_msg_heartbeat_send(
chan,
2013-03-28 18:16:29 -03:00
MAV_TYPE_GROUND_ROVER,
2012-04-30 04:17:14 -03:00
MAV_AUTOPILOT_ARDUPILOTMEGA,
base_mode,
custom_mode,
system_status);
}
static NOINLINE void send_attitude(mavlink_channel_t chan)
{
Vector3f omega = ahrs.get_gyro();
mavlink_msg_attitude_send(
chan,
2012-12-18 15:30:42 -04:00
millis(),
2012-04-30 04:17:14 -03:00
ahrs.roll,
2012-11-17 02:45:20 -04:00
ahrs.pitch,
ahrs.yaw,
2012-04-30 04:17:14 -03:00
omega.x,
omega.y,
omega.z);
}
static NOINLINE void send_extended_status1(mavlink_channel_t chan, uint16_t packet_drops)
{
2013-10-04 00:33:47 -03:00
uint32_t control_sensors_present;
2012-04-30 04:17:14 -03:00
uint32_t control_sensors_enabled;
uint32_t control_sensors_health;
2013-10-04 00:33:47 -03:00
// default sensors present
control_sensors_present = MAVLINK_SENSOR_PRESENT_DEFAULT;
2012-04-30 04:17:14 -03:00
// first what sensors/controllers we have
if (g.compass_enabled) {
2013-10-04 00:33:47 -03:00
control_sensors_present |= MAV_SYS_STATUS_SENSOR_3D_MAG; // compass present
2012-04-30 04:17:14 -03:00
}
2013-10-04 00:33:47 -03:00
if (g_gps != NULL && g_gps->status() > GPS::NO_GPS) {
control_sensors_present |= MAV_SYS_STATUS_SENSOR_GPS;
2012-04-30 04:17:14 -03:00
}
2013-10-04 00:33:47 -03:00
// all present sensors enabled by default except rate control, attitude stabilization, yaw, altitude, position control and motor output which we will set individually
control_sensors_enabled = control_sensors_present & (~MAV_SYS_STATUS_SENSOR_ANGULAR_RATE_CONTROL & ~MAV_SYS_STATUS_SENSOR_ATTITUDE_STABILIZATION & ~MAV_SYS_STATUS_SENSOR_YAW_POSITION & ~MAV_SYS_STATUS_SENSOR_XY_POSITION_CONTROL & ~MAV_SYS_STATUS_SENSOR_MOTOR_OUTPUTS);
2012-04-30 04:17:14 -03:00
switch (control_mode) {
case MANUAL:
2013-03-28 18:53:20 -03:00
case HOLD:
2012-04-30 04:17:14 -03:00
break;
2012-05-14 16:21:29 -03:00
case LEARNING:
2013-03-01 07:32:57 -04:00
case STEERING:
2013-10-04 00:33:47 -03:00
control_sensors_enabled |= MAV_SYS_STATUS_SENSOR_ANGULAR_RATE_CONTROL; // 3D angular rate control
control_sensors_enabled |= MAV_SYS_STATUS_SENSOR_ATTITUDE_STABILIZATION; // attitude stabilisation
2012-04-30 04:17:14 -03:00
break;
case AUTO:
case RTL:
2013-02-07 18:21:22 -04:00
case GUIDED:
2013-10-04 00:33:47 -03:00
control_sensors_enabled |= MAV_SYS_STATUS_SENSOR_ANGULAR_RATE_CONTROL; // 3D angular rate control
control_sensors_enabled |= MAV_SYS_STATUS_SENSOR_ATTITUDE_STABILIZATION; // attitude stabilisation
control_sensors_enabled |= MAV_SYS_STATUS_SENSOR_YAW_POSITION; // yaw position
control_sensors_enabled |= MAV_SYS_STATUS_SENSOR_XY_POSITION_CONTROL; // X/Y position control
control_sensors_enabled |= MAV_SYS_STATUS_SENSOR_MOTOR_OUTPUTS; // motor control
2012-04-30 04:17:14 -03:00
break;
case INITIALISING:
break;
}
2013-10-04 00:33:47 -03:00
// default to all healthy except compass and gps which we set individually
control_sensors_health = control_sensors_present & (~MAV_SYS_STATUS_SENSOR_3D_MAG & ~MAV_SYS_STATUS_SENSOR_GPS);
2013-12-09 02:46:57 -04:00
if (g.compass_enabled && compass.healthy() && ahrs.use_compass()) {
2013-10-04 00:33:47 -03:00
control_sensors_health |= MAV_SYS_STATUS_SENSOR_3D_MAG;
2012-12-18 15:30:42 -04:00
}
2014-01-26 18:02:39 -04:00
if (g_gps != NULL && g_gps->status() >= GPS::GPS_OK_FIX_3D) {
2013-10-04 00:33:47 -03:00
control_sensors_health |= MAV_SYS_STATUS_SENSOR_GPS;
2012-12-18 15:30:42 -04:00
}
2013-10-29 19:02:16 -03:00
if (!ins.healthy()) {
control_sensors_health &= ~(MAV_SYS_STATUS_SENSOR_3D_GYRO | MAV_SYS_STATUS_SENSOR_3D_ACCEL);
}
2012-12-18 15:30:42 -04:00
2013-10-02 03:07:28 -03:00
int16_t battery_current = -1;
int8_t battery_remaining = -1;
2012-04-30 04:17:14 -03:00
2013-10-02 03:07:28 -03:00
if (battery.monitoring() == AP_BATT_MONITOR_VOLTAGE_AND_CURRENT) {
battery_remaining = battery.capacity_remaining_pct();
battery_current = battery.current_amps() * 100;
2012-04-30 04:17:14 -03:00
}
mavlink_msg_sys_status_send(
chan,
control_sensors_present,
control_sensors_enabled,
control_sensors_health,
2013-07-23 04:07:35 -03:00
(uint16_t)(scheduler.load_average(20000) * 1000),
2013-10-02 03:07:28 -03:00
battery.voltage() * 1000, // mV
2012-04-30 04:17:14 -03:00
battery_current, // in 10mA units
battery_remaining, // in %
0, // comm drops %,
0, // comm drops in pkts,
0, 0, 0, 0);
}
static void NOINLINE send_location(mavlink_channel_t chan)
{
2012-12-18 15:30:42 -04:00
uint32_t fix_time;
// if we have a GPS fix, take the time as the last fix time. That
// allows us to correctly calculate velocities and extrapolate
// positions.
// If we don't have a GPS fix then we are dead reckoning, and will
// use the current boot time as the fix time.
2013-03-25 07:09:04 -03:00
if (g_gps->status() >= GPS::GPS_OK_FIX_2D) {
2012-12-18 15:30:42 -04:00
fix_time = g_gps->last_fix_time;
} else {
fix_time = millis();
}
2012-04-30 04:17:14 -03:00
mavlink_msg_global_position_int_send(
chan,
2012-12-18 15:30:42 -04:00
fix_time,
2012-04-30 04:17:14 -03:00
current_loc.lat, // in 1E7 degrees
current_loc.lng, // in 1E7 degrees
2013-07-10 01:02:40 -03:00
g_gps->altitude_cm * 10, // millimeters above sea level
2012-07-17 00:20:20 -03:00
(current_loc.alt - home.alt) * 10, // millimeters above ground
2012-12-18 15:30:42 -04:00
g_gps->velocity_north() * 100, // X speed cm/s (+ve North)
g_gps->velocity_east() * 100, // Y speed cm/s (+ve East)
g_gps->velocity_down() * -100, // Z speed cm/s (+ve up)
ahrs.yaw_sensor);
2012-04-30 04:17:14 -03:00
}
static void NOINLINE send_nav_controller_output(mavlink_channel_t chan)
{
mavlink_msg_nav_controller_output_send(
chan,
2013-08-25 19:46:22 -03:00
lateral_acceleration, // use nav_roll to hold demanded Y accel
2013-09-02 03:42:19 -03:00
g_gps->ground_speed_cm * 0.01f * ins.get_gyro().z, // use nav_pitch to hold actual Y accel
2013-06-16 20:50:53 -03:00
nav_controller->nav_bearing_cd() * 0.01f,
nav_controller->target_bearing_cd() * 0.01f,
2012-04-30 04:17:14 -03:00
wp_distance,
2013-09-08 21:18:31 -03:00
0,
2012-05-09 02:12:26 -03:00
groundspeed_error,
2013-06-16 20:50:53 -03:00
nav_controller->crosstrack_error());
2012-04-30 04:17:14 -03:00
}
static void NOINLINE send_gps_raw(mavlink_channel_t chan)
{
mavlink_msg_gps_raw_int_send(
chan,
2012-12-18 15:30:42 -04:00
g_gps->last_fix_time*(uint64_t)1000,
2013-03-25 07:09:04 -03:00
g_gps->status(),
2012-04-30 04:17:14 -03:00
g_gps->latitude, // in 1E7 degrees
g_gps->longitude, // in 1E7 degrees
2013-07-10 01:02:40 -03:00
g_gps->altitude_cm * 10, // in mm
2012-04-30 04:17:14 -03:00
g_gps->hdop,
65535,
2013-07-10 01:02:40 -03:00
g_gps->ground_speed_cm, // cm/s
g_gps->ground_course_cd, // 1/100 degrees,
2012-04-30 04:17:14 -03:00
g_gps->num_sats);
2013-12-21 07:27:06 -04:00
#if GPS2_ENABLE
2014-03-02 16:05:42 -04:00
if (g_gps2 != NULL && g_gps2->status() != GPS::NO_GPS) {
2013-12-21 07:27:06 -04:00
int16_t payload_space = comm_get_txspace(chan) - MAVLINK_NUM_NON_PAYLOAD_BYTES;
if (payload_space >= MAVLINK_MSG_ID_GPS2_RAW_LEN) {
mavlink_msg_gps2_raw_send(
chan,
g_gps2->last_fix_time*(uint64_t)1000,
g_gps2->status(),
g_gps2->latitude, // in 1E7 degrees
g_gps2->longitude, // in 1E7 degrees
g_gps2->altitude_cm * 10, // in mm
g_gps2->hdop,
65535,
g_gps2->ground_speed_cm, // cm/s
g_gps2->ground_course_cd, // 1/100 degrees,
g_gps2->num_sats,
0,
0);
}
}
#endif
2012-04-30 04:17:14 -03:00
}
2013-10-23 08:15:37 -03:00
static void NOINLINE send_system_time(mavlink_channel_t chan)
{
mavlink_msg_system_time_send(
chan,
g_gps->time_epoch_usec(),
hal.scheduler->millis());
}
2013-09-20 20:27:38 -03:00
#if HIL_MODE != HIL_MODE_DISABLED
2012-04-30 04:17:14 -03:00
static void NOINLINE send_servo_out(mavlink_channel_t chan)
{
// normalized values scaled to -10000 to 10000
// This is used for HIL. Do not change without discussing with
// HIL maintainers
mavlink_msg_rc_channels_scaled_send(
chan,
millis(),
0, // port 0
2013-06-03 06:33:59 -03:00
10000 * channel_steer->norm_output(),
2013-02-07 18:21:22 -04:00
0,
2013-06-03 06:33:59 -03:00
10000 * channel_throttle->norm_output(),
2013-02-07 18:21:22 -04:00
0,
2012-04-30 04:17:14 -03:00
0,
0,
0,
0,
2012-12-18 15:30:42 -04:00
receiver_rssi);
2012-04-30 04:17:14 -03:00
}
2013-09-20 20:27:38 -03:00
#endif
2012-04-30 04:17:14 -03:00
static void NOINLINE send_radio_in(mavlink_channel_t chan)
{
mavlink_msg_rc_channels_raw_send(
chan,
millis(),
0, // port
2012-12-18 15:30:42 -04:00
hal.rcin->read(CH_1),
hal.rcin->read(CH_2),
hal.rcin->read(CH_3),
hal.rcin->read(CH_4),
hal.rcin->read(CH_5),
hal.rcin->read(CH_6),
hal.rcin->read(CH_7),
hal.rcin->read(CH_8),
receiver_rssi);
2012-04-30 04:17:14 -03:00
}
static void NOINLINE send_radio_out(mavlink_channel_t chan)
{
2012-12-18 07:44:12 -04:00
#if HIL_MODE == HIL_MODE_DISABLED || HIL_SERVOS
mavlink_msg_servo_output_raw_send(
chan,
micros(),
0, // port
hal.rcout->read(0),
hal.rcout->read(1),
hal.rcout->read(2),
hal.rcout->read(3),
hal.rcout->read(4),
hal.rcout->read(5),
hal.rcout->read(6),
hal.rcout->read(7));
#else
mavlink_msg_servo_output_raw_send(
chan,
micros(),
0, // port
2013-06-03 02:12:11 -03:00
RC_Channel::rc_channel(0)->radio_out,
RC_Channel::rc_channel(1)->radio_out,
RC_Channel::rc_channel(2)->radio_out,
RC_Channel::rc_channel(3)->radio_out,
RC_Channel::rc_channel(4)->radio_out,
RC_Channel::rc_channel(5)->radio_out,
RC_Channel::rc_channel(6)->radio_out,
RC_Channel::rc_channel(7)->radio_out);
2012-12-18 07:44:12 -04:00
#endif
2012-04-30 04:17:14 -03:00
}
static void NOINLINE send_vfr_hud(mavlink_channel_t chan)
{
mavlink_msg_vfr_hud_send(
chan,
2013-07-10 01:02:40 -03:00
(float)g_gps->ground_speed_cm / 100.0,
(float)g_gps->ground_speed_cm / 100.0,
2012-11-17 02:45:20 -04:00
(ahrs.yaw_sensor / 100) % 360,
2013-11-24 20:50:50 -04:00
(uint16_t)(100 * fabsf(channel_throttle->norm_output())),
2012-04-30 04:17:14 -03:00
current_loc.alt / 100.0,
0);
}
static void NOINLINE send_raw_imu1(mavlink_channel_t chan)
{
2013-12-08 23:09:56 -04:00
const Vector3f &accel = ins.get_accel();
const Vector3f &gyro = ins.get_gyro();
const Vector3f &mag = compass.get_field();
2012-04-30 04:17:14 -03:00
mavlink_msg_raw_imu_send(
chan,
micros(),
2013-01-01 22:53:07 -04:00
accel.x * 1000.0 / GRAVITY_MSS,
accel.y * 1000.0 / GRAVITY_MSS,
accel.z * 1000.0 / GRAVITY_MSS,
2012-04-30 04:17:14 -03:00
gyro.x * 1000.0,
gyro.y * 1000.0,
gyro.z * 1000.0,
2013-12-08 23:09:56 -04:00
mag.x,
mag.y,
mag.z);
2013-12-09 02:14:40 -04:00
if (ins.get_gyro_count() <= 1 &&
ins.get_accel_count() <= 1 &&
compass.get_count() <= 1) {
return;
}
const Vector3f &accel2 = ins.get_accel(1);
const Vector3f &gyro2 = ins.get_gyro(1);
const Vector3f &mag2 = compass.get_field(1);
mavlink_msg_scaled_imu2_send(
chan,
millis(),
accel2.x * 1000.0f / GRAVITY_MSS,
accel2.y * 1000.0f / GRAVITY_MSS,
accel2.z * 1000.0f / GRAVITY_MSS,
gyro2.x * 1000.0f,
gyro2.y * 1000.0f,
gyro2.z * 1000.0f,
mag2.x,
mag2.y,
mag2.z);
2012-04-30 04:17:14 -03:00
}
static void NOINLINE send_raw_imu3(mavlink_channel_t chan)
{
2013-12-08 23:09:56 -04:00
const Vector3f &mag_offsets = compass.get_offsets();
const Vector3f &accel_offsets = ins.get_accel_offsets();
const Vector3f &gyro_offsets = ins.get_gyro_offsets();
2012-04-30 04:17:14 -03:00
mavlink_msg_sensor_offsets_send(chan,
mag_offsets.x,
mag_offsets.y,
mag_offsets.z,
compass.get_declination(),
2012-11-20 03:23:31 -04:00
0, 0,
gyro_offsets.x,
gyro_offsets.y,
gyro_offsets.z,
accel_offsets.x,
accel_offsets.y,
accel_offsets.z);
2012-04-30 04:17:14 -03:00
}
2012-05-14 15:33:03 -03:00
static void NOINLINE send_ahrs(mavlink_channel_t chan)
{
2013-12-08 23:09:56 -04:00
const Vector3f &omega_I = ahrs.get_gyro_drift();
2012-05-14 15:33:03 -03:00
mavlink_msg_ahrs_send(
chan,
omega_I.x,
omega_I.y,
omega_I.z,
0,
0,
ahrs.get_error_rp(),
ahrs.get_error_yaw());
}
// report simulator state
static void NOINLINE send_simstate(mavlink_channel_t chan)
{
2013-04-19 18:29:57 -03:00
#if CONFIG_HAL_BOARD == HAL_BOARD_AVR_SITL
2012-11-17 02:45:20 -04:00
sitl.simstate_send(chan);
2012-05-14 15:33:03 -03:00
#endif
2013-04-19 18:29:57 -03:00
}
2012-05-14 15:33:03 -03:00
static void NOINLINE send_hwstatus(mavlink_channel_t chan)
{
mavlink_msg_hwstatus_send(
chan,
2014-02-13 02:11:57 -04:00
hal.analogin->board_voltage()*1000,
2012-12-18 07:44:12 -04:00
hal.i2c->lockup_count());
2012-05-14 15:33:03 -03:00
}
2013-02-28 21:00:48 -04:00
static void NOINLINE send_rangefinder(mavlink_channel_t chan)
{
2013-03-28 21:00:41 -03:00
if (!sonar.enabled()) {
// no sonar to report
return;
}
/*
report smaller distance of two sonars if more than one enabled
*/
float distance_cm, voltage;
if (!sonar2.enabled()) {
distance_cm = sonar.distance_cm();
voltage = sonar.voltage();
} else {
float dist1 = sonar.distance_cm();
float dist2 = sonar2.distance_cm();
if (dist1 <= dist2) {
distance_cm = dist1;
voltage = sonar.voltage();
} else {
distance_cm = dist2;
voltage = sonar2.voltage();
}
}
2013-02-28 21:00:48 -04:00
mavlink_msg_rangefinder_send(
chan,
2013-03-28 21:00:41 -03:00
distance_cm * 0.01f,
voltage);
2013-02-28 21:00:48 -04:00
}
2012-04-30 04:17:14 -03:00
static void NOINLINE send_current_waypoint(mavlink_channel_t chan)
{
2014-03-10 05:44:05 -03:00
uint16_t current_cmd_index;
if (mission.state() == AP_Mission::MISSION_RUNNING) {
current_cmd_index = mission.get_current_nav_cmd().index;
}else{
current_cmd_index = AP_MISSION_CMD_INDEX_NONE;
}
mavlink_msg_mission_current_send(chan, current_cmd_index);
2012-04-30 04:17:14 -03:00
}
static void NOINLINE send_statustext(mavlink_channel_t chan)
{
2013-11-23 06:57:26 -04:00
mavlink_statustext_t *s = &gcs[chan-MAVLINK_COMM_0].pending_status;
2012-04-30 04:17:14 -03:00
mavlink_msg_statustext_send(
chan,
2012-12-18 07:44:12 -04:00
s->severity,
s->text);
2012-04-30 04:17:14 -03:00
}
2012-08-29 20:36:18 -03:00
// are we still delaying telemetry to try to avoid Xbee bricking?
static bool telemetry_delayed(mavlink_channel_t chan)
{
uint32_t tnow = millis() >> 10;
2013-01-13 05:05:14 -04:00
if (tnow > (uint32_t)g.telem_delay) {
2012-08-29 20:36:18 -03:00
return false;
}
2013-09-19 03:24:59 -03:00
if (chan == MAVLINK_COMM_0 && hal.gpio->usb_connected()) {
// this is USB telemetry, so won't be an Xbee
2012-08-29 20:36:18 -03:00
return false;
}
// we're either on the 2nd UART, or no USB cable is connected
2013-09-19 03:24:59 -03:00
// we need to delay telemetry by the TELEM_DELAY time
2012-08-29 20:36:18 -03:00
return true;
}
2012-04-30 04:17:14 -03:00
// try to send a message, return false if it won't fit in the serial tx buffer
static bool mavlink_try_send_message(mavlink_channel_t chan, enum ap_message id, uint16_t packet_drops)
{
2012-09-24 18:20:54 -03:00
int16_t payload_space = comm_get_txspace(chan) - MAVLINK_NUM_NON_PAYLOAD_BYTES;
2012-04-30 04:17:14 -03:00
2012-08-29 20:36:18 -03:00
if (telemetry_delayed(chan)) {
2012-04-30 04:17:14 -03:00
return false;
}
2013-10-27 20:33:52 -03:00
// if we don't have at least 1ms remaining before the main loop
// wants to fire then don't send a mavlink message. We want to
// prioritise the main flight control loop over communications
if (!in_mavlink_delay && scheduler.time_available_usec() < 1200) {
gcs_out_of_time = true;
return false;
}
2012-04-30 04:17:14 -03:00
switch (id) {
case MSG_HEARTBEAT:
CHECK_PAYLOAD_SIZE(HEARTBEAT);
2013-12-15 19:35:27 -04:00
gcs[chan-MAVLINK_COMM_0].last_heartbeat_time = hal.scheduler->millis();
2012-04-30 04:17:14 -03:00
send_heartbeat(chan);
return true;
case MSG_EXTENDED_STATUS1:
CHECK_PAYLOAD_SIZE(SYS_STATUS);
send_extended_status1(chan, packet_drops);
2014-02-13 07:10:11 -04:00
CHECK_PAYLOAD_SIZE(POWER_STATUS);
gcs[chan-MAVLINK_COMM_0].send_power_status();
2012-04-30 04:17:14 -03:00
break;
case MSG_EXTENDED_STATUS2:
CHECK_PAYLOAD_SIZE(MEMINFO);
2013-12-28 01:02:45 -04:00
gcs[chan-MAVLINK_COMM_0].send_meminfo();
2012-04-30 04:17:14 -03:00
break;
case MSG_ATTITUDE:
CHECK_PAYLOAD_SIZE(ATTITUDE);
send_attitude(chan);
break;
case MSG_LOCATION:
CHECK_PAYLOAD_SIZE(GLOBAL_POSITION_INT);
send_location(chan);
break;
case MSG_NAV_CONTROLLER_OUTPUT:
if (control_mode != MANUAL) {
CHECK_PAYLOAD_SIZE(NAV_CONTROLLER_OUTPUT);
send_nav_controller_output(chan);
}
break;
case MSG_GPS_RAW:
CHECK_PAYLOAD_SIZE(GPS_RAW_INT);
send_gps_raw(chan);
break;
2013-10-23 08:15:37 -03:00
case MSG_SYSTEM_TIME:
CHECK_PAYLOAD_SIZE(SYSTEM_TIME);
send_system_time(chan);
break;
2012-04-30 04:17:14 -03:00
case MSG_SERVO_OUT:
2013-09-20 20:27:38 -03:00
#if HIL_MODE != HIL_MODE_DISABLED
2012-04-30 04:17:14 -03:00
CHECK_PAYLOAD_SIZE(RC_CHANNELS_SCALED);
send_servo_out(chan);
2013-09-20 20:27:38 -03:00
#endif
2012-04-30 04:17:14 -03:00
break;
case MSG_RADIO_IN:
CHECK_PAYLOAD_SIZE(RC_CHANNELS_RAW);
send_radio_in(chan);
break;
case MSG_RADIO_OUT:
CHECK_PAYLOAD_SIZE(SERVO_OUTPUT_RAW);
send_radio_out(chan);
break;
case MSG_VFR_HUD:
CHECK_PAYLOAD_SIZE(VFR_HUD);
send_vfr_hud(chan);
break;
case MSG_RAW_IMU1:
CHECK_PAYLOAD_SIZE(RAW_IMU);
send_raw_imu1(chan);
break;
case MSG_RAW_IMU3:
CHECK_PAYLOAD_SIZE(SENSOR_OFFSETS);
send_raw_imu3(chan);
break;
case MSG_CURRENT_WAYPOINT:
2012-08-08 23:22:46 -03:00
CHECK_PAYLOAD_SIZE(MISSION_CURRENT);
2012-04-30 04:17:14 -03:00
send_current_waypoint(chan);
break;
case MSG_NEXT_PARAM:
CHECK_PAYLOAD_SIZE(PARAM_VALUE);
2013-11-23 06:57:26 -04:00
gcs[chan-MAVLINK_COMM_0].queued_param_send();
2012-04-30 04:17:14 -03:00
break;
case MSG_NEXT_WAYPOINT:
2012-08-08 23:22:46 -03:00
CHECK_PAYLOAD_SIZE(MISSION_REQUEST);
2013-11-23 06:57:26 -04:00
gcs[chan-MAVLINK_COMM_0].queued_waypoint_send();
2012-04-30 04:17:14 -03:00
break;
case MSG_STATUSTEXT:
CHECK_PAYLOAD_SIZE(STATUSTEXT);
send_statustext(chan);
break;
2012-05-14 15:33:03 -03:00
case MSG_AHRS:
CHECK_PAYLOAD_SIZE(AHRS);
send_ahrs(chan);
break;
case MSG_SIMSTATE:
CHECK_PAYLOAD_SIZE(SIMSTATE);
send_simstate(chan);
break;
case MSG_HWSTATUS:
CHECK_PAYLOAD_SIZE(HWSTATUS);
send_hwstatus(chan);
break;
2013-02-28 21:00:48 -04:00
case MSG_RANGEFINDER:
CHECK_PAYLOAD_SIZE(RANGEFINDER);
send_rangefinder(chan);
break;
2013-12-15 19:35:27 -04:00
case MSG_RAW_IMU2:
case MSG_LIMITS_STATUS:
case MSG_FENCE_STATUS:
case MSG_WIND:
// unused
break;
2012-04-30 04:17:14 -03:00
case MSG_RETRY_DEFERRED:
break; // just here to prevent a warning
}
2013-12-15 19:35:27 -04:00
2012-04-30 04:17:14 -03:00
return true;
}
#define MAX_DEFERRED_MESSAGES MSG_RETRY_DEFERRED
static struct mavlink_queue {
enum ap_message deferred_messages[MAX_DEFERRED_MESSAGES];
uint8_t next_deferred_message;
uint8_t num_deferred_messages;
2013-11-23 06:57:26 -04:00
} mavlink_queue[MAVLINK_COMM_NUM_BUFFERS];
2012-04-30 04:17:14 -03:00
// send a message using mavlink
static void mavlink_send_message(mavlink_channel_t chan, enum ap_message id, uint16_t packet_drops)
{
uint8_t i, nextid;
struct mavlink_queue *q = &mavlink_queue[(uint8_t)chan];
// see if we can send the deferred messages, if any
while (q->num_deferred_messages != 0) {
if (!mavlink_try_send_message(chan,
q->deferred_messages[q->next_deferred_message],
packet_drops)) {
break;
}
q->next_deferred_message++;
if (q->next_deferred_message == MAX_DEFERRED_MESSAGES) {
q->next_deferred_message = 0;
}
q->num_deferred_messages--;
}
if (id == MSG_RETRY_DEFERRED) {
return;
}
// this message id might already be deferred
for (i=0, nextid = q->next_deferred_message; i < q->num_deferred_messages; i++) {
if (q->deferred_messages[nextid] == id) {
// its already deferred, discard
return;
}
nextid++;
if (nextid == MAX_DEFERRED_MESSAGES) {
nextid = 0;
}
}
if (q->num_deferred_messages != 0 ||
!mavlink_try_send_message(chan, id, packet_drops)) {
// can't send it now, so defer it
if (q->num_deferred_messages == MAX_DEFERRED_MESSAGES) {
// the defer buffer is full, discard
return;
}
nextid = q->next_deferred_message + q->num_deferred_messages;
if (nextid >= MAX_DEFERRED_MESSAGES) {
nextid -= MAX_DEFERRED_MESSAGES;
}
q->deferred_messages[nextid] = id;
q->num_deferred_messages++;
}
}
void mavlink_send_text(mavlink_channel_t chan, gcs_severity severity, const char *str)
{
2012-08-29 20:36:18 -03:00
if (telemetry_delayed(chan)) {
2012-04-30 04:17:14 -03:00
return;
}
if (severity == SEVERITY_LOW) {
// send via the deferred queuing system
2013-11-23 06:57:26 -04:00
mavlink_statustext_t *s = &gcs[chan-MAVLINK_COMM_0].pending_status;
2012-12-18 07:44:12 -04:00
s->severity = (uint8_t)severity;
strncpy((char *)s->text, str, sizeof(s->text));
2012-04-30 04:17:14 -03:00
mavlink_send_message(chan, MSG_STATUSTEXT, 0);
} else {
// send immediately
mavlink_msg_statustext_send(chan, severity, str);
}
}
2013-10-20 19:32:39 -03:00
/*
default stream rates to 1Hz
*/
2012-04-30 04:17:14 -03:00
const AP_Param::GroupInfo GCS_MAVLINK::var_info[] PROGMEM = {
2013-09-11 20:51:36 -03:00
// @Param: RAW_SENS
2013-10-20 19:32:39 -03:00
// @DisplayName: Raw sensor stream rate
// @Description: Raw sensor stream rate to ground station
2013-09-11 20:51:36 -03:00
// @Units: Hz
2013-10-20 19:32:39 -03:00
// @Range: 0 10
2013-09-11 20:51:36 -03:00
// @Increment: 1
// @User: Advanced
2013-10-20 19:32:39 -03:00
AP_GROUPINFO("RAW_SENS", 0, GCS_MAVLINK, streamRates[0], 1),
2013-09-11 20:51:36 -03:00
// @Param: EXT_STAT
2013-10-20 19:32:39 -03:00
// @DisplayName: Extended status stream rate to ground station
// @Description: Extended status stream rate to ground station
2013-09-11 20:51:36 -03:00
// @Units: Hz
2013-10-20 19:32:39 -03:00
// @Range: 0 10
2013-09-11 20:51:36 -03:00
// @Increment: 1
// @User: Advanced
2013-10-20 19:32:39 -03:00
AP_GROUPINFO("EXT_STAT", 1, GCS_MAVLINK, streamRates[1], 1),
2013-09-11 20:51:36 -03:00
// @Param: RC_CHAN
2013-10-20 19:32:39 -03:00
// @DisplayName: RC Channel stream rate to ground station
// @Description: RC Channel stream rate to ground station
2013-09-11 20:51:36 -03:00
// @Units: Hz
2013-10-20 19:32:39 -03:00
// @Range: 0 10
2013-09-11 20:51:36 -03:00
// @Increment: 1
// @User: Advanced
2013-10-20 19:32:39 -03:00
AP_GROUPINFO("RC_CHAN", 2, GCS_MAVLINK, streamRates[2], 1),
2013-09-11 20:51:36 -03:00
// @Param: RAW_CTRL
2013-10-20 19:32:39 -03:00
// @DisplayName: Raw Control stream rate to ground station
// @Description: Raw Control stream rate to ground station
2013-09-11 20:51:36 -03:00
// @Units: Hz
2013-10-20 19:32:39 -03:00
// @Range: 0 10
2013-09-11 20:51:36 -03:00
// @Increment: 1
// @User: Advanced
2013-10-20 19:32:39 -03:00
AP_GROUPINFO("RAW_CTRL", 3, GCS_MAVLINK, streamRates[3], 1),
2013-09-11 20:51:36 -03:00
// @Param: POSITION
2013-10-20 19:32:39 -03:00
// @DisplayName: Position stream rate to ground station
// @Description: Position stream rate to ground station
2013-09-11 20:51:36 -03:00
// @Units: Hz
2013-10-20 19:32:39 -03:00
// @Range: 0 10
2013-09-11 20:51:36 -03:00
// @Increment: 1
// @User: Advanced
2013-10-20 19:32:39 -03:00
AP_GROUPINFO("POSITION", 4, GCS_MAVLINK, streamRates[4], 1),
2013-09-11 20:51:36 -03:00
// @Param: EXTRA1
2013-10-20 19:32:39 -03:00
// @DisplayName: Extra data type 1 stream rate to ground station
// @Description: Extra data type 1 stream rate to ground station
2013-09-11 20:51:36 -03:00
// @Units: Hz
2013-10-20 19:32:39 -03:00
// @Range: 0 10
2013-09-11 20:51:36 -03:00
// @Increment: 1
// @User: Advanced
2013-10-20 19:32:39 -03:00
AP_GROUPINFO("EXTRA1", 5, GCS_MAVLINK, streamRates[5], 1),
2013-09-11 20:51:36 -03:00
// @Param: EXTRA2
2013-10-20 19:32:39 -03:00
// @DisplayName: Extra data type 2 stream rate to ground station
// @Description: Extra data type 2 stream rate to ground station
2013-09-11 20:51:36 -03:00
// @Units: Hz
2013-10-20 19:32:39 -03:00
// @Range: 0 10
2013-09-11 20:51:36 -03:00
// @Increment: 1
// @User: Advanced
2013-10-20 19:32:39 -03:00
AP_GROUPINFO("EXTRA2", 6, GCS_MAVLINK, streamRates[6], 1),
2013-09-11 20:51:36 -03:00
// @Param: EXTRA3
2013-10-20 19:32:39 -03:00
// @DisplayName: Extra data type 3 stream rate to ground station
// @Description: Extra data type 3 stream rate to ground station
2013-09-11 20:51:36 -03:00
// @Units: Hz
2013-10-20 19:32:39 -03:00
// @Range: 0 10
2013-09-11 20:51:36 -03:00
// @Increment: 1
// @User: Advanced
2013-10-20 19:32:39 -03:00
AP_GROUPINFO("EXTRA3", 7, GCS_MAVLINK, streamRates[7], 1),
2013-09-11 20:51:36 -03:00
// @Param: PARAMS
2013-10-20 19:32:39 -03:00
// @DisplayName: Parameter stream rate to ground station
// @Description: Parameter stream rate to ground station
2013-09-11 20:51:36 -03:00
// @Units: Hz
2013-10-20 19:32:39 -03:00
// @Range: 0 10
2013-09-11 20:51:36 -03:00
// @Increment: 1
// @User: Advanced
2013-10-27 20:33:52 -03:00
AP_GROUPINFO("PARAMS", 8, GCS_MAVLINK, streamRates[8], 10),
2012-04-30 04:17:14 -03:00
AP_GROUPEND
};
void
GCS_MAVLINK::update(void)
{
// receive new packets
mavlink_message_t msg;
mavlink_status_t status;
status.packet_rx_drop_count = 0;
// process received bytes
2013-10-20 19:32:39 -03:00
uint16_t nbytes = comm_get_available(chan);
for (uint16_t i=0; i<nbytes; i++)
2012-04-30 04:17:14 -03:00
{
uint8_t c = comm_receive_ch(chan);
#if CLI_ENABLED == ENABLED
/* allow CLI to be started by hitting enter 3 times, if no
2012-12-18 15:30:42 -04:00
* heartbeat packets have been received */
2013-03-21 07:58:12 -03:00
if (mavlink_active == 0 && (millis() - _cli_timeout) < 20000 &&
comm_is_idle(chan)) {
2012-04-30 04:17:14 -03:00
if (c == '\n' || c == '\r') {
crlf_count++;
} else {
crlf_count = 0;
}
if (crlf_count == 3) {
2012-11-21 02:25:11 -04:00
run_cli(_port);
2012-04-30 04:17:14 -03:00
}
}
#endif
// Try to get a new message
if (mavlink_parse_char(chan, c, &msg, &status)) {
2012-11-20 23:11:05 -04:00
// we exclude radio packets to make it possible to use the
// CLI over the radio
2013-08-24 04:58:37 -03:00
if (msg.msgid != MAVLINK_MSG_ID_RADIO && msg.msgid != MAVLINK_MSG_ID_RADIO_STATUS) {
2012-11-20 23:11:05 -04:00
mavlink_active = true;
}
2012-04-30 04:17:14 -03:00
handleMessage(&msg);
}
}
// Update packet drops counter
packet_drops += status.packet_rx_drop_count;
2012-12-18 15:30:42 -04:00
if (!waypoint_receiving) {
2012-04-30 04:17:14 -03:00
return;
}
uint32_t tnow = millis();
if (waypoint_receiving &&
2012-11-27 18:35:09 -04:00
waypoint_request_i <= waypoint_request_last &&
2012-05-14 15:33:03 -03:00
tnow > waypoint_timelast_request + 500 + (stream_slowdown*20)) {
2012-04-30 04:17:14 -03:00
waypoint_timelast_request = tnow;
send_message(MSG_NEXT_WAYPOINT);
}
// stop waypoint receiving if timeout
if (waypoint_receiving && (millis() - waypoint_timelast_receive) > waypoint_receive_timeout){
waypoint_receiving = false;
}
}
2012-05-14 15:33:03 -03:00
// see if we should send a stream now. Called at 50Hz
bool GCS_MAVLINK::stream_trigger(enum streams stream_num)
{
2013-10-20 19:32:39 -03:00
if (stream_num >= NUM_STREAMS) {
return false;
}
float rate = (uint8_t)streamRates[stream_num].get();
2012-05-14 15:33:03 -03:00
2012-12-18 15:30:42 -04:00
// send at a much lower rate while handling waypoints and
// parameter sends
2013-10-27 20:33:52 -03:00
if ((stream_num != STREAM_PARAMS) &&
(waypoint_receiving || _queued_parameter != NULL)) {
2012-12-18 15:30:42 -04:00
rate *= 0.25;
}
if (rate <= 0) {
2012-05-14 15:33:03 -03:00
return false;
}
if (stream_ticks[stream_num] == 0) {
// we're triggering now, setup the next trigger point
if (rate > 50) {
rate = 50;
}
stream_ticks[stream_num] = (50 / rate) + stream_slowdown;
return true;
}
// count down at 50Hz
stream_ticks[stream_num]--;
return false;
}
2012-04-30 04:17:14 -03:00
void
2012-05-14 15:33:03 -03:00
GCS_MAVLINK::data_stream_send(void)
2012-04-30 04:17:14 -03:00
{
2013-10-27 20:33:52 -03:00
gcs_out_of_time = false;
2013-12-15 19:35:27 -04:00
if (!in_mavlink_delay) {
handle_log_send(DataFlash);
}
2012-05-14 15:33:03 -03:00
if (_queued_parameter != NULL) {
2013-10-20 19:32:39 -03:00
if (streamRates[STREAM_PARAMS].get() <= 0) {
2013-10-27 20:33:52 -03:00
streamRates[STREAM_PARAMS].set(10);
2012-05-14 15:33:03 -03:00
}
if (stream_trigger(STREAM_PARAMS)) {
send_message(MSG_NEXT_PARAM);
}
2012-12-18 15:30:42 -04:00
}
2013-10-27 20:33:52 -03:00
if (gcs_out_of_time) return;
2012-12-18 15:30:42 -04:00
if (in_mavlink_delay) {
#if HIL_MODE != HIL_MODE_DISABLED
// in HIL we need to keep sending servo values to ensure
// the simulator doesn't pause, otherwise our sensor
// calibration could stall
if (stream_trigger(STREAM_RAW_CONTROLLER)) {
send_message(MSG_SERVO_OUT);
}
2012-12-18 16:17:06 -04:00
if (stream_trigger(STREAM_RC_CHANNELS)) {
send_message(MSG_RADIO_OUT);
}
2012-12-18 15:30:42 -04:00
#endif
2012-12-18 16:17:06 -04:00
// don't send any other stream types while in the delay callback
2012-05-14 15:33:03 -03:00
return;
}
2012-04-30 04:17:14 -03:00
2013-10-27 20:33:52 -03:00
if (gcs_out_of_time) return;
2012-05-14 15:33:03 -03:00
if (stream_trigger(STREAM_RAW_SENSORS)) {
send_message(MSG_RAW_IMU1);
send_message(MSG_RAW_IMU3);
}
2012-04-30 04:17:14 -03:00
2013-10-27 20:33:52 -03:00
if (gcs_out_of_time) return;
2012-05-14 15:33:03 -03:00
if (stream_trigger(STREAM_EXTENDED_STATUS)) {
send_message(MSG_EXTENDED_STATUS1);
send_message(MSG_EXTENDED_STATUS2);
send_message(MSG_CURRENT_WAYPOINT);
send_message(MSG_GPS_RAW); // TODO - remove this message after location message is working
send_message(MSG_NAV_CONTROLLER_OUTPUT);
}
2012-04-30 04:17:14 -03:00
2013-10-27 20:33:52 -03:00
if (gcs_out_of_time) return;
2012-05-14 15:33:03 -03:00
if (stream_trigger(STREAM_POSITION)) {
// sent with GPS read
send_message(MSG_LOCATION);
}
2012-04-30 04:17:14 -03:00
2013-10-27 20:33:52 -03:00
if (gcs_out_of_time) return;
2012-05-14 15:33:03 -03:00
if (stream_trigger(STREAM_RAW_CONTROLLER)) {
send_message(MSG_SERVO_OUT);
}
2012-04-30 04:17:14 -03:00
2013-10-27 20:33:52 -03:00
if (gcs_out_of_time) return;
2012-05-14 15:33:03 -03:00
if (stream_trigger(STREAM_RC_CHANNELS)) {
send_message(MSG_RADIO_OUT);
send_message(MSG_RADIO_IN);
}
2012-04-30 04:17:14 -03:00
2013-10-27 20:33:52 -03:00
if (gcs_out_of_time) return;
2012-05-14 15:33:03 -03:00
if (stream_trigger(STREAM_EXTRA1)) {
send_message(MSG_ATTITUDE);
send_message(MSG_SIMSTATE);
}
2012-04-30 04:17:14 -03:00
2013-10-27 20:33:52 -03:00
if (gcs_out_of_time) return;
2012-05-14 15:33:03 -03:00
if (stream_trigger(STREAM_EXTRA2)) {
send_message(MSG_VFR_HUD);
}
2013-10-27 20:33:52 -03:00
if (gcs_out_of_time) return;
2012-05-14 15:33:03 -03:00
if (stream_trigger(STREAM_EXTRA3)) {
send_message(MSG_AHRS);
send_message(MSG_HWSTATUS);
2013-02-28 21:00:48 -04:00
send_message(MSG_RANGEFINDER);
2013-10-23 08:15:37 -03:00
send_message(MSG_SYSTEM_TIME);
2012-05-14 15:33:03 -03:00
}
2012-04-30 04:17:14 -03:00
}
void
GCS_MAVLINK::send_message(enum ap_message id)
{
mavlink_send_message(chan,id, packet_drops);
}
void
2012-12-18 07:44:12 -04:00
GCS_MAVLINK::send_text_P(gcs_severity severity, const prog_char_t *str)
2012-04-30 04:17:14 -03:00
{
mavlink_statustext_t m;
uint8_t i;
for (i=0; i<sizeof(m.text); i++) {
m.text[i] = pgm_read_byte((const prog_char *)(str++));
2013-05-20 00:51:29 -03:00
if (m.text[i] == '\0') {
break;
}
2012-04-30 04:17:14 -03:00
}
if (i < sizeof(m.text)) m.text[i] = 0;
mavlink_send_text(chan, severity, (const char *)m.text);
}
void GCS_MAVLINK::handleMessage(mavlink_message_t* msg)
{
2014-03-10 05:44:05 -03:00
struct AP_Mission::Mission_Command cmd = {}; // general purpose mission command
2012-04-30 04:17:14 -03:00
switch (msg->msgid) {
case MAVLINK_MSG_ID_REQUEST_DATA_STREAM:
{
// decode
mavlink_request_data_stream_t packet;
mavlink_msg_request_data_stream_decode(msg, &packet);
if (mavlink_check_target(packet.target_system, packet.target_component))
break;
2012-12-18 15:30:42 -04:00
int16_t freq = 0; // packet frequency
2012-04-30 04:17:14 -03:00
if (packet.start_stop == 0)
freq = 0; // stop sending
else if (packet.start_stop == 1)
freq = packet.req_message_rate; // start sending
else
break;
switch(packet.req_stream_id){
case MAV_DATA_STREAM_ALL:
2013-10-20 19:32:39 -03:00
// note that we don't set STREAM_PARAMS - that is internal only
for (uint8_t i=0; i<STREAM_PARAMS; i++) {
streamRates[i].set_and_save_ifchanged(freq);
}
2012-04-30 04:17:14 -03:00
break;
case MAV_DATA_STREAM_RAW_SENSORS:
2013-10-20 19:32:39 -03:00
streamRates[STREAM_RAW_SENSORS].set_and_save_ifchanged(freq);
2012-04-30 04:17:14 -03:00
break;
case MAV_DATA_STREAM_EXTENDED_STATUS:
2013-10-20 19:32:39 -03:00
streamRates[STREAM_EXTENDED_STATUS].set_and_save_ifchanged(freq);
2012-04-30 04:17:14 -03:00
break;
case MAV_DATA_STREAM_RC_CHANNELS:
2013-10-20 19:32:39 -03:00
streamRates[STREAM_RC_CHANNELS].set_and_save_ifchanged(freq);
2012-04-30 04:17:14 -03:00
break;
case MAV_DATA_STREAM_RAW_CONTROLLER:
2013-10-20 19:32:39 -03:00
streamRates[STREAM_RAW_CONTROLLER].set_and_save_ifchanged(freq);
2012-04-30 04:17:14 -03:00
break;
case MAV_DATA_STREAM_POSITION:
2013-10-20 19:32:39 -03:00
streamRates[STREAM_POSITION].set_and_save_ifchanged(freq);
2012-04-30 04:17:14 -03:00
break;
case MAV_DATA_STREAM_EXTRA1:
2013-10-20 19:32:39 -03:00
streamRates[STREAM_EXTRA1].set_and_save_ifchanged(freq);
2012-04-30 04:17:14 -03:00
break;
case MAV_DATA_STREAM_EXTRA2:
2013-10-20 19:32:39 -03:00
streamRates[STREAM_EXTRA2].set_and_save_ifchanged(freq);
2012-04-30 04:17:14 -03:00
break;
case MAV_DATA_STREAM_EXTRA3:
2013-10-20 19:32:39 -03:00
streamRates[STREAM_EXTRA3].set_and_save_ifchanged(freq);
2012-04-30 04:17:14 -03:00
break;
}
break;
}
case MAVLINK_MSG_ID_COMMAND_LONG:
{
// decode
mavlink_command_long_t packet;
mavlink_msg_command_long_decode(msg, &packet);
if (mavlink_check_target(packet.target_system, packet.target_component)) break;
2014-03-17 04:07:18 -03:00
uint8_t result = MAV_RESULT_UNSUPPORTED;
2012-04-30 04:17:14 -03:00
// do command
2012-12-18 07:44:12 -04:00
send_text_P(SEVERITY_LOW,PSTR("command received: "));
2012-04-30 04:17:14 -03:00
switch(packet.command) {
case MAV_CMD_NAV_RETURN_TO_LAUNCH:
set_mode(RTL);
result = MAV_RESULT_ACCEPTED;
break;
2012-05-14 15:33:03 -03:00
case MAV_CMD_MISSION_START:
set_mode(AUTO);
result = MAV_RESULT_ACCEPTED;
2012-04-30 04:17:14 -03:00
break;
case MAV_CMD_PREFLIGHT_CALIBRATION:
if (packet.param1 == 1 ||
packet.param2 == 1 ||
packet.param3 == 1) {
2012-11-17 02:45:20 -04:00
startup_INS_ground(true);
2012-04-30 04:17:14 -03:00
}
if (packet.param4 == 1) {
trim_radio();
}
result = MAV_RESULT_ACCEPTED;
break;
2012-12-18 15:30:42 -04:00
case MAV_CMD_DO_SET_MODE:
switch ((uint16_t)packet.param1) {
case MAV_MODE_MANUAL_ARMED:
case MAV_MODE_MANUAL_DISARMED:
set_mode(MANUAL);
result = MAV_RESULT_ACCEPTED;
break;
case MAV_MODE_AUTO_ARMED:
case MAV_MODE_AUTO_DISARMED:
set_mode(AUTO);
result = MAV_RESULT_ACCEPTED;
break;
case MAV_MODE_STABILIZE_DISARMED:
case MAV_MODE_STABILIZE_ARMED:
set_mode(LEARNING);
result = MAV_RESULT_ACCEPTED;
break;
2012-04-30 04:17:14 -03:00
default:
result = MAV_RESULT_UNSUPPORTED;
2012-12-18 15:30:42 -04:00
}
break;
case MAV_CMD_DO_SET_SERVO:
2014-01-20 01:05:23 -04:00
if (ServoRelayEvents.do_set_servo(packet.param1, packet.param2)) {
result = MAV_RESULT_ACCEPTED;
}
break;
case MAV_CMD_DO_REPEAT_SERVO:
if (ServoRelayEvents.do_repeat_servo(packet.param1, packet.param2, packet.param3, packet.param4*1000)) {
result = MAV_RESULT_ACCEPTED;
}
break;
case MAV_CMD_DO_SET_RELAY:
if (ServoRelayEvents.do_set_relay(packet.param1, packet.param2)) {
result = MAV_RESULT_ACCEPTED;
}
break;
case MAV_CMD_DO_REPEAT_RELAY:
if (ServoRelayEvents.do_repeat_relay(packet.param1, packet.param2, packet.param3*1000)) {
result = MAV_RESULT_ACCEPTED;
}
2012-12-18 15:30:42 -04:00
break;
case MAV_CMD_PREFLIGHT_REBOOT_SHUTDOWN:
2013-09-03 22:58:41 -03:00
if (packet.param1 == 1 || packet.param1 == 3) {
// when packet.param1 == 3 we reboot to hold in bootloader
hal.scheduler->reboot(packet.param1 == 3);
2012-12-18 15:30:42 -04:00
result = MAV_RESULT_ACCEPTED;
}
break;
default:
2012-04-30 04:17:14 -03:00
break;
}
mavlink_msg_command_ack_send(
chan,
packet.command,
result);
break;
}
case MAVLINK_MSG_ID_SET_MODE:
{
// decode
mavlink_set_mode_t packet;
mavlink_msg_set_mode_decode(msg, &packet);
if (!(packet.base_mode & MAV_MODE_FLAG_CUSTOM_MODE_ENABLED)) {
// we ignore base_mode as there is no sane way to map
// from that bitmap to a APM flight mode. We rely on
// custom_mode instead.
break;
}
switch (packet.custom_mode) {
case MANUAL:
2013-03-28 18:53:20 -03:00
case HOLD:
2012-05-14 16:21:29 -03:00
case LEARNING:
2013-03-01 07:32:57 -04:00
case STEERING:
2012-04-30 04:17:14 -03:00
case AUTO:
case RTL:
2013-02-07 18:21:22 -04:00
set_mode((enum mode)packet.custom_mode);
2012-04-30 04:17:14 -03:00
break;
}
break;
}
2012-08-08 23:22:46 -03:00
case MAVLINK_MSG_ID_MISSION_REQUEST_LIST:
2012-04-30 04:17:14 -03:00
{
// decode
2012-08-08 23:22:46 -03:00
mavlink_mission_request_list_t packet;
mavlink_msg_mission_request_list_decode(msg, &packet);
2012-04-30 04:17:14 -03:00
if (mavlink_check_target(packet.target_system, packet.target_component))
break;
// Start sending waypoints
2012-08-08 23:22:46 -03:00
mavlink_msg_mission_count_send(
2012-04-30 04:17:14 -03:00
chan,msg->sysid,
msg->compid,
2014-03-10 05:44:05 -03:00
mission.num_commands());
2012-04-30 04:17:14 -03:00
waypoint_receiving = false;
waypoint_dest_sysid = msg->sysid;
waypoint_dest_compid = msg->compid;
break;
}
// XXX read a WP from EEPROM and send it to the GCS
2012-08-08 23:22:46 -03:00
case MAVLINK_MSG_ID_MISSION_REQUEST:
2014-03-17 23:54:21 -03:00
{
handle_mission_request(mission, msg, cmd);
break;
}
2012-04-30 04:17:14 -03:00
2012-08-08 23:22:46 -03:00
case MAVLINK_MSG_ID_MISSION_ACK:
2012-04-30 04:17:14 -03:00
{
// decode
2012-08-08 23:22:46 -03:00
mavlink_mission_ack_t packet;
mavlink_msg_mission_ack_decode(msg, &packet);
2012-04-30 04:17:14 -03:00
if (mavlink_check_target(packet.target_system,packet.target_component)) break;
break;
}
case MAVLINK_MSG_ID_PARAM_REQUEST_LIST:
{
// decode
mavlink_param_request_list_t packet;
mavlink_msg_param_request_list_decode(msg, &packet);
if (mavlink_check_target(packet.target_system,packet.target_component)) break;
2013-11-08 07:29:13 -04:00
// mark the firmware version in the tlog
send_text_P(SEVERITY_LOW, PSTR(FIRMWARE_STRING));
2012-04-30 04:17:14 -03:00
2014-01-14 00:38:42 -04:00
#if defined(PX4_GIT_VERSION) && defined(NUTTX_GIT_VERSION)
send_text_P(SEVERITY_LOW, PSTR("PX4: " PX4_GIT_VERSION " NuttX: " NUTTX_GIT_VERSION));
#endif
2013-11-25 21:00:33 -04:00
// send system ID if we can
char sysid[40];
if (hal.util->get_system_id(sysid)) {
mavlink_send_text(chan, SEVERITY_LOW, sysid);
}
2013-11-08 07:29:13 -04:00
// Start sending parameters - next call to ::update will kick the first one out
2012-04-30 04:17:14 -03:00
_queued_parameter = AP_Param::first(&_queued_parameter_token, &_queued_parameter_type);
_queued_parameter_index = 0;
_queued_parameter_count = _count_parameters();
break;
}
2012-12-18 15:30:42 -04:00
case MAVLINK_MSG_ID_PARAM_REQUEST_READ:
{
// decode
mavlink_param_request_read_t packet;
mavlink_msg_param_request_read_decode(msg, &packet);
if (mavlink_check_target(packet.target_system,packet.target_component)) break;
enum ap_var_type p_type;
AP_Param *vp;
2013-01-21 01:51:50 -04:00
char param_name[AP_MAX_NAME_SIZE+1];
2012-12-18 15:30:42 -04:00
if (packet.param_index != -1) {
2013-01-08 18:45:33 -04:00
AP_Param::ParamToken token;
vp = AP_Param::find_by_index(packet.param_index, &p_type, &token);
2012-12-18 15:30:42 -04:00
if (vp == NULL) {
gcs_send_text_fmt(PSTR("Unknown parameter index %d"), packet.param_index);
break;
}
2013-04-19 04:53:07 -03:00
vp->copy_name_token(token, param_name, AP_MAX_NAME_SIZE, true);
2013-01-21 01:51:50 -04:00
param_name[AP_MAX_NAME_SIZE] = 0;
2012-12-18 15:30:42 -04:00
} else {
2013-01-21 01:51:50 -04:00
strncpy(param_name, packet.param_id, AP_MAX_NAME_SIZE);
param_name[AP_MAX_NAME_SIZE] = 0;
vp = AP_Param::find(param_name, &p_type);
2012-12-18 15:30:42 -04:00
if (vp == NULL) {
gcs_send_text_fmt(PSTR("Unknown parameter %.16s"), packet.param_id);
break;
}
}
float value = vp->cast_to_float(p_type);
mavlink_msg_param_value_send(
chan,
param_name,
value,
mav_var_type(p_type),
_count_parameters(),
packet.param_index);
break;
}
2012-08-08 23:22:46 -03:00
case MAVLINK_MSG_ID_MISSION_CLEAR_ALL:
2012-04-30 04:17:14 -03:00
{
// decode
2012-08-08 23:22:46 -03:00
mavlink_mission_clear_all_t packet;
mavlink_msg_mission_clear_all_decode(msg, &packet);
2012-04-30 04:17:14 -03:00
if (mavlink_check_target(packet.target_system, packet.target_component)) break;
2014-03-10 05:44:05 -03:00
// clear all commands
if (mission.clear()) {
// note that we don't send multiple acks, as otherwise a
// GCS that is doing a clear followed by a set may see
// the additional ACKs as ACKs of the set operations
mavlink_msg_mission_ack_send(chan, msg->sysid, msg->compid, MAV_MISSION_ACCEPTED);
}else{
// we failed to clear the mission (perhaps because we're currently running it) so send NAK
mavlink_msg_mission_ack_send(chan, msg->sysid, msg->compid, MAV_MISSION_ERROR);
}
2012-04-30 04:17:14 -03:00
break;
}
2012-08-08 23:22:46 -03:00
case MAVLINK_MSG_ID_MISSION_SET_CURRENT:
2012-04-30 04:17:14 -03:00
{
// decode
2012-08-08 23:22:46 -03:00
mavlink_mission_set_current_t packet;
mavlink_msg_mission_set_current_decode(msg, &packet);
2012-04-30 04:17:14 -03:00
if (mavlink_check_target(packet.target_system,packet.target_component)) break;
// set current command
2014-03-10 05:44:05 -03:00
if (mission.set_current_cmd(packet.seq)) {
mavlink_msg_mission_current_send(chan, mission.get_current_nav_cmd().index);
}
2012-04-30 04:17:14 -03:00
break;
}
2012-08-08 23:22:46 -03:00
case MAVLINK_MSG_ID_MISSION_COUNT:
2012-04-30 04:17:14 -03:00
{
// decode
2012-08-08 23:22:46 -03:00
mavlink_mission_count_t packet;
mavlink_msg_mission_count_decode(msg, &packet);
2012-04-30 04:17:14 -03:00
if (mavlink_check_target(packet.target_system,packet.target_component)) break;
// start waypoint receiving
2014-03-12 03:25:51 -03:00
if (packet.count > mission.num_commands_max()) {
2014-03-10 05:44:05 -03:00
// send NAK
mavlink_msg_mission_ack_send(chan, msg->sysid, msg->compid, MAV_MISSION_NO_SPACE);
break;
}
2014-03-17 04:07:18 -03:00
// new mission arriving, truncate mission to be the same length
mission.truncate(packet.count);
2012-04-30 04:17:14 -03:00
waypoint_timelast_receive = millis();
waypoint_timelast_request = 0;
waypoint_receiving = true;
waypoint_request_i = 0;
2014-03-10 05:44:05 -03:00
waypoint_request_last= packet.count; // record how many commands we expect to receive
2012-04-30 04:17:14 -03:00
break;
}
2012-11-27 18:35:09 -04:00
case MAVLINK_MSG_ID_MISSION_WRITE_PARTIAL_LIST:
{
// decode
mavlink_mission_write_partial_list_t packet;
mavlink_msg_mission_write_partial_list_decode(msg, &packet);
if (mavlink_check_target(packet.target_system,packet.target_component)) break;
// start waypoint receiving
2014-03-10 05:44:05 -03:00
if (packet.start_index > mission.num_commands() ||
packet.end_index > mission.num_commands() ||
2012-11-27 18:35:09 -04:00
packet.end_index < packet.start_index) {
2012-12-18 07:44:12 -04:00
send_text_P(SEVERITY_LOW,PSTR("flight plan update rejected"));
2012-11-27 18:35:09 -04:00
break;
}
waypoint_timelast_receive = millis();
waypoint_timelast_request = 0;
waypoint_receiving = true;
waypoint_request_i = packet.start_index;
waypoint_request_last= packet.end_index;
break;
}
2012-04-30 04:17:14 -03:00
#ifdef MAVLINK_MSG_ID_SET_MAG_OFFSETS
case MAVLINK_MSG_ID_SET_MAG_OFFSETS:
{
mavlink_set_mag_offsets_t packet;
mavlink_msg_set_mag_offsets_decode(msg, &packet);
if (mavlink_check_target(packet.target_system,packet.target_component)) break;
compass.set_offsets(Vector3f(packet.mag_ofs_x, packet.mag_ofs_y, packet.mag_ofs_z));
break;
}
#endif
// XXX receive a WP from GCS and store in EEPROM
2012-08-08 23:22:46 -03:00
case MAVLINK_MSG_ID_MISSION_ITEM:
2012-04-30 04:17:14 -03:00
{
// decode
2012-08-08 23:22:46 -03:00
mavlink_mission_item_t packet;
2012-04-30 04:17:14 -03:00
uint8_t result = MAV_MISSION_ACCEPTED;
2012-08-08 23:22:46 -03:00
mavlink_msg_mission_item_decode(msg, &packet);
2012-04-30 04:17:14 -03:00
if (mavlink_check_target(packet.target_system,packet.target_component)) break;
2014-03-10 05:44:05 -03:00
// convert mavlink packet to mission command
if (!AP_Mission::mavlink_to_mission_cmd(packet, cmd)) {
result = MAV_MISSION_ERROR;
goto mission_failed;
2012-04-30 04:17:14 -03:00
}
if(packet.current == 2){ //current = 2 is a flag to tell us this is a "guided mode" waypoint and not for the mission
2014-03-10 05:44:05 -03:00
guided_WP = cmd.content.location;
2012-04-30 04:17:14 -03:00
// add home alt if needed
2014-03-10 05:44:05 -03:00
if (guided_WP.flags.relative_alt){
2012-04-30 04:17:14 -03:00
guided_WP.alt += home.alt;
}
set_mode(GUIDED);
// make any new wp uploaded instant (in case we are already in Guided mode)
set_guided_WP();
// verify we recevied the command
2012-08-08 23:22:46 -03:00
mavlink_msg_mission_ack_send(
2012-04-30 04:17:14 -03:00
chan,
msg->sysid,
msg->compid,
0);
} else {
// Check if receiving waypoints (mission upload expected)
if (!waypoint_receiving) {
result = MAV_MISSION_ERROR;
goto mission_failed;
}
// check if this is the requested waypoint
if (packet.seq != waypoint_request_i) {
result = MAV_MISSION_INVALID_SEQUENCE;
goto mission_failed;
}
2014-03-10 05:44:05 -03:00
// if command index is within the existing list, replace the command
if (packet.seq < mission.num_commands()) {
if (mission.replace_cmd(packet.seq,cmd)) {
result = MAV_MISSION_ACCEPTED;
}else{
result = MAV_MISSION_ERROR;
goto mission_failed;
}
// if command is at the end of command list, add the command
}else if (packet.seq == mission.num_commands()) {
if (mission.add_cmd(cmd)) {
result = MAV_MISSION_ACCEPTED;
}else{
result = MAV_MISSION_ERROR;
goto mission_failed;
}
// if beyond the end of the command list, return an error
}else{
result = MAV_MISSION_ERROR;
goto mission_failed;
}
2012-04-30 04:17:14 -03:00
// update waypoint receiving state machine
waypoint_timelast_receive = millis();
waypoint_timelast_request = 0;
waypoint_request_i++;
2014-03-10 05:44:05 -03:00
if (waypoint_request_i >= waypoint_request_last) {
2012-08-08 23:22:46 -03:00
mavlink_msg_mission_ack_send(
2012-04-30 04:17:14 -03:00
chan,
msg->sysid,
msg->compid,
2014-03-10 05:44:05 -03:00
MAV_MISSION_ACCEPTED);
2012-04-30 04:17:14 -03:00
2012-12-18 07:44:12 -04:00
send_text_P(SEVERITY_LOW,PSTR("flight plan received"));
2012-04-30 04:17:14 -03:00
waypoint_receiving = false;
// XXX ignores waypoint radius for individual waypoints, can
// only set WP_RADIUS parameter
}
}
break;
mission_failed:
// we are rejecting the mission/waypoint
2012-08-08 23:22:46 -03:00
mavlink_msg_mission_ack_send(
2012-04-30 04:17:14 -03:00
chan,
msg->sysid,
msg->compid,
result);
break;
}
case MAVLINK_MSG_ID_PARAM_SET:
{
AP_Param *vp;
enum ap_var_type var_type;
// decode
mavlink_param_set_t packet;
mavlink_msg_param_set_decode(msg, &packet);
if (mavlink_check_target(packet.target_system, packet.target_component))
break;
// set parameter
2012-11-20 06:39:11 -04:00
char key[AP_MAX_NAME_SIZE+1];
strncpy(key, (char *)packet.param_id, AP_MAX_NAME_SIZE);
key[AP_MAX_NAME_SIZE] = 0;
2012-04-30 04:17:14 -03:00
// find the requested parameter
vp = AP_Param::find(key, &var_type);
if ((NULL != vp) && // exists
!isnan(packet.param_value) && // not nan
!isinf(packet.param_value)) { // not inf
// add a small amount before casting parameter values
// from float to integer to avoid truncating to the
// next lower integer value.
float rounding_addition = 0.01;
// handle variables with standard type IDs
if (var_type == AP_PARAM_FLOAT) {
((AP_Float *)vp)->set_and_save(packet.param_value);
} else if (var_type == AP_PARAM_INT32) {
if (packet.param_value < 0) rounding_addition = -rounding_addition;
2012-05-14 15:33:03 -03:00
float v = packet.param_value+rounding_addition;
2013-11-23 06:57:26 -04:00
v = constrain_float(v, -2147483648.0, 2147483647.0);
2012-05-14 15:33:03 -03:00
((AP_Int32 *)vp)->set_and_save(v);
2012-04-30 04:17:14 -03:00
} else if (var_type == AP_PARAM_INT16) {
if (packet.param_value < 0) rounding_addition = -rounding_addition;
2012-05-14 15:33:03 -03:00
float v = packet.param_value+rounding_addition;
2013-05-01 21:26:12 -03:00
v = constrain_float(v, -32768, 32767);
2012-05-14 15:33:03 -03:00
((AP_Int16 *)vp)->set_and_save(v);
2012-04-30 04:17:14 -03:00
} else if (var_type == AP_PARAM_INT8) {
if (packet.param_value < 0) rounding_addition = -rounding_addition;
2012-05-14 15:33:03 -03:00
float v = packet.param_value+rounding_addition;
2013-05-01 21:26:12 -03:00
v = constrain_float(v, -128, 127);
2012-05-14 15:33:03 -03:00
((AP_Int8 *)vp)->set_and_save(v);
2012-04-30 04:17:14 -03:00
} else {
// we don't support mavlink set on this parameter
break;
}
// Report back the new value if we accepted the change
// we send the value we actually set, which could be
// different from the value sent, in case someone sent
// a fractional value to an integer type
mavlink_msg_param_value_send(
chan,
key,
vp->cast_to_float(var_type),
mav_var_type(var_type),
_count_parameters(),
-1); // XXX we don't actually know what its index is...
2013-04-19 04:53:07 -03:00
DataFlash.Log_Write_Parameter(key, vp->cast_to_float(var_type));
2012-04-30 04:17:14 -03:00
}
break;
} // end case
case MAVLINK_MSG_ID_RC_CHANNELS_OVERRIDE:
2012-12-18 07:44:12 -04:00
{
// allow override of RC channel values for HIL
// or for complete GCS control of switch position
// and RC PWM values.
if(msg->sysid != g.sysid_my_gcs) break; // Only accept control from our gcs
mavlink_rc_channels_override_t packet;
int16_t v[8];
mavlink_msg_rc_channels_override_decode(msg, &packet);
if (mavlink_check_target(packet.target_system,packet.target_component))
2012-04-30 04:17:14 -03:00
break;
2012-12-18 07:44:12 -04:00
v[0] = packet.chan1_raw;
v[1] = packet.chan2_raw;
v[2] = packet.chan3_raw;
v[3] = packet.chan4_raw;
v[4] = packet.chan5_raw;
v[5] = packet.chan6_raw;
v[6] = packet.chan7_raw;
v[7] = packet.chan8_raw;
hal.rcin->set_overrides(v, 8);
2013-03-28 20:25:53 -03:00
failsafe.rc_override_timer = millis();
failsafe_trigger(FAILSAFE_EVENT_RC, false);
2012-12-18 07:44:12 -04:00
break;
}
2012-04-30 04:17:14 -03:00
case MAVLINK_MSG_ID_HEARTBEAT:
{
// We keep track of the last time we received a heartbeat from our GCS for failsafe purposes
if(msg->sysid != g.sysid_my_gcs) break;
2013-03-28 20:25:53 -03:00
last_heartbeat_ms = failsafe.rc_override_timer = millis();
failsafe_trigger(FAILSAFE_EVENT_GCS, false);
2012-04-30 04:17:14 -03:00
break;
}
2012-12-18 16:17:06 -04:00
#if HIL_MODE != HIL_MODE_DISABLED
2012-04-30 04:17:14 -03:00
case MAVLINK_MSG_ID_HIL_STATE:
{
mavlink_hil_state_t packet;
mavlink_msg_hil_state_decode(msg, &packet);
2012-12-20 00:16:18 -04:00
float vel = pythagorous2(packet.vx, packet.vy);
2013-01-10 14:42:24 -04:00
float cog = wrap_360_cd(ToDeg(atan2f(packet.vy, packet.vx)) * 100);
2012-04-30 04:17:14 -03:00
// set gps hil sensor
2014-02-25 06:11:06 -04:00
g_gps->setHIL(GPS::FIX_3D,
packet.time_usec/1000,
2013-01-10 14:42:24 -04:00
packet.lat*1.0e-7f, packet.lon*1.0e-7f, packet.alt*1.0e-3f,
vel*1.0e-2f, cog*1.0e-2f, 0, 10);
2012-04-30 04:17:14 -03:00
// rad/sec
Vector3f gyros;
2012-12-18 16:17:06 -04:00
gyros.x = packet.rollspeed;
gyros.y = packet.pitchspeed;
gyros.z = packet.yawspeed;
2012-12-18 15:30:42 -04:00
2012-04-30 04:17:14 -03:00
// m/s/s
Vector3f accels;
2013-01-10 14:42:24 -04:00
accels.x = packet.xacc * (GRAVITY_MSS/1000.0f);
accels.y = packet.yacc * (GRAVITY_MSS/1000.0f);
accels.z = packet.zacc * (GRAVITY_MSS/1000.0f);
2012-12-18 16:17:06 -04:00
2014-02-22 17:18:15 -04:00
ins.set_gyro(0, gyros);
2012-04-30 04:17:14 -03:00
2014-02-22 17:18:15 -04:00
ins.set_accel(0, accels);
2013-05-02 01:59:48 -03:00
compass.setHIL(packet.roll, packet.pitch, packet.yaw);
2013-06-03 22:57:59 -03:00
break;
2012-04-30 04:17:14 -03:00
}
#endif // HIL_MODE
2013-07-14 20:57:00 -03:00
#if CAMERA == ENABLED
case MAVLINK_MSG_ID_DIGICAM_CONFIGURE:
{
camera.configure_msg(msg);
break;
}
case MAVLINK_MSG_ID_DIGICAM_CONTROL:
{
camera.control_msg(msg);
break;
}
#endif // CAMERA == ENABLED
2012-04-30 04:17:14 -03:00
#if MOUNT == ENABLED
case MAVLINK_MSG_ID_MOUNT_CONFIGURE:
{
camera_mount.configure_msg(msg);
break;
}
case MAVLINK_MSG_ID_MOUNT_CONTROL:
{
camera_mount.control_msg(msg);
break;
}
case MAVLINK_MSG_ID_MOUNT_STATUS:
{
camera_mount.status_msg(msg);
break;
}
#endif // MOUNT == ENABLED
2012-05-14 15:33:03 -03:00
case MAVLINK_MSG_ID_RADIO:
2013-08-24 04:58:37 -03:00
case MAVLINK_MSG_ID_RADIO_STATUS:
2012-05-14 15:33:03 -03:00
{
mavlink_radio_t packet;
mavlink_msg_radio_decode(msg, &packet);
// use the state of the transmit buffer in the radio to
// control the stream rate, giving us adaptive software
// flow control
if (packet.txbuf < 20 && stream_slowdown < 100) {
// we are very low on space - slow down a lot
stream_slowdown += 3;
} else if (packet.txbuf < 50 && stream_slowdown < 100) {
// we are a bit low on space, slow down slightly
stream_slowdown += 1;
} else if (packet.txbuf > 95 && stream_slowdown > 10) {
// the buffer has plenty of space, speed up a lot
stream_slowdown -= 2;
} else if (packet.txbuf > 90 && stream_slowdown != 0) {
// the buffer has enough space, speed up a bit
stream_slowdown--;
}
break;
}
2014-01-14 00:10:13 -04:00
case MAVLINK_MSG_ID_LOG_REQUEST_DATA:
case MAVLINK_MSG_ID_LOG_ERASE:
in_log_download = true;
// fallthru
case MAVLINK_MSG_ID_LOG_REQUEST_LIST:
if (!in_mavlink_delay) {
handle_log_message(msg, DataFlash);
}
break;
case MAVLINK_MSG_ID_LOG_REQUEST_END:
in_log_download = false;
2013-12-15 19:35:27 -04:00
if (!in_mavlink_delay) {
handle_log_message(msg, DataFlash);
}
break;
2012-12-18 15:30:42 -04:00
default:
// forward unknown messages to the other link if there is one
2013-11-23 06:57:26 -04:00
for (uint8_t i=0; i<num_gcs; i++) {
if (gcs[i].initialised && i != (uint8_t)chan) {
mavlink_channel_t out_chan = (mavlink_channel_t)i;
// only forward if it would fit in the transmit buffer
2012-12-18 15:30:42 -04:00
if (comm_get_txspace(out_chan) > ((uint16_t)msg->len) + MAVLINK_NUM_NON_PAYLOAD_BYTES) {
_mavlink_resend_uart(out_chan, msg);
}
}
2013-11-23 06:57:26 -04:00
}
2012-12-18 15:30:42 -04:00
break;
2012-04-30 04:17:14 -03:00
} // end switch
} // end handle mavlink
/*
2012-12-18 07:44:12 -04:00
* a delay() callback that processes MAVLink packets. We set this as the
* callback in long running library initialisation routines to allow
* MAVLink to process packets while waiting for the initialisation to
* complete
*/
static void mavlink_delay_cb()
2012-04-30 04:17:14 -03:00
{
2012-12-18 07:44:12 -04:00
static uint32_t last_1hz, last_50hz, last_5s;
2014-01-13 23:40:10 -04:00
if (!gcs[0].initialised || in_mavlink_delay) return;
2012-04-30 04:17:14 -03:00
in_mavlink_delay = true;
2012-12-18 07:44:12 -04:00
uint32_t tnow = millis();
if (tnow - last_1hz > 1000) {
last_1hz = tnow;
gcs_send_message(MSG_HEARTBEAT);
gcs_send_message(MSG_EXTENDED_STATUS1);
}
if (tnow - last_50hz > 20) {
last_50hz = tnow;
gcs_update();
gcs_data_stream_send();
2013-08-29 00:14:16 -03:00
notify.update();
2012-12-18 07:44:12 -04:00
}
if (tnow - last_5s > 5000) {
last_5s = tnow;
gcs_send_text_P(SEVERITY_LOW, PSTR("Initialising APM..."));
}
check_usb_mux();
2012-04-30 04:17:14 -03:00
in_mavlink_delay = false;
}
/*
2012-12-18 15:30:42 -04:00
* send a message on both GCS links
2012-04-30 04:17:14 -03:00
*/
static void gcs_send_message(enum ap_message id)
{
2013-11-23 06:57:26 -04:00
for (uint8_t i=0; i<num_gcs; i++) {
if (gcs[i].initialised) {
gcs[i].send_message(id);
}
2012-04-30 04:17:14 -03:00
}
}
/*
2012-12-18 15:30:42 -04:00
* send data streams in the given rate range on both links
2012-04-30 04:17:14 -03:00
*/
2012-05-14 15:33:03 -03:00
static void gcs_data_stream_send(void)
2012-04-30 04:17:14 -03:00
{
2013-11-23 06:57:26 -04:00
for (uint8_t i=0; i<num_gcs; i++) {
if (gcs[i].initialised) {
gcs[i].data_stream_send();
}
2012-04-30 04:17:14 -03:00
}
}
/*
2012-12-18 15:30:42 -04:00
* look for incoming commands on the GCS links
2012-04-30 04:17:14 -03:00
*/
static void gcs_update(void)
{
2013-11-23 06:57:26 -04:00
for (uint8_t i=0; i<num_gcs; i++) {
if (gcs[i].initialised) {
gcs[i].update();
}
2012-04-30 04:17:14 -03:00
}
}
static void gcs_send_text_P(gcs_severity severity, const prog_char_t *str)
{
2013-11-23 06:57:26 -04:00
for (uint8_t i=0; i<num_gcs; i++) {
if (gcs[i].initialised) {
gcs[i].send_text_P(severity, str);
}
2012-04-30 04:17:14 -03:00
}
2013-11-23 06:57:26 -04:00
#if LOGGING_ENABLED == ENABLED
2013-05-02 20:18:33 -03:00
DataFlash.Log_Write_Message_P(str);
2013-11-23 06:57:26 -04:00
#endif
2012-04-30 04:17:14 -03:00
}
/*
2012-12-18 07:44:12 -04:00
* send a low priority formatted message to the GCS
* only one fits in the queue, so if you send more than one before the
* last one gets into the serial buffer then the old one will be lost
2012-04-30 04:17:14 -03:00
*/
2012-12-18 07:44:12 -04:00
void gcs_send_text_fmt(const prog_char_t *fmt, ...)
2012-04-30 04:17:14 -03:00
{
2012-12-18 07:44:12 -04:00
va_list arg_list;
2013-11-23 06:57:26 -04:00
gcs[0].pending_status.severity = (uint8_t)SEVERITY_LOW;
2012-12-18 07:44:12 -04:00
va_start(arg_list, fmt);
2013-11-23 06:57:26 -04:00
hal.util->vsnprintf_P((char *)gcs[0].pending_status.text,
sizeof(gcs[0].pending_status.text), fmt, arg_list);
2012-12-18 07:44:12 -04:00
va_end(arg_list);
2013-11-23 06:57:26 -04:00
#if LOGGING_ENABLED == ENABLED
DataFlash.Log_Write_Message(gcs[0].pending_status.text);
#endif
2012-04-30 04:17:14 -03:00
mavlink_send_message(MAVLINK_COMM_0, MSG_STATUSTEXT, 0);
2013-11-23 06:57:26 -04:00
for (uint8_t i=1; i<num_gcs; i++) {
if (gcs[i].initialised) {
gcs[i].pending_status = gcs[0].pending_status;
mavlink_send_message((mavlink_channel_t)i, MSG_STATUSTEXT, 0);
}
2012-04-30 04:17:14 -03:00
}
}
2012-05-14 15:33:03 -03:00
2013-10-27 20:33:52 -03:00
/**
retry any deferred messages
*/
static void gcs_retry_deferred(void)
{
gcs_send_message(MSG_RETRY_DEFERRED);
}