aboutsummaryrefslogtreecommitdiffstats
path: root/host/tests
diff options
context:
space:
mode:
Diffstat (limited to 'host/tests')
-rw-r--r--host/tests/CMakeLists.txt6
-rw-r--r--host/tests/offload_io_srv_test.cpp278
2 files changed, 284 insertions, 0 deletions
diff --git a/host/tests/CMakeLists.txt b/host/tests/CMakeLists.txt
index 8a477b181..89d0926fa 100644
--- a/host/tests/CMakeLists.txt
+++ b/host/tests/CMakeLists.txt
@@ -231,6 +231,12 @@ UHD_ADD_NONAPI_TEST(
${CMAKE_SOURCE_DIR}/lib/transport/inline_io_service.cpp
)
+UHD_ADD_NONAPI_TEST(
+ TARGET "offload_io_srv_test.cpp"
+ EXTRA_SOURCES
+ ${CMAKE_SOURCE_DIR}/lib/transport/offload_io_service.cpp
+)
+
########################################################################
# demo of a loadable module
########################################################################
diff --git a/host/tests/offload_io_srv_test.cpp b/host/tests/offload_io_srv_test.cpp
new file mode 100644
index 000000000..99bd6dd53
--- /dev/null
+++ b/host/tests/offload_io_srv_test.cpp
@@ -0,0 +1,278 @@
+//
+// Copyright 2019 Ettus Research, a National Instruments Brand
+//
+// SPDX-License-Identifier: GPL-3.0-or-later
+//
+
+#include "common/mock_link.hpp"
+#include <uhdlib/transport/offload_io_service.hpp>
+#include <boost/test/unit_test.hpp>
+#include <iostream>
+#include <atomic>
+
+using namespace uhd::transport;
+
+class mock_recv_io;
+constexpr size_t FRAME_SIZE = 1000;
+
+static mock_send_link::sptr make_send_link(size_t num_frames)
+{
+ const mock_send_link::link_params params = {FRAME_SIZE, num_frames};
+ return std::make_shared<mock_send_link>(params);
+}
+
+static mock_recv_link::sptr make_recv_link(size_t num_frames)
+{
+ const mock_recv_link::link_params params = {FRAME_SIZE, num_frames};
+ return std::make_shared<mock_recv_link>(params);
+}
+
+class mock_recv_io : public recv_io_if
+{
+public:
+ mock_recv_io(recv_link_if::sptr link) : _link(link) {}
+
+ frame_buff::uptr get_recv_buff(int32_t timeout_ms)
+ {
+ if (_frames_allocated > 0) {
+ _frames_allocated--;
+ return _link->get_recv_buff(timeout_ms);
+ }
+ return nullptr;
+ }
+
+ void release_recv_buff(frame_buff::uptr buff)
+ {
+ _link->release_recv_buff(std::move(buff));
+ }
+
+ size_t get_num_send_frames() const
+ {
+ return 0;
+ }
+
+ size_t get_num_recv_frames() const
+ {
+ return _link->get_num_recv_frames();
+ }
+
+ void allocate_frames(const size_t num_frames)
+ {
+ _frames_allocated += num_frames;
+ }
+
+private:
+ std::atomic<size_t> _frames_allocated{0};
+ recv_link_if::sptr _link;
+};
+
+class mock_send_io : public send_io_if
+{
+public:
+ mock_send_io(send_link_if::sptr link) : _link(link) {}
+
+ frame_buff::uptr get_send_buff(int32_t timeout_ms)
+ {
+ return _link->get_send_buff(timeout_ms);
+ }
+
+ void release_send_buff(frame_buff::uptr buff)
+ {
+ _link->release_send_buff(std::move(buff));
+ }
+
+ size_t get_num_send_frames() const
+ {
+ return _link->get_num_send_frames();
+ }
+
+ size_t get_num_recv_frames() const
+ {
+ return 0;
+ }
+
+private:
+ send_link_if::sptr _link;
+};
+
+class mock_io_service : public io_service
+{
+public:
+ void attach_recv_link(recv_link_if::sptr /*link*/) {}
+
+ void attach_send_link(send_link_if::sptr /*link*/) {}
+
+ send_io_if::sptr make_send_client(send_link_if::sptr send_link,
+ size_t /*num_send_frames*/,
+ send_io_if::send_callback_t /*cb*/,
+ recv_link_if::sptr /*recv_link*/,
+ size_t /*num_recv_frames*/,
+ recv_callback_t /*recv_cb*/)
+ {
+ return std::make_shared<mock_send_io>(send_link);
+ }
+
+ recv_io_if::sptr make_recv_client(recv_link_if::sptr recv_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*/)
+ {
+ auto io = std::make_shared<mock_recv_io>(recv_link);
+ _recv_io.push_back(io);
+ return io;
+ }
+
+ void allocate_recv_frames(const size_t client_idx, const size_t num_frames)
+ {
+ assert(client_idx < _recv_io.size());
+ _recv_io[client_idx]->allocate_frames(num_frames);
+ }
+
+private:
+ std::vector<std::shared_ptr<mock_recv_io>> _recv_io;
+};
+
+constexpr auto RECV_ONLY = offload_io_service::RECV_ONLY;
+constexpr auto SEND_ONLY = offload_io_service::SEND_ONLY;
+constexpr auto BOTH_SEND_AND_RECV = offload_io_service::BOTH_SEND_AND_RECV;
+
+constexpr auto POLL = offload_io_service::POLL;
+constexpr auto BLOCK = offload_io_service::BLOCK;
+using params_t = offload_io_service::params_t;
+
+std::vector<offload_io_service::wait_mode_t> wait_modes({POLL, BLOCK});
+
+BOOST_AUTO_TEST_CASE(test_construction)
+{
+ for (const auto wait_mode : wait_modes) {
+ params_t params {{}, SEND_ONLY, wait_mode};
+ auto mock_io_srv = std::make_shared<mock_io_service>();
+ auto io_srv = offload_io_service::make(mock_io_srv, params_t());
+ auto send_link = make_send_link(5);
+ io_srv->attach_send_link(send_link);
+ auto send_client =
+ io_srv->make_send_client(send_link, 5, nullptr, nullptr, 0, nullptr);
+ }
+ for (const auto wait_mode : wait_modes) {
+ params_t params {{}, RECV_ONLY, wait_mode};
+ auto mock_io_srv = std::make_shared<mock_io_service>();
+ auto io_srv = offload_io_service::make(mock_io_srv, params_t());
+ auto recv_link = make_recv_link(5);
+ io_srv->attach_recv_link(recv_link);
+ auto recv_client =
+ io_srv->make_recv_client(recv_link, 5, nullptr, nullptr, 0, nullptr);
+ }
+}
+
+BOOST_AUTO_TEST_CASE(test_construction_with_options)
+{
+ offload_io_service::params_t params;
+ params.cpu_affinity_list = {0};
+
+ auto mock_io_srv = std::make_shared<mock_io_service>();
+ auto io_srv = offload_io_service::make(mock_io_srv, params);
+ auto send_link = make_send_link(5);
+ io_srv->attach_send_link(send_link);
+ auto recv_link = make_recv_link(5);
+ io_srv->attach_recv_link(recv_link);
+ auto send_client =
+ io_srv->make_send_client(send_link, 5, nullptr, nullptr, 0, nullptr);
+ auto recv_client =
+ io_srv->make_recv_client(recv_link, 5, nullptr, nullptr, 0, nullptr);
+}
+
+BOOST_AUTO_TEST_CASE(test_send)
+{
+ for (const auto wait_mode : wait_modes) {
+ params_t params = {{}, SEND_ONLY, wait_mode};
+ auto mock_io_srv = std::make_shared<mock_io_service>();
+ auto io_srv = offload_io_service::make(mock_io_srv, params);
+ auto send_link = make_send_link(5);
+ io_srv->attach_send_link(send_link);
+ auto send_client =
+ io_srv->make_send_client(send_link, 1, nullptr, nullptr, 0, nullptr);
+
+ for (size_t i = 0; i < 10; i++) {
+ auto buff = send_client->get_send_buff(100);
+ BOOST_CHECK(buff != nullptr);
+ send_client->release_send_buff(std::move(buff));
+ }
+ send_client.reset();
+ }
+}
+
+BOOST_AUTO_TEST_CASE(test_recv)
+{
+ for (const auto wait_mode : wait_modes) {
+ params_t params = {{}, RECV_ONLY, wait_mode};
+ auto mock_io_srv = std::make_shared<mock_io_service>();
+ auto io_srv = offload_io_service::make(mock_io_srv, params);
+ auto recv_link = make_recv_link(5);
+ io_srv->attach_recv_link(recv_link);
+
+ auto recv_client =
+ io_srv->make_recv_client(recv_link, 1, nullptr, nullptr, 0, nullptr);
+
+ for (size_t i = 0; i < 10; i++) {
+ recv_link->push_back_recv_packet(
+ boost::shared_array<uint8_t>(new uint8_t[FRAME_SIZE]), FRAME_SIZE);
+ }
+ BOOST_CHECK(recv_client);
+ mock_io_srv->allocate_recv_frames(0, 10);
+
+ for (size_t i = 0; i < 10; i++) {
+ auto buff = recv_client->get_recv_buff(100);
+ BOOST_CHECK(buff != nullptr);
+ recv_client->release_recv_buff(std::move(buff));
+ }
+ recv_client.reset();
+ }
+}
+
+BOOST_AUTO_TEST_CASE(test_send_recv)
+{
+ auto mock_io_srv = std::make_shared<mock_io_service>();
+ auto io_srv = offload_io_service::make(mock_io_srv, params_t());
+ auto send_link = make_send_link(5);
+ io_srv->attach_send_link(send_link);
+ auto recv_link = make_recv_link(5);
+ io_srv->attach_recv_link(recv_link);
+
+ auto send_client =
+ io_srv->make_send_client(send_link, 1, nullptr, nullptr, 0, nullptr);
+ auto recv_client =
+ io_srv->make_recv_client(recv_link, 1, nullptr, nullptr, 0, nullptr);
+
+ for (size_t i = 0; i < 20; i++) {
+ recv_link->push_back_recv_packet(
+ boost::shared_array<uint8_t>(new uint8_t[FRAME_SIZE]), FRAME_SIZE);
+ }
+
+ for (size_t i = 0; i < 10; i++) {
+ send_client->release_send_buff(send_client->get_send_buff(100));
+ mock_io_srv->allocate_recv_frames(0, 1);
+ recv_client->release_recv_buff(recv_client->get_recv_buff(100));
+ }
+
+ auto recv_client2 =
+ io_srv->make_recv_client(recv_link, 1, nullptr, nullptr, 0, nullptr);
+ auto send_client2 =
+ io_srv->make_send_client(send_link, 1, nullptr, nullptr, 0, nullptr);
+ for (size_t i = 0; i < 5; i++) {
+ mock_io_srv->allocate_recv_frames(1, 1);
+ recv_client2->release_recv_buff(recv_client2->get_recv_buff(100));
+ send_client2->release_send_buff(send_client2->get_send_buff(100));
+ }
+ send_client2.reset();
+ recv_client2.reset();
+
+ for (size_t i = 0; i < 5; i++) {
+ mock_io_srv->allocate_recv_frames(0, 1);
+ recv_client->release_recv_buff(recv_client->get_recv_buff(100));
+ }
+
+ send_client.reset();
+ recv_client.reset();
+}