From f3d0bbe6053eba05972f29cc5a4b6965c6a65061 Mon Sep 17 00:00:00 2001 From: Julian Oes Date: Thu, 19 Dec 2024 14:34:03 +1300 Subject: [PATCH 1/4] core: fix param get regression It turns out single params are sent with index -1, so caught by the test for the _HASH_CHECK param. --- src/mavsdk/core/mavlink_parameter_client.cpp | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/src/mavsdk/core/mavlink_parameter_client.cpp b/src/mavsdk/core/mavlink_parameter_client.cpp index 19d78dad0..13055e8f3 100644 --- a/src/mavsdk/core/mavlink_parameter_client.cpp +++ b/src/mavsdk/core/mavlink_parameter_client.cpp @@ -669,7 +669,8 @@ void MavlinkParameterClient::process_param_value(const mavlink_message_t& messag << ", index: " << param_value.param_index; } - if (param_value.param_index == std::numeric_limits::max()) { + if (param_value.param_index == std::numeric_limits::max() && + safe_param_id == "_HASH_CHECK") { // Ignore PX4's _HASH_CHECK param. return; } From 120c6bbdfd90b7080aabca07d22f7ab65c684104 Mon Sep 17 00:00:00 2001 From: Julian Oes Date: Thu, 19 Dec 2024 14:34:56 +1300 Subject: [PATCH 2/4] core: return WrongType instead of Timeout In case of a param set with the wrong type, we should catch that and return with wrong type. --- src/mavsdk/core/mavlink_parameter_client.cpp | 12 ++++++++++++ 1 file changed, 12 insertions(+) diff --git a/src/mavsdk/core/mavlink_parameter_client.cpp b/src/mavsdk/core/mavlink_parameter_client.cpp index 13055e8f3..32b385b2f 100644 --- a/src/mavsdk/core/mavlink_parameter_client.cpp +++ b/src/mavsdk/core/mavlink_parameter_client.cpp @@ -706,6 +706,18 @@ void MavlinkParameterClient::process_param_value(const mavlink_message_t& messag << ", received: " << received_value; } + if (!item.param_value.is_same_type(received_value)) { + LogErr() << "Wrong type in param set"; + _timeout_handler.remove(_timeout_cookie); + work_queue_guard->pop_front(); + if (item.callback) { + auto callback = item.callback; + work_queue_guard.reset(); + callback(MavlinkParameterClient::Result::WrongType); + } + return; + } + if (item.param_value == received_value) { // This was successful. Inform the caller. _timeout_handler.remove(_timeout_cookie); From a4f806ce43cee20baf5f31adc3ee039611b3b717 Mon Sep 17 00:00:00 2001 From: Julian Oes Date: Thu, 19 Dec 2024 14:35:47 +1300 Subject: [PATCH 3/4] examples: improve params example It now morphs more into a param CLI tool. --- examples/params/params.cpp | 222 ++++++++++++++++++++++++++++++++++--- 1 file changed, 204 insertions(+), 18 deletions(-) diff --git a/examples/params/params.cpp b/examples/params/params.cpp index 9f4c7731a..47b8de249 100644 --- a/examples/params/params.cpp +++ b/examples/params/params.cpp @@ -4,34 +4,117 @@ #include #include + +#include #include #include #include #include +#include +#include using namespace mavsdk; -void usage(const std::string& bin_name) -{ - std::cerr << "Usage :" << bin_name << '\n' - << "Connection URL format should be :\n" - << " For TCP : tcp://[server_host][:server_port]\n" - << " For UDP : udp://[bind_host][:bind_port]\n" - << " For Serial : serial:///path/to/serial/dev[:baudrate]\n" - << "For example, to connect to the simulator use URL: udpin://0.0.0.0:14540\n"; -} +void get_all(Param& param); +void get(Param& param, std::string name); +void set(Param& param, std::string name, std::string value); + +class CommandLineParser { +public: + static void print_usage(const std::string& bin_name) + { + std::cerr << "Usage :" << bin_name << " [args]\n" + << "\n" + << "Connection URL format should be :\n" + << " For TCP : tcp://[server_host][:server_port]\n" + << " For UDP : udp://[bind_host][:bind_port]\n" + << " For Serial : serial:///path/to/serial/dev[:baudrate]\n" + << "For example, to connect to the simulator use URL: udpin://0.0.0.0:14540\n" + << "\n" + << "Actions:\n" + << " get_all: Print all parameters\n" + << " get : Get one param\n" + << " set : Set one param\n" + << std::flush; + } + + enum class Result { + Ok, + Invalid, + }; + + enum class Action { + GetAll, + Get, + Set, + }; + + struct Args { + std::string connection; + Action action; + std::string param_name; // Empty for GetAll + std::string value; // Only used for Set + }; + + std::pair parse(int argc, char** argv) + { + if (argc < 3) { // Need at least program_name, url, and action + std::cerr << "Insufficient arguments" << std::endl; + return {Result::Invalid, {}}; + } + + Args args{}; + args.connection = argv[1]; + + std::string action_str = argv[2]; + + if (action_str == "get_all") { + if (argc != 3) { + std::cerr << "get_all takes no additional arguments" << std::endl; + return {Result::Invalid, {}}; + } + args.action = Action::GetAll; + } else if (action_str == "get") { + if (argc != 4) { + std::cerr << "get requires a parameter name" << std::endl; + return {Result::Invalid, {}}; + } + args.action = Action::Get; + args.param_name = argv[3]; + } else if (action_str == "set") { + if (argc != 5) { + std::cerr << "set requires a parameter name and value" << std::endl; + return {Result::Invalid, {}}; + } + + args.action = Action::Set; + args.param_name = argv[3]; + args.value = argv[4]; + } else { + std::cerr << "Unknown action: " << action_str << std::endl; + return {Result::Invalid, {}}; + } + return {Result::Ok, args}; + } +}; int main(int argc, char** argv) { - if (argc != 2) { - usage(argv[0]); - return 1; + CommandLineParser parser; + auto result = parser.parse(argc, argv); + switch (result.first) { + case CommandLineParser::Result::Invalid: + CommandLineParser::print_usage(argv[0]); + return 1; + + case CommandLineParser::Result::Ok: + break; } - const std::string connection_url = argv[1]; + const auto& args = result.second; Mavsdk mavsdk{Mavsdk::Configuration{ComponentType::GroundStation}}; - const ConnectionResult connection_result = mavsdk.add_any_connection(connection_url); + const ConnectionResult connection_result = mavsdk.add_any_connection(args.connection); if (connection_result != ConnectionResult::Success) { std::cerr << "Connection failed: " << connection_result << '\n'; @@ -69,18 +152,121 @@ int main(int argc, char** argv) // Instantiate plugins. auto param = Param{system}; + switch (args.action) { + case CommandLineParser::Action::GetAll: + get_all(param); + break; + + case CommandLineParser::Action::Get: + get(param, args.param_name); + break; + + case CommandLineParser::Action::Set: + set(param, args.param_name, args.value); + break; + }; + + return 0; +} + +void get_all(Param& param) +{ // Print params once. - const auto get_all_result = param.get_all_params(); + const auto result = param.get_all_params(); std::cout << "Int params: \n"; - for (auto p : get_all_result.int_params) { + for (auto p : result.int_params) { std::cout << p.name << ": " << p.value << '\n'; } std::cout << "Float params: \n"; - for (auto p : get_all_result.float_params) { + for (auto p : result.float_params) { std::cout << p.name << ": " << p.value << '\n'; } +} - return 0; +void get(Param& param, std::string name) +{ + // This is not very pretty, but it does seem to work." + std::cerr << "Try as float param..." << std::flush; + auto result = param.get_param_float(name); + + if (result.first == Param::Result::Success) { + std::cerr << "Ok" << std::endl; + std::cout << name << ": " << result.second << '\n'; + return; + } + + if (result.first != Param::Result::WrongType) { + std::cerr << "Failed: " << result.first << std::endl; + return; + } + + std::cout << "Wrong type" << std::endl; + std::cerr << "Try as int param next..." << std::flush; + result = param.get_param_int(name); + + if (result.first == Param::Result::Success) { + std::cerr << "Ok" << std::endl; + std::cout << name << ": " << result.second << '\n'; + return; + } + + if (result.first != Param::Result::WrongType) { + std::cerr << "Failed: " << result.first << std::endl; + return; + } + + std::cerr << "Failed: " << result.first << std::endl; + std::cerr << "Neither int or float worked, should maybe try custom? (not implemented)" + << std::endl; +} + +void set(Param& param, std::string name, std::string value) +{ + // Assume integer first + int int_value; + const auto int_result = std::from_chars(value.data(), value.data() + value.size(), int_value); + if (int_result.ec == std::errc() && int_result.ptr == value.data() + value.size()) { + std::cerr << "Setting " << value << " as integer..." << std::flush; + + auto result = param.set_param_int(name, int_value); + + if (result == Param::Result::Success) { + std::cerr << "Ok" << std::endl; + return; + } + + if (result != Param::Result::WrongType) { + std::cerr << "Failed: " << result << std::endl; + return; + } + + // If we got the type wrong, we try as float next. + } + + // Try float if integer failed + float float_value; + const auto float_result = + std::from_chars(value.data(), value.data() + value.size(), float_value); + if (float_result.ec == std::errc() && float_result.ptr == value.data() + value.size()) { + std::cerr << "Setting " << value << " as float..." << std::flush; + + auto result = param.set_param_float(name, float_value); + + if (result == Param::Result::Success) { + std::cerr << "Ok" << std::endl; + return; + } + + if (result != Param::Result::WrongType) { + std::cerr << "Failed: " << result << std::endl; + return; + } + + std::cerr << "Failed: " << result << std::endl; + } + + std::cerr << "Neither int or float worked, should maybe try custom? (not implemented)" + << std::endl; } From 6a6dcd839acf173dd277883d313fb34c07552c69 Mon Sep 17 00:00:00 2001 From: Julian Oes Date: Sun, 22 Dec 2024 13:54:55 +1300 Subject: [PATCH 4/4] examples: add our from_chars for float This is required because GCC 9 and 10 don't implement std::from_chars for float yet. --- examples/params/params.cpp | 23 ++++++++++++++++++++++- 1 file changed, 22 insertions(+), 1 deletion(-) diff --git a/examples/params/params.cpp b/examples/params/params.cpp index 47b8de249..3d0522b7f 100644 --- a/examples/params/params.cpp +++ b/examples/params/params.cpp @@ -7,12 +7,15 @@ #include #include +#include #include #include #include #include #include +std::from_chars_result our_from_chars_float(const char* first, const char* last, float& value); + using namespace mavsdk; void get_all(Param& param); @@ -248,7 +251,7 @@ void set(Param& param, std::string name, std::string value) // Try float if integer failed float float_value; const auto float_result = - std::from_chars(value.data(), value.data() + value.size(), float_value); + our_from_chars_float(value.data(), value.data() + value.size(), float_value); if (float_result.ec == std::errc() && float_result.ptr == value.data() + value.size()) { std::cerr << "Setting " << value << " as float..." << std::flush; @@ -270,3 +273,21 @@ void set(Param& param, std::string name, std::string value) std::cerr << "Neither int or float worked, should maybe try custom? (not implemented)" << std::endl; } + +// GCC 9/10 don't have std::from_chars for float yet, so we have to have our own (simplified) one. +std::from_chars_result our_from_chars_float(const char* first, const char* last, float& value) +{ + float parsed_value; + int chars_read; + + // sscanf returns number of successfully parsed items (should be 1) + // %n stores the number of characters read + if (sscanf(first, "%f%n", &parsed_value, &chars_read) == 1) { + if (first + chars_read <= last) { + value = parsed_value; + return {first + chars_read, std::errc{}}; + } + } + + return {first, std::errc::invalid_argument}; +}