#include "Copter.h" // // pre-takeoff checks // // detects if the vehicle should be allowed to takeoff or not and sets the motors.blocked flag void Copter::takeoff_check() { #if HAL_WITH_ESC_TELEM && FRAME_CONFIG != HELI_FRAME // If takeoff check is disabled or vehicle is armed and flying then clear block and return if ((g2.takeoff_rpm_min <= 0) || (motors->armed() && !ap.land_complete)) { motors->set_spoolup_block(false); return; } // block takeoff when disarmed but do not display warnings if (!motors->armed()) { motors->set_spoolup_block(true); takeoff_check_warning_ms = 0; return; } // if motors have become unblocked return immediately // this ensures the motors can only be blocked immediate after arming if (!motors->get_spoolup_block()) { return; } // check ESCs are sending RPM at expected level uint32_t motor_mask = motors->get_motor_mask(); #if HAL_WITH_IO_MCU if (AP_BoardConfig::io_enabled()) { // In 4.4 and earlier ESC telemetry is always indexed from 1 for servo channels 9+ motor_mask >>= 8; } #endif const bool telem_active = AP::esc_telem().is_telemetry_active(motor_mask); const bool rpm_adequate = AP::esc_telem().are_motors_running(motor_mask, g2.takeoff_rpm_min); // if RPM is at the expected level clear block if (telem_active && rpm_adequate) { motors->set_spoolup_block(false); return; } // warn user telem inactive or rpm is inadequate every 5 seconds uint32_t now_ms = AP_HAL::millis(); if (takeoff_check_warning_ms == 0) { takeoff_check_warning_ms = now_ms; } if (now_ms - takeoff_check_warning_ms > 5000) { takeoff_check_warning_ms = now_ms; const char* prefix_str = "Takeoff blocked:"; if (!telem_active) { gcs().send_text(MAV_SEVERITY_CRITICAL, "%s waiting for ESC RPM", prefix_str); } else if (!rpm_adequate) { gcs().send_text(MAV_SEVERITY_CRITICAL, "%s ESC RPM too low", prefix_str); } } #endif }