2015-05-28 09:00:53 +02:00
|
|
|
// Copyright 2014 Dolphin Emulator Project
|
2021-07-05 03:22:19 +02:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2015-05-28 09:00:53 +02:00
|
|
|
|
|
|
|
#include <atomic>
|
|
|
|
#include <thread>
|
|
|
|
|
|
|
|
#include <gtest/gtest.h>
|
|
|
|
|
|
|
|
#include "Common/BlockingLoop.h"
|
|
|
|
|
|
|
|
TEST(BlockingLoop, MultiThreaded)
|
|
|
|
{
|
2016-06-24 10:43:46 +02:00
|
|
|
Common::BlockingLoop loop;
|
|
|
|
std::atomic<int> signaled_a(0);
|
|
|
|
std::atomic<int> received_a(0);
|
|
|
|
std::atomic<int> signaled_b(0);
|
|
|
|
std::atomic<int> received_b(0);
|
|
|
|
for (int i = 0; i < 100; i++)
|
|
|
|
{
|
|
|
|
// Invalidate the current state.
|
|
|
|
received_a.store(signaled_a.load() + 1);
|
|
|
|
received_b.store(signaled_b.load() + 123);
|
2015-05-28 09:00:53 +02:00
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
// Must not block as the loop is stopped.
|
|
|
|
loop.Wait();
|
2015-05-28 09:00:53 +02:00
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
std::thread loop_thread([&]() {
|
|
|
|
loop.Run([&]() {
|
|
|
|
received_a.store(signaled_a.load());
|
|
|
|
received_b.store(signaled_b.load());
|
|
|
|
});
|
|
|
|
});
|
2015-05-28 09:00:53 +02:00
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
// Now Wait must block.
|
|
|
|
loop.Prepare();
|
2015-05-28 09:00:53 +02:00
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
// The payload must run at least once on startup.
|
|
|
|
loop.Wait();
|
|
|
|
EXPECT_EQ(signaled_a.load(), received_a.load());
|
|
|
|
EXPECT_EQ(signaled_b.load(), received_b.load());
|
2015-05-28 09:00:53 +02:00
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
std::thread run_a_thread([&]() {
|
|
|
|
for (int j = 0; j < 100; j++)
|
|
|
|
{
|
|
|
|
for (int k = 0; k < 100; k++)
|
|
|
|
{
|
|
|
|
signaled_a++;
|
|
|
|
loop.Wakeup();
|
|
|
|
}
|
2015-05-28 09:00:53 +02:00
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
loop.Wait();
|
|
|
|
EXPECT_EQ(signaled_a.load(), received_a.load());
|
|
|
|
}
|
|
|
|
});
|
|
|
|
std::thread run_b_thread([&]() {
|
|
|
|
for (int j = 0; j < 100; j++)
|
|
|
|
{
|
|
|
|
for (int k = 0; k < 100; k++)
|
|
|
|
{
|
|
|
|
signaled_b++;
|
|
|
|
loop.Wakeup();
|
|
|
|
}
|
2015-05-28 09:00:53 +02:00
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
loop.Wait();
|
|
|
|
EXPECT_EQ(signaled_b.load(), received_b.load());
|
|
|
|
}
|
|
|
|
});
|
2015-05-28 09:00:53 +02:00
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
run_a_thread.join();
|
|
|
|
run_b_thread.join();
|
2015-05-28 09:00:53 +02:00
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
loop.Stop();
|
2015-05-28 09:00:53 +02:00
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
// Must not block
|
|
|
|
loop.Wait();
|
2015-05-28 09:00:53 +02:00
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
loop_thread.join();
|
|
|
|
}
|
2015-05-28 09:00:53 +02:00
|
|
|
}
|