From 15c058015f56cfcd0e42cf6779a6e6ef6e0da911 Mon Sep 17 00:00:00 2001 From: Martin Braun Date: Fri, 31 May 2019 21:18:15 -0700 Subject: rfnoc: Use RTTI "serialization" for stream commands A small modification to rfnoc::action_info makes it polymorphic, and instead of serializing data structures into a string, this allows creating custom action objects and identifying them via RTTI. The stream command action object is a good example for how to use this, so all the usages of stream command action objects were converted to this scheme. --- host/include/uhd/rfnoc/actions.hpp | 26 ++++++++++--- host/lib/rfnoc/actions.cpp | 24 ++++++++++++ host/tests/actions_test.cpp | 6 +-- host/tests/rfnoc_graph_mock_nodes.hpp | 70 ++++++++++++++++++----------------- 4 files changed, 82 insertions(+), 44 deletions(-) diff --git a/host/include/uhd/rfnoc/actions.hpp b/host/include/uhd/rfnoc/actions.hpp index ac454827c..611cb787c 100644 --- a/host/include/uhd/rfnoc/actions.hpp +++ b/host/include/uhd/rfnoc/actions.hpp @@ -8,6 +8,8 @@ #define INCLUDED_LIBUHD_RFNOC_ACTIONS_HPP #include +#include +#include #include #include #include @@ -24,6 +26,8 @@ namespace uhd { namespace rfnoc { struct UHD_API action_info { public: + virtual ~action_info() {} + using sptr = std::shared_ptr; //! A unique counter for this action const size_t id; @@ -34,16 +38,26 @@ public: std::vector payload; //! Factory function - static sptr make(const std::string& key="") - { - //return std::make_shared(key); - return sptr(new action_info(key)); - } + static sptr make(const std::string& key=""); -private: +protected: action_info(const std::string& key); }; +struct UHD_API stream_cmd_action_info : public action_info +{ +public: + using sptr = std::shared_ptr; + + uhd::stream_cmd_t stream_cmd; + + //! Factory function + static sptr make(const uhd::stream_cmd_t::stream_mode_t stream_mode); + +private: + stream_cmd_action_info(const uhd::stream_cmd_t::stream_mode_t stream_mode); +}; + }} /* namespace uhd::rfnoc */ #endif /* INCLUDED_LIBUHD_RFNOC_ACTIONS_HPP */ diff --git a/host/lib/rfnoc/actions.cpp b/host/lib/rfnoc/actions.cpp index 1f5f0f2f7..428a1fcb2 100644 --- a/host/lib/rfnoc/actions.cpp +++ b/host/lib/rfnoc/actions.cpp @@ -19,3 +19,27 @@ action_info::action_info(const std::string& key_) : id(action_counter++), key(ke // nop } +//! Factory function +action_info::sptr action_info::make(const std::string& key) +{ + if (key == ACTION_KEY_STREAM_CMD) { + return stream_cmd_action_info::make( + uhd::stream_cmd_t::STREAM_MODE_STOP_CONTINUOUS); + } + //return std::make_shared(key); + return sptr(new action_info(key)); +} + +stream_cmd_action_info::stream_cmd_action_info( + const uhd::stream_cmd_t::stream_mode_t stream_mode) + : action_info(ACTION_KEY_STREAM_CMD), stream_cmd(stream_mode) +{ + // nop +} + +stream_cmd_action_info::sptr stream_cmd_action_info::make( + const uhd::stream_cmd_t::stream_mode_t stream_mode) +{ + //return std::make_shared(ACTION_KEY_STREAM_CMD); + return sptr(new stream_cmd_action_info(stream_mode)); +} diff --git a/host/tests/actions_test.cpp b/host/tests/actions_test.cpp index c0344eacf..b5feb2521 100644 --- a/host/tests/actions_test.cpp +++ b/host/tests/actions_test.cpp @@ -25,10 +25,8 @@ BOOST_AUTO_TEST_CASE(test_actions_single_node) // Define some mock nodes: mock_radio_node_t mock_radio(0); - auto stream_cmd = action_info::make(STREAM_CMD_KEY); - std::string cmd_payload = "START"; - stream_cmd->payload = std::vector(cmd_payload.begin(), cmd_payload.end()); - + auto stream_cmd = + stream_cmd_action_info::make(uhd::stream_cmd_t::STREAM_MODE_START_CONTINUOUS); auto other_cmd = action_info::make("FOO"); node_accessor.send_action(&mock_radio, {res_source_info::INPUT_EDGE, 0}, stream_cmd); diff --git a/host/tests/rfnoc_graph_mock_nodes.hpp b/host/tests/rfnoc_graph_mock_nodes.hpp index a9d8d4e55..2137e3336 100644 --- a/host/tests/rfnoc_graph_mock_nodes.hpp +++ b/host/tests/rfnoc_graph_mock_nodes.hpp @@ -88,20 +88,23 @@ public: set_action_forwarding_policy(forwarding_policy_t::DROP); - register_action_handler( - "stream_cmd", [this](const res_source_info& src, action_info::sptr action) { - UHD_ASSERT_THROW(action->key == "stream_cmd"); - const std::string cmd(action->payload.begin(), action->payload.end()); - UHD_LOG_INFO(get_unique_id(), - "Received stream command: " << cmd << " to " << src.to_string()); - if (cmd == "START") { + register_action_handler(ACTION_KEY_STREAM_CMD, + [this](const res_source_info& src, action_info::sptr action) { + stream_cmd_action_info::sptr stream_cmd_action = + std::dynamic_pointer_cast(action); + UHD_ASSERT_THROW(stream_cmd_action); + uhd::stream_cmd_t::stream_mode_t stream_mode = + stream_cmd_action->stream_cmd.stream_mode; + RFNOC_LOG_INFO("Received stream command: " << stream_mode << " to " + << src.to_string() + << ", id==" << action->id); + if (stream_mode == uhd::stream_cmd_t::STREAM_MODE_START_CONTINUOUS) { UHD_LOG_INFO(get_unique_id(), "Starting Stream!"); - } else if (cmd == "STOP") { + } else if (stream_mode == uhd::stream_cmd_t::STREAM_MODE_START_CONTINUOUS) { UHD_LOG_INFO(get_unique_id(), "Stopping Stream!"); } else { - this->last_num_samps = std::stoul(cmd); - UHD_LOG_INFO(get_unique_id(), - "Streaming num samps: " << this->last_num_samps); + this->last_num_samps = stream_cmd_action->stream_cmd.num_samps; + RFNOC_LOG_INFO("Streaming num samps: " << this->last_num_samps); } }); } @@ -196,27 +199,32 @@ public: }); register_action_handler( - "stream_cmd", [this](const res_source_info& src, action_info::sptr action) { + ACTION_KEY_STREAM_CMD, [this](const res_source_info& src, action_info::sptr action) { res_source_info dst_edge{ res_source_info::invert_edge(src.type), src.instance}; - auto new_action = action_info::make(action->key); - std::string cmd(action->payload.begin(), action->payload.end()); - if (cmd == "START" || cmd == "STOP") { - new_action->payload = action->payload; - } else { - unsigned long long num_samps = std::stoull(cmd); + stream_cmd_action_info::sptr stream_cmd_action = + std::dynamic_pointer_cast(action); + UHD_ASSERT_THROW(stream_cmd_action); + uhd::stream_cmd_t::stream_mode_t stream_mode = + stream_cmd_action->stream_cmd.stream_mode; + RFNOC_LOG_INFO("Received stream command: " << stream_mode << " to " + << src.to_string() + << ", id==" << action->id); + auto new_action = stream_cmd_action_info::make(stream_mode); + new_action->stream_cmd = stream_cmd_action->stream_cmd; + if (stream_mode == uhd::stream_cmd_t::STREAM_MODE_NUM_SAMPS_AND_DONE + || stream_mode == uhd::stream_cmd_t::STREAM_MODE_NUM_SAMPS_AND_MORE) { if (src.type == res_source_info::OUTPUT_EDGE) { - num_samps *= _decim.get(); + RFNOC_LOG_INFO("Multiplying num_samps by " << _decim.get()); + new_action->stream_cmd.num_samps *= _decim.get(); } else { - num_samps /= _decim.get(); + RFNOC_LOG_INFO("Dividing num_samps by " << _decim.get()); + new_action->stream_cmd.num_samps /= _decim.get(); } - std::string new_cmd = std::to_string(num_samps); - new_action->payload.insert( - new_action->payload.begin(), new_cmd.begin(), new_cmd.end()); } - UHD_LOG_INFO(get_unique_id(), - "Forwarding stream_cmd, decim is " << _decim.get()); + RFNOC_LOG_INFO("Forwarding stream_cmd, num_samps is " + << new_action->stream_cmd.num_samps << ", id==" << new_action->id); post_action(dst_edge, new_action); }); } @@ -328,15 +336,9 @@ public: void issue_stream_cmd(uhd::stream_cmd_t stream_cmd, const size_t chan) { - std::string cmd = - stream_cmd.stream_mode == uhd::stream_cmd_t::STREAM_MODE_START_CONTINUOUS - ? "START" - : stream_cmd.stream_mode == uhd::stream_cmd_t::STREAM_MODE_STOP_CONTINUOUS - ? "STOP" - : std::to_string(stream_cmd.num_samps); - auto scmd = action_info::make("stream_cmd"); - scmd->payload.insert(scmd->payload.begin(), cmd.begin(), cmd.end()); - + auto scmd = + stream_cmd_action_info::make(uhd::stream_cmd_t::STREAM_MODE_START_CONTINUOUS); + scmd->stream_cmd = stream_cmd; post_action({res_source_info::INPUT_EDGE, chan}, scmd); } -- cgit v1.2.3