// // Copyright 2019 Ettus Research, a National Instruments brand // // SPDX-License-Identifier: GPL-3.0-or-later // #ifndef INCLUDED_UHDLIB_TRANSPORT_INLINE_IO_SERVICE_HPP #define INCLUDED_UHDLIB_TRANSPORT_INLINE_IO_SERVICE_HPP #include #include #include namespace uhd { namespace transport { class inline_recv_mux; class inline_recv_cb; /*! * Single-threaded I/O service * Note this is not an appropriate io_service to use with polling-mode drivers, * since such drivers require a thread to poll them and not block (i.e. * timeouts are not allowed at the link interface) */ class inline_io_service : public virtual io_service, public std::enable_shared_from_this { public: using sptr = std::shared_ptr; static sptr make(void) { return sptr(new inline_io_service()); } ~inline_io_service(); void attach_recv_link(recv_link_if::sptr link); void attach_send_link(send_link_if::sptr link); recv_io_if::sptr make_recv_client(recv_link_if::sptr data_link, size_t num_recv_frames, recv_callback_t cb, send_link_if::sptr fc_link, size_t num_send_frames, recv_io_if::fc_callback_t fc_cb); send_io_if::sptr make_send_client(send_link_if::sptr send_link, size_t num_send_frames, send_io_if::send_callback_t send_cb, recv_link_if::sptr recv_link, size_t num_recv_frames, recv_callback_t recv_cb); private: friend class inline_recv_io; friend class inline_send_io; inline_io_service() = default; inline_io_service(const inline_io_service&) = delete; /*! * Senders are free to mux a send_link, but the total reserved send_frames * must be less than or equal to the link's capacity * * \param link the link used for sending data * \param num_frames number of frames to reserve for this connection */ void connect_sender(send_link_if* link, size_t num_frames); /*! * Disconnect the sender and free resources * * \param link the link that was used for sending data * \param num_frames number of frames to release (same as reservation) */ void disconnect_sender(send_link_if* link, size_t num_frames); /*! * Connect a receiver to the link and reserve resources * \param link the recv link to use for getting data * \param cb a callback for processing received data * \param num_frames the number of frames to reserve for this receiver */ void connect_receiver(recv_link_if* link, inline_recv_cb* cb, size_t num_frames); /*! * Disconnect the receiver from the provided link and free resources * \param link the recv link that was used for reception * \param cb the callback to disassociate * \param num_frames the number of frames that was reserved for the cb */ void disconnect_receiver(recv_link_if* link, inline_recv_cb* cb, size_t num_frames); /* * Function to perform recv operations on a link, which is potentially * muxed. Packets are forwarded to the appropriate mux or callback. * * \param recv_io_cb the callback+interface initiating the operation * \param recv_link link to perform receive on * \param timeout_ms timeout to wait for a buffer on the link * \return a frame_buff uptr with either a buffer with data or no buffer */ frame_buff::uptr recv( inline_recv_cb* recv_io_cb, recv_link_if* recv_link, int32_t timeout_ms); /* Track whether link is muxed, the callback, and buffer reservations */ std::unordered_map> _recv_tbl; /* Track how many send_frames have been reserved for each link */ std::unordered_map _send_tbl; /* Shared ptr kept to avoid untimely release */ std::list _send_links; std::list _recv_links; }; }} // namespace uhd::transport #endif /* INCLUDED_UHDLIB_TRANSPORT_INLINE_IO_SERVICE_HPP */