blob: c98462df335c2f20e42268f7b48da4d24e4be1b7 [file] [log] [blame]
#include <poll.h>
#include <array>
#include <chrono>
#include <optional>
#include <stdplus/io_uring.hpp>
#include <string_view>
#include <gmock/gmock.h>
#include <gtest/gtest.h>
namespace stdplus
{
using testing::_;
TEST(Convert, ChronoToKTS)
{
const auto ns = 700;
const auto s = 5;
const auto kts =
chronoToKTS(std::chrono::nanoseconds(ns) + std::chrono::seconds(s));
EXPECT_EQ(kts.tv_sec, s);
EXPECT_EQ(kts.tv_nsec, ns);
}
class MockHandler : public IoUring::CQEHandler
{
public:
MOCK_METHOD(void, handleCQE, (io_uring_cqe&), (noexcept, override));
};
class IoUringTest : public testing::Test
{
protected:
static void SetUpTestCase()
{
io_uring r;
if (io_uring_queue_init(1, &r, 0) == -ENOSYS)
{
// Not supported, skip running this test
exit(77);
}
io_uring_queue_exit(&r);
}
std::array<testing::StrictMock<MockHandler>, 2> h;
IoUring ring;
void testFdWrite(int fd, int flags,
std::optional<int> expected_res = std::nullopt)
{
auto& sqe = ring.getSQE();
std::string_view data = "Test\n";
io_uring_prep_write(&sqe, fd, data.data(), data.size(), 0);
sqe.flags |= flags;
ring.setHandler(sqe, &h[0]);
ring.submit();
ring.wait();
EXPECT_CALL(h[0], handleCQE(_)).WillOnce([&](io_uring_cqe& cqe) {
EXPECT_EQ(cqe.res, expected_res.value_or(data.size()));
});
ring.process();
testing::Mock::VerifyAndClearExpectations(&h[0]);
}
};
TEST_F(IoUringTest, NullHandler)
{
auto& sqe = ring.getSQE();
io_uring_prep_nop(&sqe);
ring.submit();
ring.process();
}
TEST_F(IoUringTest, HandlerCalled)
{
{
auto& sqe = ring.getSQE();
io_uring_prep_nop(&sqe);
ring.setHandler(sqe, &h[0]);
}
// Nothing should happen without submission
ring.process();
{
auto& sqe = ring.getSQE();
io_uring_prep_nop(&sqe);
ring.setHandler(sqe, &h[1]);
}
// Handle all of the outstanding requests
ring.submit();
EXPECT_CALL(h[0], handleCQE(_));
EXPECT_CALL(h[1], handleCQE(_));
ring.process();
testing::Mock::VerifyAndClearExpectations(&h[0]);
testing::Mock::VerifyAndClearExpectations(&h[1]);
}
TEST_F(IoUringTest, HandlerReplacement)
{
auto& sqe = ring.getSQE();
io_uring_prep_nop(&sqe);
ring.setHandler(sqe, &h[0]);
// Setting a new handler should cancel the old one
EXPECT_CALL(h[0], handleCQE(_));
ring.setHandler(sqe, &h[1]);
testing::Mock::VerifyAndClearExpectations(&h[0]);
// Setting a null handler should cancel the old one
EXPECT_CALL(h[1], handleCQE(_));
ring.setHandler(sqe, nullptr);
testing::Mock::VerifyAndClearExpectations(&h[1]);
// Set it back twice and make sure it isn't recognized idempotently
ring.setHandler(sqe, &h[1]);
ring.setHandler(sqe, &h[1]);
// Make sure it still works
ring.submit();
EXPECT_CALL(h[1], handleCQE(_));
ring.process();
testing::Mock::VerifyAndClearExpectations(&h[1]);
}
TEST_F(IoUringTest, EventFd)
{
auto& efd = ring.getEventFd();
for (size_t i = 0; i < h.size(); ++i)
{
auto& sqe = ring.getSQE();
io_uring_prep_nop(&sqe);
ring.setHandler(sqe, &h[i]);
ring.submit();
}
// Our event fd should become ready
pollfd pfd;
pfd.fd = efd.get();
pfd.events = POLLIN;
ASSERT_EQ(1, poll(&pfd, 1, 100));
// Handle all of the outstanding requests
EXPECT_CALL(h[0], handleCQE(_));
EXPECT_CALL(h[1], handleCQE(_));
ring.processEvents();
testing::Mock::VerifyAndClearExpectations(&h[0]);
testing::Mock::VerifyAndClearExpectations(&h[1]);
// Our event fd should be empty
ASSERT_EQ(0, poll(&pfd, 1, 100));
}
TEST_F(IoUringTest, Wait)
{
auto& sqe = ring.getSQE();
auto kts = chronoToKTS(std::chrono::milliseconds(1));
io_uring_prep_timeout(&sqe, &kts, 0, 0);
ring.setHandler(sqe, &h[0]);
ring.submit();
ring.wait(std::chrono::seconds(1));
EXPECT_CALL(h[0], handleCQE(_));
}
TEST_F(IoUringTest, HandleCalledOnDestroy)
{
auto& sqe = ring.getSQE();
io_uring_prep_nop(&sqe);
ring.setHandler(sqe, &h[0]);
EXPECT_CALL(h[0], handleCQE(_));
}
TEST_F(IoUringTest, RegisterFiles)
{
std::optional<IoUring::FileHandle> fh;
// Slots are always allocated linearly and re-used if invalidated
fh = ring.registerFile(0);
EXPECT_GT(ring.getFiles().size(), 1);
EXPECT_EQ(*fh, 0);
fh = ring.registerFile(1);
EXPECT_EQ(*fh, 1);
EXPECT_EQ(ring.getFiles()[1], 1);
// The first handle should have dropped and can be replaced
fh = ring.registerFile(2);
EXPECT_EQ(*fh, 0);
EXPECT_EQ(ring.getFiles()[0], 2);
// We should be able to write to stderr via the fixed file and regular fd
testFdWrite(2, 0);
testFdWrite(*fh, IOSQE_FIXED_FILE);
// Without registration we should only be able to write to the regular fd
fh.reset();
testFdWrite(2, 0);
testFdWrite(*fh, IOSQE_FIXED_FILE, -EBADF);
std::vector<IoUring::FileHandle> fhs;
EXPECT_LT(ring.getFiles().size(), 9);
ring.reserveFiles(9);
EXPECT_EQ(ring.getFiles().size(), 9);
for (size_t i = 0; i < 9; ++i)
{
fhs.emplace_back(ring.registerFile(2));
testFdWrite(fhs.back(), IOSQE_FIXED_FILE);
}
EXPECT_EQ(ring.getFiles().size(), 9);
fhs.emplace_back(ring.registerFile(2));
testFdWrite(fhs.back(), IOSQE_FIXED_FILE);
EXPECT_GE(ring.getFiles().size(), 10);
}
} // namespace stdplus