// // Copyright 2011 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 . // #include #include "../lib/transport/super_send_packet_handler.hpp" #include #include #include #include #include #define BOOST_CHECK_TS_CLOSE(a, b) \ BOOST_CHECK_CLOSE((a).get_real_secs(), (b).get_real_secs(), 0.001) /*********************************************************************** * A dummy managed send buffer for testing **********************************************************************/ class dummy_msb : public uhd::transport::managed_send_buffer{ public: void commit(size_t len){ if (len == 0) return; *_len = len; } sptr get_new(boost::shared_array mem, size_t *len){ _mem = mem; _len = len; return make_managed_buffer(this); } private: void *get_buff(void) const{return _mem.get();} size_t get_size(void) const{return *_len;} boost::shared_array _mem; size_t *_len; }; /*********************************************************************** * A dummy transport class to fill with fake data **********************************************************************/ class dummy_send_xport_class{ public: dummy_send_xport_class(const uhd::otw_type_t &otw_type){ _otw_type = otw_type; } void pop_front_packet( uhd::transport::vrt::if_packet_info_t &ifpi ){ ifpi.num_packet_words32 = _lens.front()/sizeof(boost::uint32_t); if (_otw_type.byteorder == uhd::otw_type_t::BO_BIG_ENDIAN){ uhd::transport::vrt::if_hdr_unpack_be(reinterpret_cast(_mems.front().get()), ifpi); } if (_otw_type.byteorder == uhd::otw_type_t::BO_LITTLE_ENDIAN){ uhd::transport::vrt::if_hdr_unpack_le(reinterpret_cast(_mems.front().get()), ifpi); } _mems.pop_front(); _lens.pop_front(); } uhd::transport::managed_send_buffer::sptr get_send_buff(double){ _msbs.push_back(dummy_msb()); _mems.push_back(boost::shared_array(new char[1000])); _lens.push_back(1000); uhd::transport::managed_send_buffer::sptr mrb = _msbs.back().get_new(_mems.back(), &_lens.back()); return mrb; } private: std::list > _mems; std::list _lens; std::list _msbs; //list means no-realloc uhd::otw_type_t _otw_type; }; //////////////////////////////////////////////////////////////////////// BOOST_AUTO_TEST_CASE(test_sph_send_one_channel_one_packet_mode){ //////////////////////////////////////////////////////////////////////// uhd::otw_type_t otw_type; otw_type.width = 16; otw_type.shift = 0; otw_type.byteorder = uhd::otw_type_t::BO_BIG_ENDIAN; dummy_send_xport_class dummy_send_xport(otw_type); static const double TICK_RATE = 100e6; static const double SAMP_RATE = 10e6; static const size_t NUM_PKTS_TO_TEST = 30; //create the super send packet handler uhd::transport::sph::send_packet_handler handler(1); handler.set_vrt_packer(&uhd::transport::vrt::if_hdr_pack_be); handler.set_tick_rate(TICK_RATE); handler.set_samp_rate(SAMP_RATE); handler.set_xport_chan_get_buff(0, boost::bind(&dummy_send_xport_class::get_send_buff, &dummy_send_xport, _1)); handler.set_converter(otw_type); handler.set_max_samples_per_packet(20); //allocate metadata and buffer std::vector > buff(20); uhd::tx_metadata_t metadata; metadata.has_time_spec = true; metadata.time_spec = uhd::time_spec_t(0.0); //generate the test data for (size_t i = 0; i < NUM_PKTS_TO_TEST; i++){ metadata.start_of_burst = (i == 0); metadata.end_of_burst = (i == NUM_PKTS_TO_TEST-1); const size_t num_sent = handler.send( &buff.front(), 10 + i%10, metadata, uhd::io_type_t::COMPLEX_FLOAT32, uhd::device::SEND_MODE_ONE_PACKET, 1.0 ); BOOST_CHECK_EQUAL(num_sent, 10 + i%10); metadata.time_spec += uhd::time_spec_t(0, num_sent, SAMP_RATE); } //check the sent packets size_t num_accum_samps = 0; uhd::transport::vrt::if_packet_info_t ifpi; for (size_t i = 0; i < NUM_PKTS_TO_TEST; i++){ std::cout << "data check " << i << std::endl; dummy_send_xport.pop_front_packet(ifpi); BOOST_CHECK_EQUAL(ifpi.num_payload_words32, 10+i%10); BOOST_CHECK(ifpi.has_tsi); BOOST_CHECK(ifpi.has_tsf); BOOST_CHECK_EQUAL(ifpi.tsi, 0); BOOST_CHECK_EQUAL(ifpi.tsf, num_accum_samps*TICK_RATE/SAMP_RATE); BOOST_CHECK_EQUAL(ifpi.sob, i == 0); BOOST_CHECK_EQUAL(ifpi.eob, i == NUM_PKTS_TO_TEST-1); num_accum_samps += ifpi.num_payload_words32; } } //////////////////////////////////////////////////////////////////////// BOOST_AUTO_TEST_CASE(test_sph_send_one_channel_full_buffer_mode){ //////////////////////////////////////////////////////////////////////// uhd::otw_type_t otw_type; otw_type.width = 16; otw_type.shift = 0; otw_type.byteorder = uhd::otw_type_t::BO_BIG_ENDIAN; dummy_send_xport_class dummy_send_xport(otw_type); static const double TICK_RATE = 100e6; static const double SAMP_RATE = 10e6; static const size_t NUM_PKTS_TO_TEST = 30; //create the super send packet handler uhd::transport::sph::send_packet_handler handler(1); handler.set_vrt_packer(&uhd::transport::vrt::if_hdr_pack_be); handler.set_tick_rate(TICK_RATE); handler.set_samp_rate(SAMP_RATE); handler.set_xport_chan_get_buff(0, boost::bind(&dummy_send_xport_class::get_send_buff, &dummy_send_xport, _1)); handler.set_converter(otw_type); handler.set_max_samples_per_packet(20); //allocate metadata and buffer std::vector > buff(20*NUM_PKTS_TO_TEST); uhd::tx_metadata_t metadata; metadata.start_of_burst = true; metadata.end_of_burst = true; metadata.has_time_spec = true; metadata.time_spec = uhd::time_spec_t(0.0); //generate the test data const size_t num_sent = handler.send( &buff.front(), buff.size(), metadata, uhd::io_type_t::COMPLEX_FLOAT32, uhd::device::SEND_MODE_FULL_BUFF, 1.0 ); BOOST_CHECK_EQUAL(num_sent, buff.size()); //check the sent packets size_t num_accum_samps = 0; uhd::transport::vrt::if_packet_info_t ifpi; for (size_t i = 0; i < NUM_PKTS_TO_TEST; i++){ std::cout << "data check " << i << std::endl; dummy_send_xport.pop_front_packet(ifpi); BOOST_CHECK_EQUAL(ifpi.num_payload_words32, 20); BOOST_CHECK(ifpi.has_tsi); BOOST_CHECK(ifpi.has_tsf); BOOST_CHECK_EQUAL(ifpi.tsi, 0); BOOST_CHECK_EQUAL(ifpi.tsf, num_accum_samps*TICK_RATE/SAMP_RATE); BOOST_CHECK_EQUAL(ifpi.sob, i == 0); BOOST_CHECK_EQUAL(ifpi.eob, i == NUM_PKTS_TO_TEST-1); num_accum_samps += ifpi.num_payload_words32; } }