#include "Plane.h" void Plane::failsafe_short_on_event(enum failsafe_state fstype, mode_reason_t reason) { // This is how to handle a short loss of control signal failsafe. failsafe.state = fstype; failsafe.short_timer_ms = millis(); gcs().send_text(MAV_SEVERITY_WARNING, "Failsafe. Short event on: type=%u/reason=%u", fstype, reason); switch(control_mode) { case MANUAL: case STABILIZE: case ACRO: case FLY_BY_WIRE_A: case AUTOTUNE: case FLY_BY_WIRE_B: case CRUISE: case TRAINING: failsafe.saved_mode = control_mode; failsafe.saved_mode_set = true; if(g.fs_action_short == FS_ACTION_SHORT_FBWA) { set_mode(FLY_BY_WIRE_A, reason); } else { set_mode(CIRCLE, reason); } break; case QSTABILIZE: case QLOITER: case QHOVER: failsafe.saved_mode = control_mode; failsafe.saved_mode_set = true; set_mode(QLAND, reason); break; case AUTO: case AVOID_ADSB: case GUIDED: case LOITER: if(g.fs_action_short != FS_ACTION_SHORT_BESTGUESS) { failsafe.saved_mode = control_mode; failsafe.saved_mode_set = true; if(g.fs_action_short == FS_ACTION_SHORT_FBWA) { set_mode(FLY_BY_WIRE_A, reason); } else { set_mode(CIRCLE, reason); } } break; case CIRCLE: case RTL: case QLAND: case QRTL: default: break; } gcs().send_text(MAV_SEVERITY_INFO, "Flight mode = %u", (unsigned)control_mode); } void Plane::failsafe_long_on_event(enum failsafe_state fstype, mode_reason_t reason) { // This is how to handle a long loss of control signal failsafe. gcs().send_text(MAV_SEVERITY_WARNING, "Failsafe. Long event on: type=%u/reason=%u", fstype, reason); // If the GCS is locked up we allow control to revert to RC hal.rcin->clear_overrides(); failsafe.state = fstype; switch(control_mode) { case MANUAL: case STABILIZE: case ACRO: case FLY_BY_WIRE_A: case AUTOTUNE: case FLY_BY_WIRE_B: case CRUISE: case TRAINING: case CIRCLE: if(g.fs_action_long == FS_ACTION_LONG_PARACHUTE) { #if PARACHUTE == ENABLED parachute_release(); #endif } else if (g.fs_action_long == FS_ACTION_LONG_GLIDE) { set_mode(FLY_BY_WIRE_A, reason); } else { set_mode(RTL, reason); } break; case QSTABILIZE: case QHOVER: case QLOITER: set_mode(QLAND, reason); break; case AUTO: case AVOID_ADSB: case GUIDED: case LOITER: if(g.fs_action_long == FS_ACTION_LONG_PARACHUTE) { #if PARACHUTE == ENABLED parachute_release(); #endif } else if (g.fs_action_long == FS_ACTION_LONG_GLIDE) { set_mode(FLY_BY_WIRE_A, reason); } else if (g.fs_action_long == FS_ACTION_LONG_RTL) { set_mode(RTL, reason); } break; case RTL: case QLAND: case QRTL: default: break; } gcs().send_text(MAV_SEVERITY_INFO, "Flight mode = %u", (unsigned)control_mode); } void Plane::failsafe_short_off_event(mode_reason_t reason) { // We're back in radio contact gcs().send_text(MAV_SEVERITY_WARNING, "Failsafe. Short event off: reason=%u", reason); failsafe.state = FAILSAFE_NONE; // re-read the switch so we can return to our preferred mode // -------------------------------------------------------- if (control_mode == CIRCLE && failsafe.saved_mode_set) { failsafe.saved_mode_set = false; set_mode(failsafe.saved_mode, reason); } } void Plane::failsafe_long_off_event(mode_reason_t reason) { // We're back in radio contact gcs().send_text(MAV_SEVERITY_WARNING, "Failsafe. Long event off: reason=%u", reason); failsafe.state = FAILSAFE_NONE; } void Plane::low_battery_event(void) { if (failsafe.low_battery) { return; } gcs().send_text(MAV_SEVERITY_WARNING, "Low battery %.2fV used %.0f mAh", (double)battery.voltage(), (double)battery.current_total_mah()); if (flight_stage != AP_Vehicle::FixedWing::FLIGHT_LAND) { set_mode(RTL, MODE_REASON_BATTERY_FAILSAFE); aparm.throttle_cruise.load(); } failsafe.low_battery = true; AP_Notify::flags.failsafe_battery = true; } void Plane::update_events(void) { ServoRelayEvents.update_events(); }