diff options
author | Ciro Nishiguchi <ciro.nishiguchi@ni.com> | 2019-10-28 14:28:33 -0500 |
---|---|---|
committer | Martin Braun <martin.braun@ettus.com> | 2019-11-26 12:21:33 -0800 |
commit | a8e286b106f19c37d6cf20de886c65f3c04da162 (patch) | |
tree | 5c1baf4310abc0e4e082c960dffabd18a9fb6a3e /host/lib/include/uhdlib/rfnoc | |
parent | 10b9d2688b5bcb150eec786a9ef7473f1c1c28ac (diff) | |
download | uhd-a8e286b106f19c37d6cf20de886c65f3c04da162.tar.gz uhd-a8e286b106f19c37d6cf20de886c65f3c04da162.tar.bz2 uhd-a8e286b106f19c37d6cf20de886c65f3c04da162.zip |
rfnoc: Make polling I/O service not block on flow control
Add a new method to io_service::send_io to check whether the destination
is ready for data, to make it possible to poll send_io rather than block
waiting for flow control credits.
Diffstat (limited to 'host/lib/include/uhdlib/rfnoc')
-rw-r--r-- | host/lib/include/uhdlib/rfnoc/chdr_tx_data_xport.hpp | 49 |
1 files changed, 34 insertions, 15 deletions
diff --git a/host/lib/include/uhdlib/rfnoc/chdr_tx_data_xport.hpp b/host/lib/include/uhdlib/rfnoc/chdr_tx_data_xport.hpp index 3226ba59b..47293f44a 100644 --- a/host/lib/include/uhdlib/rfnoc/chdr_tx_data_xport.hpp +++ b/host/lib/include/uhdlib/rfnoc/chdr_tx_data_xport.hpp @@ -188,7 +188,11 @@ public: */ buff_t::uptr get_send_buff(const int32_t timeout_ms) { - return _send_io->get_send_buff(timeout_ms); + if (_send_io->wait_for_dest_ready(_frame_size, timeout_ms)) { + return _send_io->get_send_buff(timeout_ms); + } else { + return nullptr; + } } /*! @@ -318,27 +322,23 @@ private: * \param buff the frame buffer to release * \param send_link the send link for flow control messages */ - void _send_callback(buff_t::uptr& buff, transport::send_link_if* send_link) + void _send_callback(buff_t::uptr buff, transport::send_link_if* send_link) { // If the packet size is not a multiple of the word size, then we will // still occupy an integer multiple of word size bytes in the FPGA, so // we need to calculate appropriately. const size_t packet_size_rounded = _round_pkt_size(buff->packet_size()); + send_link->release_send_buff(std::move(buff)); - if (_fc_state.dest_has_space(packet_size_rounded)) { - send_link->release_send_buff(std::move(buff)); - buff = nullptr; - - _fc_state.data_sent(packet_size_rounded); + _fc_state.data_sent(packet_size_rounded); - if (_fc_state.get_fc_resync_req_pending() - && _fc_state.dest_has_space(chdr::strc_payload::MAX_PACKET_SIZE)) { - const auto& xfer_counts = _fc_state.get_xfer_counts(); - const size_t strc_size = - _round_pkt_size(_fc_sender.send_strc_resync(send_link, xfer_counts)); - _fc_state.clear_fc_resync_req_pending(); - _fc_state.data_sent(strc_size); - } + if (_fc_state.get_fc_resync_req_pending() + && _fc_state.dest_has_space(chdr::strc_payload::MAX_PACKET_SIZE)) { + const auto& xfer_counts = _fc_state.get_xfer_counts(); + const size_t strc_size = + _round_pkt_size(_fc_sender.send_strc_resync(send_link, xfer_counts)); + _fc_state.clear_fc_resync_req_pending(); + _fc_state.data_sent(strc_size); } } @@ -347,6 +347,22 @@ private: return ((pkt_size_bytes + _chdr_w_bytes - 1) / _chdr_w_bytes) * _chdr_w_bytes; } + /*! + * Flow control callback for I/O service + * + * The I/O service invokes this callback in the send_io::wait_for_dest_ready + * method. + * + * \param num_bytes The number of bytes in the packet to be sent + * \return Whether there are enough flow control credits for num_bytes + */ + bool _fc_callback(const size_t num_bytes) + { + // No need to round num_bytes since the transport always checks for + // enough space for a full frame. + return _fc_state.dest_has_space(num_bytes); + } + // Interface to the I/O service transport::send_io_if::sptr _send_io; @@ -379,6 +395,9 @@ private: //! The CHDR width in bytes. size_t _chdr_w_bytes; + + //! The size of the send frame + size_t _frame_size; }; }} // namespace uhd::rfnoc |