123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326 |
- /*
- MAVLink logfile transfer functions
- */
- /*
- This program is free software: you can redistribute it and/or modify
- it under the terms of the GNU General Public License as published by
- the Free Software Foundation, either version 3 of the License, or
- (at your option) any later version.
- This program is distributed in the hope that it will be useful,
- but WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- GNU General Public License for more details.
- You should have received a copy of the GNU General Public License
- along with this program. If not, see <http://www.gnu.org/licenses/>.
- */
- #include <AP_HAL/AP_HAL.h>
- #include <AP_Logger/AP_Logger.h>
- #include <GCS_MAVLink/GCS.h> // for LOG_ENTRY
- extern const AP_HAL::HAL& hal;
- // We avoid doing log messages when timing is critical:
- bool AP_Logger::should_handle_log_message()
- {
- if (!WritesEnabled()) {
- // this is currently used as a proxy for "in_mavlink_delay"
- return false;
- }
- if (vehicle_is_armed()) {
- return false;
- }
- return true;
- }
- /**
- handle all types of log download requests from the GCS
- */
- void AP_Logger::handle_log_message(GCS_MAVLINK &link, const mavlink_message_t &msg)
- {
- if (!should_handle_log_message()) {
- return;
- }
- switch (msg.msgid) {
- case MAVLINK_MSG_ID_LOG_REQUEST_LIST:
- handle_log_request_list(link, msg);
- break;
- case MAVLINK_MSG_ID_LOG_REQUEST_DATA:
- handle_log_request_data(link, msg);
- break;
- case MAVLINK_MSG_ID_LOG_ERASE:
- handle_log_request_erase(link, msg);
- break;
- case MAVLINK_MSG_ID_LOG_REQUEST_END:
- handle_log_request_end(link, msg);
- break;
- }
- }
- /**
- handle all types of log download requests from the GCS
- */
- void AP_Logger::handle_log_request_list(GCS_MAVLINK &link, const mavlink_message_t &msg)
- {
- WITH_SEMAPHORE(_log_send_sem);
- if (_log_sending_link != nullptr) {
- link.send_text(MAV_SEVERITY_INFO, "Log download in progress");
- return;
- }
- mavlink_log_request_list_t packet;
- mavlink_msg_log_request_list_decode(&msg, &packet);
- _log_num_logs = get_num_logs();
- if (_log_num_logs == 0) {
- _log_next_list_entry = 0;
- _log_last_list_entry = 0;
- } else {
- _log_next_list_entry = packet.start;
- _log_last_list_entry = packet.end;
- if (_log_last_list_entry > _log_num_logs) {
- _log_last_list_entry = _log_num_logs;
- }
- if (_log_next_list_entry < 1) {
- _log_next_list_entry = 1;
- }
- }
- transfer_activity = LISTING;
- _log_sending_link = &link;
- handle_log_send_listing();
- }
- /**
- handle request for log data
- */
- void AP_Logger::handle_log_request_data(GCS_MAVLINK &link, const mavlink_message_t &msg)
- {
- WITH_SEMAPHORE(_log_send_sem);
- if (_log_sending_link != nullptr) {
- // some GCS (e.g. MAVProxy) attempt to stream request_data
- // messages when they're filling gaps in the downloaded logs.
- // This channel check avoids complaining to them, at the cost
- // of silently dropping any repeated attempts to start logging
- if (_log_sending_link->get_chan() != link.get_chan()) {
- link.send_text(MAV_SEVERITY_INFO, "Log download in progress");
- }
- return;
- }
- mavlink_log_request_data_t packet;
- mavlink_msg_log_request_data_decode(&msg, &packet);
- // consider opening or switching logs:
- if (transfer_activity != SENDING || _log_num_data != packet.id) {
- uint16_t num_logs = get_num_logs();
- if (packet.id > num_logs || packet.id < 1) {
- // request for an invalid log; cancel any current download
- transfer_activity = IDLE;
- return;
- }
- uint32_t time_utc, size;
- get_log_info(packet.id, size, time_utc);
- _log_num_data = packet.id;
- _log_data_size = size;
- uint32_t end;
- get_log_boundaries(packet.id, _log_data_page, end);
- }
- _log_data_offset = packet.ofs;
- if (_log_data_offset >= _log_data_size) {
- _log_data_remaining = 0;
- } else {
- _log_data_remaining = _log_data_size - _log_data_offset;
- }
- if (_log_data_remaining > packet.count) {
- _log_data_remaining = packet.count;
- }
- transfer_activity = SENDING;
- _log_sending_link = &link;
- handle_log_send();
- }
- /**
- handle request to erase log data
- */
- void AP_Logger::handle_log_request_erase(GCS_MAVLINK &link, const mavlink_message_t &msg)
- {
- // mavlink_log_erase_t packet;
- // mavlink_msg_log_erase_decode(&msg, &packet);
- EraseAll();
- }
- /**
- handle request to stop transfer and resume normal logging
- */
- void AP_Logger::handle_log_request_end(GCS_MAVLINK &link, const mavlink_message_t &msg)
- {
- WITH_SEMAPHORE(_log_send_sem);
- mavlink_log_request_end_t packet;
- mavlink_msg_log_request_end_decode(&msg, &packet);
- transfer_activity = IDLE;
- _log_sending_link = nullptr;
- }
- /**
- trigger sending of log messages if there are some pending
- */
- void AP_Logger::handle_log_send()
- {
- WITH_SEMAPHORE(_log_send_sem);
- if (_log_sending_link == nullptr) {
- return;
- }
- if (hal.util->get_soft_armed()) {
- // might be flying
- return;
- }
- switch (transfer_activity) {
- case IDLE:
- break;
- case LISTING:
- handle_log_send_listing();
- break;
- case SENDING:
- handle_log_sending();
- break;
- }
- }
- void AP_Logger::handle_log_sending()
- {
- WITH_SEMAPHORE(_log_send_sem);
- #if CONFIG_HAL_BOARD == HAL_BOARD_SITL
- // assume USB speeds in SITL for the purposes of log download
- const uint8_t num_sends = 40;
- #else
- uint8_t num_sends = 1;
- if (_log_sending_link->is_high_bandwidth() && hal.gpio->usb_connected()) {
- // when on USB we can send a lot more data
- num_sends = 250;
- } else if (_log_sending_link->have_flow_control()) {
- #if CONFIG_HAL_BOARD == HAL_BOARD_LINUX
- num_sends = 80;
- #else
- num_sends = 10;
- #endif
- }
- #endif
- for (uint8_t i=0; i<num_sends; i++) {
- if (transfer_activity != SENDING) {
- // may have completed sending data
- break;
- }
- if (!handle_log_send_data()) {
- break;
- }
- }
- }
- /**
- trigger sending of log messages if there are some pending
- */
- void AP_Logger::handle_log_send_listing()
- {
- WITH_SEMAPHORE(_log_send_sem);
- if (!HAVE_PAYLOAD_SPACE(_log_sending_link->get_chan(), LOG_ENTRY)) {
- // no space
- return;
- }
- if (AP_HAL::millis() - _log_sending_link->get_last_heartbeat_time() > 3000) {
- // give a heartbeat a chance
- return;
- }
- uint32_t size, time_utc;
- if (_log_next_list_entry == 0) {
- size = 0;
- time_utc = 0;
- } else {
- get_log_info(_log_next_list_entry, size, time_utc);
- }
- mavlink_msg_log_entry_send(_log_sending_link->get_chan(),
- _log_next_list_entry,
- _log_num_logs,
- _log_last_list_entry,
- time_utc,
- size);
- if (_log_next_list_entry == _log_last_list_entry) {
- transfer_activity = IDLE;
- _log_sending_link = nullptr;
- } else {
- _log_next_list_entry++;
- }
- }
- /**
- trigger sending of log data if there are some pending
- */
- bool AP_Logger::handle_log_send_data()
- {
- WITH_SEMAPHORE(_log_send_sem);
- if (!HAVE_PAYLOAD_SPACE(_log_sending_link->get_chan(), LOG_DATA)) {
- // no space
- return false;
- }
- if (AP_HAL::millis() - _log_sending_link->get_last_heartbeat_time() > 3000) {
- // give a heartbeat a chance
- return false;
- }
- int16_t ret = 0;
- uint32_t len = _log_data_remaining;
- mavlink_log_data_t packet;
- if (len > MAVLINK_MSG_LOG_DATA_FIELD_DATA_LEN) {
- len = MAVLINK_MSG_LOG_DATA_FIELD_DATA_LEN;
- }
- ret = get_log_data(_log_num_data, _log_data_page, _log_data_offset, len, packet.data);
- if (ret < 0) {
- // report as EOF on error
- ret = 0;
- }
- if (ret < MAVLINK_MSG_LOG_DATA_FIELD_DATA_LEN) {
- memset(&packet.data[ret], 0, MAVLINK_MSG_LOG_DATA_FIELD_DATA_LEN-ret);
- }
- packet.ofs = _log_data_offset;
- packet.id = _log_num_data;
- packet.count = ret;
- _mav_finalize_message_chan_send(_log_sending_link->get_chan(),
- MAVLINK_MSG_ID_LOG_DATA,
- (const char *)&packet,
- MAVLINK_MSG_ID_LOG_DATA_MIN_LEN,
- MAVLINK_MSG_ID_LOG_DATA_LEN,
- MAVLINK_MSG_ID_LOG_DATA_CRC);
- _log_data_offset += len;
- _log_data_remaining -= len;
- if (ret < MAVLINK_MSG_LOG_DATA_FIELD_DATA_LEN || _log_data_remaining == 0) {
- transfer_activity = IDLE;
- _log_sending_link = nullptr;
- }
- return true;
- }
|