From bbee8312dd9e1b5369878bc7e3317a03d1fd68a3 Mon Sep 17 00:00:00 2001 From: Andrew Tridgell Date: Mon, 2 Jan 2023 07:49:26 +1100 Subject: [PATCH] GCS_MAVLink: send FTP replies directly from the FTP thread this saves over 700 bytes of memory while also making ftp faster --- libraries/GCS_MAVLink/GCS.h | 3 +- libraries/GCS_MAVLink/GCS_Common.cpp | 3 +- libraries/GCS_MAVLink/GCS_FTP.cpp | 66 ++++++++++------------------ 3 files changed, 24 insertions(+), 48 deletions(-) diff --git a/libraries/GCS_MAVLink/GCS.h b/libraries/GCS_MAVLink/GCS.h index bfb058030c..947768e469 100644 --- a/libraries/GCS_MAVLink/GCS.h +++ b/libraries/GCS_MAVLink/GCS.h @@ -890,7 +890,6 @@ private: struct ftp_state { ObjectBuffer *requests; - ObjectBuffer *replies; // session specific info, currently only support a single session over all links int fd = -1; @@ -907,7 +906,7 @@ private: bool ftp_init(void); void handle_file_transfer_protocol(const mavlink_message_t &msg); - void send_ftp_replies(void); + bool send_ftp_reply(const pending_ftp &reply); void ftp_worker(void); void ftp_push_replies(pending_ftp &reply); diff --git a/libraries/GCS_MAVLink/GCS_Common.cpp b/libraries/GCS_MAVLink/GCS_Common.cpp index dc3d16ab7e..0f9e338bd9 100644 --- a/libraries/GCS_MAVLink/GCS_Common.cpp +++ b/libraries/GCS_MAVLink/GCS_Common.cpp @@ -1044,7 +1044,7 @@ uint16_t GCS_MAVLINK::get_reschedule_interval_ms(const deferred_message_bucket_t // we are sending requests for waypoints, penalize streams: interval_ms *= 4; } - if (ftp.replies && AP_HAL::millis() - ftp.last_send_ms < 500) { + if (AP_HAL::millis() - ftp.last_send_ms < 500) { // we are sending ftp replies interval_ms *= 4; } @@ -1221,7 +1221,6 @@ void GCS_MAVLINK::update_send() AP::logger().handle_log_send(); } #endif - send_ftp_replies(); if (!deferred_messages_initialised) { initialise_message_intervals_from_streamrates(); diff --git a/libraries/GCS_MAVLink/GCS_FTP.cpp b/libraries/GCS_MAVLink/GCS_FTP.cpp index da5c71331a..fff430c369 100644 --- a/libraries/GCS_MAVLink/GCS_FTP.cpp +++ b/libraries/GCS_MAVLink/GCS_FTP.cpp @@ -47,10 +47,6 @@ bool GCS_MAVLINK::ftp_init(void) { if (ftp.requests == nullptr) { goto failed; } - ftp.replies = new ObjectBuffer(30); - if (ftp.replies == nullptr) { - goto failed; - } if (!hal.scheduler->thread_create(FUNCTOR_BIND_MEMBER(&GCS_MAVLINK::ftp_worker, void), "FTP", 2560, AP_HAL::Scheduler::PRIORITY_IO, 0)) { @@ -62,8 +58,6 @@ bool GCS_MAVLINK::ftp_init(void) { failed: delete ftp.requests; ftp.requests = nullptr; - delete ftp.replies; - ftp.replies = nullptr; gcs().send_text(MAV_SEVERITY_WARNING, "failed to initialize MAVFTP"); return false; @@ -96,50 +90,34 @@ void GCS_MAVLINK::handle_file_transfer_protocol(const mavlink_message_t &msg) { } } -void GCS_MAVLINK::send_ftp_replies(void) +bool GCS_MAVLINK::send_ftp_reply(const pending_ftp &reply) { /* provide same banner we would give with old param download */ - if (ftp.need_banner_send_mask & (1U<is_empty()) { - return; - } - - for (uint8_t i = 0; i < 20; i++) { - if (!HAVE_PAYLOAD_SPACE(chan, FILE_TRANSFER_PROTOCOL)) { - return; - } - if ((i > 0) && comm_get_txspace(chan) < (2 * (packet_overhead() + MAVLINK_MSG_ID_FILE_TRANSFER_PROTOCOL_LEN))) { - // if this isn't the first packet we have to leave deadspace for the next message - return; - } - - struct pending_ftp reply; - uint8_t payload[251] = {}; - if (ftp.replies->peek(reply) && (reply.chan == chan)) { - put_le16_ptr(payload, reply.seq_number); - payload[2] = reply.session; - payload[3] = static_cast(reply.opcode); - payload[4] = reply.size; - payload[5] = static_cast(reply.req_opcode); - payload[6] = reply.burst_complete ? 1 : 0; - put_le32_ptr(&payload[8], reply.offset); - memcpy(&payload[12], reply.data, sizeof(reply.data)); - mavlink_msg_file_transfer_protocol_send( - reply.chan, - 0, reply.sysid, reply.compid, - payload); - ftp.replies->pop(); - ftp.last_send_ms = AP_HAL::millis(); - } else { - return; - } + WITH_SEMAPHORE(comm_chan_lock(reply.chan)); + if (!HAVE_PAYLOAD_SPACE(chan, FILE_TRANSFER_PROTOCOL)) { + return false; } + uint8_t payload[251] = {}; + put_le16_ptr(payload, reply.seq_number); + payload[2] = reply.session; + payload[3] = static_cast(reply.opcode); + payload[4] = reply.size; + payload[5] = static_cast(reply.req_opcode); + payload[6] = reply.burst_complete ? 1 : 0; + put_le32_ptr(&payload[8], reply.offset); + memcpy(&payload[12], reply.data, sizeof(reply.data)); + mavlink_msg_file_transfer_protocol_send( + reply.chan, + 0, reply.sysid, reply.compid, + payload); + ftp.last_send_ms = AP_HAL::millis(); + return true; } void GCS_MAVLINK::ftp_error(struct pending_ftp &response, FTP_ERROR error) { @@ -168,7 +146,7 @@ void GCS_MAVLINK::ftp_error(struct pending_ftp &response, FTP_ERROR error) { // send our response back out to the system void GCS_MAVLINK::ftp_push_replies(pending_ftp &reply) { - while (!ftp.replies->push(reply)) { // we must fit the response, keep shoving it in + while (!send_ftp_reply(reply)) { hal.scheduler->delay(2); } }