#include "xray_buffer_queue.h"
#include "gmock/gmock.h"
#include "gtest/gtest.h"
#include <atomic>
#include <future>
#include <thread>
#include <unistd.h>
namespace __xray {
namespace {
static constexpr size_t kSize = 4096;
using ::testing::Eq;
TEST(BufferQueueTest, API) {
bool Success = false;
BufferQueue Buffers(kSize, 1, Success);
ASSERT_TRUE(Success);
}
TEST(BufferQueueTest, GetAndRelease) {
bool Success = false;
BufferQueue Buffers(kSize, 1, Success);
ASSERT_TRUE(Success);
BufferQueue::Buffer Buf;
ASSERT_EQ(Buffers.getBuffer(Buf), BufferQueue::ErrorCode::Ok);
ASSERT_NE(nullptr, Buf.Data);
ASSERT_EQ(Buffers.releaseBuffer(Buf), BufferQueue::ErrorCode::Ok);
ASSERT_EQ(nullptr, Buf.Data);
}
TEST(BufferQueueTest, GetUntilFailed) {
bool Success = false;
BufferQueue Buffers(kSize, 1, Success);
ASSERT_TRUE(Success);
BufferQueue::Buffer Buf0;
EXPECT_EQ(Buffers.getBuffer(Buf0), BufferQueue::ErrorCode::Ok);
BufferQueue::Buffer Buf1;
EXPECT_EQ(BufferQueue::ErrorCode::NotEnoughMemory, Buffers.getBuffer(Buf1));
EXPECT_EQ(Buffers.releaseBuffer(Buf0), BufferQueue::ErrorCode::Ok);
}
TEST(BufferQueueTest, ReleaseUnknown) {
bool Success = false;
BufferQueue Buffers(kSize, 1, Success);
ASSERT_TRUE(Success);
BufferQueue::Buffer Buf;
Buf.Data = reinterpret_cast<void *>(0xdeadbeef);
Buf.Size = kSize;
Buf.Generation = Buffers.generation();
BufferQueue::Buffer Known;
EXPECT_THAT(Buffers.getBuffer(Known), Eq(BufferQueue::ErrorCode::Ok));
EXPECT_THAT(Buffers.releaseBuffer(Buf),
Eq(BufferQueue::ErrorCode::UnrecognizedBuffer));
EXPECT_THAT(Buffers.releaseBuffer(Known), Eq(BufferQueue::ErrorCode::Ok));
}
TEST(BufferQueueTest, ErrorsWhenFinalising) {
bool Success = false;
BufferQueue Buffers(kSize, 2, Success);
ASSERT_TRUE(Success);
BufferQueue::Buffer Buf;
ASSERT_EQ(Buffers.getBuffer(Buf), BufferQueue::ErrorCode::Ok);
ASSERT_NE(nullptr, Buf.Data);
ASSERT_EQ(Buffers.finalize(), BufferQueue::ErrorCode::Ok);
BufferQueue::Buffer OtherBuf;
ASSERT_EQ(BufferQueue::ErrorCode::QueueFinalizing,
Buffers.getBuffer(OtherBuf));
ASSERT_EQ(BufferQueue::ErrorCode::QueueFinalizing, Buffers.finalize());
ASSERT_EQ(Buffers.releaseBuffer(Buf), BufferQueue::ErrorCode::Ok);
}
TEST(BufferQueueTest, MultiThreaded) {
bool Success = false;
BufferQueue Buffers(kSize, 100, Success);
ASSERT_TRUE(Success);
auto F = [&] {
BufferQueue::Buffer B;
while (true) {
auto EC = Buffers.getBuffer(B);
if (EC != BufferQueue::ErrorCode::Ok)
return;
Buffers.releaseBuffer(B);
}
};
auto T0 = std::async(std::launch::async, F);
auto T1 = std::async(std::launch::async, F);
auto T2 = std::async(std::launch::async, [&] {
while (Buffers.finalize() != BufferQueue::ErrorCode::Ok)
;
});
F();
}
TEST(BufferQueueTest, Apply) {
bool Success = false;
BufferQueue Buffers(kSize, 10, Success);
ASSERT_TRUE(Success);
auto Count = 0;
BufferQueue::Buffer B;
for (int I = 0; I < 10; ++I) {
ASSERT_EQ(Buffers.getBuffer(B), BufferQueue::ErrorCode::Ok);
ASSERT_EQ(Buffers.releaseBuffer(B), BufferQueue::ErrorCode::Ok);
}
Buffers.apply([&](const BufferQueue::Buffer &B) { ++Count; });
ASSERT_EQ(Count, 10);
}
TEST(BufferQueueTest, GenerationalSupport) {
bool Success = false;
BufferQueue Buffers(kSize, 10, Success);
ASSERT_TRUE(Success);
BufferQueue::Buffer B0;
ASSERT_EQ(Buffers.getBuffer(B0), BufferQueue::ErrorCode::Ok);
ASSERT_EQ(Buffers.finalize(),
BufferQueue::ErrorCode::Ok);
ASSERT_EQ(Buffers.init(kSize, 10), BufferQueue::ErrorCode::Ok);
BufferQueue::Buffer B1;
ASSERT_EQ(Buffers.getBuffer(B1), BufferQueue::ErrorCode::Ok);
ASSERT_NE(B0.Generation, B1.Generation);
auto PrevGen = B1.Generation;
EXPECT_EQ(Buffers.releaseBuffer(B0), BufferQueue::ErrorCode::Ok);
EXPECT_EQ(Buffers.releaseBuffer(B1), BufferQueue::ErrorCode::Ok);
ASSERT_EQ(Buffers.finalize(), BufferQueue::ErrorCode::Ok);
ASSERT_EQ(Buffers.init(kSize, 10), BufferQueue::ErrorCode::Ok);
EXPECT_EQ(Buffers.getBuffer(B0), BufferQueue::ErrorCode::Ok);
EXPECT_EQ(Buffers.getBuffer(B1), BufferQueue::ErrorCode::Ok);
EXPECT_NE(B0.Generation, PrevGen);
EXPECT_EQ(B1.Generation, B1.Generation);
ASSERT_EQ(Buffers.finalize(), BufferQueue::ErrorCode::Ok);
EXPECT_EQ(Buffers.releaseBuffer(B0), BufferQueue::ErrorCode::Ok);
EXPECT_EQ(Buffers.releaseBuffer(B1), BufferQueue::ErrorCode::Ok);
}
TEST(BufferQueueTest, GenerationalSupportAcrossThreads) {
bool Success = false;
BufferQueue Buffers(kSize, 10, Success);
ASSERT_TRUE(Success);
std::atomic<int> Counter{0};
auto Process = [&] {
thread_local BufferQueue::Buffer B;
ASSERT_EQ(Buffers.getBuffer(B), BufferQueue::ErrorCode::Ok);
auto FirstGen = B.Generation;
Counter.fetch_add(1, std::memory_order_acq_rel);
while (!Buffers.finalizing()) {
Buffers.releaseBuffer(B);
Buffers.getBuffer(B);
}
Counter.fetch_sub(1, std::memory_order_acq_rel);
if (B.Data != nullptr)
Buffers.releaseBuffer(B);
while (Buffers.getBuffer(B) != BufferQueue::ErrorCode::Ok)
;
Counter.fetch_add(1, std::memory_order_acq_rel);
EXPECT_NE(FirstGen, B.Generation);
EXPECT_EQ(Buffers.releaseBuffer(B), BufferQueue::ErrorCode::Ok);
Counter.fetch_sub(1, std::memory_order_acq_rel);
};
std::thread T0(Process), T1(Process);
while (Counter.load(std::memory_order_acquire) != 2)
;
Buffers.finalize();
while (Counter.load(std::memory_order_acquire) != 0)
;
EXPECT_EQ(Buffers.init(kSize, 10), BufferQueue::ErrorCode::Ok);
T0.join();
T1.join();
ASSERT_EQ(Counter.load(std::memory_order_acquire), 0);
}
}
}