2011-09-08 22:29:39 -03:00
|
|
|
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*-
|
|
|
|
|
|
|
|
|
2013-07-19 01:11:16 -03:00
|
|
|
static void failsafe_short_on_event(enum failsafe_state fstype)
|
2011-09-08 22:29:39 -03:00
|
|
|
{
|
2012-08-16 21:50:15 -03:00
|
|
|
// This is how to handle a short loss of control signal failsafe.
|
2013-07-19 01:11:16 -03:00
|
|
|
failsafe.state = fstype;
|
|
|
|
failsafe.ch3_timer_ms = millis();
|
2011-12-09 12:06:45 -04:00
|
|
|
gcs_send_text_P(SEVERITY_LOW, PSTR("Failsafe - Short event on, "));
|
2012-08-16 21:50:15 -03:00
|
|
|
switch(control_mode)
|
|
|
|
{
|
|
|
|
case MANUAL:
|
|
|
|
case STABILIZE:
|
2013-07-10 10:25:38 -03:00
|
|
|
case ACRO:
|
2012-12-04 02:32:37 -04:00
|
|
|
case FLY_BY_WIRE_A:
|
|
|
|
case FLY_BY_WIRE_B:
|
2013-07-13 07:05:53 -03:00
|
|
|
case CRUISE:
|
2012-12-04 02:32:37 -04:00
|
|
|
case TRAINING:
|
2013-07-20 04:08:58 -03:00
|
|
|
failsafe.saved_mode = control_mode;
|
|
|
|
failsafe.saved_mode_set = 1;
|
2013-09-13 21:30:13 -03:00
|
|
|
if(g.short_fs_action == 2) {
|
|
|
|
set_mode(FLY_BY_WIRE_A);
|
|
|
|
} else {
|
|
|
|
set_mode(CIRCLE);
|
|
|
|
}
|
2012-08-16 21:50:15 -03:00
|
|
|
break;
|
|
|
|
|
|
|
|
case AUTO:
|
|
|
|
case GUIDED:
|
|
|
|
case LOITER:
|
2013-09-13 21:30:13 -03:00
|
|
|
if(g.short_fs_action != 0) {
|
2013-07-20 04:08:58 -03:00
|
|
|
failsafe.saved_mode = control_mode;
|
|
|
|
failsafe.saved_mode_set = 1;
|
2013-09-13 21:30:13 -03:00
|
|
|
if(g.short_fs_action == 2) {
|
|
|
|
set_mode(FLY_BY_WIRE_A);
|
|
|
|
} else {
|
|
|
|
set_mode(CIRCLE);
|
|
|
|
}
|
2012-08-16 21:50:15 -03:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CIRCLE:
|
|
|
|
case RTL:
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2011-09-18 05:59:59 -03:00
|
|
|
gcs_send_text_fmt(PSTR("flight mode = %u"), (unsigned)control_mode);
|
2011-09-08 22:29:39 -03:00
|
|
|
}
|
|
|
|
|
2013-07-19 01:11:16 -03:00
|
|
|
static void failsafe_long_on_event(enum failsafe_state fstype)
|
2011-09-08 22:29:39 -03:00
|
|
|
{
|
2012-08-16 21:50:15 -03:00
|
|
|
// This is how to handle a long loss of control signal failsafe.
|
|
|
|
gcs_send_text_P(SEVERITY_LOW, PSTR("Failsafe - Long event on, "));
|
2012-12-04 18:22:21 -04:00
|
|
|
// If the GCS is locked up we allow control to revert to RC
|
|
|
|
hal.rcin->clear_overrides();
|
2013-07-19 01:11:16 -03:00
|
|
|
failsafe.state = fstype;
|
2012-08-16 21:50:15 -03:00
|
|
|
switch(control_mode)
|
|
|
|
{
|
|
|
|
case MANUAL:
|
|
|
|
case STABILIZE:
|
2013-07-10 10:25:38 -03:00
|
|
|
case ACRO:
|
2012-12-04 02:32:37 -04:00
|
|
|
case FLY_BY_WIRE_A:
|
|
|
|
case FLY_BY_WIRE_B:
|
2013-07-13 07:05:53 -03:00
|
|
|
case CRUISE:
|
2012-12-04 02:32:37 -04:00
|
|
|
case TRAINING:
|
2012-08-16 21:50:15 -03:00
|
|
|
case CIRCLE:
|
2013-09-13 21:30:13 -03:00
|
|
|
if(g.long_fs_action == 2) {
|
|
|
|
set_mode(FLY_BY_WIRE_A);
|
|
|
|
} else {
|
|
|
|
set_mode(RTL);
|
|
|
|
}
|
2012-08-16 21:50:15 -03:00
|
|
|
break;
|
|
|
|
|
|
|
|
case AUTO:
|
|
|
|
case GUIDED:
|
|
|
|
case LOITER:
|
2013-09-13 21:30:13 -03:00
|
|
|
if(g.long_fs_action == 2) {
|
|
|
|
set_mode(FLY_BY_WIRE_A);
|
|
|
|
} else if (g.long_fs_action == 1) {
|
2012-08-16 21:50:15 -03:00
|
|
|
set_mode(RTL);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case RTL:
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2011-12-09 12:06:45 -04:00
|
|
|
gcs_send_text_fmt(PSTR("flight mode = %u"), (unsigned)control_mode);
|
2011-09-08 22:29:39 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
static void failsafe_short_off_event()
|
|
|
|
{
|
2012-08-16 21:50:15 -03:00
|
|
|
// We're back in radio contact
|
|
|
|
gcs_send_text_P(SEVERITY_LOW, PSTR("Failsafe - Short event off"));
|
2013-07-19 01:11:16 -03:00
|
|
|
failsafe.state = FAILSAFE_NONE;
|
2011-09-08 22:29:39 -03:00
|
|
|
|
2012-08-16 21:50:15 -03:00
|
|
|
// re-read the switch so we can return to our preferred mode
|
|
|
|
// --------------------------------------------------------
|
2013-07-20 04:08:58 -03:00
|
|
|
if (control_mode == CIRCLE && failsafe.saved_mode_set) {
|
|
|
|
failsafe.saved_mode_set = 0;
|
|
|
|
set_mode(failsafe.saved_mode);
|
2012-03-02 21:50:46 -04:00
|
|
|
}
|
2011-09-08 22:29:39 -03:00
|
|
|
}
|
|
|
|
|
2012-02-13 17:53:54 -04:00
|
|
|
void low_battery_event(void)
|
2011-09-08 22:29:39 -03:00
|
|
|
{
|
2013-09-29 09:54:39 -03:00
|
|
|
if (failsafe.low_battery) {
|
2013-05-22 07:33:57 -03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
gcs_send_text_fmt(PSTR("Low Battery %.2fV Used %.0f mAh"),
|
2013-09-29 09:54:39 -03:00
|
|
|
battery.voltage(), battery.current_total_mah());
|
2012-08-16 21:50:15 -03:00
|
|
|
set_mode(RTL);
|
2013-06-26 07:48:45 -03:00
|
|
|
aparm.throttle_cruise.load();
|
2013-09-29 09:54:39 -03:00
|
|
|
failsafe.low_battery = true;
|
2013-09-17 21:50:34 -03:00
|
|
|
AP_Notify::flags.failsafe_battery = true;
|
2011-09-08 22:29:39 -03:00
|
|
|
}
|
|
|
|
|
2012-09-16 20:24:00 -03:00
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// repeating event control
|
|
|
|
|
|
|
|
/*
|
|
|
|
update state for MAV_CMD_DO_REPEAT_SERVO and MAV_CMD_DO_REPEAT_RELAY
|
|
|
|
*/
|
|
|
|
static void update_events(void)
|
2011-09-08 22:29:39 -03:00
|
|
|
{
|
2012-09-16 20:24:00 -03:00
|
|
|
if (event_state.repeat == 0 || (millis() - event_state.start_time_ms) < event_state.delay_ms) {
|
2012-08-16 21:50:15 -03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-01-19 23:52:05 -04:00
|
|
|
event_state.start_time_ms = millis();
|
2012-08-16 21:50:15 -03:00
|
|
|
|
2014-01-19 23:52:05 -04:00
|
|
|
switch (event_state.type) {
|
|
|
|
case EVENT_TYPE_SERVO:
|
|
|
|
hal.rcout->enable_ch(event_state.channel-1);
|
|
|
|
if (event_state.repeat & 1) {
|
|
|
|
servo_write(event_state.channel-1, event_state.undo_value);
|
|
|
|
} else {
|
|
|
|
servo_write(event_state.channel-1, event_state.servo_value);
|
2012-09-16 20:24:00 -03:00
|
|
|
}
|
2014-01-19 23:52:05 -04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case EVENT_TYPE_RELAY:
|
|
|
|
if (event_state.delay_ms >= 1000) {
|
|
|
|
// don't spam the GCS with messages too fast
|
|
|
|
gcs_send_text_fmt(PSTR("Relay toggle"));
|
2012-08-16 21:50:15 -03:00
|
|
|
}
|
2014-01-19 23:52:05 -04:00
|
|
|
relay.toggle(event_state.channel);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (event_state.repeat > 0) {
|
|
|
|
event_state.repeat--;
|
|
|
|
} else {
|
|
|
|
event_state.repeat ^= 1;
|
2012-08-16 21:50:15 -03:00
|
|
|
}
|
2011-09-08 22:29:39 -03:00
|
|
|
}
|