// // Copyright 2018 Ettus Research, a National Instruments Company // // SPDX-License-Identifier: GPL-3.0-or-later // #ifndef INCLUDED_MOCK_XPORT_HPP #define INCLUDED_MOCK_XPORT_HPP #include #include #include #include #include #include #include #include #include #include #include #include #include /*********************************************************************** * Transport mockups **********************************************************************/ /*! A single transport class that implements send() and recv() * * Tx and Rx are separate. We can access the other end of the FIFOs from * this class. */ static constexpr size_t SEND_BUFF_SIZE = 1024; static constexpr size_t RECV_BUFF_SIZE = 1024; /*********************************************************************** * Dummy managed buffers for testing **********************************************************************/ class mock_msb : public uhd::transport::managed_send_buffer { public: void release(void) { /* nop */ } sptr get_new(boost::shared_array mem, size_t* len) { _mem = mem; return make(this, mem.get(), *len); } private: boost::shared_array _mem; }; class mock_mrb : public uhd::transport::managed_recv_buffer { public: void release(void) { /* nop */ } sptr get_new(boost::shared_array mem, size_t len) { _mem = mem; return make(this, _mem.get(), len); } private: boost::shared_array _mem; }; class mock_zero_copy : public uhd::transport::zero_copy_if { public: typedef boost::shared_ptr sptr; mock_zero_copy(uhd::transport::vrt::if_packet_info_t::link_type_t type); uhd::transport::managed_recv_buffer::sptr get_recv_buff(double); uhd::transport::managed_send_buffer::sptr get_send_buff(double); size_t get_num_recv_frames(void) const { return 1; } size_t get_num_send_frames(void) const { return 1; } size_t get_recv_frame_size(void) const { return RECV_BUFF_SIZE; } size_t get_send_frame_size(void) const { return SEND_BUFF_SIZE; } void set_simulate_io_error(bool status) { _simulate_io_error = status; } template void push_back_recv_packet( uhd::transport::vrt::if_packet_info_t& ifpi, const std::vector& otw_data ); template void push_back_inline_message_packet( uhd::transport::vrt::if_packet_info_t& ifpi, const uint32_t message ); template void pop_send_packet( uhd::transport::vrt::if_packet_info_t &ifpi ); private: std::list> _tx_mems; std::list _tx_lens; std::list> _rx_mems; std::list _rx_lens; std::vector> _msbs; std::vector> _mrbs; uhd::transport::vrt::if_packet_info_t::link_type_t _link_type; bool _simulate_io_error = false; }; template void mock_zero_copy::push_back_recv_packet( uhd::transport::vrt::if_packet_info_t& ifpi, const std::vector& otw_data ) { using namespace uhd::transport; UHD_ASSERT_THROW( ifpi.num_payload_words32 * sizeof(uint32_t) == otw_data.size() * sizeof(T)); const size_t max_hdr_len = _link_type == vrt::if_packet_info_t::LINK_TYPE_CHDR ? vrt::chdr::max_if_hdr_words64*sizeof(uint64_t) : (vrt::max_if_hdr_words32 + 1/*tlr*/)*sizeof(uint32_t); const size_t max_pkt_len = ifpi.num_payload_words32*sizeof(uint32_t)+max_hdr_len; UHD_ASSERT_THROW(max_pkt_len <= RECV_BUFF_SIZE); // Create recv buffer _rx_mems.push_back(boost::shared_array(new uint8_t[max_pkt_len])); uint32_t* rx_buff_ptr = reinterpret_cast(_rx_mems.back().get()); // Copy header if (endianness == uhd::ENDIANNESS_BIG) { if (_link_type == vrt::if_packet_info_t::LINK_TYPE_CHDR) { uhd::transport::vrt::chdr::if_hdr_pack_be(rx_buff_ptr, ifpi); } else { uhd::transport::vrt::if_hdr_pack_be(rx_buff_ptr, ifpi); } } else { if (_link_type == vrt::if_packet_info_t::LINK_TYPE_CHDR) { uhd::transport::vrt::chdr::if_hdr_pack_le(rx_buff_ptr, ifpi); } else { uhd::transport::vrt::if_hdr_pack_le(rx_buff_ptr, ifpi); } } // Copy data uint32_t* data_ptr = (rx_buff_ptr + ifpi.num_header_words32); std::copy(otw_data.begin(), otw_data.end(), reinterpret_cast(data_ptr)); _rx_lens.push_back(ifpi.num_packet_words32*sizeof(uint32_t)); } template void mock_zero_copy::push_back_inline_message_packet( uhd::transport::vrt::if_packet_info_t& ifpi, const uint32_t message ) { const std::vector data { message | uhd::byteswap(message) }; push_back_recv_packet(ifpi, data); } template void mock_zero_copy::pop_send_packet( uhd::transport::vrt::if_packet_info_t &ifpi ) { using namespace uhd::transport; ifpi.num_packet_words32 = _tx_lens.front()/sizeof(uint32_t); uint32_t* tx_buff_ptr = reinterpret_cast(_tx_mems.front().get()); if (endianness == uhd::ENDIANNESS_BIG) { if (_link_type == vrt::if_packet_info_t::LINK_TYPE_CHDR) { uhd::transport::vrt::chdr::if_hdr_unpack_be(tx_buff_ptr, ifpi); } else { uhd::transport::vrt::if_hdr_unpack_be(tx_buff_ptr, ifpi); } } else { if (_link_type == vrt::if_packet_info_t::LINK_TYPE_CHDR) { uhd::transport::vrt::chdr::if_hdr_unpack_le(tx_buff_ptr, ifpi); } else { uhd::transport::vrt::if_hdr_unpack_le(tx_buff_ptr, ifpi); } } _tx_mems.pop_front(); _tx_lens.pop_front(); } #endif /*INCLUDED_MOCK_XPORT_HPP*/