// // Copyright 2019 Ettus Research, a National Instruments Brand // // SPDX-License-Identifier: GPL-3.0-or-later // #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include // FIXME remove when rfnoc_device is ready #include using namespace uhd::rfnoc; namespace { const std::string LOG_ID("RFNOC::GRAPH"); } class rfnoc_graph_impl : public rfnoc_graph { public: /************************************************************************** * Structors *************************************************************************/ rfnoc_graph_impl(const uhd::device_addr_t& dev_addr) : _block_registry(std::make_unique()) , _graph(std::make_unique()) { setup_graph(dev_addr); _init_sep_map(); _init_static_connections(); } ~rfnoc_graph_impl() { _graph.reset(); _block_registry->shutdown(); } /************************************************************************** * Block Discovery/Retrieval *************************************************************************/ std::vector find_blocks(const std::string& block_id_hint) const { return _block_registry->find_blocks(block_id_hint); } bool has_block(const block_id_t& block_id) const { return _block_registry->has_block(block_id); } noc_block_base::sptr get_block(const block_id_t& block_id) const { return _block_registry->get_block(block_id); } /************************************************************************** * Graph Connections *************************************************************************/ void connect(const block_id_t& src_blk, size_t src_port, const block_id_t& dst_blk, size_t dst_port, bool skip_property_propagation) { if (!has_block(src_blk)) { throw uhd::lookup_error( std::string("Cannot connect blocks, source block not found: ") + src_blk.to_string()); } if (!has_block(dst_blk)) { throw uhd::lookup_error( std::string("Cannot connect blocks, destination block not found: ") + dst_blk.to_string()); } auto edge_type = _physical_connect(src_blk, src_port, dst_blk, dst_port); _connect(get_block(src_blk), src_port, get_block(dst_blk), dst_port, edge_type, skip_property_propagation); } void connect(uhd::tx_streamer::sptr streamer, size_t strm_port, const block_id_t& dst_blk, size_t dst_port) { // Verify the streamer was created by us auto rfnoc_streamer = boost::dynamic_pointer_cast(streamer); if (!rfnoc_streamer) { throw uhd::type_error("Streamer is not rfnoc capable"); } // Verify src_blk even exists in this graph if (!has_block(dst_blk)) { throw uhd::lookup_error( std::string("Cannot connect block to streamer, source block not found: ") + dst_blk.to_string()); } // Verify src_blk has an SEP upstream graph_edge_t dst_static_edge = _assert_edge( _get_static_edge( [dst_blk_id = dst_blk.to_string(), dst_port](const graph_edge_t& edge) { return edge.dst_blockid == dst_blk_id && edge.dst_port == dst_port; }), dst_blk.to_string()); if (block_id_t(dst_static_edge.src_blockid).get_block_name() != NODE_ID_SEP) { const std::string err_msg = dst_blk.to_string() + ":" + std::to_string(dst_port) + " is not connected to an SEP! Routing impossible."; UHD_LOG_ERROR(LOG_ID, err_msg); throw uhd::routing_error(err_msg); } // Now get the name and address of the SEP const std::string sep_block_id = dst_static_edge.src_blockid; const sep_addr_t sep_addr = _sep_map.at(sep_block_id); const sw_buff_t pyld_fmt = bits_to_sw_buff(rfnoc_streamer->get_otw_item_comp_bit_width()); const sw_buff_t mdata_fmt = BUFF_U64; auto xport = _gsm->create_host_to_device_data_stream(sep_addr, pyld_fmt, mdata_fmt, rfnoc_streamer->get_stream_args().args); rfnoc_streamer->connect_channel(strm_port, std::move(xport)); //// If this worked, then also connect the streamer in the BGL graph auto dst = get_block(dst_blk); graph_edge_t edge_info(strm_port, dst_port, graph_edge_t::TX_STREAM, true); _graph->connect(rfnoc_streamer.get(), dst.get(), edge_info); } void connect(const block_id_t& src_blk, size_t src_port, uhd::rx_streamer::sptr streamer, size_t strm_port) { // Verify the streamer was created by us auto rfnoc_streamer = boost::dynamic_pointer_cast(streamer); if (!rfnoc_streamer) { throw uhd::type_error("Streamer is not rfnoc capable"); } // Verify src_blk even exists in this graph if (!has_block(src_blk)) { throw uhd::lookup_error( std::string("Cannot connect block to streamer, source block not found: ") + src_blk.to_string()); } // Verify src_blk has an SEP downstream graph_edge_t src_static_edge = _assert_edge( _get_static_edge( [src_blk_id = src_blk.to_string(), src_port](const graph_edge_t& edge) { return edge.src_blockid == src_blk_id && edge.src_port == src_port; }), src_blk.to_string()); if (block_id_t(src_static_edge.dst_blockid).get_block_name() != NODE_ID_SEP) { const std::string err_msg = src_blk.to_string() + ":" + std::to_string(src_port) + " is not connected to an SEP! Routing impossible."; UHD_LOG_ERROR(LOG_ID, err_msg); throw uhd::routing_error(err_msg); } // Now get the name and address of the SEP const std::string sep_block_id = src_static_edge.dst_blockid; const sep_addr_t sep_addr = _sep_map.at(sep_block_id); const sw_buff_t pyld_fmt = bits_to_sw_buff(rfnoc_streamer->get_otw_item_comp_bit_width()); const sw_buff_t mdata_fmt = BUFF_U64; auto xport = _gsm->create_device_to_host_data_stream(sep_addr, pyld_fmt, mdata_fmt, rfnoc_streamer->get_stream_args().args); rfnoc_streamer->connect_channel(strm_port, std::move(xport)); // If this worked, then also connect the streamer in the BGL graph auto src = get_block(src_blk); graph_edge_t edge_info(src_port, strm_port, graph_edge_t::RX_STREAM, true); _graph->connect(src.get(), rfnoc_streamer.get(), edge_info); } uhd::rx_streamer::sptr create_rx_streamer( const size_t num_chans, const uhd::stream_args_t& args) { return boost::make_shared(num_chans, args); } uhd::tx_streamer::sptr create_tx_streamer( const size_t num_chans, const uhd::stream_args_t& args) { return boost::make_shared(num_chans, args); } std::shared_ptr get_mb_controller(const size_t mb_index = 0) { if (!_mb_controllers.count(mb_index)) { throw uhd::index_error( std::string("Could not get mb controller for motherboard index ") + std::to_string(mb_index)); } return _mb_controllers.at(mb_index); } size_t get_num_mboards() const { return _num_mboards; } std::vector enumerate_active_connections() { return _graph->enumerate_edges(); } std::vector enumerate_static_connections() const { return _static_edges; } void commit() { _graph->commit(); } void release() { _graph->release(); } private: /************************************************************************** * Device Setup *************************************************************************/ void setup_graph(const uhd::device_addr_t& dev_addr) { // Phase I: Initialize the motherboards auto dev = uhd::device::make(dev_addr); _device = boost::dynamic_pointer_cast(dev); if (!_device) { throw uhd::key_error(std::string("Found no RFNoC devices for ----->\n") + dev_addr.to_pp_string()); } _tree = _device->get_tree(); _num_mboards = _tree->list("/mboards").size(); for (size_t i = 0; i < _num_mboards; ++i) { _mb_controllers.emplace(i, _device->get_mb_controller(i)); } // Create a graph stream manager // FIXME get these from mb_iface or something static const chdr::chdr_packet_factory pkt_factory( CHDR_W_64, uhd::ENDIANNESS_BIG); epid_allocator::sptr epid_alloc = std::make_shared(); // Create a collection of link definitions: (ID, MB) pairs std::vector> links; // TODO fix device_id we're creating links.push_back(std::make_pair(100, &_device->get_mb_iface(0))); try { _gsm = graph_stream_manager::make(pkt_factory, epid_alloc, links); } catch (uhd::io_error& ex) { UHD_LOG_ERROR( "RFNOC::GRAPH", "IO Error during GSM initialization. " << ex.what()); throw; } // Configure endpoint_manager, make sure all routes are established // FIXME // Enumerate blocks, load them into the block registry // Iterate through the mboards for (size_t mb_idx = 0; mb_idx < get_num_mboards(); ++mb_idx) { // Setup the interfaces for this mboard and get some configuration info mb_iface& mb = _device->get_mb_iface(mb_idx); // Ask GSM to allow us to talk to our remote mb sep_addr_t ctrl_sep_addr(mb.get_remote_device_id(), 0); _gsm->connect_host_to_device(ctrl_sep_addr); // Grab and stash the Client Zero for this mboard detail::client_zero::sptr mb_cz = _gsm->get_client_zero(ctrl_sep_addr); // Client zero port numbers are based on the control xbar numbers, // which have the client 0 interface first, followed by stream // endpoints, and then the blocks. _client_zeros.emplace(mb_idx, mb_cz); const size_t num_blocks = mb_cz->get_num_blocks(); const size_t first_block_port = 1 + mb_cz->get_num_stream_endpoints(); /* Flush and reset each block in the mboard * We do this before we enumerate the blocks to ensure they're in a clean * state before we construct their block controller, and so that we don't * reset any setting that the block controller writes */ _flush_and_reset_mboard(mb_idx, mb_cz, num_blocks, first_block_port); // Make a map to count the number of each block we have std::unordered_map block_count_map; // Iterate through and register each of the blocks in this mboard for (size_t portno = 0; portno < num_blocks; ++portno) { auto noc_id = mb_cz->get_noc_id(portno + first_block_port); auto block_factory_info = factory::get_block_factory(noc_id); auto block_info = mb_cz->get_block_info(portno + first_block_port); block_id_t block_id(mb_idx, block_factory_info.block_name, block_count_map[block_factory_info.block_name]++); // Get access to the clock interface objects. We have some rules // here: // - The ctrlport clock must always be provided through the // BSP via mb_iface // - The timebase clock can be set to "graph", which means the // block takes care of the timebase itself (via property // propagation). In that case, we generate a clock iface // object on the fly here. // - In all other cases, the BSP must provide us that clock // iface object through the mb_iface auto ctrlport_clk_iface = mb.get_clock_iface(block_factory_info.ctrlport_clk); auto tb_clk_iface = (block_factory_info.timebase_clk == CLOCK_KEY_GRAPH) ? std::make_shared(CLOCK_KEY_GRAPH) : mb.get_clock_iface(block_factory_info.timebase_clk); // A "graph" clock is always "running" if (block_factory_info.timebase_clk == CLOCK_KEY_GRAPH) { tb_clk_iface->set_running(true); } auto block_reg_iface = _gsm->get_block_register_iface(ctrl_sep_addr, portno, *ctrlport_clk_iface.get(), *tb_clk_iface.get()); auto make_args_uptr = std::make_unique(); make_args_uptr->noc_id = noc_id; make_args_uptr->block_id = block_id; make_args_uptr->num_input_ports = block_info.num_inputs; make_args_uptr->num_output_ports = block_info.num_outputs; make_args_uptr->reg_iface = block_reg_iface; make_args_uptr->tb_clk_iface = tb_clk_iface; make_args_uptr->ctrlport_clk_iface = ctrlport_clk_iface; make_args_uptr->mb_control = (factory::has_requested_mb_access(noc_id) ? _mb_controllers.at(mb_idx) : nullptr); const uhd::fs_path block_path( uhd::fs_path("/blocks") / block_id.to_string()); _tree->create(block_path / "noc_id").set(noc_id); make_args_uptr->tree = _tree->subtree(block_path); make_args_uptr->args = dev_addr; // TODO filter the device args _block_registry->register_block( block_factory_info.factory_fn(std::move(make_args_uptr))); _xbar_block_config[block_id.to_string()] = { portno, noc_id, block_id.get_block_count()}; _port_block_map.insert({{mb_idx, portno + first_block_port}, block_id}); } } UHD_LOG_TRACE("RFNOC::GRAPH", "Initializing properties on all blocks..."); _block_registry->init_props(); } void _init_sep_map() { for (size_t mb_idx = 0; mb_idx < get_num_mboards(); ++mb_idx) { auto remote_device_id = _device->get_mb_iface(mb_idx).get_remote_device_id(); auto& cz = _client_zeros.at(mb_idx); for (size_t sep_idx = 0; sep_idx < cz->get_num_stream_endpoints(); ++sep_idx) { // Register ID in _port_block_map block_id_t id(mb_idx, NODE_ID_SEP, sep_idx); _port_block_map.insert({{mb_idx, sep_idx + 1}, id}); _sep_map.insert({id.to_string(), sep_addr_t(remote_device_id, sep_idx)}); } } } void _init_static_connections() { UHD_LOG_TRACE("RFNOC::GRAPH", "Identifying static connections..."); for (auto& kv_cz : _client_zeros) { auto& adjacency_list = kv_cz.second->get_adjacency_list(); for (auto& edge : adjacency_list) { // Assemble edge auto graph_edge = graph_edge_t(); UHD_ASSERT_THROW( _port_block_map.count({kv_cz.first, edge.src_blk_index})); graph_edge.src_blockid = _port_block_map.at({kv_cz.first, edge.src_blk_index}); UHD_ASSERT_THROW( _port_block_map.count({kv_cz.first, edge.dst_blk_index})); graph_edge.dst_blockid = _port_block_map.at({kv_cz.first, edge.dst_blk_index}); graph_edge.src_port = edge.src_blk_port; graph_edge.dst_port = edge.dst_blk_port; graph_edge.edge = graph_edge_t::edge_t::STATIC; _static_edges.push_back(graph_edge); UHD_LOG_TRACE( "RFNOC::GRAPH", "Static connection: " << graph_edge.to_string()); } } } /************************************************************************** * Helpers *************************************************************************/ /*! Internal connection helper * * Make the connections in the _graph, and set up property propagation * Prerequisite: \p src_blk and \p dst_blk need to point to valid nodes */ void _connect(std::shared_ptr src_blk, size_t src_port, std::shared_ptr dst_blk, size_t dst_port, graph_edge_t::edge_t edge_type, bool skip_property_propagation) { graph_edge_t edge_info( src_port, dst_port, edge_type, not skip_property_propagation); edge_info.src_blockid = src_blk->get_unique_id(); edge_info.dst_blockid = dst_blk->get_unique_id(); _graph->connect(src_blk.get(), dst_blk.get(), edge_info); } /*! Internal physical connection helper * * Make the connections in the physical device * * \throws uhd::routing_error * if the blocks are statically connected to something else */ graph_edge_t::edge_t _physical_connect(const block_id_t& src_blk, size_t src_port, const block_id_t& dst_blk, size_t dst_port) { const std::string src_blk_info = src_blk.to_string() + ":" + std::to_string(src_port); const std::string dst_blk_info = dst_blk.to_string() + ":" + std::to_string(dst_port); // Find the static edge for src_blk:src_port graph_edge_t src_static_edge = _assert_edge( _get_static_edge( [src_blk_id = src_blk.to_string(), src_port](const graph_edge_t& edge) { return edge.src_blockid == src_blk_id && edge.src_port == src_port; }), src_blk_info); // Now see if it's already connected to the destination if (src_static_edge.dst_blockid == dst_blk.to_string() && src_static_edge.dst_port == dst_port) { UHD_LOG_TRACE(LOG_ID, "Blocks " << src_blk_info << " and " << dst_blk_info << " are already statically connected, no physical connection " "required."); return graph_edge_t::STATIC; } // If they're not statically connected, the source *must* be connected // to an SEP, or this route is impossible if (block_id_t(src_static_edge.dst_blockid).get_block_name() != NODE_ID_SEP) { const std::string err_msg = src_blk_info + " is neither statically connected to " + dst_blk_info + " nor to an SEP! Routing impossible."; UHD_LOG_ERROR(LOG_ID, err_msg); throw uhd::routing_error(err_msg); } // OK, now we know which source SEP we have const std::string src_sep_info = src_static_edge.dst_blockid; const sep_addr_t src_sep_addr = _sep_map.at(src_sep_info); // Now find the static edge for the destination SEP auto dst_static_edge = _assert_edge( _get_static_edge( [dst_blk_id = dst_blk.to_string(), dst_port](const graph_edge_t& edge) { return edge.dst_blockid == dst_blk_id && edge.dst_port == dst_port; }), dst_blk_info); // If they're not statically connected, the source *must* be connected // to an SEP, or this route is impossible if (block_id_t(dst_static_edge.src_blockid).get_block_name() != NODE_ID_SEP) { const std::string err_msg = dst_blk_info + " is neither statically connected to " + src_blk_info + " nor to an SEP! Routing impossible."; UHD_LOG_ERROR(LOG_ID, err_msg); throw uhd::routing_error(err_msg); } // OK, now we know which destination SEP we have const std::string dst_sep_info = dst_static_edge.src_blockid; const sep_addr_t dst_sep_addr = _sep_map.at(dst_sep_info); // Now all we need to do is dynamically connect those SEPs auto strm_info = _gsm->create_device_to_device_data_stream( dst_sep_addr, src_sep_addr, false, 0.1, 0.0, false); UHD_LOGGER_DEBUG(LOG_ID) << boost::format("Data stream between EPID %d and EPID %d established " "where downstream buffer can hold %lu bytes and %u packets") % std::get<0>(strm_info).first % std::get<0>(strm_info).second % std::get<1>(strm_info).bytes % std::get<1>(strm_info).packets; return graph_edge_t::DYNAMIC; } //! Flush and reset each connected port on the mboard void _flush_and_reset_mboard(size_t mb_idx, detail::client_zero::sptr mb_cz, const size_t num_blocks, const size_t first_block_port) { UHD_LOG_TRACE("RFNOC::GRAPH", std::string("Flushing and resetting blocks on mboard ") + std::to_string(mb_idx)); mb_cz->complete_flush_all_blocks(); // Reset for (size_t portno = 0; portno < num_blocks; ++portno) { auto block_portno = portno + first_block_port; mb_cz->reset_chdr(block_portno); mb_cz->reset_ctrl(block_portno); } } /*! Find the static edge that matches \p pred * * \throws uhd::assertion_error if the edge can't be found. So be careful! */ template boost::optional _get_static_edge(UnaryPredicate&& pred) { auto edge_it = std::find_if(_static_edges.cbegin(), _static_edges.cend(), pred); if (edge_it == _static_edges.cend()) { return boost::none; } return *edge_it; } /*! Make sure an optional edge info is valid, or throw. */ graph_edge_t _assert_edge( boost::optional edge_o, const std::string& blk_info) { if (!bool(edge_o)) { const std::string err_msg = std::string("Cannot connect block ") + blk_info + ", port is unconnected in the FPGA!"; UHD_LOG_ERROR("RFNOC::GRAPH", err_msg); throw uhd::routing_error(err_msg); } return edge_o.get(); } /************************************************************************** * Attributes *************************************************************************/ //! Reference to the underlying device implementation detail::rfnoc_device::sptr _device; //! Number of motherboards, this is technically redundant but useful for // easy lookups. size_t _num_mboards; //! Reference to the property tree uhd::property_tree::sptr _tree; //! Registry for the blocks (it's a separate class) std::unique_ptr _block_registry; /*! Registry for the actual block connections on the crossbar * When we register blocks in the _block_registry, we also need to store some * information in this map so we can easily figure out which crossbar port a block * controller is connected to * \p keys are the string representation of the block ID * \p values are the block crossbar information structs * TODO: change from string block IDs to block_id_t with COOL custom hashing */ std::unordered_map _xbar_block_config; //! Reference to the graph std::unique_ptr _graph; //! Stash a list of motherboard controllers std::unordered_map _mb_controllers; //! Stash of the client zeros for all motherboards std::unordered_map _client_zeros; //! Map a pair (motherboard index, control crossbar port) to an RFNoC block // or SEP std::map, block_id_t> _port_block_map; //! Map SEP block ID (e.g. 0/SEP#0) onto a sep_addr_t std::unordered_map _sep_map; //! List of statically connected edges. Includes SEPs too! std::vector _static_edges; //! uptr to graph stream manager graph_stream_manager::uptr _gsm; }; /* class rfnoc_graph_impl */ /****************************************************************************** * Factory *****************************************************************************/ rfnoc_graph::sptr rfnoc_graph::make(const uhd::device_addr_t& device_addr) { return std::make_shared(device_addr); }