// // Copyright 2019 Ettus Research, a National Instruments Brand // // SPDX-License-Identifier: GPL-3.0-or-later // #ifndef INCLUDED_UHD_TRANSPORT_NIRIO_LINK_HPP #define INCLUDED_UHD_TRANSPORT_NIRIO_LINK_HPP #include #include #include #include #include #include #include #include #include #include #include namespace uhd { namespace transport { using fifo_data_t = uint64_t; /*! NI-RIO frame_buff * * NI-RIO internally manages the memory, so these frame_buff objects just need * a way of internally manipulating the _data pointer */ class nirio_frame_buff : public frame_buff { public: nirio_frame_buff() { _data = nullptr; } fifo_data_t** get_fifo_ptr_ref() { return reinterpret_cast(&_data); }; }; class nirio_adapter_info : public adapter_info { public: nirio_adapter_info(const std::string& resource) : _resource(resource) {} ~nirio_adapter_info() {} std::string to_string() { return std::string("NIRIO:") + _resource; } bool operator==(const nirio_adapter_info& rhs) const { return (_resource == rhs._resource); } private: const std::string _resource; }; /*! Link object to talking to NI-RIO USRPs (X310 variants using PCIe) * * \b Note: This link cannot release frame buffers out of order, which means it * can't be used with an IO service that does that. */ class nirio_link : public recv_link_base, public send_link_base { public: using sptr = std::shared_ptr; ~nirio_link(); /*! Make a new NI-RIO link. * * \param addr a string representing the destination address * \param port a string representing the destination port * \param params Values for frame sizes, num frames, and buffer sizes * \param[out] recv_socket_buff_size Returns the recv socket buffer size * \param[out] send_socket_buff_size Returns the send socket buffer size */ static sptr make(uhd::niusrprio::niusrprio_session::sptr fpga_session, const uint32_t instance, const link_params_t& params, const uhd::device_addr_t& hints); /*! * Get the physical adapter ID used for this link */ adapter_id_t get_send_adapter_id() const { return _adapter_id; } /*! * Get the physical adapter ID used for this link */ adapter_id_t get_recv_adapter_id() const { return _adapter_id; } /*! * Returns whether this link type supports releasing the frame buffers * in an order different from that in which they were acquired. */ bool supports_send_buff_out_of_order() const { return false; } /*! * Returns whether this link type supports releasing the frame buffers * in an order different from that in which they were acquired. */ bool supports_recv_buff_out_of_order() const { return false; } private: using recv_link_base_t = recv_link_base; using send_link_base_t = send_link_base; // Friend declarations to allow base classes to call private methods friend recv_link_base_t; friend send_link_base_t; nirio_link(uhd::niusrprio::niusrprio_session::sptr fpga_session, uint32_t instance, const link_params_t& params); /************************************************************************** * NI-RIO specific helpers *************************************************************************/ void _flush_rx_buff(); void _wait_until_stream_ready(); /************************************************************************** * recv_link/send_link API *************************************************************************/ // Methods called by recv_link_base UHD_FORCE_INLINE size_t get_recv_buff_derived(frame_buff& buff, int32_t timeout_ms) { using namespace uhd::niusrprio; nirio_status status = 0; size_t elems_acquired = 0; size_t elems_remaining = 0; // This will modify the data pointer in buff if successful: fifo_data_t** data_ptr = static_cast(buff).get_fifo_ptr_ref(); nirio_status_chain(_recv_fifo->acquire(*data_ptr, _link_params.recv_frame_size / sizeof(fifo_data_t), static_cast(timeout_ms * 1000), elems_acquired, elems_remaining), status); const size_t length = elems_acquired * sizeof(fifo_data_t); if (nirio_status_not_fatal(status)) { return length; } else if (status == NiRio_Status_CommunicationTimeout) { nirio_status_to_exception(status, "NI-RIO PCIe data transfer failed."); } return 0; // zero for timeout or error. } UHD_FORCE_INLINE void release_recv_buff_derived(frame_buff& /*buff*/) { _recv_fifo->release(_link_params.recv_frame_size / sizeof(fifo_data_t)); } // Methods called by send_link_base UHD_FORCE_INLINE bool get_send_buff_derived(frame_buff& buff, int32_t timeout_ms) { using namespace uhd::niusrprio; nirio_status status = 0; size_t elems_acquired = 0; size_t elems_remaining = 0; // This will modify the data pointer in buff if successful: fifo_data_t** data_ptr = static_cast(buff).get_fifo_ptr_ref(); nirio_status_chain(_send_fifo->acquire(*data_ptr, _link_params.send_frame_size / sizeof(fifo_data_t), static_cast(timeout_ms * 1000), elems_acquired, elems_remaining), status); // const size_t length = elems_acquired * sizeof(fifo_data_t); if (nirio_status_not_fatal(status)) { return true; } else if (status == NiRio_Status_CommunicationTimeout) { nirio_status_to_exception(status, "NI-RIO PCIe data transfer failed."); } return false; } UHD_FORCE_INLINE void release_send_buff_derived(frame_buff& /*buff*/) { _send_fifo->release(_link_params.send_frame_size / sizeof(fifo_data_t)); } /************************************************************************** * Private attributes *************************************************************************/ //! Reference to the NI-RIO session niusrprio::niusrprio_session::sptr _fpga_session; //! DMA channel index const uint32_t _fifo_instance; //! Recv and send FIFO objects uhd::niusrprio::nirio_fifo::sptr _recv_fifo, _send_fifo; const link_params_t _link_params; std::vector _recv_buffs; std::vector _send_buffs; adapter_id_t _adapter_id; }; }} // namespace uhd::transport #endif /* INCLUDED_UHD_TRANSPORT_NIRIO_LINK_HPP */