// // Copyright 2014-2016 Ettus Research LLC // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with this program. If not, see . // // Provides streaming-related functions which are used by device3 objects. #include "device3_impl.hpp" #include #include #include #include #include #include #include "../common/async_packet_handler.hpp" #include "../../transport/super_recv_packet_handler.hpp" #include "../../transport/super_send_packet_handler.hpp" #include "../../rfnoc/rx_stream_terminator.hpp" #include "../../rfnoc/tx_stream_terminator.hpp" #include #include #define UHD_STREAMER_LOG() UHD_LOGV(never) using namespace uhd; using namespace uhd::usrp; using namespace uhd::transport; //! CVITA uses 12-Bit sequence numbers static const boost::uint32_t HW_SEQ_NUM_MASK = 0xfff; /*********************************************************************** * Helper functions for get_?x_stream() **********************************************************************/ static uhd::stream_args_t sanitize_stream_args(const uhd::stream_args_t &args_) { uhd::stream_args_t args = args_; if (args.channels.empty()) { args.channels = std::vector(1, 0); } return args; } static void check_stream_sig_compatible(const rfnoc::stream_sig_t &stream_sig, stream_args_t &args, const std::string &tx_rx) { if (args.otw_format.empty()) { if (stream_sig.item_type.empty()) { throw uhd::runtime_error(str( boost::format("[%s Streamer] No otw_format defined!") % tx_rx )); } else { args.otw_format = stream_sig.item_type; } } else if (not stream_sig.item_type.empty() and stream_sig.item_type != args.otw_format) { throw uhd::runtime_error(str( boost::format("[%s Streamer] Conflicting OTW types defined: args.otw_format = '%s' <=> stream_sig.item_type = '%s'") % tx_rx % args.otw_format % stream_sig.item_type )); } const size_t bpi = convert::get_bytes_per_item(args.otw_format); // bytes per item if (stream_sig.packet_size) { if (args.args.has_key("spp")) { size_t args_spp = args.args.cast("spp", 0); if (args_spp * bpi != stream_sig.packet_size) { throw uhd::runtime_error(str( boost::format("[%s Streamer] Conflicting packet sizes defined: args yields %d bytes but stream_sig.packet_size is %d bytes") % tx_rx % (args_spp * bpi) % stream_sig.packet_size )); } } else { args.args["spp"] = str(boost::format("%d") % (stream_sig.packet_size / bpi)); } } } /*! \brief Returns a list of rx or tx channels for a streamer. * * If the given stream args contain instructions to set up channels, * those are used. Otherwise, the current device's channel definition * is consulted. * * \param args_ Stream args. * \param[out] chan_list The list of channels in the correct order. * \param[out] chan_args Channel args for every channel. `chan_args.size() == chan_list.size()` */ void generate_channel_list( const uhd::stream_args_t &args_, std::vector &chan_list, std::vector &chan_args ) { uhd::stream_args_t args = args_; BOOST_FOREACH(const size_t chan_idx, args.channels) { //// Find block ID for this channel: if (args.args.has_key(str(boost::format("block_id%d") % chan_idx))) { chan_list.push_back( uhd::rfnoc::block_id_t( args.args.pop(str(boost::format("block_id%d") % chan_idx)) ) ); chan_args.push_back(args.args); } else if (args.args.has_key("block_id")) { chan_list.push_back(args.args.get("block_id")); chan_args.push_back(args.args); chan_args.back().pop("block_id"); } else { throw uhd::runtime_error(str( boost::format("Cannot create streamers: No block_id specified for channel %d.") % chan_idx )); } //// Find block port for this channel if (args.args.has_key(str(boost::format("block_port%d") % chan_idx))) { chan_args.back()["block_port"] = args.args.pop(str(boost::format("block_port%d") % chan_idx)); } else if (args.args.has_key("block_port")) { // We have to write it again, because the chan args from the // property tree might have overwritten this chan_args.back()["block_port"] = args.args.get("block_port"); } } } /*********************************************************************** * RX Flow Control Functions **********************************************************************/ //! Stores the state of RX flow control struct rx_fc_cache_t { rx_fc_cache_t(): last_seq_in(0){} size_t last_seq_in; }; /*! Determine the size of the flow control window in number of packets. * * This value depends on three things: * - The packet size (in bytes), P * - The size of the software buffer (in bytes), B * - The desired buffer fullness, F * * The FC window size is thus X = floor(B*F/P). * * \param pkt_size The maximum packet size in bytes * \param sw_buff_size Software buffer size in bytes * \param rx_args If this has a key 'recv_buff_fullness', this value will * be used for said fullness. Must be between 0.01 and 1. * * \returns The size of the flow control window in number of packets */ static size_t get_rx_flow_control_window( size_t pkt_size, size_t sw_buff_size, const device_addr_t& rx_args ) { double fullness_factor = rx_args.cast( "recv_buff_fullness", uhd::rfnoc::DEFAULT_FC_RX_SW_BUFF_FULL_FACTOR ); if (fullness_factor < 0.01 || fullness_factor > 1) { throw uhd::value_error("recv_buff_fullness must be in [0.01, 1] inclusive (1% to 100%)"); } size_t window_in_pkts = (static_cast(sw_buff_size * fullness_factor) / pkt_size); if (rx_args.has_key("max_recv_window")) { window_in_pkts = std::min( window_in_pkts, rx_args.cast("max_recv_window", window_in_pkts) ); } if (window_in_pkts == 0) { throw uhd::value_error("recv_buff_size must be larger than the recv_frame_size."); } UHD_ASSERT_THROW(size_t(sw_buff_size * fullness_factor) >= pkt_size * window_in_pkts); return window_in_pkts; } /*! Send out RX flow control packets. * * For an rx stream, this function takes care of sending back * a flow control packet to the source telling it which * packets have been consumed. * * This function should only be called by the function handling * the rx stream, usually recv() in super_recv_packet_handler. * * \param sid The SID that goes into this packet. This is the reversed() * version of the data stream's SID. * \param xport A transport object over which to send the data * \param big_endian Endianness of the transport * \param seq32_state Pointer to a variable that saves the 32-Bit state * of the sequence numbers, since we only have 12 Bit * sequence numbers in CHDR. * \param last_seq The value to send: The last consumed packet's sequence number. */ static void handle_rx_flowctrl( const sid_t &sid, zero_copy_if::sptr xport, endianness_t endianness, boost::shared_ptr fc_cache, const size_t last_seq ) { static const size_t RXFC_PACKET_LEN_IN_WORDS = 2; static const size_t RXFC_CMD_CODE_OFFSET = 0; static const size_t RXFC_SEQ_NUM_OFFSET = 1; managed_send_buffer::sptr buff = xport->get_send_buff(0.0); if (not buff) { throw uhd::runtime_error("handle_rx_flowctrl timed out getting a send buffer"); } boost::uint32_t *pkt = buff->cast(); // Recover sequence number. The sequence numbers handled by the streamers // are 12 Bits, but we want to know the 32-Bit sequence number. size_t &seq32 = fc_cache->last_seq_in; const size_t seq12 = seq32 & HW_SEQ_NUM_MASK; if (last_seq < seq12) seq32 += (HW_SEQ_NUM_MASK + 1); seq32 &= ~HW_SEQ_NUM_MASK; seq32 |= last_seq; // Super-verbose mode: //static size_t fc_pkt_count = 0; //UHD_MSG(status) << "sending flow ctrl packet " << fc_pkt_count++ << ", acking " << str(boost::format("%04d\tseq_sw==0x%08x") % last_seq % seq32) << std::endl; //load packet info vrt::if_packet_info_t packet_info; packet_info.packet_type = vrt::if_packet_info_t::PACKET_TYPE_FC; packet_info.num_payload_words32 = RXFC_PACKET_LEN_IN_WORDS; packet_info.num_payload_bytes = packet_info.num_payload_words32*sizeof(boost::uint32_t); packet_info.packet_count = seq32; packet_info.sob = false; packet_info.eob = false; packet_info.sid = sid.get(); packet_info.has_sid = true; packet_info.has_cid = false; packet_info.has_tsi = false; packet_info.has_tsf = false; packet_info.has_tlr = false; if (endianness == ENDIANNESS_BIG) { // Load Header: vrt::chdr::if_hdr_pack_be(pkt, packet_info); // Load Payload: (the sequence number) pkt[packet_info.num_header_words32+RXFC_CMD_CODE_OFFSET] = uhd::htonx(0); pkt[packet_info.num_header_words32+RXFC_SEQ_NUM_OFFSET] = uhd::htonx(seq32); } else { // Load Header: vrt::chdr::if_hdr_pack_le(pkt, packet_info); // Load Payload: (the sequence number) pkt[packet_info.num_header_words32+RXFC_CMD_CODE_OFFSET] = uhd::htowx(0); pkt[packet_info.num_header_words32+RXFC_SEQ_NUM_OFFSET] = uhd::htowx(seq32); } //std::cout << " SID=" << std::hex << sid << " hdr bits=" << packet_info.packet_type << " seq32=" << seq32 << std::endl; //std::cout << "num_packet_words32: " << packet_info.num_packet_words32 << std::endl; //for (size_t i = 0; i < packet_info.num_packet_words32; i++) { //std::cout << str(boost::format("0x%08x") % pkt[i]) << " "; //if (i % 2) { //std::cout << std::endl; //} //} //send the buffer over the interface buff->commit(sizeof(boost::uint32_t)*(packet_info.num_packet_words32)); } /*********************************************************************** * TX Flow Control Functions **********************************************************************/ //! Stores the state of TX flow control struct tx_fc_cache_t { tx_fc_cache_t(void): stream_channel(0), device_channel(0), last_seq_out(0), last_seq_ack(0), seq_queue(1){} size_t stream_channel; size_t device_channel; size_t last_seq_out; size_t last_seq_ack; uhd::transport::bounded_buffer seq_queue; boost::shared_ptr async_queue; boost::shared_ptr old_async_queue; }; /*! Return the size of the flow control window in packets. * * If the return value of this function is F, the last tx'd packet * has index N and the last ack'd packet has index M, the amount of * FC credit we have is C = F + M - N (i.e. we can send C more packets * before getting another ack). * * Note: If `send_buff_size` is set in \p tx_hints, this will * override hw_buff_size_. */ static size_t get_tx_flow_control_window( size_t pkt_size, const double hw_buff_size_, const device_addr_t& tx_hints ) { double hw_buff_size = tx_hints.cast("send_buff_size", hw_buff_size_); size_t window_in_pkts = (static_cast(hw_buff_size) / pkt_size); if (window_in_pkts == 0) { throw uhd::value_error("send_buff_size must be larger than the send_frame_size."); } return window_in_pkts; } static managed_send_buffer::sptr get_tx_buff_with_flowctrl( task::sptr /*holds ref*/, boost::shared_ptr fc_cache, zero_copy_if::sptr xport, size_t fc_window, const double timeout ){ while (true) { // delta is the amount of FC credit we've used up const size_t delta = (fc_cache->last_seq_out & HW_SEQ_NUM_MASK) - (fc_cache->last_seq_ack & HW_SEQ_NUM_MASK); // If we want to send another packet, we must have FC credit left if ((delta & HW_SEQ_NUM_MASK) < fc_window) break; // If credit is all used up, we check seq_queue for more. const bool ok = fc_cache->seq_queue.pop_with_timed_wait(fc_cache->last_seq_ack, timeout); if (not ok) { return managed_send_buffer::sptr(); //timeout waiting for flow control } } managed_send_buffer::sptr buff = xport->get_send_buff(timeout); if (buff) { fc_cache->last_seq_out++; //update seq, this will actually be a send } return buff; } #define DEVICE3_ASYNC_EVENT_CODE_FLOW_CTRL 0 /*! Handle incoming messages. If they're flow control, update the TX FC cache. * Otherwise, send them to the async message queue for the user to poll. * * This is run inside a uhd::task as long as this streamer lives. */ static void handle_tx_async_msgs( boost::shared_ptr fc_cache, zero_copy_if::sptr xport, endianness_t endianness, boost::function get_tick_rate ) { managed_recv_buffer::sptr buff = xport->get_recv_buff(); if (not buff) return; //extract packet info vrt::if_packet_info_t if_packet_info; if_packet_info.num_packet_words32 = buff->size()/sizeof(boost::uint32_t); const boost::uint32_t *packet_buff = buff->cast(); //unpacking can fail boost::uint32_t (*endian_conv)(boost::uint32_t) = uhd::ntohx; try { if (endianness == ENDIANNESS_BIG) { vrt::chdr::if_hdr_unpack_be(packet_buff, if_packet_info); endian_conv = uhd::ntohx; } else { vrt::chdr::if_hdr_unpack_le(packet_buff, if_packet_info); endian_conv = uhd::wtohx; } } catch(const std::exception &ex) { UHD_MSG(error) << "Error parsing async message packet: " << ex.what() << std::endl; return; } double tick_rate = get_tick_rate(); if (tick_rate == rfnoc::tick_node_ctrl::RATE_UNDEFINED) { tick_rate = 1; } //fill in the async metadata async_metadata_t metadata; load_metadata_from_buff( endian_conv, metadata, if_packet_info, packet_buff, tick_rate, fc_cache->stream_channel ); // TODO: Shouldn't we be polling if_packet_info.packet_type == PACKET_TYPE_FC? // Thing is, on X300, packet_type == 0, so that wouldn't work. But it seems it should. //The FC response and the burst ack are two indicators that the radio //consumed packets. Use them to update the FC metadata if (metadata.event_code == DEVICE3_ASYNC_EVENT_CODE_FLOW_CTRL) { const size_t seq = metadata.user_payload[0]; fc_cache->seq_queue.push_with_pop_on_full(seq); } //FC responses don't propagate up to the user so filter them here if (metadata.event_code != DEVICE3_ASYNC_EVENT_CODE_FLOW_CTRL) { fc_cache->async_queue->push_with_pop_on_full(metadata); metadata.channel = fc_cache->device_channel; fc_cache->old_async_queue->push_with_pop_on_full(metadata); standard_async_msg_prints(metadata); } } /*********************************************************************** * Async Data **********************************************************************/ bool device3_impl::recv_async_msg( async_metadata_t &async_metadata, double timeout ) { return _async_md->pop_with_timed_wait(async_metadata, timeout); } /*********************************************************************** * Receive streamer **********************************************************************/ void device3_impl::update_rx_streamers(double /* rate */) { BOOST_FOREACH(const std::string &block_id, _rx_streamers.keys()) { UHD_STREAMER_LOG() << "[Device3] updating RX streamer to " << block_id << std::endl; boost::shared_ptr my_streamer = boost::dynamic_pointer_cast(_rx_streamers[block_id].lock()); if (my_streamer) { double tick_rate = my_streamer->get_terminator()->get_tick_rate(); if (tick_rate == rfnoc::tick_node_ctrl::RATE_UNDEFINED) { tick_rate = 1.0; } my_streamer->set_tick_rate(tick_rate); double samp_rate = my_streamer->get_terminator()->get_output_samp_rate(); if (samp_rate == rfnoc::rate_node_ctrl::RATE_UNDEFINED) { samp_rate = 1.0; } // This formula is not derived by any scientific means -- we just need to // increase the failure threshold as we increase rates. For 1 Msps, we use // the default. const size_t alignment_failure_factor = std::max(size_t(1), size_t(samp_rate * 1000 / tick_rate)); double scaling = my_streamer->get_terminator()->get_output_scale_factor(); if (scaling == rfnoc::scalar_node_ctrl::SCALE_UNDEFINED) { scaling = 1/32767.; } UHD_STREAMER_LOG() << " New tick_rate == " << tick_rate << " New samp_rate == " << samp_rate << " New scaling == " << scaling << std::endl; my_streamer->set_tick_rate(tick_rate); my_streamer->set_samp_rate(samp_rate); // 1000 packets is the default alignment failure threshold my_streamer->set_alignment_failure_threshold(1000 * alignment_failure_factor); my_streamer->set_scale_factor(scaling); } } } rx_streamer::sptr device3_impl::get_rx_stream(const stream_args_t &args_) { boost::mutex::scoped_lock lock(_transport_setup_mutex); stream_args_t args = sanitize_stream_args(args_); // I. Generate the channel list std::vector chan_list; std::vector chan_args; generate_channel_list(args, chan_list, chan_args); // Note: All 'args.args' are merged into chan_args now. // II. Iterate over all channels boost::shared_ptr my_streamer; // The terminator's lifetime is coupled to the streamer. // There is only one terminator. If the streamer has multiple channels, // it will be connected to each upstream block. rfnoc::rx_stream_terminator::sptr recv_terminator = rfnoc::rx_stream_terminator::make(); for (size_t stream_i = 0; stream_i < chan_list.size(); stream_i++) { // Get block ID and mb index uhd::rfnoc::block_id_t block_id = chan_list[stream_i]; UHD_STREAMER_LOG() << "[RX Streamer] chan " << stream_i << " connecting to " << block_id << std::endl; // Update args so args.args is always valid for this particular channel: args.args = chan_args[stream_i]; size_t mb_index = block_id.get_device_no(); size_t suggested_block_port = args.args.cast("block_port", rfnoc::ANY_PORT); // Access to this channel's block control uhd::rfnoc::source_block_ctrl_base::sptr blk_ctrl = boost::dynamic_pointer_cast(get_block_ctrl(block_id)); // Connect the terminator with this channel's block. size_t block_port = blk_ctrl->connect_downstream( recv_terminator, suggested_block_port, args.args ); const size_t terminator_port = recv_terminator->connect_upstream(blk_ctrl); blk_ctrl->set_downstream_port(block_port, terminator_port); recv_terminator->set_upstream_port(terminator_port, block_port); // Check if the block connection is compatible (spp and item type) check_stream_sig_compatible(blk_ctrl->get_output_signature(block_port), args, "RX"); // Setup the DSP transport hints device_addr_t rx_hints = get_rx_hints(mb_index); //allocate sid and create transport uhd::sid_t stream_address = blk_ctrl->get_address(block_port); UHD_STREAMER_LOG() << "[RX Streamer] creating rx stream " << rx_hints.to_string() << std::endl; both_xports_t xport = make_transport(stream_address, RX_DATA, rx_hints); UHD_STREAMER_LOG() << std::hex << "[RX Streamer] data_sid = " << xport.send_sid << std::dec << " actual recv_buff_size = " << xport.recv_buff_size << std::endl; // Configure the block blk_ctrl->set_destination(xport.send_sid.get_src(), block_port); blk_ctrl->sr_write(uhd::rfnoc::SR_RESP_OUT_DST_SID, xport.send_sid.get_src(), block_port); UHD_STREAMER_LOG() << "[RX Streamer] resp_out_dst_sid == " << xport.send_sid.get_src() << std::endl; // Find all upstream radio nodes and set their response in SID to the host std::vector > upstream_radio_nodes = blk_ctrl->find_upstream_node(); UHD_STREAMER_LOG() << "[RX Streamer] Number of upstream radio nodes: " << upstream_radio_nodes.size() << std::endl; BOOST_FOREACH(const boost::shared_ptr &node, upstream_radio_nodes) { node->sr_write(uhd::rfnoc::SR_RESP_OUT_DST_SID, xport.send_sid.get_src(), block_port); } // To calculate the max number of samples per packet, we assume the maximum header length // to avoid fragmentation should the entire header be used. const size_t bpp = xport.recv->get_recv_frame_size() - stream_options.rx_max_len_hdr; // bytes per packet const size_t bpi = convert::get_bytes_per_item(args.otw_format); // bytes per item const size_t spp = std::min(args.args.cast("spp", bpp/bpi), bpp/bpi); // samples per packet UHD_STREAMER_LOG() << "[RX Streamer] spp == " << spp << std::endl; //make the new streamer given the samples per packet if (not my_streamer) my_streamer = boost::make_shared(spp); my_streamer->resize(chan_list.size()); //init some streamer stuff std::string conv_endianness; if (get_transport_endianness(mb_index) == ENDIANNESS_BIG) { my_streamer->set_vrt_unpacker(&vrt::chdr::if_hdr_unpack_be); conv_endianness = "be"; } else { my_streamer->set_vrt_unpacker(&vrt::chdr::if_hdr_unpack_le); conv_endianness = "le"; } //set the converter uhd::convert::id_type id; id.input_format = args.otw_format + "_item32_" + conv_endianness; id.num_inputs = 1; id.output_format = args.cpu_format; id.num_outputs = 1; my_streamer->set_converter(id); //flow control setup const size_t pkt_size = spp * bpi + stream_options.rx_max_len_hdr; const size_t fc_window = get_rx_flow_control_window(pkt_size, xport.recv_buff_size, rx_hints); const size_t fc_handle_window = std::max(1, fc_window / stream_options.rx_fc_request_freq); UHD_STREAMER_LOG()<< "[RX Streamer] Flow Control Window (minus one) = " << fc_window-1 << ", Flow Control Handler Window = " << fc_handle_window << std::endl; blk_ctrl->configure_flow_control_out( fc_window-1, // Leave one space for overrun packets TODO make this obsolete block_port ); //Give the streamer a functor to get the recv_buffer //bind requires a zero_copy_if::sptr to add a streamer->xport lifetime dependency my_streamer->set_xport_chan_get_buff( stream_i, boost::bind(&zero_copy_if::get_recv_buff, xport.recv, _1), true /*flush*/ ); //Give the streamer a functor to handle overruns //bind requires a weak_ptr to break the a streamer->streamer circular dependency //Using "this" is OK because we know that this device3_impl will outlive the streamer my_streamer->set_overflow_handler( stream_i, boost::bind( &uhd::rfnoc::rx_stream_terminator::handle_overrun, recv_terminator, boost::weak_ptr(my_streamer), stream_i ) ); //Give the streamer a functor to send flow control messages //handle_rx_flowctrl is static and has no lifetime issues boost::shared_ptr fc_cache(new rx_fc_cache_t()); my_streamer->set_xport_handle_flowctrl( stream_i, boost::bind( &handle_rx_flowctrl, xport.send_sid, xport.send, get_transport_endianness(mb_index), fc_cache, _1 ), fc_handle_window, true/*init*/ ); //Give the streamer a functor issue stream cmd //bind requires a shared pointer to add a streamer->framer lifetime dependency my_streamer->set_issue_stream_cmd( stream_i, boost::bind(&uhd::rfnoc::source_block_ctrl_base::issue_stream_cmd, blk_ctrl, _1, block_port) ); // Tell the streamer which SID is valid for this channel my_streamer->set_xport_chan_sid(stream_i, true, xport.send_sid); } // Connect the terminator to the streamer my_streamer->set_terminator(recv_terminator); // Notify all blocks in this chain that they are connected to an active streamer recv_terminator->set_rx_streamer(true, 0); // Store a weak pointer to prevent a streamer->device3_impl->streamer circular dependency. // Note that we store the streamer only once, and use its terminator's // ID to do so. _rx_streamers[recv_terminator->unique_id()] = boost::weak_ptr(my_streamer); // Sets tick rate, samp rate and scaling on this streamer. // A registered terminator is required to do this. update_rx_streamers(); post_streamer_hooks(RX_DIRECTION); return my_streamer; } /*********************************************************************** * Transmit streamer **********************************************************************/ void device3_impl::update_tx_streamers(double /* rate */) { BOOST_FOREACH(const std::string &block_id, _tx_streamers.keys()) { UHD_STREAMER_LOG() << "[Device3] updating TX streamer: " << block_id << std::endl; boost::shared_ptr my_streamer = boost::dynamic_pointer_cast(_tx_streamers[block_id].lock()); if (my_streamer) { double tick_rate = my_streamer->get_terminator()->get_tick_rate(); if (tick_rate == rfnoc::tick_node_ctrl::RATE_UNDEFINED) { tick_rate = 1.0; } double samp_rate = my_streamer->get_terminator()->get_input_samp_rate(); if (samp_rate == rfnoc::rate_node_ctrl::RATE_UNDEFINED) { samp_rate = 1.0; } double scaling = my_streamer->get_terminator()->get_input_scale_factor(); if (scaling == rfnoc::scalar_node_ctrl::SCALE_UNDEFINED) { scaling = 32767.; } UHD_STREAMER_LOG() << " New tick_rate == " << tick_rate << " New samp_rate == " << samp_rate << " New scaling == " << scaling << std::endl; my_streamer->set_tick_rate(tick_rate); my_streamer->set_samp_rate(samp_rate); my_streamer->set_scale_factor(scaling); } } } tx_streamer::sptr device3_impl::get_tx_stream(const uhd::stream_args_t &args_) { boost::mutex::scoped_lock lock(_transport_setup_mutex); stream_args_t args = sanitize_stream_args(args_); // I. Generate the channel list std::vector chan_list; std::vector chan_args; generate_channel_list(args, chan_list, chan_args); // Note: All 'args.args' are merged into chan_args now. //shared async queue for all channels in streamer boost::shared_ptr async_md(new async_md_type(1000/*messages deep*/)); // II. Iterate over all channels boost::shared_ptr my_streamer; // The terminator's lifetime is coupled to the streamer. // There is only one terminator. If the streamer has multiple channels, // it will be connected to each downstream block. rfnoc::tx_stream_terminator::sptr send_terminator = rfnoc::tx_stream_terminator::make(); for (size_t stream_i = 0; stream_i < chan_list.size(); stream_i++) { // Get block ID and mb index uhd::rfnoc::block_id_t block_id = chan_list[stream_i]; // Update args so args.args is always valid for this particular channel: args.args = chan_args[stream_i]; size_t mb_index = block_id.get_device_no(); size_t suggested_block_port = args.args.cast("block_port", rfnoc::ANY_PORT); // Access to this channel's block control uhd::rfnoc::sink_block_ctrl_base::sptr blk_ctrl = boost::dynamic_pointer_cast(get_block_ctrl(block_id)); // Connect the terminator with this channel's block. // This will throw if the connection is not possible. size_t block_port = blk_ctrl->connect_upstream( send_terminator, suggested_block_port, args.args ); const size_t terminator_port = send_terminator->connect_downstream(blk_ctrl); blk_ctrl->set_upstream_port(block_port, terminator_port); send_terminator->set_downstream_port(terminator_port, block_port); // Check if the block connection is compatible (spp and item type) check_stream_sig_compatible(blk_ctrl->get_input_signature(block_port), args, "TX"); // Setup the dsp transport hints device_addr_t tx_hints = get_tx_hints(mb_index); //allocate sid and create transport uhd::sid_t stream_address = blk_ctrl->get_address(block_port); UHD_STREAMER_LOG() << "[TX Streamer] creating tx stream " << tx_hints.to_string() << std::endl; both_xports_t xport = make_transport(stream_address, TX_DATA, tx_hints); UHD_STREAMER_LOG() << std::hex << "[TX Streamer] data_sid = " << xport.send_sid << std::dec << std::endl; // To calculate the max number of samples per packet, we assume the maximum header length // to avoid fragmentation should the entire header be used. const size_t bpp = tx_hints.cast("bpp", xport.send->get_send_frame_size()) - stream_options.tx_max_len_hdr; const size_t bpi = convert::get_bytes_per_item(args.otw_format); // bytes per item const size_t spp = std::min(args.args.cast("spp", bpp/bpi), bpp/bpi); // samples per packet UHD_STREAMER_LOG() << "[TX Streamer] spp == " << spp << std::endl; //make the new streamer given the samples per packet if (not my_streamer) my_streamer = boost::make_shared(spp); my_streamer->resize(chan_list.size()); //init some streamer stuff std::string conv_endianness; if (get_transport_endianness(mb_index) == ENDIANNESS_BIG) { my_streamer->set_vrt_packer(&vrt::chdr::if_hdr_pack_be); conv_endianness = "be"; } else { my_streamer->set_vrt_packer(&vrt::chdr::if_hdr_pack_le); conv_endianness = "le"; } //set the converter uhd::convert::id_type id; id.input_format = args.cpu_format; id.num_inputs = 1; id.output_format = args.otw_format + "_item32_" + conv_endianness; id.num_outputs = 1; my_streamer->set_converter(id); //flow control setup const size_t pkt_size = spp * bpi + stream_options.tx_max_len_hdr; // For flow control, this value is used to determine the window size in *packets* size_t fc_window = get_tx_flow_control_window( pkt_size, // This is the maximum packet size blk_ctrl->get_fifo_size(block_port), tx_hints // This can override the value reported by the block! ); const size_t fc_handle_window = std::max(1, fc_window / stream_options.tx_fc_response_freq); UHD_STREAMER_LOG() << "[TX Streamer] Flow Control Window = " << fc_window << ", Flow Control Handler Window = " << fc_handle_window << std::endl; blk_ctrl->configure_flow_control_in( stream_options.tx_fc_response_cycles, fc_handle_window, /*pkts*/ block_port ); boost::shared_ptr fc_cache(new tx_fc_cache_t()); fc_cache->stream_channel = stream_i; fc_cache->device_channel = mb_index; fc_cache->async_queue = async_md; fc_cache->old_async_queue = _async_md; boost::function tick_rate_retriever = boost::bind( &rfnoc::tick_node_ctrl::get_tick_rate, send_terminator, std::set< rfnoc::node_ctrl_base::sptr >() // Need to specify default args with bind ); task::sptr task = task::make( boost::bind( &handle_tx_async_msgs, fc_cache, xport.recv, get_transport_endianness(mb_index), tick_rate_retriever ) ); blk_ctrl->sr_write(uhd::rfnoc::SR_RESP_IN_DST_SID, xport.recv_sid.get_dst(), block_port); UHD_STREAMER_LOG() << "[TX Streamer] resp_in_dst_sid == " << boost::format("0x%04X") % xport.recv_sid.get_dst() << std::endl; // Find all downstream radio nodes and set their response in SID to the host std::vector > downstream_radio_nodes = blk_ctrl->find_downstream_node(); UHD_STREAMER_LOG() << "[TX Streamer] Number of downstream radio nodes: " << downstream_radio_nodes.size() << std::endl; BOOST_FOREACH(const boost::shared_ptr &node, downstream_radio_nodes) { node->sr_write(uhd::rfnoc::SR_RESP_IN_DST_SID, xport.send_sid.get_src(), block_port); } //Give the streamer a functor to get the send buffer //get_tx_buff_with_flowctrl is static so bind has no lifetime issues //xport.send (sptr) is required to add streamer->data-transport lifetime dependency //task (sptr) is required to add a streamer->async-handler lifetime dependency my_streamer->set_xport_chan_get_buff( stream_i, boost::bind(&get_tx_buff_with_flowctrl, task, fc_cache, xport.send, fc_window, _1) ); //Give the streamer a functor handled received async messages my_streamer->set_async_receiver( boost::bind(&async_md_type::pop_with_timed_wait, async_md, _1, _2) ); my_streamer->set_xport_chan_sid(stream_i, true, xport.send_sid); // CHDR does not support trailers my_streamer->set_enable_trailer(false); } // Connect the terminator to the streamer my_streamer->set_terminator(send_terminator); // Notify all blocks in this chain that they are connected to an active streamer send_terminator->set_tx_streamer(true, 0); // Store a weak pointer to prevent a streamer->device3_impl->streamer circular dependency. // Note that we store the streamer only once, and use its terminator's // ID to do so. _tx_streamers[send_terminator->unique_id()] = boost::weak_ptr(my_streamer); // Sets tick rate, samp rate and scaling on this streamer // A registered terminator is required to do this. update_tx_streamers(); post_streamer_hooks(TX_DIRECTION); return my_streamer; }