From 1a6368331bf441290b0d08ac233c9e5050021493 Mon Sep 17 00:00:00 2001 From: Ciro Nishiguchi Date: Tue, 27 Aug 2019 16:20:25 -0500 Subject: tests: Add benchmark of streamer and chdr xports --- host/tests/CMakeLists.txt | 34 ++- host/tests/streamer_benchmark.cpp | 507 ++++++++++++++++++++++++++++++++++++++ 2 files changed, 534 insertions(+), 7 deletions(-) create mode 100644 host/tests/streamer_benchmark.cpp diff --git a/host/tests/CMakeLists.txt b/host/tests/CMakeLists.txt index ea069410b..d790b214e 100644 --- a/host/tests/CMakeLists.txt +++ b/host/tests/CMakeLists.txt @@ -107,7 +107,6 @@ foreach(benchmark_source ${benchmark_sources}) UHD_INSTALL(TARGETS ${benchmark_name} RUNTIME DESTINATION ${PKG_LIB_DIR}/tests COMPONENT tests) endforeach(benchmark_source) - ############################################################################### # Add a unit test that requires linkage to internal parts of UHD which are not # API @@ -165,15 +164,36 @@ UHD_ADD_NONAPI_TEST( ) UHD_ADD_NONAPI_TEST( - TARGET "nocscript_ftable_test.cpp" + TARGET "streamer_benchmark.cpp" EXTRA_SOURCES - ${CMAKE_SOURCE_DIR}/lib/rfnoc/nocscript/function_table.cpp - ${CMAKE_SOURCE_DIR}/lib/rfnoc/nocscript/expression.cpp - INCLUDE_DIRS - ${CMAKE_BINARY_DIR}/lib/rfnoc/nocscript/ - ${CMAKE_SOURCE_DIR}/lib/rfnoc/nocscript/ + ${CMAKE_SOURCE_DIR}/lib/rfnoc/chdr_packet.cpp + ${CMAKE_SOURCE_DIR}/lib/rfnoc/chdr_types.cpp + ${CMAKE_SOURCE_DIR}/lib/rfnoc/chdr_ctrl_xport.cpp + ${CMAKE_SOURCE_DIR}/lib/rfnoc/chdr_rx_data_xport.cpp + ${CMAKE_SOURCE_DIR}/lib/rfnoc/chdr_tx_data_xport.cpp + ${CMAKE_SOURCE_DIR}/lib/transport/inline_io_service.cpp + NOAUTORUN # Don't register for auto-run ) +#UHD_ADD_NONAPI_TEST( + #TARGET "nocscript_expr_test.cpp" + #EXTRA_SOURCES + #"${CMAKE_SOURCE_DIR}/lib/rfnoc/nocscript/expression.cpp" + #INCLUDE_DIRS + #${CMAKE_BINARY_DIR}/lib/rfnoc/nocscript/ + #${CMAKE_SOURCE_DIR}/lib/rfnoc/nocscript/ +#) + +#UHD_ADD_NONAPI_TEST( + #TARGET "nocscript_ftable_test.cpp" + #EXTRA_SOURCES + #${CMAKE_SOURCE_DIR}/lib/rfnoc/nocscript/function_table.cpp + #${CMAKE_SOURCE_DIR}/lib/rfnoc/nocscript/expression.cpp + #INCLUDE_DIRS + #${CMAKE_BINARY_DIR}/lib/rfnoc/nocscript/ + #${CMAKE_SOURCE_DIR}/lib/rfnoc/nocscript/ +#) + UHD_ADD_NONAPI_TEST( TARGET "nocscript_parser_test.cpp" EXTRA_SOURCES diff --git a/host/tests/streamer_benchmark.cpp b/host/tests/streamer_benchmark.cpp new file mode 100644 index 000000000..46ba6add0 --- /dev/null +++ b/host/tests/streamer_benchmark.cpp @@ -0,0 +1,507 @@ +// +// Copyright 2019 Ettus Research, a National Instruments Brand +// +// SPDX-License-Identifier: GPL-3.0-or-later +// + +#include "../common/mock_link.hpp" +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +namespace po = boost::program_options; +using namespace uhd; +using namespace uhd::rfnoc; +using namespace uhd::transport; + +static const double TICK_RATE = 100e6; +static const double SAMP_RATE = 10e6; +static const size_t FRAME_SIZE = 1000; +static const double SCALE_FACTOR = 2; + +/*! + * Mock rx data xport which doesn't do anything, doesn't use I/O services or + * links, to benchmark rx_streamer_impl with minimal other overhead. + */ +class mock_rx_data_xport +{ +public: + using uptr = std::unique_ptr; + + struct buff_t + { + using uptr = std::unique_ptr; + std::vector data; + }; + + struct packet_info_t + { + bool eob = false; + bool has_tsf = false; + uint64_t tsf = 0; + size_t payload_bytes = 0; + const void* payload = nullptr; + }; + + mock_rx_data_xport(const size_t buff_size) : _buff_size(buff_size) + { + _buff = std::make_unique(); + _buff->data.resize(buff_size); + + _packet_info.has_tsf = true; + _packet_info.tsf = 1000; + _packet_info.payload_bytes = buff_size; + _packet_info.payload = _buff->data.data(); + } + + std::tuple get_recv_buff( + const int32_t /*timeout_ms*/) + { + const bool seq_error = false; + + return std::make_tuple(std::move(_buff), _packet_info, seq_error); + } + + void release_recv_buff(buff_t::uptr buff) + { + _buff = std::move(buff); + } + + size_t get_max_payload_size() const + { + return _buff_size; + } + +private: + size_t _buff_size; + buff_t::uptr _buff; + packet_info_t _packet_info; +}; + +/*! + * Mock tx data xport which doesn't do anything, doesn't use I/O services or + * links, to benchmark tx_streamer_impl with minimal other overhead. + */ +class mock_tx_data_xport +{ +public: + using uptr = std::unique_ptr; + + struct buff_t + { + using uptr = std::unique_ptr; + std::vector data; + + void set_packet_size(const size_t) {} + }; + + struct packet_info_t + { + bool eob = false; + bool has_tsf = false; + uint64_t tsf = 0; + size_t payload_bytes = 0; + }; + + mock_tx_data_xport(const size_t buff_size) : _buff_size(buff_size) + { + _buff = std::make_unique(); + _buff->data.resize(buff_size); + } + + buff_t::uptr get_send_buff(const int32_t /*timeout_ms*/) + { + return std::move(_buff); + } + + std::pair write_packet_header( + buff_t::uptr& buff, const packet_info_t& info) + { + uint8_t* data = buff->data.data(); + *(reinterpret_cast(data)) = info; + return std::make_pair(data + sizeof(info), sizeof(info) + info.payload_bytes); + } + + void release_send_buff(buff_t::uptr buff) + { + _buff = std::move(buff); + } + + size_t get_max_payload_size() const + { + return _buff_size - sizeof(packet_info_t); + } + +private: + size_t _buff_size; + buff_t::uptr _buff; +}; + +/*! + * Mock rx streamer, configured to ignore sequence errors + */ +template +class mock_rx_streamer : public rx_streamer_impl +{ +public: + mock_rx_streamer(const size_t num_chans, const uhd::stream_args_t& stream_args) + : rx_streamer_impl(num_chans, stream_args) + { + } + + void set_tick_rate(double rate) + { + rx_streamer_impl::set_tick_rate(rate); + } + + void set_samp_rate(double rate) + { + rx_streamer_impl::set_samp_rate(rate); + } + + void set_scale_factor(const size_t chan, const double scale_factor) + { + rx_streamer_impl::set_scale_factor(chan, scale_factor); + } + + void issue_stream_cmd(const stream_cmd_t& /*stream_cmd*/) {} +}; + +/*! + * Mock tx streamer + */ +template +class mock_tx_streamer : public tx_streamer_impl +{ +public: + mock_tx_streamer(const size_t num_chans, const uhd::stream_args_t& stream_args) + : tx_streamer_impl(num_chans, stream_args) + { + } + + void set_tick_rate(double rate) + { + tx_streamer_impl::set_tick_rate(rate); + } + + void set_samp_rate(double rate) + { + tx_streamer_impl::set_samp_rate(rate); + } + + void set_scale_factor(const size_t chan, const double scale_factor) + { + tx_streamer_impl::set_scale_factor(chan, scale_factor); + } + + bool recv_async_msg( + uhd::async_metadata_t& /*async_metadata*/, double /*timeout = 0.1*/) + { + return false; + } +}; + +using rx_streamer_mock_xport = mock_rx_streamer; +using tx_streamer_mock_xport = mock_tx_streamer; +using rx_streamer_mock_link = mock_rx_streamer; +using tx_streamer_mock_link = mock_tx_streamer; + +/*! + * Helper functions + */ +static boost::shared_ptr make_rx_streamer_mock_xport( + const size_t spp, const std::string& format) +{ + const uhd::stream_args_t stream_args(format, "sc16"); + auto streamer = boost::make_shared(1, stream_args); + streamer->set_tick_rate(TICK_RATE); + streamer->set_samp_rate(SAMP_RATE); + streamer->set_scale_factor(0, SCALE_FACTOR); + + const size_t bpi = convert::get_bytes_per_item(format); + const size_t frame_size = bpi * spp; + + auto xport = std::make_unique(frame_size); + + streamer->connect_channel(0, std::move(xport)); + + return streamer; +} + +static boost::shared_ptr make_tx_streamer_mock_xport( + const size_t spp, const std::string& format) +{ + const uhd::stream_args_t stream_args(format, "sc16"); + auto streamer = boost::make_shared(1, stream_args); + streamer->set_tick_rate(TICK_RATE); + streamer->set_samp_rate(SAMP_RATE); + streamer->set_scale_factor(0, SCALE_FACTOR); + + const size_t bpi = convert::get_bytes_per_item(format); + const size_t frame_size = bpi * spp + sizeof(mock_tx_data_xport::packet_info_t); + + auto xport = std::make_unique(frame_size); + + streamer->connect_channel(0, std::move(xport)); + + return streamer; +} + +static boost::shared_ptr make_rx_streamer_mock_link( + const size_t spp, const std::string& format) +{ + const uhd::stream_args_t stream_args(format, "sc16"); + auto streamer = boost::make_shared(1, stream_args); + streamer->set_tick_rate(TICK_RATE); + streamer->set_samp_rate(SAMP_RATE); + streamer->set_scale_factor(0, SCALE_FACTOR); + + const chdr::chdr_packet_factory pkt_factory(CHDR_W_64, ENDIANNESS_BIG); + const sep_id_pair_t epids = {0, 1}; + const stream_buff_params_t buff_capacity = {UINT64_MAX, UINT32_MAX}; + const stream_buff_params_t fc_freq = {UINT64_MAX, UINT32_MAX}; + const chdr_rx_data_xport::fc_params_t fc_params{buff_capacity, fc_freq}; + + const size_t bpi = convert::get_bytes_per_item(format); + const size_t frame_size = bpi * spp + 16; + + const mock_recv_link::link_params recv_params = {frame_size, 1}; + const mock_send_link::link_params send_params = {frame_size, 1}; + + auto recv_link = std::make_shared(recv_params, true); + auto send_link = std::make_shared(send_params, true); + + boost::shared_array recv_frame(new uint8_t[frame_size]); + auto pkt = pkt_factory.make_generic(); + chdr::chdr_header header; + header.set_pkt_type(chdr::PKT_TYPE_DATA_WITH_TS); + header.set_length(frame_size); + header.set_dst_epid(epids.second); + pkt->refresh(recv_frame.get(), header, 1000 /*tsf*/); + + recv_link->push_back_recv_packet(recv_frame, frame_size); + + auto io_srv = inline_io_service::make(); + io_srv->attach_recv_link(recv_link); + io_srv->attach_send_link(send_link); + + auto xport = std::make_unique(io_srv, + recv_link, + send_link, + pkt_factory, + epids, + send_link->get_num_send_frames(), + fc_params); + + streamer->connect_channel(0, std::move(xport)); + return streamer; +} + +static boost::shared_ptr make_tx_streamer_mock_link( + const size_t spp, const std::string& format) +{ + const uhd::stream_args_t stream_args(format, "sc16"); + auto streamer = boost::make_shared(1, stream_args); + streamer->set_tick_rate(TICK_RATE); + streamer->set_samp_rate(SAMP_RATE); + streamer->set_scale_factor(0, SCALE_FACTOR); + + const chdr::chdr_packet_factory pkt_factory(CHDR_W_64, ENDIANNESS_BIG); + const sep_id_pair_t epids = {0, 1}; + const stream_buff_params_t buff_capacity = {UINT64_MAX, UINT32_MAX}; + const chdr_tx_data_xport::fc_params_t fc_params{buff_capacity}; + + const size_t bpi = convert::get_bytes_per_item(format); + const size_t frame_size = bpi * spp + 16; + + const mock_recv_link::link_params recv_params = {frame_size, 1}; + const mock_send_link::link_params send_params = {frame_size, 1}; + + auto recv_link = std::make_shared(recv_params, true); + auto send_link = std::make_shared(send_params, true); + + auto io_srv = inline_io_service::make(); + io_srv->attach_recv_link(recv_link); + io_srv->attach_send_link(send_link); + + auto xport = std::make_unique(io_srv, + recv_link, + send_link, + pkt_factory, + epids, + send_link->get_num_send_frames(), + fc_params); + + streamer->connect_channel(0, std::move(xport)); + return streamer; +} + +/*! + * Benchmark of rx streamer + */ +void benchmark_rx_streamer( + rx_streamer::sptr streamer, const size_t spp, const std::string& format) +{ + // Allocate buffer + const size_t bpi = convert::get_bytes_per_item(format); + std::vector buffer(spp * bpi); + std::vector buffers; + buffers.push_back(buffer.data()); + + // Run benchmark + uhd::rx_metadata_t md; + + const auto start_time = std::chrono::steady_clock::now(); + const size_t iterations = 1e7; + + for (size_t i = 0; i < iterations; i++) { + streamer->recv(buffers, spp, md, 1.0, true); + } + + const auto end_time = std::chrono::steady_clock::now(); + const std::chrono::duration elapsed_time(end_time - start_time); + const double time_per_packet = elapsed_time.count() / iterations; + + std::cout << format << ": " << time_per_packet / spp * 1e9 << " ns/sample, " + << time_per_packet * 1e9 << " ns/packet\n"; +} + +/*! + * Benchmark of tx streamer + */ +void benchmark_tx_streamer(tx_streamer::sptr streamer, + const size_t spp, + const std::string& format, + bool use_time_spec) +{ + // Allocate buffer + const size_t bpi = convert::get_bytes_per_item(format); + std::vector buffer(spp * bpi); + std::vector buffers; + buffers.push_back(buffer.data()); + + // Run benchmark + uhd::tx_metadata_t md; + md.has_time_spec = use_time_spec; + + const auto start_time = std::chrono::steady_clock::now(); + const size_t iterations = 1e7; + + for (size_t i = 0; i < iterations; i++) { + if (use_time_spec) { + md.time_spec = uhd::time_spec_t(i, 0.0); + } + streamer->send(buffers, spp, md, 1.0); + } + + const auto end_time = std::chrono::steady_clock::now(); + const std::chrono::duration elapsed_time(end_time - start_time); + const double time_per_packet = elapsed_time.count() / iterations; + + std::cout << format << ": " << time_per_packet / spp * 1e9 << " ns/sample, " + << time_per_packet * 1e9 << " ns/packet\n"; +} + +int UHD_SAFE_MAIN(int argc, char* argv[]) +{ + po::options_description desc("Allowed options"); + desc.add_options()("help", "help message"); + + po::variables_map vm; + po::store(po::parse_command_line(argc, argv, desc), vm); + po::notify(vm); + + // Print the help message + if (vm.count("help")) { + std::cout << boost::format("UHD Streamer Benchmark %s") % desc << std::endl; + std::cout << " Benchmark of send and receive streamer functions\n" + " All benchmarks use mock transport objects. No\n" + " parameters are needed to run this benchmark.\n" + << std::endl; + return EXIT_FAILURE; + } + + const char* formats[] = {"sc16", "fc32", "fc64"}; + constexpr size_t spp = 1000; + std::cout << "spp: " << spp << "\n"; + + std::cout << "----------------------------------------------------------\n"; + std::cout << "Benchmark of recv with mock transport \n"; + std::cout << " \n"; + std::cout << " Measures time spent in the rx streamer only. \n"; + std::cout << "----------------------------------------------------------\n"; + + std::cout << "*** with timespec ***\n"; + for (size_t i = 0; i < std::extent::value; i++) { + auto streamer = make_rx_streamer_mock_xport(spp, formats[i]); + benchmark_rx_streamer(streamer, spp, formats[i]); + } + + std::cout << "\n"; + std::cout << "----------------------------------------------------------\n"; + std::cout << "Benchmark of send with mock transport \n"; + std::cout << " \n"; + std::cout << " Measures time time spent in the tx streamer only. \n"; + std::cout << "----------------------------------------------------------\n"; + + std::cout << "*** without timespec ***\n"; + for (size_t i = 0; i < std::extent::value; i++) { + auto streamer = make_tx_streamer_mock_xport(spp, formats[i]); + benchmark_tx_streamer(streamer, spp, formats[i], false); + } + std::cout << "\n"; + + std::cout << "*** with timespec ***\n"; + for (size_t i = 0; i < std::extent::value; i++) { + auto streamer = make_tx_streamer_mock_xport(spp, formats[i]); + benchmark_tx_streamer(streamer, spp, formats[i], true); + } + std::cout << "\n"; + + std::cout << "----------------------------------------------------------\n"; + std::cout << "Benchmark of recv with mock link \n"; + std::cout << " \n"; + std::cout << " Measures time time spent in the rx streamer, I/O \n"; + std::cout << " service, and chdr data xport. \n"; + std::cout << "----------------------------------------------------------\n"; + + std::cout << "*** with timespec ***\n"; + for (size_t i = 0; i < std::extent::value; i++) { + auto streamer = make_rx_streamer_mock_link(spp, formats[i]); + benchmark_rx_streamer(streamer, spp, formats[i]); + } + std::cout << "\n"; + + std::cout << "----------------------------------------------------------\n"; + std::cout << "Benchmark of send with mock link \n"; + std::cout << " \n"; + std::cout << " Measures time time spent in the tx streamer, I/O \n"; + std::cout << " service, and chdr data xport. \n"; + std::cout << "----------------------------------------------------------\n"; + + std::cout << "*** without timespec ***\n"; + for (size_t i = 0; i < std::extent::value; i++) { + auto streamer = make_tx_streamer_mock_link(spp, formats[i]); + benchmark_tx_streamer(streamer, spp, formats[i], false); + } + std::cout << "\n"; + + std::cout << "*** with timespec ***\n"; + for (size_t i = 0; i < std::extent::value; i++) { + auto streamer = make_tx_streamer_mock_link(spp, formats[i]); + benchmark_tx_streamer(streamer, spp, formats[i], true); + } + std::cout << "\n"; + + return EXIT_SUCCESS; +} -- cgit v1.2.3