From efb1d5a4729ea892ea03b8d0265aae9e8fadfff1 Mon Sep 17 00:00:00 2001 From: Martin Braun Date: Sun, 24 Mar 2019 16:54:11 -0700 Subject: rfnoc: Add properties, nodes, and accessors Adds the following classes: - uhd::rfnoc::node_t, the base class for RFNoC nodes - uhd::rfnoc::node_accessor_t, a class to access private properties - uhd::rfnoc::res_source_info, a struct that identifies where properties come from - uhd::rfnoc::property_t, and property_base_t (its parent) - uhd::rfnoc::prop_accessor_t, a class to access properties Add always dirty property (dirtifier). Also adds unit tests for properties. --- host/include/uhd/rfnoc/CMakeLists.txt | 3 + host/include/uhd/rfnoc/dirtifier.hpp | 54 ++++++ host/include/uhd/rfnoc/node.hpp | 206 +++++++++++++++++++++++ host/include/uhd/rfnoc/node.ipp | 78 +++++++++ host/include/uhd/rfnoc/property.hpp | 257 +++++++++++++++++++++++++++++ host/include/uhd/rfnoc/property.ipp | 25 +++ host/include/uhd/rfnoc/res_source_info.hpp | 89 ++++++++++ 7 files changed, 712 insertions(+) create mode 100644 host/include/uhd/rfnoc/dirtifier.hpp create mode 100644 host/include/uhd/rfnoc/node.hpp create mode 100644 host/include/uhd/rfnoc/node.ipp create mode 100644 host/include/uhd/rfnoc/property.hpp create mode 100644 host/include/uhd/rfnoc/property.ipp create mode 100644 host/include/uhd/rfnoc/res_source_info.hpp (limited to 'host/include') diff --git a/host/include/uhd/rfnoc/CMakeLists.txt b/host/include/uhd/rfnoc/CMakeLists.txt index 7b5b69f4b..bc5d7f054 100644 --- a/host/include/uhd/rfnoc/CMakeLists.txt +++ b/host/include/uhd/rfnoc/CMakeLists.txt @@ -13,9 +13,12 @@ if(ENABLE_RFNOC) blockdef.hpp block_id.hpp constants.hpp + dirtifier.hpp graph.hpp node_ctrl_base.hpp node_ctrl_base.ipp + node.hpp + node.ipp rate_node_ctrl.hpp scalar_node_ctrl.hpp sink_block_ctrl_base.hpp diff --git a/host/include/uhd/rfnoc/dirtifier.hpp b/host/include/uhd/rfnoc/dirtifier.hpp new file mode 100644 index 000000000..cc4430d3d --- /dev/null +++ b/host/include/uhd/rfnoc/dirtifier.hpp @@ -0,0 +1,54 @@ +// +// Copyright 2019 Ettus Research, a National Instruments Brand +// +// SPDX-License-Identifier: GPL-3.0-or-later +// + +#ifndef INCLUDED_LIBUHD_DIRTIFYIER_HPP +#define INCLUDED_LIBUHD_DIRTIFYIER_HPP + +#include + +namespace uhd { namespace rfnoc { + +/*! This is a special class for property that is always dirty. This is useful + * to force property resolutions in certain cases. + * + * Note: This has nothing to do with 'dirtify' in the CGI/graphics sense. + */ +class dirtifier_t : public property_base_t +{ +public: + dirtifier_t() + : property_base_t("__ALWAYS_DIRTY__", res_source_info(res_source_info::FRAMEWORK)) + { + // nop + } + + //! This property is always dirty + bool is_dirty() const { return true; } + + //! This property is never equal to anything else + bool equal(property_base_t*) const { return false; } + + //! Always dirty, so this can be called as often as we like + void force_dirty() {} + +private: + //! This property cannot be marked clean, but nothing happens if you try + void mark_clean() {} + + //! The value from this property cannot be forwarded + void forward(property_base_t*) + { + throw uhd::type_error("Cannot forward to or from dirtifier property!"); + } + + //! This property never has the same type as another type + 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/node.hpp b/host/include/uhd/rfnoc/node.hpp new file mode 100644 index 000000000..5f8a15f4f --- /dev/null +++ b/host/include/uhd/rfnoc/node.hpp @@ -0,0 +1,206 @@ +// +// Copyright 2019 Ettus Research, a National Instruments Brand +// +// SPDX-License-Identifier: GPL-3.0-or-later +// + +#ifndef INCLUDED_LIBUHD_RFNOC_NODE_HPP +#define INCLUDED_LIBUHD_RFNOC_NODE_HPP + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +namespace uhd { namespace rfnoc { + +/*! The base class for all nodes within an RFNoC graph + * + * The block supports the following types of data access: + * - High-level property access + * - Action execution + */ +class UHD_API node_t +{ +public: + using resolver_fn_t = std::function; + + /****************************************** + * Basic Operations + ******************************************/ + //! Return a unique identifier string for this node. In every RFNoC graph, + // no two nodes cannot have the same ID. + // + // \returns The unique ID as a string + virtual std::string get_unique_id() const; + + /*! Return the number of input ports for this block. + * + * This function needs to be overridden. + * + * \return noc_id The number of ports + */ + virtual size_t get_num_input_ports() const = 0; + + /*! Return the number of output ports for this block. + * + * This function needs to be overridden. + * + * \return noc_id The number of ports + */ + virtual size_t get_num_output_ports() const = 0; + + /****************************************** + * Property Specific + ******************************************/ + + /*! Return the names of all possible user properties that can be + * accessed for this block. + * + * Note that the type of the property is not auto-detectable. + * + * \returns A vector of all possible IDs of user properties supported by + * this block. + */ + std::vector get_property_ids() const; + + /*! Set a specific user property that belongs to this block. + * + * Setting a user property will trigger a property resolution. This means + * that changing this block can have effects on other nodes. + * + * If the property does not exist, or if the property can be determined to + * be of a different type than \p prop_data_t due to the usage of runtime + * type information (RTTI), a lookup_error is thrown. + * + * \param prop_data_t The data type of the property + * \param id The identifier of the property to write. To find out which + * values of \p id are valid, call get_property_ids() + * \param instance The instance number of this property + * \param val The new value of the property. + */ + template + void set_property( + const std::string& id, const prop_data_t& val, const size_t instance = 0); + + /*! Get the value of a specific block argument. \p The type of an argument + * must be known at compile time. + * + * If the property does not exist, or if the property can be determined to + * be of a different type than \p prop_data_t due to the usage of runtime + * type information (RTTI), a lookup_error is thrown. + * + * Note: Despite this being a "getter", this function is not declared const. + * This is because internally, it can resolve properties, which may cause + * changes within the object. + * + * \param prop_data_t The data type of the property + * \param id The identifier of the property to write. + * \param instance The instance number of this property + * \return The value of the property. + * \throws uhd::lookup_error if the property can't be found. + */ + template + const prop_data_t& get_property( + const std::string& id, const size_t instance = 0) /* mutable */; + + /****************************************** + * Action Specific + ******************************************/ + // TBW + +protected: + /****************************************** + * Internal Registration Functions + ******************************************/ + + /*! Register a property for this block + * + * \param prop A reference to the property + * + * \throws uhd::key_error if another property with the same ID and source + * type is already registered + */ + void register_property(property_base_t* prop); + + /*! Add a resolver function to this block. A resolver function is used to + * reconcile state changes in the block, and is triggered by a + * user or other upstream/downstream blocks. A block may have multiple + * resolvers. + * + * NOTE: Multiple resolvers may share properties for reading and a + * resolver may read multiple properties + * NOTE: The framework will perform run-time validation to + * ensure read/write property access is not violated. This means the + * resolver function can only read values that are specified in the + * \p inputs list, and only write values that are specified in the + * \p outputs list. + * + * \param inputs The properties that this resolver will read + * \param outputs The properties that this resolver will write to + * \param resolver_fn The resolver function + */ + void add_property_resolver(std::set&& inputs, + std::set&& outputs, + resolver_fn_t&& resolver_fn); + + /*! Handle a request to perform an action. The default action handler + * ignores user action and forwards port actions. + * + * \param handler The function that is called to handle the action + */ + //void register_action_handler(std::function< + //void(const action_info& info, const res_source_info& src) + //> handler); + + /****************************************** + * Internal action forwarding + ******************************************/ + // TBW + // + +private: + + /*! Return a reference to a property, if it exists. + * + * \returns A reference to the property, if it exists, or nullptr otherwise + */ + property_base_t* _find_property( + res_source_info src_info, const std::string& id) const; + + /*! RAII-Style property access + */ + uhd::utils::scope_exit::uptr _request_property_access( + property_base_t* prop, property_base_t::access_t access) const; + + /****** Attributes *******************************************************/ + //! Mutex to lock access to the property registry + mutable std::mutex _prop_mutex; + + //! Stores a reference to every registered property (Property Registry) + std::unordered_map, + std::hash > + _props; + + using property_resolver_t = std::tuple, + std::set, + resolver_fn_t> ; + //! Stores the list of property resolvers + std::vector _prop_resolvers; + +}; // class node_t + +}} /* namespace uhd::rfnoc */ + +#include + +#endif /* INCLUDED_LIBUHD_RFNOC_NODE_HPP */ diff --git a/host/include/uhd/rfnoc/node.ipp b/host/include/uhd/rfnoc/node.ipp new file mode 100644 index 000000000..7de69063c --- /dev/null +++ b/host/include/uhd/rfnoc/node.ipp @@ -0,0 +1,78 @@ +// +// Copyright 2019 Ettus Research, a National Instruments Brand +// +// SPDX-License-Identifier: GPL-3.0-or-later +// + +#ifndef INCLUDED_LIBUHD_RFNOC_NODE_IPP +#define INCLUDED_LIBUHD_RFNOC_NODE_IPP + +#include +#include + +namespace { + +template +uhd::rfnoc::property_t* _assert_prop( + uhd::rfnoc::property_base_t* prop_base_ptr, + const std::string& node_id, + const std::string& prop_id) +{ + // First check if the pointer is valid at all: + if (prop_base_ptr == nullptr) { + throw uhd::lookup_error( + str(boost::format("[%s] Unknown user property: `%s'") % node_id % prop_id)); + } + + // Next, check if we can cast the pointer to the desired type: + auto prop_ptr = + dynamic_cast*>(prop_base_ptr); + if (!prop_ptr) { + throw uhd::type_error(str( + boost::format( + "[%s] Found user property `%s', but could not cast to requested type `%s'!") + % node_id % prop_id % boost::units::detail::demangle(typeid(prop_data_t).name()) )); + } + + // All is good, we now return the raw pointer that has been validated. + return prop_ptr; +} + +} // namespace + +namespace uhd { namespace rfnoc { + +template +void node_t::set_property( + const std::string& id, const prop_data_t& val, const size_t instance) +{ + res_source_info src_info{res_source_info::USER, instance}; + UHD_LOG_TRACE(get_unique_id(), "Setting property `" << id << "`"); + auto prop_ptr = + _assert_prop(_find_property(src_info, id), get_unique_id(), id); + { + auto prop_access = _request_property_access(prop_ptr, property_base_t::RW); + prop_ptr->set(val); + } + + // resolve_all() TODO +} + +template +const prop_data_t& node_t::get_property(const std::string& id, const size_t instance) +{ + res_source_info src_info{res_source_info::USER, instance}; + + // resolve_all() TODO + auto prop_ptr = _assert_prop( + _find_property(src_info, id), get_unique_id(), id); + + auto prop_access = _request_property_access(prop_ptr, property_base_t::RO); + return prop_ptr->get(); +} + + +}} /* namespace uhd::rfnoc */ + +#endif /* INCLUDED_LIBUHD_RFNOC_NODE_IPP */ + diff --git a/host/include/uhd/rfnoc/property.hpp b/host/include/uhd/rfnoc/property.hpp new file mode 100644 index 000000000..a85c978e9 --- /dev/null +++ b/host/include/uhd/rfnoc/property.hpp @@ -0,0 +1,257 @@ +// +// Copyright 2019 Ettus Research, a National Instruments Brand +// +// SPDX-License-Identifier: GPL-3.0-or-later +// + +#ifndef INCLUDED_LIBUHD_PROPERTY_HPP +#define INCLUDED_LIBUHD_PROPERTY_HPP + +#include +#include +#include +#include +#include + + +namespace uhd { namespace rfnoc { + +// Forward declaration, separates includes +class prop_accessor_t; + +/*! Base class for properties + * + */ +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. + }; + + property_base_t(const std::string& id, const res_source_info& source_info) + : _id(id), _source_info(source_info) + { + } + + //! Gets the ID (name) of this property + const std::string& get_id() const + { + return _id; + } + + //! Return the source info for this property + const res_source_info& get_src_info() const + { + return _source_info; + } + + //! Query this property's dirty flag. + // + // If it's true, that means this property was recently changed, but changes + // have not propagated yet and still need resolving. + virtual bool is_dirty() const = 0; + + //! Returns true if this property can be read. + bool read_access_granted() const + { + return static_cast(_access_mode) & 0x1; + } + + //! Returns true if this property can be written to. + bool write_access_granted() const + { + return static_cast(_access_mode) & 0x2; + } + + //! Return the current access mode + access_t get_access_mode() const + { + return _access_mode; + } + + //! Return true if rhs has the same type and value + virtual bool equal(property_base_t* rhs) const = 0; + + //! Create a copy of this property + // + // The copy must have the same type, value, and ID. However, it is often + // desirable to have a new source information, so that can be overridden. + // + // The cleanliness state of \p original is not preserved. The new property + // will have the same cleanliness state as any other new property. + // + virtual std::unique_ptr clone(res_source_info) + { + throw uhd::not_implemented_error("Cloning is not available for this property."); + } + + virtual void force_dirty() = 0; + +private: + friend class prop_accessor_t; + + //! Reset the dirty bit. See also dirty_tracked::mark_clean() + virtual void mark_clean() = 0; + + //! Forward the value of this property to another one + virtual void forward(property_base_t* next_prop) = 0; + + //! Compare property types + // + // Note: This uses RTTI to evaluate type equality + virtual bool is_type_equal(property_base_t* other_prop) const = 0; + + /*** Attributes **********************************************************/ + //! Stores an ID string for this property. They don't need to be unique. + const std::string _id; + + //! Stores the source info for this property. + const res_source_info _source_info; + + //! Access mode. Note that the prop_accessor_t is the only one who can + // write this. + access_t _access_mode = RO; +}; + +/*! + * An encapsulation class for a block property. + */ +template +class property_t : public property_base_t +{ +public: + //! We want to be good C++ citizens + using value_type = data_t; + + property_t(const std::string& id, data_t&& value, const res_source_info& source_info); + + property_t( + const std::string& id, const data_t& value, const res_source_info& source_info); + + property_t(const property_t& prop) = default; + + //! Returns the dirty state of this property + // + // If true, this means the value was recently changed, but it wasn't marked + // clean yet. + bool is_dirty() const + { + return _data.is_dirty(); + } + + bool equal(property_base_t* rhs) const + { + if (!is_type_equal(rhs)) { + return false; + } + return get() == dynamic_cast*>(rhs)->get(); + } + + std::unique_ptr clone(res_source_info new_src_info) override + { + return std::unique_ptr(new property_t( + get_id(), get(), new_src_info)); + } + + //! Returns the source info for the property + // const res_source_info& get_src_info() const = 0; + + //! Set the value of this property + // + // \throws uhd::access_error if the current access mode is not RW or RWLOCKED + // \throws uhd::resolve_error if the property is RWLOCKED but the new value + // doesn't match + void set(const data_t& value) + { + if (write_access_granted()) { + _data = value; + } else if (get_access_mode() == RWLOCKED) { + if (_data.get() != value) { + throw uhd::resolve_error(std::string("Attempting to overwrite property `") + + get_id() + "@" + get_src_info().to_string() + + "' with a new value after it was locked!"); + } + } else { + throw uhd::access_error(std::string("Attempting to write to property `") + + get_id() + "' without access privileges!"); + } + } + + void force_dirty() + { + if (write_access_granted()) { + _data.force_dirty(); + } else if (get_access_mode() == RWLOCKED) { + if (!_data.is_dirty()) { + throw uhd::resolve_error(std::string("Attempting to overwrite property `") + + get_id() + + "' with dirty flag after it was locked!"); + } + } else { + throw uhd::access_error(std::string("Attempting to flag dirty property `") + + get_id() + "' without access privileges!"); + } + } + + //! Get the value of this property + const data_t& get() const + { + if (read_access_granted()) { + return _data; + } + throw uhd::access_error(std::string("Attempting to read property `") + get_id() + + "' without access privileges!"); + } + + operator const data_t&() const + { + return get(); + } + + bool operator==(const data_t& rhs) + { + return get() == rhs; + } + + property_t& operator=(const data_t& value) + { + set(value); + return *this; + } + +private: + void mark_clean() + { + _data.mark_clean(); + } + + void forward(property_base_t* next_prop) + { + property_t* prop_ptr = dynamic_cast*>(next_prop); + if (prop_ptr == nullptr) { + throw uhd::type_error(std::string("Unable to cast property ") + + next_prop->get_id() + " to the same type as property " + + get_id()); + } + + prop_ptr->set(get()); + } + + bool is_type_equal(property_base_t* other_prop) const + { + return dynamic_cast*>(other_prop) != nullptr; + } + + dirty_tracked _data; +}; // class property_t + +}} /* namespace uhd::rfnoc */ + +#include + +#endif /* INCLUDED_LIBUHD_PROPERTY_HPP */ + diff --git a/host/include/uhd/rfnoc/property.ipp b/host/include/uhd/rfnoc/property.ipp new file mode 100644 index 000000000..101aa5d7e --- /dev/null +++ b/host/include/uhd/rfnoc/property.ipp @@ -0,0 +1,25 @@ +// +// Copyright 2019 Ettus Research, a National Instruments Brand +// +// SPDX-License-Identifier: GPL-3.0-or-later +// + +#ifndef INCLUDED_LIBUHD_PROPERTY_IPP +#define INCLUDED_LIBUHD_PROPERTY_IPP + +template +uhd::rfnoc::property_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)) +{ + // nop +} + +template +uhd::rfnoc::property_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) +{ + // nop +} + +#endif /* INCLUDED_LIBUHD_PROPERTY_IPP */ + diff --git a/host/include/uhd/rfnoc/res_source_info.hpp b/host/include/uhd/rfnoc/res_source_info.hpp new file mode 100644 index 000000000..37371009d --- /dev/null +++ b/host/include/uhd/rfnoc/res_source_info.hpp @@ -0,0 +1,89 @@ +// +// Copyright 2019 Ettus Research, a National Instruments Brand +// +// SPDX-License-Identifier: GPL-3.0-or-later +// + +#ifndef INCLUDED_LIBUHD_RES_SRC_INFO_HPP +#define INCLUDED_LIBUHD_RES_SRC_INFO_HPP + +#include +#include +#include + +namespace uhd { namespace rfnoc { + +/*! Describes the source of a particular resource (property or action) +*/ +struct res_source_info { + /*! Source type + */ + enum source_t { + USER, ///< The user API sources this resource + INPUT_EDGE, ///< An input edge sources this resource + OUTPUT_EDGE, ///< An input edge sources this resource + FRAMEWORK ///< This is a special resource, only accessed by the framework + }; + + // No default ctor: The source type must be specified + res_source_info() = delete; + + res_source_info(source_t source_type, size_t instance_ = 0) + : type(source_type), instance(instance_) + { + // nop + } + + //! The type of source (user or edge) + source_t type; + + //! The instance of the source. For resource that is sourced by a edge, it + // corresponds to the port number + size_t instance = 0; + + bool operator==(const res_source_info& rhs) const + { + return rhs.type == type && rhs.instance == instance; + } + + //! Returns a string representation of the source + std::string to_string() const + { + const std::string type_repr = type == USER ? "USER" : + type == INPUT_EDGE ? "INPUT_EDGE" : + type == OUTPUT_EDGE ? "OUTPUT_EDGE" : "INVALID"; + return type_repr + ":" + std::to_string(instance); + } + + /*! Convenience function to invert the type value if it's an edge. + * + * - Will assert that the edge is actually either INPUT_EDGE or OUTPUT_EDGE + * - Then, returns the opposite of what it was before + * + * \throws uhd::assertion_error if \p edge_direction is neither INPUT_EDGE + * nor OUTPUT_EDGE + */ + static source_t invert_edge(const source_t edge_direction) + { + UHD_ASSERT_THROW(edge_direction == INPUT_EDGE || edge_direction == OUTPUT_EDGE); + return edge_direction == INPUT_EDGE ? OUTPUT_EDGE : INPUT_EDGE; + } +}; + +}} /* namespace uhd::rfnoc */ + +namespace std { +template <> +struct hash +{ + size_t operator()(const uhd::rfnoc::res_source_info& src_info) const + { + const size_t hash_type = std::hash{}(src_info.type); + const size_t hash_inst = std::hash{}(src_info.instance); + return hash_type ^ (hash_inst << 1); + } +}; +} // namespace std + +#endif /* INCLUDED_LIBUHD_RES_SRC_INFO_HPP */ + -- cgit v1.2.3