diff options
Diffstat (limited to 'host/lib')
-rw-r--r-- | host/lib/include/uhdlib/rfnoc/epid_allocator.hpp | 9 | ||||
-rw-r--r-- | host/lib/include/uhdlib/rfnoc/graph_stream_manager.hpp | 50 | ||||
-rw-r--r-- | host/lib/include/uhdlib/rfnoc/link_stream_manager.hpp | 89 | ||||
-rw-r--r-- | host/lib/include/uhdlib/rfnoc/mgmt_portal.hpp | 31 | ||||
-rw-r--r-- | host/lib/include/uhdlib/rfnoc/rfnoc_common.hpp | 18 | ||||
-rw-r--r-- | host/lib/rfnoc/epid_allocator.cpp | 14 | ||||
-rw-r--r-- | host/lib/rfnoc/graph_stream_manager.cpp | 104 | ||||
-rw-r--r-- | host/lib/rfnoc/link_stream_manager.cpp | 98 | ||||
-rw-r--r-- | host/lib/rfnoc/mgmt_portal.cpp | 76 |
9 files changed, 408 insertions, 81 deletions
diff --git a/host/lib/include/uhdlib/rfnoc/epid_allocator.hpp b/host/lib/include/uhdlib/rfnoc/epid_allocator.hpp index c92ca013d..ec23dcb50 100644 --- a/host/lib/include/uhdlib/rfnoc/epid_allocator.hpp +++ b/host/lib/include/uhdlib/rfnoc/epid_allocator.hpp @@ -33,12 +33,19 @@ public: */ sep_id_t allocate_epid(const sep_addr_t& addr); + /*! \brief Get a pre-allocated EPID. Throws an exception is not allocated + * + * \param addr The physical address (device, instance) of the stream endpoint + * \return The allocated EPID + */ + sep_id_t get_epid(const sep_addr_t& addr); + /*! \brief Lookup an EPID and return the address associated with it. * * \param epid The allocated EPID * \return The physical address (device, instance) of the stream endpoint */ - sep_addr_t lookup_epid(const sep_id_t& epid) const; + sep_addr_t lookup_addr(const sep_id_t& epid) const; /*! \brief Deallocate the specified EPID. * diff --git a/host/lib/include/uhdlib/rfnoc/graph_stream_manager.hpp b/host/lib/include/uhdlib/rfnoc/graph_stream_manager.hpp index 81657453d..120b0e0f8 100644 --- a/host/lib/include/uhdlib/rfnoc/graph_stream_manager.hpp +++ b/host/lib/include/uhdlib/rfnoc/graph_stream_manager.hpp @@ -15,6 +15,7 @@ #include <functional> #include <memory> #include <set> +#include <tuple> namespace uhd { namespace rfnoc { @@ -41,25 +42,34 @@ public: */ virtual const std::set<sep_addr_t>& get_reachable_endpoints() const = 0; - /*! \brief Initialize a control endpoint to the specified destination + /*! \brief Connect the host to the specified destination and create a control stream * * \param dst_addr The physical address of the destination endpoint * \param via_device The preference for the device to take to get to the destination * \return A pair (source, destination) endpoint IDs for the control stream */ - virtual sep_id_pair_t init_ctrl_stream( - sep_addr_t dst_addr, device_id_t via_device = NULL_DEVICE_ID) = 0; + virtual sep_id_pair_t connect_host_to_device( + sep_addr_t dst_addr, device_id_t host_device = NULL_DEVICE_ID) = 0; + + /*! \brief Connect two remote endpoints to each other + * + * \param dst_addr The physical address of the destination endpoint + * \param src_addr The physical address of the source endpoint + * \return A pair (source, destination) endpoint IDs for the src/dst + */ + virtual sep_id_pair_t connect_device_to_device( + sep_addr_t dst_addr, sep_addr_t src_addr) = 0; /*! \brief Get a register iface (ctrlport endpoint) to a particular block * - * \param dst_epid The endpoint ID of the destination + * \param dst_addr The physical address of the destination endpoint * \param block_index The index of the block in the device * \param client_clk The clock that is driving the ctrlport slave * \param timebase_clk The clock that is driving the timebase * \param via_device The preference for the device to take to get to the destination * \return An interface to the ctrlport endpoint */ - virtual ctrlport_endpoint::sptr get_block_register_iface(sep_id_t dst_epid, + virtual ctrlport_endpoint::sptr get_block_register_iface(sep_addr_t dst_addr, uint16_t block_index, const clock_iface& client_clk, const clock_iface& timebase_clk, @@ -72,8 +82,36 @@ public: * \return An interface to the client zero instance */ virtual detail::client_zero::sptr get_client_zero( - sep_id_t dst_epid, device_id_t via_device = NULL_DEVICE_ID) const = 0; + sep_addr_t dst_addr, device_id_t via_device = NULL_DEVICE_ID) const = 0; + /*! Configure a flow controlled data stream from the endpoint with ID src_epid to the + * endpoint with ID dst_epid + * + * \param dst_addr The physical address of the destination endpoint + * \param src_addr The physical address of the source endpoint + * \param lossy_xport Is the transport lossy? + * \param fc_freq_ratio Flow control response frequency as a ratio of the buff params + * \param fc_headroom_ratio Flow control headroom as a ratio of the buff params + * \param reset Optionally reset the stream + */ + virtual std::tuple<sep_id_pair_t, stream_buff_params_t> + create_device_to_device_data_stream(const sep_addr_t dst_addr, + const sep_addr_t src_addr, + const bool lossy_xport, + const double fc_freq_ratio, + const double fc_headroom_ratio, + const bool reset = false) = 0; + + // TODO: Implement functions to get graph-wide streamers + + /*! + * \brief Create a graph_stream_manager and return a unique_ptr to it + * + * \param pkt_factory A factory for generating CHDR packets + * \param epid_alloc The allocator for all EPIDs in the graph + * \param links Pairs of host devices and motherboards that should be connected + * \return A unique_ptr to the newly-created graph_stream_manager + */ static uptr make(const chdr::chdr_packet_factory& pkt_factory, const epid_allocator::sptr& epid_alloc, const std::vector<std::pair<device_id_t, mb_iface*>>& links); diff --git a/host/lib/include/uhdlib/rfnoc/link_stream_manager.hpp b/host/lib/include/uhdlib/rfnoc/link_stream_manager.hpp index c31bf8987..4ff69bb3e 100644 --- a/host/lib/include/uhdlib/rfnoc/link_stream_manager.hpp +++ b/host/lib/include/uhdlib/rfnoc/link_stream_manager.hpp @@ -13,12 +13,19 @@ #include <uhdlib/rfnoc/mb_iface.hpp> #include <functional> #include <memory> +#include <set> namespace uhd { namespace rfnoc { /*! A class that is responsible managing all data endpoints, control endpoints and client - * zero instances accessible via a physical link. There must be one instance this this - * class per physical link (Ethernet cable, PCIe connection, etc) + * zero instances accessible via a logical link between the host device and + * motherboard. + * + * Note that each transport adapter on the host has its own set of streaming endpoints, + * and thus, the host's device_id_t uniquely identifies the host-side transport adapter + * to use for packet transmission/reception. + * + * There must be one instance of this class per logical link. */ class link_stream_manager { @@ -39,12 +46,30 @@ public: */ virtual const std::set<sep_addr_t>& get_reachable_endpoints() const = 0; - /*! \brief Initialize a control endpoint to the specified destination + /*! \brief Connect the host to the specified destination and init a control endpoint * * \param dst_addr The physical address of the destination endpoint * \return A pair (source, destination) endpoint IDs for the control stream */ - virtual sep_id_pair_t init_ctrl_stream(sep_addr_t dst_addr) = 0; + virtual sep_id_pair_t connect_host_to_device(sep_addr_t dst_addr) = 0; + + /*! \brief Check if the two specified endpoints can be connected remotely + * + * \param dst_addr The physical address of the destination endpoint + * \param src_addr The physical address of the source endpoint + * \return true if the endpoints can be connected + */ + virtual bool can_connect_device_to_device( + sep_addr_t dst_addr, sep_addr_t src_addr) const = 0; + + /*! \brief Connect two remote endpoints to each other + * + * \param dst_addr The physical address of the destination endpoint + * \param src_addr The physical address of the source endpoint + * \return A pair (source, destination) endpoint IDs for the src/dst + */ + virtual sep_id_pair_t connect_device_to_device( + sep_addr_t dst_addr, sep_addr_t src_addr) = 0; /*! \brief Get a register iface (ctrlport endpoint) to a particular block * @@ -66,15 +91,61 @@ public: */ virtual detail::client_zero::sptr get_client_zero(sep_id_t dst_epid) const = 0; - /*! \brief Create a data stream + /*! Configure a flow controlled data stream from the endpoint with ID src_epid to the + * endpoint with ID dst_epid * * \param dst_epid The endpoint ID of the destination - * \param vc The virtual channel - * \param xport_args The transport argument + * \param src_epid The endpoint ID of the source + * \param lossy_xport Is the transport lossy? + * \param fc_freq_ratio Flow control response frequency as a ratio of the buff params + * \param fc_headroom_ratio Flow control headroom as a ratio of the buff params + * \param reset Optionally reset the stream + */ + virtual stream_buff_params_t create_device_to_device_data_stream( + const sep_id_t& dst_epid, + const sep_id_t& src_epid, + const bool lossy_xport, + const double fc_freq_ratio, + const double fc_headroom_ratio, + const bool reset = false) = 0; + + /*! \brief Create a data stream going from the host to the device + * + * \param dst_addr The address of the destination stream endpoint + * \param lossy_xport Is the transport lossy? + * \param pyld_buff_fmt Datatype of SW buffer that holds the data payload + * \param mdata_buff_fmt Datatype of SW buffer that holds the data metadata + * \param fc_freq_ratio Flow control response frequency as a ratio of the buff params + * \param fc_headroom_ratio Flow control headroom as a ratio of the buff params + * \param xport_args The transport arguments + * \return An transport instance + */ + virtual chdr_data_xport_t create_host_to_device_data_stream(const sep_addr_t dst_addr, + const bool lossy_xport, + const sw_buff_t pyld_buff_fmt, + const sw_buff_t mdata_buff_fmt, + const double fc_freq_ratio, + const double fc_headroom_ratio, + const device_addr_t& xport_args) = 0; + + /*! \brief Create a data stream going from the device to the host + * + * \param dst_addr The address of the destination stream endpoint + * \param lossy_xport Is the transport lossy? + * \param pyld_buff_fmt Datatype of SW buffer that holds the data payload + * \param mdata_buff_fmt Datatype of SW buffer that holds the data metadata + * \param fc_freq_ratio Flow control response frequency as a ratio of the buff params + * \param fc_headroom_ratio Flow control headroom as a ratio of the buff params + * \param xport_args The transport arguments * \return An transport instance */ - virtual chdr_data_xport_t create_data_stream( - sep_addr_t dst_addr, sep_vc_t vc, const device_addr_t& xport_args) = 0; + virtual chdr_data_xport_t create_device_to_host_data_stream(const sep_addr_t src_addr, + const bool lossy_xport, + const sw_buff_t pyld_buff_fmt, + const sw_buff_t mdata_buff_fmt, + const double fc_freq_ratio, + const double fc_headroom_ratio, + const device_addr_t& xport_args) = 0; static uptr make(const chdr::chdr_packet_factory& pkt_factory, mb_iface& mb_if, diff --git a/host/lib/include/uhdlib/rfnoc/mgmt_portal.hpp b/host/lib/include/uhdlib/rfnoc/mgmt_portal.hpp index 12f40f172..1412d0e3d 100644 --- a/host/lib/include/uhdlib/rfnoc/mgmt_portal.hpp +++ b/host/lib/include/uhdlib/rfnoc/mgmt_portal.hpp @@ -9,6 +9,7 @@ #include <uhdlib/rfnoc/chdr_types.hpp> #include <memory> +#include <set> namespace uhd { namespace rfnoc { namespace mgmt { @@ -26,13 +27,6 @@ public: using xport_cfg_fn_t = std::function<void( device_id_t devid, uint16_t inst, uint8_t subtype, chdr::mgmt_hop_t& hop)>; - //! Flow control buffer configuration parameters - struct buff_params_t - { - uint64_t bytes; - uint32_t packets; - }; - //! Information about a stream endpoint struct sep_info_t { @@ -50,9 +44,6 @@ public: sep_addr_t addr; }; - //! The data type of the buffer used to capture/generate data - enum sw_buff_t { BUFF_U64 = 0, BUFF_U32 = 1, BUFF_U16 = 2, BUFF_U8 = 3 }; - virtual ~mgmt_portal() = 0; //! Get addresses for all stream endpoints reachable from this SW mgmt portal @@ -90,6 +81,14 @@ public: // virtual void setup_local_route(const sep_id_t& dst_epid) = 0; + //! Can a route from between the source and destination endpoints be established? + // + // \param dst_epid The endpoint ID of the destination + // \param src_epid The endpoint ID of the source + // + virtual bool can_remote_route( + const sep_addr_t& dst_addr, const sep_addr_t& src_addr) const = 0; + //! Setup a route from between the source and destination endpoints // // After a route is established, it should be possible for the source to send packets @@ -121,8 +120,8 @@ public: const bool lossy_xport, const sw_buff_t pyld_buff_fmt, const sw_buff_t mdata_buff_fmt, - const buff_params_t& fc_freq, - const buff_params_t& fc_headroom, + const stream_buff_params_t& fc_freq, + const stream_buff_params_t& fc_headroom, const bool reset = false) = 0; //! Finish configuring a flow controlled receive data stream from the endpoint with @@ -130,7 +129,7 @@ public: // // \param epid The endpoint ID of the data source // - virtual buff_params_t config_local_rx_stream_commit( + virtual stream_buff_params_t config_local_rx_stream_commit( const sep_id_t& epid, const double timeout = 0.2) = 0; //! Configure a flow controlled transmit data stream from this SW mgmt portal to the @@ -156,11 +155,11 @@ public: // \param fc_freq Flow control response frequency parameters // \param fc_freq Flow control headroom parameters // - virtual buff_params_t config_remote_stream(const sep_id_t& dst_epid, + virtual stream_buff_params_t config_remote_stream(const sep_id_t& dst_epid, const sep_id_t& src_epid, const bool lossy_xport, - const buff_params_t& fc_freq, - const buff_params_t& fc_headroom, + const stream_buff_params_t& fc_freq, + const stream_buff_params_t& fc_headroom, const bool reset = false, const double timeout = 0.2) = 0; diff --git a/host/lib/include/uhdlib/rfnoc/rfnoc_common.hpp b/host/lib/include/uhdlib/rfnoc/rfnoc_common.hpp index bac510f04..c08c8d74a 100644 --- a/host/lib/include/uhdlib/rfnoc/rfnoc_common.hpp +++ b/host/lib/include/uhdlib/rfnoc/rfnoc_common.hpp @@ -56,16 +56,26 @@ static constexpr sep_addr_t NULL_DEVICE_ADDR{NULL_DEVICE_ID, 0}; static constexpr sep_id_t NULL_EPID = 0; +//! Flow control buffer configuration parameters +struct stream_buff_params_t +{ + uint64_t bytes; + uint32_t packets; +}; + +//! The data type of the buffer used to capture/generate data +enum sw_buff_t { BUFF_U64 = 0, BUFF_U32 = 1, BUFF_U16 = 2, BUFF_U8 = 3 }; + // TODO: Update these struct chdr_ctrl_xport_t { chdr_ctrl_xport_t() = default; uhd::transport::zero_copy_if::sptr recv; uhd::transport::zero_copy_if::sptr send; - size_t recv_buff_size = 0; - size_t send_buff_size = 0; - sep_id_t src_epid = 0; - sep_id_t dst_epid = 0; + stream_buff_params_t recv_buff_params{0, 0}; + stream_buff_params_t send_buff_params{0, 0}; + sep_id_t src_epid{0}; + sep_id_t dst_epid{0}; }; using chdr_data_xport_t = chdr_ctrl_xport_t; diff --git a/host/lib/rfnoc/epid_allocator.cpp b/host/lib/rfnoc/epid_allocator.cpp index 984b1716e..97a30dc64 100644 --- a/host/lib/rfnoc/epid_allocator.cpp +++ b/host/lib/rfnoc/epid_allocator.cpp @@ -27,7 +27,19 @@ sep_id_t epid_allocator::allocate_epid(const sep_addr_t& addr) } } -sep_addr_t epid_allocator::lookup_epid(const sep_id_t& epid) const +sep_id_t epid_allocator::get_epid(const sep_addr_t& addr) +{ + std::lock_guard<std::mutex> lock(_mutex); + + if (_epid_map.count(addr) != 0) { + return _epid_map.at(addr); + } else { + throw uhd::lookup_error( + "An EPID has not been allocated for the requested endpoint"); + } +} + +sep_addr_t epid_allocator::lookup_addr(const sep_id_t& epid) const { std::lock_guard<std::mutex> lock(_mutex); diff --git a/host/lib/rfnoc/graph_stream_manager.cpp b/host/lib/rfnoc/graph_stream_manager.cpp index 06f78dad4..f2024786a 100644 --- a/host/lib/rfnoc/graph_stream_manager.cpp +++ b/host/lib/rfnoc/graph_stream_manager.cpp @@ -61,32 +61,120 @@ public: return retval; } - virtual sep_id_pair_t init_ctrl_stream( - sep_addr_t dst_addr, device_id_t via_device = NULL_DEVICE_ID) + virtual sep_id_pair_t connect_host_to_device( + sep_addr_t dst_addr, device_id_t host_device = NULL_DEVICE_ID) { - return _link_mgrs.at(_check_dst_and_find_src(dst_addr, via_device)) - ->init_ctrl_stream(dst_addr); + UHD_LOGGER_DEBUG("RFNOC::GRAPH") + << boost::format("Connecting the Host to Endpoint %d:%d through Device " + "%d (0 = no preference)... ") + % dst_addr.first % dst_addr.second % host_device; + + // When we connect, we setup a route and fire up a control stream between + // the endpoints + device_id_t gateway = _check_dst_and_find_src(dst_addr, host_device); + sep_id_pair_t epid_pair = + _link_mgrs.at(gateway)->connect_host_to_device(dst_addr); + UHD_LOGGER_DEBUG("RFNOC::GRAPH") + << boost::format("Connection to Endpoint %d:%d completed through Device %d. " + "Using EPIDs %d -> %d.") + % dst_addr.first % dst_addr.second % gateway % epid_pair.first + % epid_pair.second; + + return epid_pair; } - virtual ctrlport_endpoint::sptr get_block_register_iface(sep_id_t dst_epid, + virtual sep_id_pair_t connect_device_to_device( + sep_addr_t dst_addr, sep_addr_t src_addr) + { + UHD_LOGGER_DEBUG("RFNOC::GRAPH") + << boost::format("Connecting the Endpoint %d:%d to Endpoint %d:%d...") + % src_addr.first % src_addr.second % dst_addr.first % dst_addr.second; + + // Iterate through all link managers and check if they are capable of connecting + // the requested endpoints. If no one can connect then the endpoints may actually + // not share a common crossbar or we don't have enough connectivity in the + // software session to reach the common crossbar. + for (auto& kv : _link_mgrs) { + if (kv.second->can_connect_device_to_device(dst_addr, src_addr)) { + sep_id_pair_t epid_pair = + kv.second->connect_device_to_device(dst_addr, src_addr); + UHD_LOGGER_DEBUG("RFNOC::GRAPH") + << boost::format("Connection from Endpoint %d:%d to Endpoint %d:%d " + "completed through Device %d. Using " + "EPIDs %d -> %d.") + % src_addr.first % src_addr.second % dst_addr.first + % dst_addr.second % kv.first % epid_pair.first + % epid_pair.second; + return epid_pair; + } + } + throw uhd::routing_error("The specified destination is unreachable from the " + "specified source endpoint"); + } + + virtual ctrlport_endpoint::sptr get_block_register_iface(sep_addr_t dst_addr, uint16_t block_index, const clock_iface& client_clk, const clock_iface& timebase_clk, device_id_t via_device = NULL_DEVICE_ID) { - sep_addr_t dst_addr = _epid_alloc->lookup_epid(dst_epid); + // We must be connected to dst_addr before getting a register iface + sep_id_t dst_epid = _epid_alloc->get_epid(dst_addr); return _link_mgrs.at(_check_dst_and_find_src(dst_addr, via_device)) ->get_block_register_iface(dst_epid, block_index, client_clk, timebase_clk); } virtual detail::client_zero::sptr get_client_zero( - sep_id_t dst_epid, device_id_t via_device = NULL_DEVICE_ID) const + sep_addr_t dst_addr, device_id_t via_device = NULL_DEVICE_ID) const { - sep_addr_t dst_addr = _epid_alloc->lookup_epid(dst_epid); + // We must be connected to dst_addr before getting a client zero + sep_id_t dst_epid = _epid_alloc->get_epid(dst_addr); return _link_mgrs.at(_check_dst_and_find_src(dst_addr, via_device)) ->get_client_zero(dst_epid); } + virtual std::tuple<sep_id_pair_t, stream_buff_params_t> + create_device_to_device_data_stream(const sep_addr_t dst_addr, + const sep_addr_t src_addr, + const bool lossy_xport, + const double fc_freq_ratio, + const double fc_headroom_ratio, + const bool reset = false) + { + UHD_LOGGER_DEBUG("RFNOC::GRAPH") + << boost::format( + "Initializing data stream from Endpoint %d:%d to Endpoint %d:%d...") + % src_addr.first % src_addr.second % dst_addr.first % dst_addr.second; + + // Iterate through all link managers and check if they are capable of connecting + // the requested endpoints. If no one can connect then the endpoints may actually + // not share a common crossbar or we don't have enough connectivity in the + // software session to reach the common crossbar. + for (auto& kv : _link_mgrs) { + if (kv.second->can_connect_device_to_device(dst_addr, src_addr)) { + sep_id_pair_t epid_pair = + kv.second->connect_device_to_device(dst_addr, src_addr); + UHD_LOGGER_DEBUG("RFNOC::GRAPH") + << boost::format("Connection from Endpoint %d:%d to Endpoint %d:%d " + "completed through Device %d. Using " + "EPIDs %d -> %d.") + % src_addr.first % src_addr.second % dst_addr.first + % dst_addr.second % kv.first % epid_pair.first + % epid_pair.second; + stream_buff_params_t buff_params = + kv.second->create_device_to_device_data_stream(epid_pair.second, + epid_pair.first, + lossy_xport, + fc_freq_ratio, + fc_headroom_ratio, + reset); + return std::make_tuple(epid_pair, buff_params); + } + } + throw uhd::routing_error("The specified destination is unreachable from the " + "specified source endpoint"); + } + private: device_id_t _check_dst_and_find_src(sep_addr_t dst_addr, device_id_t via_device) const { diff --git a/host/lib/rfnoc/link_stream_manager.cpp b/host/lib/rfnoc/link_stream_manager.cpp index 0faec94e3..bd330e313 100644 --- a/host/lib/rfnoc/link_stream_manager.cpp +++ b/host/lib/rfnoc/link_stream_manager.cpp @@ -6,7 +6,6 @@ #include <uhd/exception.hpp> #include <uhd/transport/muxed_zero_copy_if.hpp> -#include <uhd/utils/log.hpp> #include <uhdlib/rfnoc/chdr_ctrl_endpoint.hpp> #include <uhdlib/rfnoc/link_stream_manager.hpp> #include <uhdlib/rfnoc/mgmt_portal.hpp> @@ -22,6 +21,8 @@ using namespace uhd::rfnoc::detail; constexpr sep_inst_t SEP_INST_MGMT_CTRL = 0; constexpr sep_inst_t SEP_INST_DATA_BASE = 1; +constexpr double STREAM_SETUP_TIMEOUT = 0.2; + link_stream_manager::~link_stream_manager() = default; class link_stream_manager_impl : public link_stream_manager @@ -109,15 +110,18 @@ public: return _mgmt_portal->get_reachable_endpoints(); } - virtual sep_id_pair_t init_ctrl_stream(sep_addr_t dst_addr) + virtual bool can_connect_device_to_device( + sep_addr_t dst_addr, sep_addr_t src_addr) const + { + return _mgmt_portal->can_remote_route(dst_addr, src_addr); + } + + virtual sep_id_pair_t connect_host_to_device(sep_addr_t dst_addr) { + _ensure_ep_is_reachable(dst_addr); + // Allocate EPIDs sep_id_t dst_epid = _epid_alloc->allocate_epid(dst_addr); - UHD_LOGGER_DEBUG("RFNOC::LINK_MGR") - << boost::format("Initializing control stream to Endpoint %d:%d with EPID " - "%d...") - % dst_addr.first % dst_addr.second % dst_epid; - _ensure_ep_is_reachable(dst_addr); // Make sure that the software side of the endpoint is initialized and reachable if (_ctrl_ep == nullptr) { @@ -142,6 +146,25 @@ public: return sep_id_pair_t(_my_mgmt_ctrl_epid, dst_epid); } + virtual sep_id_pair_t connect_device_to_device( + sep_addr_t dst_addr, sep_addr_t src_addr) + { + _ensure_ep_is_reachable(dst_addr); + _ensure_ep_is_reachable(src_addr); + + // Allocate EPIDs + sep_id_t dst_epid = _epid_alloc->allocate_epid(dst_addr); + sep_id_t src_epid = _epid_alloc->allocate_epid(src_addr); + + // Initialize endpoints + _mgmt_portal->initialize_endpoint(dst_addr, dst_epid); + _mgmt_portal->initialize_endpoint(src_addr, src_epid); + + _mgmt_portal->setup_remote_route(dst_epid, src_epid); + + return sep_id_pair_t(src_epid, dst_epid); + } + virtual ctrlport_endpoint::sptr get_block_register_iface(sep_id_t dst_epid, uint16_t block_index, const clock_iface& client_clk, @@ -179,8 +202,54 @@ public: return _client_zero_map.at(dst_epid); } - virtual chdr_data_xport_t create_data_stream( - sep_addr_t dst_addr, sep_vc_t /*vc*/, const device_addr_t& xport_args) + virtual stream_buff_params_t create_device_to_device_data_stream( + const sep_id_t& dst_epid, + const sep_id_t& src_epid, + const bool lossy_xport, + const double fc_freq_ratio, + const double fc_headroom_ratio, + const bool reset = false) + { + // We assume that the devices are already connected (because this API requires + // EPIDs) + + // Setup a stream + stream_buff_params_t buff_params = _mgmt_portal->config_remote_stream(dst_epid, + src_epid, + lossy_xport, + stream_buff_params_t{1, 1}, // Dummy frequency + stream_buff_params_t{0, 0}, // Dummy headroom + false, + STREAM_SETUP_TIMEOUT); + + // Compute FC frequency and headroom based on buff parameters + stream_buff_params_t fc_freq{ + static_cast<uint64_t>(std::ceil(double(buff_params.bytes) * fc_freq_ratio)), + static_cast<uint32_t>( + std::ceil(double(buff_params.packets) * fc_freq_ratio))}; + stream_buff_params_t fc_headroom{ + static_cast<uint64_t>( + std::ceil(double(buff_params.bytes) * fc_headroom_ratio)), + static_cast<uint32_t>( + std::ceil(double(buff_params.packets) * fc_headroom_ratio))}; + + // Reconfigure flow control using the new frequency and headroom + return _mgmt_portal->config_remote_stream(dst_epid, + src_epid, + lossy_xport, + fc_freq, + fc_headroom, + reset, + STREAM_SETUP_TIMEOUT); + } + + virtual chdr_data_xport_t create_host_to_device_data_stream(const sep_addr_t dst_addr, + const bool lossy_xport, + const sw_buff_t pyld_buff_fmt, + const sw_buff_t mdata_buff_fmt, + const double fc_freq_ratio, + const double fc_headroom_ratio, + const device_addr_t& xport_args) { // Create a new source endpoint and EPID sep_addr_t sw_epid_addr(_my_device_id, SEP_INST_DATA_BASE + (_data_ep_inst++)); @@ -219,6 +288,17 @@ public: return xport; } + virtual chdr_data_xport_t create_device_to_host_data_stream(const sep_addr_t src_addr, + const bool lossy_xport, + const sw_buff_t pyld_buff_fmt, + const sw_buff_t mdata_buff_fmt, + const double fc_freq_ratio, + const double fc_headroom_ratio, + const device_addr_t& xport_args) + { + // TODO: Implement me + } + private: void _ensure_ep_is_reachable(const sep_addr_t& ep_addr_) { diff --git a/host/lib/rfnoc/mgmt_portal.cpp b/host/lib/rfnoc/mgmt_portal.cpp index deb9f9166..ba54f83d0 100644 --- a/host/lib/rfnoc/mgmt_portal.cpp +++ b/host/lib/rfnoc/mgmt_portal.cpp @@ -43,10 +43,8 @@ constexpr uint32_t RESET_AND_FLUSH_ISTRM = (1 << 1); constexpr uint32_t RESET_AND_FLUSH_CTRL = (1 << 2); constexpr uint32_t RESET_AND_FLUSH_ALL = 0x7; -constexpr uint32_t BUILD_CTRL_STATUS_WORD(bool cfg_start, - bool xport_lossy, - mgmt_portal::sw_buff_t pyld_buff_fmt, - mgmt_portal::sw_buff_t mdata_buff_fmt) +constexpr uint32_t BUILD_CTRL_STATUS_WORD( + bool cfg_start, bool xport_lossy, sw_buff_t pyld_buff_fmt, sw_buff_t mdata_buff_fmt) { return (cfg_start ? 1 : 0) | (xport_lossy ? 2 : 0) | (static_cast<uint32_t>(pyld_buff_fmt) << 2) @@ -346,29 +344,52 @@ public: % dst_epid % to_string(node_addr))); } - virtual void setup_remote_route(const sep_id_t& dst_epid, const sep_id_t& src_epid) + virtual bool can_remote_route( + const sep_addr_t& dst_addr, const sep_addr_t& src_addr) const { std::lock_guard<std::recursive_mutex> lock(_mutex); + if ((_discovered_ep_set.count(dst_addr) == 0) + || (_discovered_ep_set.count(src_addr) == 0)) { + // Can't route to/from something if we don't know about it + return false; + } + + UHD_ASSERT_THROW(_node_addr_map.count(node_id_t(dst_addr)) > 0); + UHD_ASSERT_THROW(_node_addr_map.count(node_id_t(src_addr)) > 0); + // Lookup the src and dst node address using the endpoint ID - const node_addr_t& dst_node_addr = _lookup_sep_node_addr(dst_epid); - const node_addr_t& src_node_addr = _lookup_sep_node_addr(src_epid); + const node_addr_t& dst_node_addr = _node_addr_map.at(node_id_t(dst_addr)); + const node_addr_t& src_node_addr = _node_addr_map.at(node_id_t(src_addr)); // Find a common parent (could be faster than n^2 but meh, this is easier) - bool found_common_parent = false; for (const auto& dnode : dst_node_addr) { for (const auto& snode : src_node_addr) { - found_common_parent = - ((dnode.first == snode.first) && dnode.first.type == NODE_TYPE_XBAR); - if (found_common_parent) - break; + if (dnode.first == snode.first && dnode.first.type == NODE_TYPE_XBAR) { + return true; + } } - if (found_common_parent) - break; } - if (!found_common_parent) { - throw uhd::routing_error("setup_remote_route: Route setup failed. The " - "endpoints don't share a common crossbar parent."); + return false; + } + + virtual void setup_remote_route(const sep_id_t& dst_epid, const sep_id_t& src_epid) + { + std::lock_guard<std::recursive_mutex> lock(_mutex); + + if (not is_endpoint_initialized(dst_epid)) { + throw uhd::routing_error("Route setup failed. The destination endpoint was " + "not initialized and bound to an EPID"); + } + if (not is_endpoint_initialized(src_epid)) { + throw uhd::routing_error("Route setup failed. The source endpoint was " + "not initialized and bound to an EPID"); + } + + if (not can_remote_route( + _epid_addr_map.at(dst_epid), _epid_addr_map.at(src_epid))) { + throw uhd::routing_error("Route setup failed. The endpoints don't share a " + "common crossbar parent."); } // If we setup local routes from this host to both the source and destination @@ -393,8 +414,8 @@ public: const bool lossy_xport, const sw_buff_t pyld_buff_fmt, const sw_buff_t mdata_buff_fmt, - const buff_params_t& fc_freq, - const buff_params_t& fc_headroom, + const stream_buff_params_t& fc_freq, + const stream_buff_params_t& fc_headroom, const bool reset = false) { std::lock_guard<std::recursive_mutex> lock(_mutex); @@ -433,7 +454,7 @@ public: (boost::format("Initiated RX stream setup for EPID=%d") % epid)); } - virtual buff_params_t config_local_rx_stream_commit( + virtual stream_buff_params_t config_local_rx_stream_commit( const sep_id_t& epid, const double timeout = 0.2) { std::lock_guard<std::recursive_mutex> lock(_mutex); @@ -487,11 +508,11 @@ public: (boost::format("Finished TX stream setup for EPID=%d") % epid)); } - virtual buff_params_t config_remote_stream(const sep_id_t& dst_epid, + virtual stream_buff_params_t config_remote_stream(const sep_id_t& dst_epid, const sep_id_t& src_epid, const bool lossy_xport, - const buff_params_t& fc_freq, - const buff_params_t& fc_headroom, + const stream_buff_params_t& fc_freq, + const stream_buff_params_t& fc_headroom, const bool reset = false, const double timeout = 0.2) { @@ -743,8 +764,8 @@ private: // Functions void _push_ostrm_flow_control_config(const bool lossy_xport, const sw_buff_t pyld_buff_fmt, const sw_buff_t mdata_buff_fmt, - const buff_params_t& fc_freq, - const buff_params_t& fc_headroom, + const stream_buff_params_t& fc_freq, + const stream_buff_params_t& fc_headroom, mgmt_hop_t& hop) { // Validate flow control parameters @@ -780,7 +801,8 @@ private: // Functions } // Send/recv a management transaction that will get the output stream status - std::tuple<uint32_t, buff_params_t> _get_ostrm_status(const node_addr_t& node_addr) + std::tuple<uint32_t, stream_buff_params_t> _get_ostrm_status( + const node_addr_t& node_addr) { // Build a management transaction to first get to the node mgmt_payload status_xact; @@ -824,7 +846,7 @@ private: // Functions mgmt_op_t::cfg_payload cap_bytes_hi = rhop.get_op(3).get_op_payload(); mgmt_op_t::cfg_payload cap_pkts = rhop.get_op(4).get_op_payload(); - buff_params_t buff_params; + stream_buff_params_t buff_params; buff_params.bytes = static_cast<uint64_t>(cap_bytes_lo.data) | (static_cast<uint64_t>(cap_bytes_hi.data) << 32); buff_params.packets = static_cast<uint32_t>(cap_pkts.data); |