aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--host/include/uhd/rfnoc/dirtifier.hpp22
-rw-r--r--host/include/uhd/rfnoc/property.hpp47
-rw-r--r--host/include/uhd/rfnoc/property.ipp22
-rw-r--r--host/lib/include/uhdlib/rfnoc/prop_accessor.hpp2
-rw-r--r--host/lib/rfnoc/node.cpp9
5 files changed, 84 insertions, 18 deletions
diff --git a/host/include/uhd/rfnoc/dirtifier.hpp b/host/include/uhd/rfnoc/dirtifier.hpp
index cc4430d3d..e31b26595 100644
--- a/host/include/uhd/rfnoc/dirtifier.hpp
+++ b/host/include/uhd/rfnoc/dirtifier.hpp
@@ -26,10 +26,22 @@ public:
}
//! This property is always dirty
- bool is_dirty() const { return true; }
+ bool is_dirty() const
+ {
+ return true;
+ }
+
+ //! This property is always invalid
+ bool is_valid() const
+ {
+ return false;
+ }
//! This property is never equal to anything else
- bool equal(property_base_t*) const { return false; }
+ bool equal(property_base_t*) const
+ {
+ return false;
+ }
//! Always dirty, so this can be called as often as we like
void force_dirty() {}
@@ -45,10 +57,12 @@ private:
}
//! This property never has the same type as another type
- bool is_type_equal(property_base_t*) const { return false; }
+ bool is_type_equal(property_base_t*) const
+ {
+ return false;
+ }
};
}} /* namespace uhd::rfnoc */
#endif /* INCLUDED_LIBUHD_DIRTIFYIER_HPP */
-
diff --git a/host/include/uhd/rfnoc/property.hpp b/host/include/uhd/rfnoc/property.hpp
index a85c978e9..d01613c58 100644
--- a/host/include/uhd/rfnoc/property.hpp
+++ b/host/include/uhd/rfnoc/property.hpp
@@ -10,8 +10,8 @@
#include <uhd/exception.hpp>
#include <uhd/rfnoc/res_source_info.hpp>
#include <uhd/utils/dirty_tracked.hpp>
-#include <string>
#include <memory>
+#include <string>
namespace uhd { namespace rfnoc {
@@ -27,9 +27,10 @@ class UHD_API property_base_t
public:
enum access_t {
NONE, //!< Neither reading nor writing to this property is permitted
- RO = 0x1, //!< Read-Only
- RW = 0x3, //!< Read-Write
- RWLOCKED = 0x5 //!< Write is locked. This lets you call set(), but only if the value is unchanged.
+ RO = 0x1, //!< Read-Only
+ RW = 0x3, //!< Read-Write
+ RWLOCKED = 0x5 //!< Write is locked. This lets you call set(), but only if the
+ //!< value is unchanged.
};
property_base_t(const std::string& id, const res_source_info& source_info)
@@ -55,6 +56,12 @@ public:
// have not propagated yet and still need resolving.
virtual bool is_dirty() const = 0;
+ //! Query this property's valid flag.
+ //
+ // If it's false, that means this property has a default value that should
+ // NOT be forwarded.
+ virtual bool is_valid() const = 0;
+
//! Returns true if this property can be read.
bool read_access_granted() const
{
@@ -132,6 +139,8 @@ public:
property_t(
const std::string& id, const data_t& value, const res_source_info& source_info);
+ property_t(const std::string& id, const res_source_info& source_info);
+
property_t(const property_t<data_t>& prop) = default;
//! Returns the dirty state of this property
@@ -143,6 +152,15 @@ public:
return _data.is_dirty();
}
+ //! Query this property's valid flag.
+ //
+ // If it's false, that means this property has a default value that should
+ // NOT be used.
+ bool is_valid() const
+ {
+ return _valid;
+ }
+
bool equal(property_base_t* rhs) const
{
if (!is_type_equal(rhs)) {
@@ -153,8 +171,8 @@ public:
std::unique_ptr<property_base_t> clone(res_source_info new_src_info) override
{
- return std::unique_ptr<property_base_t>(new property_t<data_t>(
- get_id(), get(), new_src_info));
+ return std::unique_ptr<property_base_t>(
+ new property_t<data_t>(get_id(), get(), new_src_info));
}
//! Returns the source info for the property
@@ -168,7 +186,8 @@ public:
void set(const data_t& value)
{
if (write_access_granted()) {
- _data = value;
+ _data = value;
+ _valid = true;
} else if (get_access_mode() == RWLOCKED) {
if (_data.get() != value) {
throw uhd::resolve_error(std::string("Attempting to overwrite property `")
@@ -198,8 +217,16 @@ public:
}
//! Get the value of this property
+ //
+ // \throws uhd::access_error if either the property is flagged as invalid,
+ // or if no read access was granted.
const data_t& get() const
{
+ if (!is_valid()) {
+ throw uhd::access_error(std::string("Attempting to read property `")
+ + get_id() + "@" + get_src_info().to_string()
+ + "' before it was initialized!");
+ }
if (read_access_granted()) {
return _data;
}
@@ -231,6 +258,10 @@ private:
void forward(property_base_t* next_prop)
{
+ if (not _valid) {
+ throw uhd::resolve_error(
+ std::string("Unable to forward invalid property ") + get_id());
+ }
property_t<data_t>* prop_ptr = dynamic_cast<property_t<data_t>*>(next_prop);
if (prop_ptr == nullptr) {
throw uhd::type_error(std::string("Unable to cast property ")
@@ -247,6 +278,7 @@ private:
}
dirty_tracked<data_t> _data;
+ bool _valid;
}; // class property_t
}} /* namespace uhd::rfnoc */
@@ -254,4 +286,3 @@ private:
#include <uhd/rfnoc/property.ipp>
#endif /* INCLUDED_LIBUHD_PROPERTY_HPP */
-
diff --git a/host/include/uhd/rfnoc/property.ipp b/host/include/uhd/rfnoc/property.ipp
index 101aa5d7e..87b4394e6 100644
--- a/host/include/uhd/rfnoc/property.ipp
+++ b/host/include/uhd/rfnoc/property.ipp
@@ -8,18 +8,30 @@
#define INCLUDED_LIBUHD_PROPERTY_IPP
template <typename data_t>
-uhd::rfnoc::property_t<data_t>::property_t(const std::string& id, data_t&& data, const uhd::rfnoc::res_source_info& source_info)
- : uhd::rfnoc::property_base_t(id, source_info), _data(std::forward<data_t>(data))
+uhd::rfnoc::property_t<data_t>::property_t(
+ const std::string& id, data_t&& data, const uhd::rfnoc::res_source_info& source_info)
+ : uhd::rfnoc::property_base_t(id, source_info)
+ , _data(std::forward<data_t>(data))
+ , _valid(true)
{
// nop
}
template <typename data_t>
-uhd::rfnoc::property_t<data_t>::property_t(const std::string& id, const data_t& data, const uhd::rfnoc::res_source_info& source_info)
- : uhd::rfnoc::property_base_t(id, source_info), _data(data)
+uhd::rfnoc::property_t<data_t>::property_t(const std::string& id,
+ const data_t& data,
+ const uhd::rfnoc::res_source_info& source_info)
+ : uhd::rfnoc::property_base_t(id, source_info), _data(data), _valid(true)
{
// nop
}
-#endif /* INCLUDED_LIBUHD_PROPERTY_IPP */
+template <typename data_t>
+uhd::rfnoc::property_t<data_t>::property_t(
+ const std::string& id, const uhd::rfnoc::res_source_info& source_info)
+ : uhd::rfnoc::property_base_t(id, source_info), _valid(false)
+{
+ // nop
+}
+#endif /* INCLUDED_LIBUHD_PROPERTY_IPP */
diff --git a/host/lib/include/uhdlib/rfnoc/prop_accessor.hpp b/host/lib/include/uhdlib/rfnoc/prop_accessor.hpp
index a62f54620..2f39cbbec 100644
--- a/host/lib/include/uhdlib/rfnoc/prop_accessor.hpp
+++ b/host/lib/include/uhdlib/rfnoc/prop_accessor.hpp
@@ -65,7 +65,7 @@ public:
*
* Note: This method will grant temporary write access to the destination
* property!
- * If \p safe is set to true, it'll only allow RWLOCKED forwarding, i.e.,
+ * If \p safe is set to true, it'll only allow RWLOCKED forwarding, i.e.,
* the new value cannot be different.
*
* \throws uhd::type_error if types mismatch
diff --git a/host/lib/rfnoc/node.cpp b/host/lib/rfnoc/node.cpp
index b3ad2b380..cdb69e158 100644
--- a/host/lib/rfnoc/node.cpp
+++ b/host/lib/rfnoc/node.cpp
@@ -444,6 +444,15 @@ void node_t::forward_edge_property(
"Incoming edge property: `" << incoming_prop->get_id() << "`, source info: "
<< incoming_prop->get_src_info().to_string());
+ // Don't forward properties that are not yet valid
+ if (!incoming_prop->is_valid()) {
+ UHD_LOG_TRACE(get_unique_id(),
+ "Skipped empty edge property: `"
+ << incoming_prop->get_id()
+ << "`, source info: " << incoming_prop->get_src_info().to_string());
+ return;
+ }
+
// The source type of my local prop (it's the opposite of the source type
// of incoming_prop)
const auto prop_src_type =