6538e8c9ae
Delete unneeded orphan comment replace get_last_txbuf() with a predicate Make txbuf flow control threashold consistent between Parameter download and FTP and keep it in range where we are also slowing down normal streams Delay sending text banner until after first FTP response to reduce latency on slow links Don't let flow control delay setting ftp.last_send_ms so as to slow down normal streams as soon as possible to improve FTP response time
475 lines
14 KiB
C++
475 lines
14 KiB
C++
/*
|
|
GCS MAVLink functions related to parameter handling
|
|
|
|
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 "GCS_config.h"
|
|
|
|
#if HAL_GCS_ENABLED
|
|
|
|
#include <AP_HAL/AP_HAL.h>
|
|
|
|
#include "GCS.h"
|
|
#include <AP_Logger/AP_Logger.h>
|
|
#include <AP_BoardConfig/AP_BoardConfig.h>
|
|
|
|
extern const AP_HAL::HAL& hal;
|
|
|
|
// queue of pending parameter requests and replies
|
|
ObjectBuffer<GCS_MAVLINK::pending_param_request> GCS_MAVLINK::param_requests(20);
|
|
ObjectBuffer<GCS_MAVLINK::pending_param_reply> GCS_MAVLINK::param_replies(5);
|
|
|
|
bool GCS_MAVLINK::param_timer_registered;
|
|
|
|
/**
|
|
* @brief Send the next pending parameter, called from deferred message
|
|
* handling code
|
|
*/
|
|
void
|
|
GCS_MAVLINK::queued_param_send()
|
|
{
|
|
// send parameter async replies
|
|
uint8_t async_replies_sent_count = send_parameter_async_replies();
|
|
|
|
// now send the streaming parameters (from PARAM_REQUEST_LIST)
|
|
if (_queued_parameter == nullptr) {
|
|
// .... or not....
|
|
return;
|
|
}
|
|
|
|
const uint32_t tnow = AP_HAL::millis();
|
|
const uint32_t tstart = AP_HAL::micros();
|
|
|
|
// use at most 30% of bandwidth on parameters
|
|
const uint32_t link_bw = _port->bw_in_bytes_per_second();
|
|
|
|
uint32_t bytes_allowed = link_bw * (tnow - _queued_parameter_send_time_ms) / 3333;
|
|
const uint16_t size_for_one_param_value_msg = MAVLINK_MSG_ID_PARAM_VALUE_LEN + packet_overhead();
|
|
if (bytes_allowed < size_for_one_param_value_msg) {
|
|
bytes_allowed = size_for_one_param_value_msg;
|
|
}
|
|
if (bytes_allowed > txspace()) {
|
|
bytes_allowed = txspace();
|
|
}
|
|
uint32_t count = bytes_allowed / size_for_one_param_value_msg;
|
|
|
|
// when we don't have flow control we really need to keep the
|
|
// param download very slow, or it tends to stall
|
|
if (!have_flow_control() && count > 5) {
|
|
count = 5;
|
|
}
|
|
if (async_replies_sent_count >= count) {
|
|
return;
|
|
}
|
|
count -= async_replies_sent_count;
|
|
|
|
while (count && _queued_parameter != nullptr && last_txbuf_is_greater(33)) {
|
|
char param_name[AP_MAX_NAME_SIZE];
|
|
_queued_parameter->copy_name_token(_queued_parameter_token, param_name, sizeof(param_name), true);
|
|
|
|
mavlink_msg_param_value_send(
|
|
chan,
|
|
param_name,
|
|
_queued_parameter->cast_to_float(_queued_parameter_type),
|
|
mav_param_type(_queued_parameter_type),
|
|
_queued_parameter_count,
|
|
_queued_parameter_index);
|
|
|
|
_queued_parameter = AP_Param::next_scalar(&_queued_parameter_token, &_queued_parameter_type);
|
|
_queued_parameter_index++;
|
|
|
|
if (AP_HAL::micros() - tstart > 1000) {
|
|
// don't use more than 1ms sending blocks of parameters
|
|
break;
|
|
}
|
|
count--;
|
|
}
|
|
_queued_parameter_send_time_ms = tnow;
|
|
}
|
|
|
|
/*
|
|
return true if a channel has flow control
|
|
*/
|
|
bool GCS_MAVLINK::have_flow_control(void)
|
|
{
|
|
if (_port == nullptr) {
|
|
return false;
|
|
}
|
|
|
|
if (_port->get_flow_control() != AP_HAL::UARTDriver::FLOW_CONTROL_DISABLE) {
|
|
return true;
|
|
}
|
|
|
|
if (chan == MAVLINK_COMM_0) {
|
|
// assume USB console has flow control
|
|
return hal.gpio->usb_connected();
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
/*
|
|
handle a request to change stream rate. Note that copter passes in
|
|
save==false so we don't want the save to happen when the user connects the
|
|
ground station.
|
|
*/
|
|
void GCS_MAVLINK::handle_request_data_stream(const mavlink_message_t &msg)
|
|
{
|
|
mavlink_request_data_stream_t packet;
|
|
mavlink_msg_request_data_stream_decode(&msg, &packet);
|
|
|
|
int16_t freq = 0; // packet frequency
|
|
|
|
if (packet.start_stop == 0)
|
|
freq = 0; // stop sending
|
|
else if (packet.start_stop == 1)
|
|
freq = packet.req_message_rate; // start sending
|
|
else
|
|
return;
|
|
|
|
// if stream_id is still NUM_STREAMS at the end of this switch
|
|
// block then either we set stream rates for all streams, or we
|
|
// were asked to set the streamrate for an unrecognised stream
|
|
streams stream_id = NUM_STREAMS;
|
|
switch (packet.req_stream_id) {
|
|
case MAV_DATA_STREAM_ALL:
|
|
for (uint8_t i=0; i<NUM_STREAMS; i++) {
|
|
if (i == STREAM_PARAMS) {
|
|
// don't touch parameter streaming rate; it is
|
|
// considered "internal".
|
|
continue;
|
|
}
|
|
if (persist_streamrates()) {
|
|
streamRates[i].set_and_save_ifchanged(freq);
|
|
} else {
|
|
streamRates[i].set(freq);
|
|
}
|
|
initialise_message_intervals_for_stream((streams)i);
|
|
}
|
|
break;
|
|
case MAV_DATA_STREAM_RAW_SENSORS:
|
|
stream_id = STREAM_RAW_SENSORS;
|
|
break;
|
|
case MAV_DATA_STREAM_EXTENDED_STATUS:
|
|
stream_id = STREAM_EXTENDED_STATUS;
|
|
break;
|
|
case MAV_DATA_STREAM_RC_CHANNELS:
|
|
stream_id = STREAM_RC_CHANNELS;
|
|
break;
|
|
case MAV_DATA_STREAM_RAW_CONTROLLER:
|
|
stream_id = STREAM_RAW_CONTROLLER;
|
|
break;
|
|
case MAV_DATA_STREAM_POSITION:
|
|
stream_id = STREAM_POSITION;
|
|
break;
|
|
case MAV_DATA_STREAM_EXTRA1:
|
|
stream_id = STREAM_EXTRA1;
|
|
break;
|
|
case MAV_DATA_STREAM_EXTRA2:
|
|
stream_id = STREAM_EXTRA2;
|
|
break;
|
|
case MAV_DATA_STREAM_EXTRA3:
|
|
stream_id = STREAM_EXTRA3;
|
|
break;
|
|
}
|
|
|
|
if (stream_id == NUM_STREAMS) {
|
|
// asked to set rate on unknown stream (or all were set already)
|
|
return;
|
|
}
|
|
|
|
AP_Int16 *rate = &streamRates[stream_id];
|
|
|
|
if (rate != nullptr) {
|
|
if (persist_streamrates()) {
|
|
rate->set_and_save_ifchanged(freq);
|
|
} else {
|
|
rate->set(freq);
|
|
}
|
|
initialise_message_intervals_for_stream(stream_id);
|
|
}
|
|
}
|
|
|
|
void GCS_MAVLINK::handle_param_request_list(const mavlink_message_t &msg)
|
|
{
|
|
if (!params_ready()) {
|
|
return;
|
|
}
|
|
|
|
mavlink_param_request_list_t packet;
|
|
mavlink_msg_param_request_list_decode(&msg, &packet);
|
|
|
|
// requesting parameters is a convenient way to get extra information
|
|
send_banner();
|
|
|
|
// Start sending parameters - next call to ::update will kick the first one out
|
|
_queued_parameter = AP_Param::first(&_queued_parameter_token, &_queued_parameter_type);
|
|
_queued_parameter_index = 0;
|
|
_queued_parameter_count = AP_Param::count_parameters();
|
|
_queued_parameter_send_time_ms = AP_HAL::millis(); // avoid initial flooding
|
|
}
|
|
|
|
void GCS_MAVLINK::handle_param_request_read(const mavlink_message_t &msg)
|
|
{
|
|
if (param_requests.space() == 0) {
|
|
// we can't process this right now, drop it
|
|
return;
|
|
}
|
|
|
|
mavlink_param_request_read_t packet;
|
|
mavlink_msg_param_request_read_decode(&msg, &packet);
|
|
|
|
/*
|
|
we reserve some space for sending parameters if the client ever
|
|
fails to get a parameter due to lack of space
|
|
*/
|
|
uint32_t saved_reserve_param_space_start_ms = reserve_param_space_start_ms;
|
|
reserve_param_space_start_ms = 0; // bypass packet_overhead_chan reservation checking
|
|
if (!HAVE_PAYLOAD_SPACE(chan, PARAM_VALUE)) {
|
|
reserve_param_space_start_ms = AP_HAL::millis();
|
|
} else {
|
|
reserve_param_space_start_ms = saved_reserve_param_space_start_ms;
|
|
}
|
|
|
|
struct pending_param_request req;
|
|
req.chan = chan;
|
|
req.param_index = packet.param_index;
|
|
memcpy(req.param_name, packet.param_id, MIN(sizeof(packet.param_id), sizeof(req.param_name)));
|
|
req.param_name[AP_MAX_NAME_SIZE] = 0;
|
|
|
|
// queue it for processing by io timer
|
|
param_requests.push(req);
|
|
|
|
// speaking of which, we'd best make sure it is running:
|
|
if (!param_timer_registered) {
|
|
param_timer_registered = true;
|
|
hal.scheduler->register_io_process(FUNCTOR_BIND_MEMBER(&GCS_MAVLINK::param_io_timer, void));
|
|
}
|
|
}
|
|
|
|
void GCS_MAVLINK::handle_param_set(const mavlink_message_t &msg)
|
|
{
|
|
mavlink_param_set_t packet;
|
|
mavlink_msg_param_set_decode(&msg, &packet);
|
|
enum ap_var_type var_type;
|
|
|
|
// set parameter
|
|
AP_Param *vp;
|
|
char key[AP_MAX_NAME_SIZE+1];
|
|
strncpy(key, (char *)packet.param_id, AP_MAX_NAME_SIZE);
|
|
key[AP_MAX_NAME_SIZE] = 0;
|
|
|
|
// find existing param so we can get the old value
|
|
uint16_t parameter_flags = 0;
|
|
vp = AP_Param::find(key, &var_type, ¶meter_flags);
|
|
if (vp == nullptr || isnan(packet.param_value) || isinf(packet.param_value)) {
|
|
return;
|
|
}
|
|
|
|
float old_value = vp->cast_to_float(var_type);
|
|
|
|
if (parameter_flags & AP_PARAM_FLAG_INTERNAL_USE_ONLY) {
|
|
// the user can set BRD_OPTIONS to enable set of internal
|
|
// parameters, for developer testing or unusual use cases
|
|
if (AP_BoardConfig::allow_set_internal_parameters()) {
|
|
parameter_flags &= ~AP_PARAM_FLAG_INTERNAL_USE_ONLY;
|
|
}
|
|
}
|
|
|
|
if ((parameter_flags & AP_PARAM_FLAG_INTERNAL_USE_ONLY) || vp->is_read_only()) {
|
|
gcs().send_text(MAV_SEVERITY_WARNING, "Param write denied (%s)", key);
|
|
// send the readonly value
|
|
send_parameter_value(key, var_type, old_value);
|
|
return;
|
|
}
|
|
|
|
// set the value
|
|
vp->set_float(packet.param_value, var_type);
|
|
|
|
/*
|
|
we force the save if the value is not equal to the old
|
|
value. This copes with the use of override values in
|
|
constructors, such as PID elements. Otherwise a set to the
|
|
default value which differs from the constructor value doesn't
|
|
save the change
|
|
*/
|
|
bool force_save = !is_equal(packet.param_value, old_value);
|
|
|
|
// save the change
|
|
vp->save(force_save);
|
|
|
|
if (force_save && (parameter_flags & AP_PARAM_FLAG_ENABLE)) {
|
|
AP_Param::invalidate_count();
|
|
}
|
|
|
|
#if HAL_LOGGING_ENABLED
|
|
AP_Logger *logger = AP_Logger::get_singleton();
|
|
if (logger != nullptr) {
|
|
logger->Write_Parameter(key, vp->cast_to_float(var_type));
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void GCS_MAVLINK::send_parameter_value(const char *param_name, ap_var_type param_type, float param_value)
|
|
{
|
|
if (!HAVE_PAYLOAD_SPACE(chan, PARAM_VALUE)) {
|
|
return;
|
|
}
|
|
mavlink_msg_param_value_send(
|
|
chan,
|
|
param_name,
|
|
param_value,
|
|
mav_param_type(param_type),
|
|
AP_Param::count_parameters(),
|
|
-1);
|
|
}
|
|
|
|
/*
|
|
send a parameter value message to all active MAVLink connections
|
|
*/
|
|
void GCS::send_parameter_value(const char *param_name, ap_var_type param_type, float param_value)
|
|
{
|
|
mavlink_param_value_t packet{};
|
|
const uint8_t to_copy = MIN(ARRAY_SIZE(packet.param_id), strlen(param_name));
|
|
memcpy(packet.param_id, param_name, to_copy);
|
|
packet.param_value = param_value;
|
|
packet.param_type = GCS_MAVLINK::mav_param_type(param_type);
|
|
packet.param_count = AP_Param::count_parameters();
|
|
packet.param_index = -1;
|
|
|
|
gcs().send_to_active_channels(MAVLINK_MSG_ID_PARAM_VALUE,
|
|
(const char *)&packet);
|
|
|
|
#if HAL_LOGGING_ENABLED
|
|
// also log to AP_Logger
|
|
AP_Logger *logger = AP_Logger::get_singleton();
|
|
if (logger != nullptr) {
|
|
logger->Write_Parameter(param_name, param_value);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
|
|
/*
|
|
timer callback for async parameter requests
|
|
*/
|
|
void GCS_MAVLINK::param_io_timer(void)
|
|
{
|
|
struct pending_param_request req;
|
|
|
|
// this is mostly a no-op, but doing this here means we won't
|
|
// block the main thread counting parameters (~30ms on PH)
|
|
AP_Param::count_parameters();
|
|
|
|
if (param_replies.space() == 0) {
|
|
// no room
|
|
return;
|
|
}
|
|
|
|
if (!param_requests.pop(req)) {
|
|
// nothing to do
|
|
return;
|
|
}
|
|
|
|
struct pending_param_reply reply;
|
|
AP_Param *vp;
|
|
|
|
if (req.param_index != -1) {
|
|
AP_Param::ParamToken token {};
|
|
vp = AP_Param::find_by_index(req.param_index, &reply.p_type, &token);
|
|
if (vp == nullptr) {
|
|
return;
|
|
}
|
|
vp->copy_name_token(token, reply.param_name, AP_MAX_NAME_SIZE, true);
|
|
} else {
|
|
strncpy(reply.param_name, req.param_name, AP_MAX_NAME_SIZE+1);
|
|
vp = AP_Param::find(req.param_name, &reply.p_type);
|
|
if (vp == nullptr) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
reply.chan = req.chan;
|
|
reply.param_name[AP_MAX_NAME_SIZE] = 0;
|
|
reply.value = vp->cast_to_float(reply.p_type);
|
|
reply.param_index = req.param_index;
|
|
reply.count = AP_Param::count_parameters();
|
|
|
|
// queue for transmission
|
|
param_replies.push(reply);
|
|
}
|
|
|
|
/*
|
|
send replies to PARAM_REQUEST_READ
|
|
*/
|
|
uint8_t GCS_MAVLINK::send_parameter_async_replies()
|
|
{
|
|
uint8_t async_replies_sent_count = 0;
|
|
|
|
while (async_replies_sent_count < 5) {
|
|
struct pending_param_reply reply;
|
|
if (!param_replies.peek(reply)) {
|
|
return async_replies_sent_count;
|
|
}
|
|
|
|
/*
|
|
we reserve some space for sending parameters if the client ever
|
|
fails to get a parameter due to lack of space
|
|
*/
|
|
uint32_t saved_reserve_param_space_start_ms = reserve_param_space_start_ms;
|
|
reserve_param_space_start_ms = 0; // bypass packet_overhead_chan reservation checking
|
|
if (!HAVE_PAYLOAD_SPACE(reply.chan, PARAM_VALUE)) {
|
|
reserve_param_space_start_ms = AP_HAL::millis();
|
|
return async_replies_sent_count;
|
|
}
|
|
reserve_param_space_start_ms = saved_reserve_param_space_start_ms;
|
|
|
|
mavlink_msg_param_value_send(
|
|
reply.chan,
|
|
reply.param_name,
|
|
reply.value,
|
|
mav_param_type(reply.p_type),
|
|
reply.count,
|
|
reply.param_index);
|
|
|
|
_queued_parameter_send_time_ms = AP_HAL::millis();
|
|
async_replies_sent_count++;
|
|
|
|
if (!param_replies.pop()) {
|
|
// internal error...
|
|
return async_replies_sent_count;
|
|
}
|
|
}
|
|
return async_replies_sent_count;
|
|
}
|
|
|
|
void GCS_MAVLINK::handle_common_param_message(const mavlink_message_t &msg)
|
|
{
|
|
switch (msg.msgid) {
|
|
case MAVLINK_MSG_ID_PARAM_REQUEST_LIST:
|
|
handle_param_request_list(msg);
|
|
break;
|
|
case MAVLINK_MSG_ID_PARAM_SET:
|
|
handle_param_set(msg);
|
|
break;
|
|
case MAVLINK_MSG_ID_PARAM_REQUEST_READ:
|
|
handle_param_request_read(msg);
|
|
break;
|
|
}
|
|
}
|
|
|
|
#endif // HAL_GCS_ENABLED
|