/* failsafe support Andrew Tridgell, December 2011 */ #include "Rover.h" /* our failsafe strategy is to detect main loop lockup and switch to passing inputs straight from the RC inputs to RC outputs. */ /* this failsafe_check function is called from the core timer interrupt at 1kHz. */ void Rover::failsafe_check() { static uint16_t last_mainLoop_count; static uint32_t last_timestamp; static bool in_failsafe; const uint32_t tnow = AP_HAL::micros(); if (mainLoop_count != last_mainLoop_count) { // the main loop is running, all is OK last_mainLoop_count = mainLoop_count; last_timestamp = tnow; in_failsafe = false; return; } if (tnow - last_timestamp > 200000) { // we have gone at least 0.2 seconds since the main loop // ran. That means we're in trouble, or perhaps are in // an initialisation routine or log erase. Start passing RC // inputs through to outputs in_failsafe = true; } if (in_failsafe && tnow - last_timestamp > 20000 && channel_throttle->read() >= static_cast(g.fs_throttle_value)) { // pass RC inputs to outputs every 20ms last_timestamp = tnow; hal.rcin->clear_overrides(); for (uint8_t ch = 0; ch < 4; ch++) { hal.rcout->write(ch, hal.rcin->read(ch)); } SRV_Channels::copy_radio_in_out(SRV_Channel::k_manual, true); } } /* called to set/unset a failsafe event. */ void Rover::failsafe_trigger(uint8_t failsafe_type, bool on) { uint8_t old_bits = failsafe.bits; if (on) { failsafe.bits |= failsafe_type; } else { failsafe.bits &= ~failsafe_type; } if (old_bits == 0 && failsafe.bits != 0) { // 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(MAV_SEVERITY_INFO, "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(MAV_SEVERITY_WARNING, "Failsafe trigger 0x%x", static_cast(failsafe.triggered)); switch (g.fs_action) { case 0: break; case 1: set_mode(RTL); break; case 2: set_mode(HOLD); break; } } } #if ADVANCED_FAILSAFE == ENABLED /* check for AFS failsafe check */ void Rover::afs_fs_check(void) { // perform AFS failsafe checks g2.afs.check(rover.last_heartbeat_ms, false, failsafe.last_valid_rc_ms); // Rover don't have fence } #endif