#include #include #include "gtest/gtest.h" #include "data_structures/ring_buffer.hpp" #include "threading/sync/spinlock.hpp" TEST(RingBuffer, MultithreadedUsage) { auto test_f = [](int producer_count, int elems_per_producer, int producer_sleep_ms, int consumer_count, int consumer_sleep_ms) { std::unordered_set consumed; SpinLock consumed_lock; RingBuffer buffer; std::vector producers; for (int i = 0; i < producer_count; i++) producers.emplace_back( [i, elems_per_producer, producer_sleep_ms, &buffer]() { for (int j = 0; j < elems_per_producer; j++) { std::this_thread::sleep_for( std::chrono::milliseconds(producer_sleep_ms)); buffer.emplace(j + i * elems_per_producer); } }); std::vector consumers; size_t elem_total_count = producer_count * elems_per_producer; for (int i = 0; i < consumer_count; i++) consumers.emplace_back([elem_total_count, consumer_sleep_ms, &buffer, &consumed, &consumed_lock]() { while (true) { std::this_thread::sleep_for( std::chrono::milliseconds(consumer_sleep_ms)); std::lock_guard guard(consumed_lock); if (consumed.size() == elem_total_count) break; auto value = buffer.pop(); if (value) consumed.emplace(*value); } }); for (auto &producer : producers) producer.join(); for (auto &consumer : consumers) consumer.join(); return !buffer.pop() && consumed.size() == elem_total_count; }; // Many slow producers, many fast consumers. EXPECT_TRUE(test_f(10, 200, 3, 10, 0)); // Many fast producers, many slow consumers. EXPECT_TRUE(test_f(10, 200, 0, 10, 3)); // One slower producer, many consumers. EXPECT_TRUE(test_f(1, 500, 3, 10, 0)); // Many producers, one slow consumer. EXPECT_TRUE(test_f(10, 200, 0, 1, 3)); } TEST(RingBuffer, ComplexValues) { RingBuffer, 10> buffer; std::vector element; for (int i = 0 ; i < 5 ; i++) { element.emplace_back(i); buffer.emplace(element); } element.clear(); for (int i = 0 ; i < 5 ; i++) { element.emplace_back(i); EXPECT_EQ(*buffer.pop(), element); } EXPECT_FALSE(buffer.pop()); } TEST(RingBuffer, NonCopyable) { RingBuffer, 10> buffer; buffer.emplace(new std::string("string")); buffer.emplace(new std::string("kifla")); EXPECT_EQ(**buffer.pop(), "string"); EXPECT_EQ(**buffer.pop(), "kifla"); EXPECT_FALSE(buffer.pop()); std::unique_ptr a(new std::string("bla")); }