mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-18 06:38:29 -04:00
Rover: new failsafe logic
this obeys FS_TIMEOUT and FS_ACTION
This commit is contained in:
parent
0ed3061d32
commit
caaf32211e
@ -293,16 +293,14 @@ static bool rc_override_active = false;
|
|||||||
// Failsafe
|
// Failsafe
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
// A tracking variable for type of failsafe active
|
// A tracking variable for type of failsafe active
|
||||||
// Used for failsafe based on loss of RC signal or GCS signal
|
// Used for failsafe based on loss of RC signal or GCS signal. See
|
||||||
static int16_t failsafe;
|
// FAILSAFE_EVENT_*
|
||||||
// Used to track if the value on channel 3 (throtttle) has fallen below the failsafe threshold
|
static struct {
|
||||||
// RC receiver should be set up to output a low throttle value when signal is lost
|
uint8_t bits;
|
||||||
static bool ch3_failsafe;
|
uint32_t rc_override_timer;
|
||||||
|
uint32_t start_time;
|
||||||
// A timer used to track how long since we have received the last GCS heartbeat or rc override message
|
uint8_t triggered;
|
||||||
static uint32_t rc_override_fs_timer = 0;
|
} failsafe;
|
||||||
// A timer used to track how long we have been in a "short failsafe" condition due to loss of RC signal
|
|
||||||
static uint32_t ch3_failsafe_timer = 0;
|
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
// LED output
|
// LED output
|
||||||
@ -667,6 +665,7 @@ static void medium_loop()
|
|||||||
// This case deals with the GPS
|
// This case deals with the GPS
|
||||||
//-------------------------------
|
//-------------------------------
|
||||||
case 0:
|
case 0:
|
||||||
|
failsafe_trigger(FAILSAFE_EVENT_GCS, last_heartbeat_ms != 0 && (millis() - last_heartbeat_ms) > 2000);
|
||||||
medium_loopCounter++;
|
medium_loopCounter++;
|
||||||
update_GPS();
|
update_GPS();
|
||||||
|
|
||||||
@ -744,7 +743,6 @@ static void slow_loop()
|
|||||||
switch (slow_loopCounter){
|
switch (slow_loopCounter){
|
||||||
case 0:
|
case 0:
|
||||||
slow_loopCounter++;
|
slow_loopCounter++;
|
||||||
check_long_failsafe();
|
|
||||||
superslow_loopCounter++;
|
superslow_loopCounter++;
|
||||||
if(superslow_loopCounter >=200) { // 200 = Execute every minute
|
if(superslow_loopCounter >=200) { // 200 = Execute every minute
|
||||||
#if HIL_MODE != HIL_MODE_ATTITUDE
|
#if HIL_MODE != HIL_MODE_ATTITUDE
|
||||||
|
@ -25,7 +25,7 @@ static NOINLINE void send_heartbeat(mavlink_channel_t chan)
|
|||||||
uint8_t system_status = MAV_STATE_ACTIVE;
|
uint8_t system_status = MAV_STATE_ACTIVE;
|
||||||
uint32_t custom_mode = control_mode;
|
uint32_t custom_mode = control_mode;
|
||||||
|
|
||||||
if (failsafe != FAILSAFE_NONE) {
|
if (failsafe.triggered != 0) {
|
||||||
system_status = MAV_STATE_CRITICAL;
|
system_status = MAV_STATE_CRITICAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1631,7 +1631,8 @@ void GCS_MAVLINK::handleMessage(mavlink_message_t* msg)
|
|||||||
|
|
||||||
hal.rcin->set_overrides(v, 8);
|
hal.rcin->set_overrides(v, 8);
|
||||||
|
|
||||||
rc_override_fs_timer = millis();
|
failsafe.rc_override_timer = millis();
|
||||||
|
failsafe_trigger(FAILSAFE_EVENT_RC, false);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1639,7 +1640,8 @@ void GCS_MAVLINK::handleMessage(mavlink_message_t* msg)
|
|||||||
{
|
{
|
||||||
// We keep track of the last time we received a heartbeat from our GCS for failsafe purposes
|
// 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;
|
if(msg->sysid != g.sysid_my_gcs) break;
|
||||||
last_heartbeat_ms = rc_override_fs_timer = millis();
|
last_heartbeat_ms = failsafe.rc_override_timer = millis();
|
||||||
|
failsafe_trigger(FAILSAFE_EVENT_GCS, false);
|
||||||
pmTest1++;
|
pmTest1++;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -241,16 +241,16 @@ const AP_Param::Info var_info[] PROGMEM = {
|
|||||||
// @Param: FS_ACTION
|
// @Param: FS_ACTION
|
||||||
// @DisplayName: Failsafe Action
|
// @DisplayName: Failsafe Action
|
||||||
// @Description: What to do on a failsafe event
|
// @Description: What to do on a failsafe event
|
||||||
// @Values: 0:Nothing,1:RTL,2:STOP
|
// @Values: 0:Nothing,1:RTL,2:HOLD
|
||||||
// @User: Standard
|
// @User: Standard
|
||||||
GSCALAR(fs_action, "FS_ACTION", 0),
|
GSCALAR(fs_action, "FS_ACTION", 1),
|
||||||
|
|
||||||
// @Param: FS_TIMEOUT
|
// @Param: FS_TIMEOUT
|
||||||
// @DisplayName: Failsafe timeout
|
// @DisplayName: Failsafe timeout
|
||||||
// @Description: How long a failsafe event need to happen for before we trigger the failsafe action
|
// @Description: How long a failsafe event need to happen for before we trigger the failsafe action
|
||||||
// @Units: seconds
|
// @Units: seconds
|
||||||
// @User: Standard
|
// @User: Standard
|
||||||
GSCALAR(fs_timeout, "FS_TIMEOUT", 10),
|
GSCALAR(fs_timeout, "FS_TIMEOUT", 5),
|
||||||
|
|
||||||
// @Param: FS_THR_ENABLE
|
// @Param: FS_THR_ENABLE
|
||||||
// @DisplayName: Throttle Failsafe Enable
|
// @DisplayName: Throttle Failsafe Enable
|
||||||
|
@ -157,11 +157,21 @@ static void set_servos(void)
|
|||||||
// do a direct pass through of radio values
|
// do a direct pass through of radio values
|
||||||
g.channel_steer.radio_out = hal.rcin->read(CH_STEER);
|
g.channel_steer.radio_out = hal.rcin->read(CH_STEER);
|
||||||
g.channel_throttle.radio_out = hal.rcin->read(CH_THROTTLE);
|
g.channel_throttle.radio_out = hal.rcin->read(CH_THROTTLE);
|
||||||
|
if (failsafe.bits & FAILSAFE_EVENT_THROTTLE) {
|
||||||
|
// suppress throttle if in failsafe and manual
|
||||||
|
g.channel_throttle.radio_out = g.channel_throttle.radio_trim;
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
g.channel_steer.calc_pwm();
|
g.channel_steer.calc_pwm();
|
||||||
g.channel_throttle.servo_out = constrain_int16(g.channel_throttle.servo_out,
|
g.channel_throttle.servo_out = constrain_int16(g.channel_throttle.servo_out,
|
||||||
g.throttle_min.get(),
|
g.throttle_min.get(),
|
||||||
g.throttle_max.get());
|
g.throttle_max.get());
|
||||||
|
|
||||||
|
if ((failsafe.bits & FAILSAFE_EVENT_THROTTLE) && control_mode < AUTO) {
|
||||||
|
// suppress throttle if in failsafe
|
||||||
|
g.channel_throttle.servo_out = 0;
|
||||||
|
}
|
||||||
|
|
||||||
// convert 0 to 100% into PWM
|
// convert 0 to 100% into PWM
|
||||||
g.channel_throttle.calc_pwm();
|
g.channel_throttle.calc_pwm();
|
||||||
|
|
||||||
|
@ -14,16 +14,6 @@
|
|||||||
#define DEBUG 0
|
#define DEBUG 0
|
||||||
#define SERVO_MAX 4500 // This value represents 45 degrees and is just an arbitrary representation of servo max travel.
|
#define SERVO_MAX 4500 // This value represents 45 degrees and is just an arbitrary representation of servo max travel.
|
||||||
|
|
||||||
// failsafe
|
|
||||||
// ----------------------
|
|
||||||
#define FAILSAFE_NONE 0
|
|
||||||
#define FAILSAFE_SHORT 1
|
|
||||||
#define FAILSAFE_LONG 2
|
|
||||||
#define FAILSAFE_GCS 3
|
|
||||||
#define FAILSAFE_SHORT_TIME 1500 // Miliiseconds
|
|
||||||
#define FAILSAFE_LONG_TIME 20000 // Miliiseconds
|
|
||||||
|
|
||||||
|
|
||||||
// active altitude sensor
|
// active altitude sensor
|
||||||
// ----------------------
|
// ----------------------
|
||||||
#define SONAR 0
|
#define SONAR 0
|
||||||
@ -73,6 +63,11 @@ enum mode {
|
|||||||
INITIALISING=16
|
INITIALISING=16
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// types of failsafe events
|
||||||
|
#define FAILSAFE_EVENT_THROTTLE (1<<0)
|
||||||
|
#define FAILSAFE_EVENT_GCS (1<<1)
|
||||||
|
#define FAILSAFE_EVENT_RC (1<<2)
|
||||||
|
|
||||||
// Commands - Note that APM now uses a subset of the MAVLink protocol commands. See enum MAV_CMD in the GCS_Mavlink library
|
// Commands - Note that APM now uses a subset of the MAVLink protocol commands. See enum MAV_CMD in the GCS_Mavlink library
|
||||||
#define CMD_BLANK 0 // there is no command stored in the mem location requested
|
#define CMD_BLANK 0 // there is no command stored in the mem location requested
|
||||||
#define NO_COMMAND 0
|
#define NO_COMMAND 0
|
||||||
|
@ -1,33 +1,6 @@
|
|||||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*-
|
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*-
|
||||||
|
|
||||||
|
|
||||||
static void failsafe_long_on_event(int fstype)
|
|
||||||
{
|
|
||||||
// This is how to handle a long loss of control signal failsafe.
|
|
||||||
gcs_send_text_P(SEVERITY_LOW, PSTR("Failsafe - Long event on, "));
|
|
||||||
hal.rcin->clear_overrides();
|
|
||||||
failsafe = fstype;
|
|
||||||
switch(control_mode)
|
|
||||||
{
|
|
||||||
case MANUAL:
|
|
||||||
case LEARNING:
|
|
||||||
case STEERING:
|
|
||||||
set_mode(RTL);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case AUTO:
|
|
||||||
case GUIDED:
|
|
||||||
set_mode(RTL);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case RTL:
|
|
||||||
case HOLD:
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
gcs_send_text_fmt(PSTR("flight mode = %u"), (unsigned)control_mode);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void update_events(void) // Used for MAV_CMD_DO_REPEAT_SERVO and MAV_CMD_DO_REPEAT_RELAY
|
static void update_events(void) // Used for MAV_CMD_DO_REPEAT_SERVO and MAV_CMD_DO_REPEAT_RELAY
|
||||||
{
|
{
|
||||||
if(event_repeat == 0 || (millis() - event_timer) < event_delay)
|
if(event_repeat == 0 || (millis() - event_timer) < event_delay)
|
||||||
|
@ -1,10 +1,5 @@
|
|||||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*-
|
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*-
|
||||||
|
|
||||||
//Function that will read the radio data, limit servos and trigger a failsafe
|
|
||||||
// ----------------------------------------------------------------------------
|
|
||||||
static uint8_t failsafeCounter = 0; // we wait a second to take over the throttle and send the rover circling
|
|
||||||
|
|
||||||
|
|
||||||
static void init_rc_in()
|
static void init_rc_in()
|
||||||
{
|
{
|
||||||
// set rc channel ranges
|
// set rc channel ranges
|
||||||
@ -117,41 +112,9 @@ static void control_failsafe(uint16_t pwm)
|
|||||||
|
|
||||||
// Check for failsafe condition based on loss of GCS control
|
// Check for failsafe condition based on loss of GCS control
|
||||||
if (rc_override_active) {
|
if (rc_override_active) {
|
||||||
if(millis() - rc_override_fs_timer > FAILSAFE_SHORT_TIME) {
|
failsafe_trigger(FAILSAFE_EVENT_RC, (millis() - failsafe.rc_override_timer) > 1500);
|
||||||
ch3_failsafe = true;
|
|
||||||
} else {
|
|
||||||
ch3_failsafe = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
//Check for failsafe and debounce funky reads
|
|
||||||
} else if (g.fs_throttle_enabled) {
|
} else if (g.fs_throttle_enabled) {
|
||||||
if (pwm < (unsigned)g.fs_throttle_value){
|
failsafe_trigger(FAILSAFE_EVENT_THROTTLE, pwm < (uint16_t)g.fs_throttle_value);
|
||||||
// we detect a failsafe from radio
|
|
||||||
// throttle has dropped below the mark
|
|
||||||
failsafeCounter++;
|
|
||||||
if (failsafeCounter == 9){
|
|
||||||
gcs_send_text_fmt(PSTR("MSG FS ON %u"), (unsigned)pwm);
|
|
||||||
}else if(failsafeCounter == 10) {
|
|
||||||
ch3_failsafe = true;
|
|
||||||
}else if (failsafeCounter > 10){
|
|
||||||
failsafeCounter = 11;
|
|
||||||
}
|
|
||||||
|
|
||||||
}else if(failsafeCounter > 0){
|
|
||||||
// we are no longer in failsafe condition
|
|
||||||
// but we need to recover quickly
|
|
||||||
failsafeCounter--;
|
|
||||||
if (failsafeCounter > 3){
|
|
||||||
failsafeCounter = 3;
|
|
||||||
}
|
|
||||||
if (failsafeCounter == 1){
|
|
||||||
gcs_send_text_fmt(PSTR("MSG FS OFF %u"), (unsigned)pwm);
|
|
||||||
}else if(failsafeCounter == 0) {
|
|
||||||
ch3_failsafe = false;
|
|
||||||
}else if (failsafeCounter <0){
|
|
||||||
failsafeCounter = -1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -337,25 +337,45 @@ static void set_mode(enum mode mode)
|
|||||||
Log_Write_Mode(control_mode);
|
Log_Write_Mode(control_mode);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void check_long_failsafe()
|
/*
|
||||||
|
called to set/unset a failsafe event.
|
||||||
|
*/
|
||||||
|
static void failsafe_trigger(uint8_t failsafe_type, bool on)
|
||||||
{
|
{
|
||||||
// only act on changes
|
uint8_t old_bits = failsafe.bits;
|
||||||
// -------------------
|
if (on) {
|
||||||
if(failsafe != FAILSAFE_LONG && failsafe != FAILSAFE_GCS){
|
failsafe.bits |= failsafe_type;
|
||||||
if(rc_override_active && millis() - rc_override_fs_timer > FAILSAFE_LONG_TIME) {
|
|
||||||
failsafe_long_on_event(FAILSAFE_LONG);
|
|
||||||
}
|
|
||||||
if(! rc_override_active && failsafe == FAILSAFE_SHORT && millis() - ch3_failsafe_timer > FAILSAFE_LONG_TIME) {
|
|
||||||
failsafe_long_on_event(FAILSAFE_LONG);
|
|
||||||
}
|
|
||||||
if (g.fs_gcs_enabled && millis() - rc_override_fs_timer > FAILSAFE_LONG_TIME) {
|
|
||||||
failsafe_long_on_event(FAILSAFE_GCS);
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
// We do not change state but allow for user to change mode
|
failsafe.bits &= ~failsafe_type;
|
||||||
if(failsafe == FAILSAFE_GCS && millis() - rc_override_fs_timer < FAILSAFE_SHORT_TIME) failsafe = FAILSAFE_NONE;
|
}
|
||||||
if(failsafe == FAILSAFE_LONG && rc_override_active && millis() - rc_override_fs_timer < FAILSAFE_SHORT_TIME) failsafe = FAILSAFE_NONE;
|
if (old_bits == 0 && failsafe.bits != 0) {
|
||||||
if(failsafe == FAILSAFE_LONG && !rc_override_active && !ch3_failsafe) failsafe = FAILSAFE_NONE;
|
// a failsafe event has started
|
||||||
|
failsafe.start_time = millis();
|
||||||
|
}
|
||||||
|
if (failsafe.triggered != 0 && failsafe.bits == 0) {
|
||||||
|
// a failsafe event has ended
|
||||||
|
gcs_send_text_fmt(PSTR("Failsafe ended"));
|
||||||
|
}
|
||||||
|
|
||||||
|
failsafe.triggered &= failsafe.bits;
|
||||||
|
|
||||||
|
if (failsafe.triggered == 0 &&
|
||||||
|
failsafe.bits != 0 &&
|
||||||
|
millis() - failsafe.start_time > g.fs_timeout*1000 &&
|
||||||
|
control_mode != RTL &&
|
||||||
|
control_mode != HOLD) {
|
||||||
|
failsafe.triggered = failsafe.bits;
|
||||||
|
gcs_send_text_fmt(PSTR("Failsafe trigger 0x%x"), (unsigned)failsafe.triggered);
|
||||||
|
switch (g.fs_action) {
|
||||||
|
case 0:
|
||||||
|
break;
|
||||||
|
case 1:
|
||||||
|
set_mode(RTL);
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
set_mode(HOLD);
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user