aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorMartin Braun <martin.braun@ettus.com>2019-05-31 21:18:15 -0700
committerMartin Braun <martin.braun@ettus.com>2019-11-26 11:49:18 -0800
commit15c058015f56cfcd0e42cf6779a6e6ef6e0da911 (patch)
tree82a4c9311fea7c2eafab371e00bfef00e05596b8
parent73171698b55133aaeab461781476b02c4416da8f (diff)
downloaduhd-15c058015f56cfcd0e42cf6779a6e6ef6e0da911.tar.gz
uhd-15c058015f56cfcd0e42cf6779a6e6ef6e0da911.tar.bz2
uhd-15c058015f56cfcd0e42cf6779a6e6ef6e0da911.zip
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.
-rw-r--r--host/include/uhd/rfnoc/actions.hpp26
-rw-r--r--host/lib/rfnoc/actions.cpp24
-rw-r--r--host/tests/actions_test.cpp6
-rw-r--r--host/tests/rfnoc_graph_mock_nodes.hpp70
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 <uhd/config.hpp>
+#include <uhd/rfnoc/defaults.hpp>
+#include <uhd/types/stream_cmd.hpp>
#include <string>
#include <vector>
#include <memory>
@@ -24,6 +26,8 @@ namespace uhd { namespace rfnoc {
struct UHD_API action_info
{
public:
+ virtual ~action_info() {}
+
using sptr = std::shared_ptr<action_info>;
//! A unique counter for this action
const size_t id;
@@ -34,16 +38,26 @@ public:
std::vector<uint8_t> payload;
//! Factory function
- static sptr make(const std::string& key="")
- {
- //return std::make_shared<action_info>(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<stream_cmd_action_info>;
+
+ 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<action_info>(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_info>(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<uint8_t>(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<stream_cmd_action_info>(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<stream_cmd_action_info>(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);
}