Merge remote-tracking branch 'bwr/linux_threads' into canary
This commit is contained in:
commit
1a1dd4ef25
|
@ -0,0 +1,10 @@
|
||||||
|
# Ignore HighResolutionTimer custom event
|
||||||
|
handle SIG34 nostop noprint
|
||||||
|
# Ignore PosixTimer custom event
|
||||||
|
handle SIG35 nostop noprint
|
||||||
|
# Ignore PosixThread exit event
|
||||||
|
handle SIG32 nostop noprint
|
||||||
|
# Ignore PosixThread suspend event
|
||||||
|
handle SIG36 nostop noprint
|
||||||
|
# Ignore PosixThread user callback event
|
||||||
|
handle SIG37 nostop noprint
|
|
@ -70,8 +70,8 @@ std::unique_ptr<EmulatorWindow> EmulatorWindow::Create(Emulator* emulator) {
|
||||||
std::unique_ptr<EmulatorWindow> emulator_window(new EmulatorWindow(emulator));
|
std::unique_ptr<EmulatorWindow> emulator_window(new EmulatorWindow(emulator));
|
||||||
|
|
||||||
emulator_window->loop()->PostSynchronous([&emulator_window]() {
|
emulator_window->loop()->PostSynchronous([&emulator_window]() {
|
||||||
xe::threading::set_name("Win32 Loop");
|
xe::threading::set_name("Windowing Loop");
|
||||||
xe::Profiler::ThreadEnter("Win32 Loop");
|
xe::Profiler::ThreadEnter("Windowing Loop");
|
||||||
|
|
||||||
if (!emulator_window->Initialize()) {
|
if (!emulator_window->Initialize()) {
|
||||||
xe::FatalError("Failed to initialize main window");
|
xe::FatalError("Failed to initialize main window");
|
||||||
|
|
|
@ -133,7 +133,7 @@ X_STATUS XmaDecoder::Setup(kernel::KernelState* kernel_state) {
|
||||||
WorkerThreadMain();
|
WorkerThreadMain();
|
||||||
return 0;
|
return 0;
|
||||||
}));
|
}));
|
||||||
worker_thread_->set_name("XMA Decoder Worker");
|
worker_thread_->set_name("XMA Decoder");
|
||||||
worker_thread_->set_can_debugger_suspend(true);
|
worker_thread_->set_can_debugger_suspend(true);
|
||||||
worker_thread_->Create();
|
worker_thread_->Create();
|
||||||
|
|
||||||
|
|
|
@ -72,7 +72,7 @@ class Logger {
|
||||||
|
|
||||||
write_thread_ =
|
write_thread_ =
|
||||||
xe::threading::Thread::Create({}, [this]() { WriteThread(); });
|
xe::threading::Thread::Create({}, [this]() { WriteThread(); });
|
||||||
write_thread_->set_name("xe::FileLogSink Writer");
|
write_thread_->set_name("Logging Writer");
|
||||||
}
|
}
|
||||||
|
|
||||||
~Logger() {
|
~Logger() {
|
||||||
|
|
|
@ -0,0 +1,965 @@
|
||||||
|
/**
|
||||||
|
******************************************************************************
|
||||||
|
* Xenia : Xbox 360 Emulator Research Project *
|
||||||
|
******************************************************************************
|
||||||
|
* Copyright 2018 Ben Vanik. All rights reserved. *
|
||||||
|
* Released under the BSD license - see LICENSE in the root for more details. *
|
||||||
|
******************************************************************************
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <array>
|
||||||
|
|
||||||
|
#include "xenia/base/threading.h"
|
||||||
|
|
||||||
|
#include "third_party/catch/include/catch.hpp"
|
||||||
|
|
||||||
|
namespace xe {
|
||||||
|
namespace base {
|
||||||
|
namespace test {
|
||||||
|
using namespace threading;
|
||||||
|
using namespace std::chrono_literals;
|
||||||
|
|
||||||
|
TEST_CASE("Fence") {
|
||||||
|
std::unique_ptr<threading::Fence> pFence;
|
||||||
|
std::unique_ptr<threading::HighResolutionTimer> pTimer;
|
||||||
|
|
||||||
|
// Signal without wait
|
||||||
|
pFence = std::make_unique<threading::Fence>();
|
||||||
|
pFence->Signal();
|
||||||
|
|
||||||
|
// Signal once and wait
|
||||||
|
pFence = std::make_unique<threading::Fence>();
|
||||||
|
pFence->Signal();
|
||||||
|
pFence->Wait();
|
||||||
|
|
||||||
|
// Signal twice and wait
|
||||||
|
pFence = std::make_unique<threading::Fence>();
|
||||||
|
pFence->Signal();
|
||||||
|
pFence->Signal();
|
||||||
|
pFence->Wait();
|
||||||
|
|
||||||
|
// Test to synchronize multiple threads
|
||||||
|
std::atomic<int> started(0);
|
||||||
|
std::atomic<int> finished(0);
|
||||||
|
pFence = std::make_unique<threading::Fence>();
|
||||||
|
auto func = [&pFence, &started, &finished] {
|
||||||
|
started.fetch_add(1);
|
||||||
|
pFence->Wait();
|
||||||
|
finished.fetch_add(1);
|
||||||
|
};
|
||||||
|
|
||||||
|
auto threads = std::array<std::thread, 5>({
|
||||||
|
std::thread(func),
|
||||||
|
std::thread(func),
|
||||||
|
std::thread(func),
|
||||||
|
std::thread(func),
|
||||||
|
std::thread(func),
|
||||||
|
});
|
||||||
|
|
||||||
|
Sleep(10ms);
|
||||||
|
REQUIRE(finished.load() == 0);
|
||||||
|
|
||||||
|
// TODO(bwrsandman): Check if this is correct behaviour: looping with Sleep
|
||||||
|
// is the only way to get fence to signal all threads on windows
|
||||||
|
for (int i = 0; i < threads.size(); ++i) {
|
||||||
|
Sleep(1ms);
|
||||||
|
pFence->Signal();
|
||||||
|
}
|
||||||
|
REQUIRE(started.load() == threads.size());
|
||||||
|
|
||||||
|
for (auto& t : threads) t.join();
|
||||||
|
REQUIRE(finished.load() == threads.size());
|
||||||
|
} // namespace test
|
||||||
|
|
||||||
|
TEST_CASE("Get number of logical processors") {
|
||||||
|
auto count = std::thread::hardware_concurrency();
|
||||||
|
REQUIRE(logical_processor_count() == count);
|
||||||
|
REQUIRE(logical_processor_count() == count);
|
||||||
|
REQUIRE(logical_processor_count() == count);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("Enable process to set thread affinity") {
|
||||||
|
EnableAffinityConfiguration();
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("Yield Current Thread", "MaybeYield") {
|
||||||
|
// Run to see if there are any errors
|
||||||
|
MaybeYield();
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("Sync with Memory Barrier", "SyncMemory") {
|
||||||
|
// Run to see if there are any errors
|
||||||
|
SyncMemory();
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("Sleep Current Thread", "Sleep") {
|
||||||
|
auto wait_time = 5ms;
|
||||||
|
auto start = std::chrono::steady_clock::now();
|
||||||
|
Sleep(wait_time);
|
||||||
|
auto duration = std::chrono::steady_clock::now() - start;
|
||||||
|
REQUIRE(duration >= wait_time);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("Sleep Current Thread in Alertable State", "Sleep") {
|
||||||
|
auto wait_time = 5ms;
|
||||||
|
auto start = std::chrono::steady_clock::now();
|
||||||
|
auto result = threading::AlertableSleep(wait_time);
|
||||||
|
auto duration = std::chrono::steady_clock::now() - start;
|
||||||
|
REQUIRE(duration >= wait_time);
|
||||||
|
REQUIRE(result == threading::SleepResult::kSuccess);
|
||||||
|
|
||||||
|
// TODO(bwrsandman): Test a Thread to return kAlerted.
|
||||||
|
// Need callback to call extended I/O function (ReadFileEx or WriteFileEx)
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("TlsHandle") {
|
||||||
|
// Test Allocate
|
||||||
|
auto handle = threading::AllocateTlsHandle();
|
||||||
|
|
||||||
|
// Test Free
|
||||||
|
REQUIRE(threading::FreeTlsHandle(handle));
|
||||||
|
REQUIRE(!threading::FreeTlsHandle(handle));
|
||||||
|
REQUIRE(!threading::FreeTlsHandle(threading::kInvalidTlsHandle));
|
||||||
|
|
||||||
|
// Test setting values
|
||||||
|
handle = threading::AllocateTlsHandle();
|
||||||
|
REQUIRE(threading::GetTlsValue(handle) == 0);
|
||||||
|
uint32_t value = 0xDEADBEEF;
|
||||||
|
threading::SetTlsValue(handle, reinterpret_cast<uintptr_t>(&value));
|
||||||
|
auto p_received_value = threading::GetTlsValue(handle);
|
||||||
|
REQUIRE(threading::GetTlsValue(handle) != 0);
|
||||||
|
auto received_value = *reinterpret_cast<uint32_t*>(p_received_value);
|
||||||
|
REQUIRE(received_value == value);
|
||||||
|
|
||||||
|
uintptr_t non_thread_local_value = 0;
|
||||||
|
auto thread = Thread::Create({}, [&non_thread_local_value, &handle] {
|
||||||
|
non_thread_local_value = threading::GetTlsValue(handle);
|
||||||
|
});
|
||||||
|
|
||||||
|
auto result = Wait(thread.get(), false, 5ms);
|
||||||
|
REQUIRE(result == WaitResult::kSuccess);
|
||||||
|
REQUIRE(non_thread_local_value == 0);
|
||||||
|
|
||||||
|
// Cleanup
|
||||||
|
REQUIRE(threading::FreeTlsHandle(handle));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("HighResolutionTimer") {
|
||||||
|
// The wait time is 500ms with an interval of 50ms
|
||||||
|
// Smaller values are not as precise and fail the test
|
||||||
|
const auto wait_time = 50ms;
|
||||||
|
|
||||||
|
// Time the actual sleep duration
|
||||||
|
{
|
||||||
|
const auto interval = 5ms;
|
||||||
|
uint64_t counter = 0;
|
||||||
|
auto start = std::chrono::steady_clock::now();
|
||||||
|
auto cb = [&counter] { ++counter; };
|
||||||
|
auto pTimer = HighResolutionTimer::CreateRepeating(interval, cb);
|
||||||
|
Sleep(wait_time);
|
||||||
|
pTimer.reset();
|
||||||
|
auto duration = std::chrono::steady_clock::now() - start;
|
||||||
|
|
||||||
|
// Should have run as many times as wait_time / timer_interval plus or
|
||||||
|
// minus 1 due to imprecision of Sleep
|
||||||
|
REQUIRE(duration.count() >= wait_time.count());
|
||||||
|
auto ratio = static_cast<uint64_t>(duration / interval);
|
||||||
|
REQUIRE(counter >= ratio - 1);
|
||||||
|
REQUIRE(counter <= ratio + 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Test concurrent timers
|
||||||
|
{
|
||||||
|
const auto interval1 = 10ms;
|
||||||
|
const auto interval2 = 20ms;
|
||||||
|
uint64_t counter1 = 0;
|
||||||
|
uint64_t counter2 = 0;
|
||||||
|
auto start = std::chrono::steady_clock::now();
|
||||||
|
auto cb1 = [&counter1] { ++counter1; };
|
||||||
|
auto cb2 = [&counter2] { ++counter2; };
|
||||||
|
auto pTimer1 = HighResolutionTimer::CreateRepeating(interval1, cb1);
|
||||||
|
auto pTimer2 = HighResolutionTimer::CreateRepeating(interval2, cb2);
|
||||||
|
Sleep(wait_time);
|
||||||
|
pTimer1.reset();
|
||||||
|
pTimer2.reset();
|
||||||
|
auto duration = std::chrono::steady_clock::now() - start;
|
||||||
|
|
||||||
|
// Should have run as many times as wait_time / timer_interval plus or
|
||||||
|
// minus 1 due to imprecision of Sleep
|
||||||
|
REQUIRE(duration.count() >= wait_time.count());
|
||||||
|
auto ratio1 = static_cast<uint64_t>(duration / interval1);
|
||||||
|
auto ratio2 = static_cast<uint64_t>(duration / interval2);
|
||||||
|
REQUIRE(counter1 >= ratio1 - 1);
|
||||||
|
REQUIRE(counter1 <= ratio1 + 1);
|
||||||
|
REQUIRE(counter2 >= ratio2 - 1);
|
||||||
|
REQUIRE(counter2 <= ratio2 + 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO(bwrsandman): Check on which thread callbacks are executed when
|
||||||
|
// spawned from differing threads
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("Wait on Multiple Handles", "Wait") {
|
||||||
|
auto mutant = Mutant::Create(true);
|
||||||
|
auto semaphore = Semaphore::Create(10, 10);
|
||||||
|
auto event_ = Event::CreateManualResetEvent(false);
|
||||||
|
auto thread = Thread::Create({}, [&mutant, &semaphore, &event_] {
|
||||||
|
event_->Set();
|
||||||
|
Wait(mutant.get(), false, 2ms);
|
||||||
|
semaphore->Release(1, nullptr);
|
||||||
|
Wait(mutant.get(), false, 2ms);
|
||||||
|
mutant->Release();
|
||||||
|
});
|
||||||
|
|
||||||
|
std::vector<WaitHandle*> handles = {
|
||||||
|
mutant.get(),
|
||||||
|
semaphore.get(),
|
||||||
|
event_.get(),
|
||||||
|
thread.get(),
|
||||||
|
};
|
||||||
|
|
||||||
|
auto any_result = WaitAny(handles, false, 10ms);
|
||||||
|
REQUIRE(any_result.first == WaitResult::kSuccess);
|
||||||
|
REQUIRE(any_result.second == 0);
|
||||||
|
|
||||||
|
auto all_result = WaitAll(handles, false, 10ms);
|
||||||
|
REQUIRE(all_result == WaitResult::kSuccess);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("Signal and Wait") {
|
||||||
|
WaitResult result;
|
||||||
|
auto mutant = Mutant::Create(true);
|
||||||
|
auto event_ = Event::CreateAutoResetEvent(false);
|
||||||
|
auto thread = Thread::Create({}, [&mutant, &event_] {
|
||||||
|
Wait(mutant.get(), false);
|
||||||
|
event_->Set();
|
||||||
|
});
|
||||||
|
result = Wait(event_.get(), false, 5ms);
|
||||||
|
REQUIRE(result == WaitResult::kTimeout);
|
||||||
|
result = SignalAndWait(mutant.get(), event_.get(), false, 5ms);
|
||||||
|
REQUIRE(result == WaitResult::kSuccess);
|
||||||
|
result = Wait(thread.get(), false, 5ms);
|
||||||
|
REQUIRE(result == WaitResult::kSuccess);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("Wait on Event", "Event") {
|
||||||
|
auto evt = Event::CreateAutoResetEvent(false);
|
||||||
|
WaitResult result;
|
||||||
|
|
||||||
|
// Call wait on unset Event
|
||||||
|
result = Wait(evt.get(), false, 5ms);
|
||||||
|
REQUIRE(result == WaitResult::kTimeout);
|
||||||
|
|
||||||
|
// Call wait on set Event
|
||||||
|
evt->Set();
|
||||||
|
result = Wait(evt.get(), false, 5ms);
|
||||||
|
REQUIRE(result == WaitResult::kSuccess);
|
||||||
|
|
||||||
|
// Call wait on now consumed Event
|
||||||
|
result = Wait(evt.get(), false, 5ms);
|
||||||
|
REQUIRE(result == WaitResult::kTimeout);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("Reset Event", "Event") {
|
||||||
|
auto evt = Event::CreateAutoResetEvent(false);
|
||||||
|
WaitResult result;
|
||||||
|
|
||||||
|
// Call wait on reset Event
|
||||||
|
evt->Set();
|
||||||
|
evt->Reset();
|
||||||
|
result = Wait(evt.get(), false, 5ms);
|
||||||
|
REQUIRE(result == WaitResult::kTimeout);
|
||||||
|
|
||||||
|
// Test resetting the unset event
|
||||||
|
evt->Reset();
|
||||||
|
result = Wait(evt.get(), false, 5ms);
|
||||||
|
REQUIRE(result == WaitResult::kTimeout);
|
||||||
|
|
||||||
|
// Test setting the reset event
|
||||||
|
evt->Set();
|
||||||
|
result = Wait(evt.get(), false, 5ms);
|
||||||
|
REQUIRE(result == WaitResult::kSuccess);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("Wait on Multiple Events", "Event") {
|
||||||
|
auto events = std::array<std::unique_ptr<Event>, 4>{
|
||||||
|
Event::CreateAutoResetEvent(false),
|
||||||
|
Event::CreateAutoResetEvent(false),
|
||||||
|
Event::CreateAutoResetEvent(false),
|
||||||
|
Event::CreateManualResetEvent(false),
|
||||||
|
};
|
||||||
|
|
||||||
|
std::array<char, 8> order = {0};
|
||||||
|
std::atomic_uint index(0);
|
||||||
|
auto sign_in = [&order, &index](uint32_t id) {
|
||||||
|
auto i = index.fetch_add(1, std::memory_order::memory_order_relaxed);
|
||||||
|
order[i] = static_cast<char>('0' + id);
|
||||||
|
};
|
||||||
|
|
||||||
|
auto threads = std::array<std::thread, 4>{
|
||||||
|
std::thread([&events, &sign_in] {
|
||||||
|
auto res = WaitAll({events[1].get(), events[3].get()}, false, 10ms);
|
||||||
|
if (res == WaitResult::kSuccess) {
|
||||||
|
sign_in(1);
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
std::thread([&events, &sign_in] {
|
||||||
|
auto res = WaitAny({events[0].get(), events[2].get()}, false, 10ms);
|
||||||
|
if (res.first == WaitResult::kSuccess) {
|
||||||
|
sign_in(2);
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
std::thread([&events, &sign_in] {
|
||||||
|
auto res = WaitAll({events[0].get(), events[2].get(), events[3].get()},
|
||||||
|
false, 10ms);
|
||||||
|
if (res == WaitResult::kSuccess) {
|
||||||
|
sign_in(3);
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
std::thread([&events, &sign_in] {
|
||||||
|
auto res = WaitAny({events[1].get(), events[3].get()}, false, 10ms);
|
||||||
|
if (res.first == WaitResult::kSuccess) {
|
||||||
|
sign_in(4);
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
};
|
||||||
|
|
||||||
|
Sleep(1ms);
|
||||||
|
events[3]->Set(); // Signals thread id=4 and stays on for 1 and 3
|
||||||
|
Sleep(1ms);
|
||||||
|
events[1]->Set(); // Signals thread id=1
|
||||||
|
Sleep(1ms);
|
||||||
|
events[0]->Set(); // Signals thread id=2
|
||||||
|
Sleep(1ms);
|
||||||
|
events[2]->Set(); // Partial signals thread id=3
|
||||||
|
events[0]->Set(); // Signals thread id=3
|
||||||
|
|
||||||
|
for (auto& t : threads) {
|
||||||
|
t.join();
|
||||||
|
}
|
||||||
|
|
||||||
|
INFO(order.data());
|
||||||
|
REQUIRE(order[0] == '4');
|
||||||
|
// TODO(bwrsandman): Order is not always maintained on linux
|
||||||
|
// REQUIRE(order[1] == '1');
|
||||||
|
// REQUIRE(order[2] == '2');
|
||||||
|
// REQUIRE(order[3] == '3');
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("Wait on Semaphore", "Semaphore") {
|
||||||
|
WaitResult result;
|
||||||
|
std::unique_ptr<Semaphore> sem;
|
||||||
|
int previous_count = 0;
|
||||||
|
|
||||||
|
// Wait on semaphore with no room
|
||||||
|
sem = Semaphore::Create(0, 5);
|
||||||
|
result = Wait(sem.get(), false, 1ms);
|
||||||
|
REQUIRE(result == WaitResult::kTimeout);
|
||||||
|
|
||||||
|
// Add room in semaphore
|
||||||
|
REQUIRE(sem->Release(2, &previous_count));
|
||||||
|
REQUIRE(previous_count == 0);
|
||||||
|
REQUIRE(sem->Release(1, &previous_count));
|
||||||
|
REQUIRE(previous_count == 2);
|
||||||
|
result = Wait(sem.get(), false, 1ms);
|
||||||
|
REQUIRE(result == WaitResult::kSuccess);
|
||||||
|
REQUIRE(sem->Release(1, &previous_count));
|
||||||
|
REQUIRE(previous_count == 2);
|
||||||
|
|
||||||
|
// Set semaphore over maximum_count
|
||||||
|
sem = Semaphore::Create(5, 5);
|
||||||
|
previous_count = -1;
|
||||||
|
REQUIRE_FALSE(sem->Release(1, &previous_count));
|
||||||
|
REQUIRE(previous_count == -1);
|
||||||
|
REQUIRE_FALSE(sem->Release(10, &previous_count));
|
||||||
|
REQUIRE(previous_count == -1);
|
||||||
|
sem = Semaphore::Create(0, 5);
|
||||||
|
REQUIRE_FALSE(sem->Release(10, &previous_count));
|
||||||
|
REQUIRE(previous_count == -1);
|
||||||
|
REQUIRE_FALSE(sem->Release(10, &previous_count));
|
||||||
|
REQUIRE(previous_count == -1);
|
||||||
|
|
||||||
|
// Test invalid Release parameters
|
||||||
|
REQUIRE_FALSE(sem->Release(0, &previous_count));
|
||||||
|
REQUIRE(previous_count == -1);
|
||||||
|
REQUIRE_FALSE(sem->Release(-1, &previous_count));
|
||||||
|
REQUIRE(previous_count == -1);
|
||||||
|
|
||||||
|
// Wait on fully available semaphore
|
||||||
|
sem = Semaphore::Create(5, 5);
|
||||||
|
result = Wait(sem.get(), false, 1ms);
|
||||||
|
REQUIRE(result == WaitResult::kSuccess);
|
||||||
|
result = Wait(sem.get(), false, 1ms);
|
||||||
|
REQUIRE(result == WaitResult::kSuccess);
|
||||||
|
result = Wait(sem.get(), false, 1ms);
|
||||||
|
REQUIRE(result == WaitResult::kSuccess);
|
||||||
|
result = Wait(sem.get(), false, 1ms);
|
||||||
|
REQUIRE(result == WaitResult::kSuccess);
|
||||||
|
result = Wait(sem.get(), false, 1ms);
|
||||||
|
REQUIRE(result == WaitResult::kSuccess);
|
||||||
|
result = Wait(sem.get(), false, 1ms);
|
||||||
|
REQUIRE(result == WaitResult::kTimeout);
|
||||||
|
|
||||||
|
// Semaphore between threads
|
||||||
|
sem = Semaphore::Create(5, 5);
|
||||||
|
Sleep(1ms);
|
||||||
|
// Occupy the semaphore with 5 threads
|
||||||
|
auto func = [&sem] {
|
||||||
|
auto res = Wait(sem.get(), false, 10ms);
|
||||||
|
Sleep(50ms);
|
||||||
|
if (res == WaitResult::kSuccess) {
|
||||||
|
sem->Release(1, nullptr);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
auto threads = std::array<std::thread, 5>{
|
||||||
|
std::thread(func), std::thread(func), std::thread(func),
|
||||||
|
std::thread(func), std::thread(func),
|
||||||
|
};
|
||||||
|
// Give threads time to acquire semaphore
|
||||||
|
Sleep(1ms);
|
||||||
|
// Attempt to acquire full semaphore with current (6th) thread
|
||||||
|
result = Wait(sem.get(), false, 2ms);
|
||||||
|
REQUIRE(result == WaitResult::kTimeout);
|
||||||
|
// Give threads time to release semaphore
|
||||||
|
for (auto& t : threads) {
|
||||||
|
t.join();
|
||||||
|
}
|
||||||
|
result = Wait(sem.get(), false, 1ms);
|
||||||
|
REQUIRE(result == WaitResult::kSuccess);
|
||||||
|
sem->Release(1, &previous_count);
|
||||||
|
REQUIRE(previous_count == 4);
|
||||||
|
|
||||||
|
// Test invalid construction parameters
|
||||||
|
// These are invalid according to documentation
|
||||||
|
// TODO(bwrsandman): Many of these invalid invocations succeed
|
||||||
|
sem = Semaphore::Create(-1, 5);
|
||||||
|
// REQUIRE(sem.get() == nullptr);
|
||||||
|
sem = Semaphore::Create(10, 5);
|
||||||
|
// REQUIRE(sem.get() == nullptr);
|
||||||
|
sem = Semaphore::Create(0, 0);
|
||||||
|
// REQUIRE(sem.get() == nullptr);
|
||||||
|
sem = Semaphore::Create(0, -1);
|
||||||
|
// REQUIRE(sem.get() == nullptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("Wait on Multiple Semaphores", "Semaphore") {
|
||||||
|
WaitResult all_result;
|
||||||
|
std::pair<WaitResult, size_t> any_result;
|
||||||
|
int previous_count;
|
||||||
|
std::unique_ptr<Semaphore> sem0, sem1;
|
||||||
|
|
||||||
|
// Test Wait all which should fail
|
||||||
|
sem0 = Semaphore::Create(0, 5);
|
||||||
|
sem1 = Semaphore::Create(5, 5);
|
||||||
|
all_result = WaitAll({sem0.get(), sem1.get()}, false, 1ms);
|
||||||
|
REQUIRE(all_result == WaitResult::kTimeout);
|
||||||
|
previous_count = -1;
|
||||||
|
REQUIRE(sem0->Release(1, &previous_count));
|
||||||
|
REQUIRE(previous_count == 0);
|
||||||
|
previous_count = -1;
|
||||||
|
REQUIRE_FALSE(sem1->Release(1, &previous_count));
|
||||||
|
REQUIRE(previous_count == -1);
|
||||||
|
|
||||||
|
// Test Wait all again which should succeed
|
||||||
|
sem0 = Semaphore::Create(1, 5);
|
||||||
|
sem1 = Semaphore::Create(5, 5);
|
||||||
|
all_result = WaitAll({sem0.get(), sem1.get()}, false, 1ms);
|
||||||
|
REQUIRE(all_result == WaitResult::kSuccess);
|
||||||
|
previous_count = -1;
|
||||||
|
REQUIRE(sem0->Release(1, &previous_count));
|
||||||
|
REQUIRE(previous_count == 0);
|
||||||
|
previous_count = -1;
|
||||||
|
REQUIRE(sem1->Release(1, &previous_count));
|
||||||
|
REQUIRE(previous_count == 4);
|
||||||
|
|
||||||
|
// Test Wait Any which should fail
|
||||||
|
sem0 = Semaphore::Create(0, 5);
|
||||||
|
sem1 = Semaphore::Create(0, 5);
|
||||||
|
any_result = WaitAny({sem0.get(), sem1.get()}, false, 1ms);
|
||||||
|
REQUIRE(any_result.first == WaitResult::kTimeout);
|
||||||
|
REQUIRE(any_result.second == 0);
|
||||||
|
previous_count = -1;
|
||||||
|
REQUIRE(sem0->Release(1, &previous_count));
|
||||||
|
REQUIRE(previous_count == 0);
|
||||||
|
previous_count = -1;
|
||||||
|
REQUIRE(sem1->Release(1, &previous_count));
|
||||||
|
REQUIRE(previous_count == 0);
|
||||||
|
|
||||||
|
// Test Wait Any which should succeed
|
||||||
|
sem0 = Semaphore::Create(0, 5);
|
||||||
|
sem1 = Semaphore::Create(5, 5);
|
||||||
|
any_result = WaitAny({sem0.get(), sem1.get()}, false, 1ms);
|
||||||
|
REQUIRE(any_result.first == WaitResult::kSuccess);
|
||||||
|
REQUIRE(any_result.second == 1);
|
||||||
|
previous_count = -1;
|
||||||
|
REQUIRE(sem0->Release(1, &previous_count));
|
||||||
|
REQUIRE(previous_count == 0);
|
||||||
|
previous_count = -1;
|
||||||
|
REQUIRE(sem1->Release(1, &previous_count));
|
||||||
|
REQUIRE(previous_count == 4);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("Wait on Mutant", "Mutant") {
|
||||||
|
WaitResult result;
|
||||||
|
std::unique_ptr<Mutant> mut;
|
||||||
|
|
||||||
|
// Release on initially owned mutant
|
||||||
|
mut = Mutant::Create(true);
|
||||||
|
REQUIRE(mut->Release());
|
||||||
|
REQUIRE_FALSE(mut->Release());
|
||||||
|
|
||||||
|
// Release on initially not-owned mutant
|
||||||
|
mut = Mutant::Create(false);
|
||||||
|
REQUIRE_FALSE(mut->Release());
|
||||||
|
|
||||||
|
// Wait on initially owned mutant
|
||||||
|
mut = Mutant::Create(true);
|
||||||
|
result = Wait(mut.get(), false, 1ms);
|
||||||
|
REQUIRE(result == WaitResult::kSuccess);
|
||||||
|
REQUIRE(mut->Release());
|
||||||
|
REQUIRE(mut->Release());
|
||||||
|
REQUIRE_FALSE(mut->Release());
|
||||||
|
|
||||||
|
// Wait on initially not owned mutant
|
||||||
|
mut = Mutant::Create(false);
|
||||||
|
result = Wait(mut.get(), false, 1ms);
|
||||||
|
REQUIRE(result == WaitResult::kSuccess);
|
||||||
|
REQUIRE(mut->Release());
|
||||||
|
REQUIRE_FALSE(mut->Release());
|
||||||
|
|
||||||
|
// Multiple waits (or locks)
|
||||||
|
mut = Mutant::Create(false);
|
||||||
|
for (int i = 0; i < 10; ++i) {
|
||||||
|
result = Wait(mut.get(), false, 1ms);
|
||||||
|
REQUIRE(result == WaitResult::kSuccess);
|
||||||
|
}
|
||||||
|
for (int i = 0; i < 10; ++i) {
|
||||||
|
REQUIRE(mut->Release());
|
||||||
|
}
|
||||||
|
REQUIRE_FALSE(mut->Release());
|
||||||
|
|
||||||
|
// Test mutants on other threads
|
||||||
|
auto thread1 = std::thread([&mut] {
|
||||||
|
Sleep(5ms);
|
||||||
|
mut = Mutant::Create(true);
|
||||||
|
Sleep(100ms);
|
||||||
|
mut->Release();
|
||||||
|
});
|
||||||
|
Sleep(10ms);
|
||||||
|
REQUIRE_FALSE(mut->Release());
|
||||||
|
Sleep(10ms);
|
||||||
|
result = Wait(mut.get(), false, 50ms);
|
||||||
|
REQUIRE(result == WaitResult::kTimeout);
|
||||||
|
thread1.join();
|
||||||
|
result = Wait(mut.get(), false, 1ms);
|
||||||
|
REQUIRE(result == WaitResult::kSuccess);
|
||||||
|
REQUIRE(mut->Release());
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("Wait on Multiple Mutants", "Mutant") {
|
||||||
|
WaitResult all_result;
|
||||||
|
std::pair<WaitResult, size_t> any_result;
|
||||||
|
std::unique_ptr<Mutant> mut0, mut1;
|
||||||
|
|
||||||
|
// Test which should fail for WaitAll and WaitAny
|
||||||
|
auto thread0 = std::thread([&mut0, &mut1] {
|
||||||
|
mut0 = Mutant::Create(true);
|
||||||
|
mut1 = Mutant::Create(true);
|
||||||
|
Sleep(5ms);
|
||||||
|
mut0->Release();
|
||||||
|
mut1->Release();
|
||||||
|
});
|
||||||
|
Sleep(1ms);
|
||||||
|
all_result = WaitAll({mut0.get(), mut1.get()}, false, 1ms);
|
||||||
|
REQUIRE(all_result == WaitResult::kTimeout);
|
||||||
|
REQUIRE_FALSE(mut0->Release());
|
||||||
|
REQUIRE_FALSE(mut1->Release());
|
||||||
|
any_result = WaitAny({mut0.get(), mut1.get()}, false, 1ms);
|
||||||
|
REQUIRE(any_result.first == WaitResult::kTimeout);
|
||||||
|
REQUIRE(any_result.second == 0);
|
||||||
|
REQUIRE_FALSE(mut0->Release());
|
||||||
|
REQUIRE_FALSE(mut1->Release());
|
||||||
|
thread0.join();
|
||||||
|
|
||||||
|
// Test which should fail for WaitAll but not WaitAny
|
||||||
|
auto thread1 = std::thread([&mut0, &mut1] {
|
||||||
|
mut0 = Mutant::Create(true);
|
||||||
|
mut1 = Mutant::Create(false);
|
||||||
|
Sleep(5ms);
|
||||||
|
mut0->Release();
|
||||||
|
});
|
||||||
|
Sleep(1ms);
|
||||||
|
all_result = WaitAll({mut0.get(), mut1.get()}, false, 1ms);
|
||||||
|
REQUIRE(all_result == WaitResult::kTimeout);
|
||||||
|
REQUIRE_FALSE(mut0->Release());
|
||||||
|
REQUIRE_FALSE(mut1->Release());
|
||||||
|
any_result = WaitAny({mut0.get(), mut1.get()}, false, 1ms);
|
||||||
|
REQUIRE(any_result.first == WaitResult::kSuccess);
|
||||||
|
REQUIRE(any_result.second == 1);
|
||||||
|
REQUIRE_FALSE(mut0->Release());
|
||||||
|
REQUIRE(mut1->Release());
|
||||||
|
thread1.join();
|
||||||
|
|
||||||
|
// Test which should pass for WaitAll and WaitAny
|
||||||
|
auto thread2 = std::thread([&mut0, &mut1] {
|
||||||
|
mut0 = Mutant::Create(false);
|
||||||
|
mut1 = Mutant::Create(false);
|
||||||
|
Sleep(5ms);
|
||||||
|
});
|
||||||
|
Sleep(1ms);
|
||||||
|
all_result = WaitAll({mut0.get(), mut1.get()}, false, 1ms);
|
||||||
|
REQUIRE(all_result == WaitResult::kSuccess);
|
||||||
|
REQUIRE(mut0->Release());
|
||||||
|
REQUIRE(mut1->Release());
|
||||||
|
any_result = WaitAny({mut0.get(), mut1.get()}, false, 1ms);
|
||||||
|
REQUIRE(any_result.first == WaitResult::kSuccess);
|
||||||
|
REQUIRE(any_result.second == 0);
|
||||||
|
REQUIRE(mut0->Release());
|
||||||
|
REQUIRE_FALSE(mut1->Release());
|
||||||
|
thread2.join();
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("Wait on Timer", "Timer") {
|
||||||
|
WaitResult result;
|
||||||
|
std::unique_ptr<Timer> timer;
|
||||||
|
|
||||||
|
// Test Manual Reset
|
||||||
|
timer = Timer::CreateManualResetTimer();
|
||||||
|
result = Wait(timer.get(), false, 1ms);
|
||||||
|
REQUIRE(result == WaitResult::kTimeout);
|
||||||
|
REQUIRE(timer->SetOnce(1ms)); // Signals it
|
||||||
|
result = Wait(timer.get(), false, 2ms);
|
||||||
|
REQUIRE(result == WaitResult::kSuccess);
|
||||||
|
result = Wait(timer.get(), false, 1ms);
|
||||||
|
REQUIRE(result == WaitResult::kSuccess); // Did not reset
|
||||||
|
|
||||||
|
// Test Synchronization
|
||||||
|
timer = Timer::CreateSynchronizationTimer();
|
||||||
|
result = Wait(timer.get(), false, 1ms);
|
||||||
|
REQUIRE(result == WaitResult::kTimeout);
|
||||||
|
REQUIRE(timer->SetOnce(1ms)); // Signals it
|
||||||
|
result = Wait(timer.get(), false, 2ms);
|
||||||
|
REQUIRE(result == WaitResult::kSuccess);
|
||||||
|
result = Wait(timer.get(), false, 1ms);
|
||||||
|
REQUIRE(result == WaitResult::kTimeout); // Did reset
|
||||||
|
|
||||||
|
// TODO(bwrsandman): This test unexpectedly fails under windows
|
||||||
|
// Test long due time
|
||||||
|
// timer = Timer::CreateSynchronizationTimer();
|
||||||
|
// REQUIRE(timer->SetOnce(10s));
|
||||||
|
// result = Wait(timer.get(), false, 10ms); // Still signals under windows
|
||||||
|
// REQUIRE(result == WaitResult::kTimeout);
|
||||||
|
|
||||||
|
// Test Repeating
|
||||||
|
REQUIRE(timer->SetRepeating(100us, 1ms));
|
||||||
|
for (int i = 0; i < 10; ++i) {
|
||||||
|
result = Wait(timer.get(), false, 2ms);
|
||||||
|
INFO(i);
|
||||||
|
REQUIRE(result == WaitResult::kSuccess);
|
||||||
|
}
|
||||||
|
MaybeYield();
|
||||||
|
Sleep(1ms); // Skip a few events
|
||||||
|
for (int i = 0; i < 10; ++i) {
|
||||||
|
result = Wait(timer.get(), false, 2ms);
|
||||||
|
REQUIRE(result == WaitResult::kSuccess);
|
||||||
|
}
|
||||||
|
// Cancel it
|
||||||
|
timer->Cancel();
|
||||||
|
result = Wait(timer.get(), false, 2ms);
|
||||||
|
REQUIRE(result == WaitResult::kTimeout);
|
||||||
|
MaybeYield();
|
||||||
|
Sleep(1ms); // Skip a few events
|
||||||
|
result = Wait(timer.get(), false, 2ms);
|
||||||
|
REQUIRE(result == WaitResult::kTimeout);
|
||||||
|
// Cancel with SetOnce
|
||||||
|
REQUIRE(timer->SetRepeating(1ms, 1ms));
|
||||||
|
for (int i = 0; i < 10; ++i) {
|
||||||
|
result = Wait(timer.get(), false, 2ms);
|
||||||
|
REQUIRE(result == WaitResult::kSuccess);
|
||||||
|
}
|
||||||
|
REQUIRE(timer->SetOnce(100us));
|
||||||
|
result = Wait(timer.get(), false, 2ms);
|
||||||
|
REQUIRE(result == WaitResult::kSuccess); // Signal from Set Once
|
||||||
|
result = Wait(timer.get(), false, 2ms);
|
||||||
|
REQUIRE(result == WaitResult::kTimeout); // No more signals from repeating
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("Wait on Multiple Timers", "Timer") {
|
||||||
|
WaitResult all_result;
|
||||||
|
std::pair<WaitResult, size_t> any_result;
|
||||||
|
|
||||||
|
auto timer0 = Timer::CreateSynchronizationTimer();
|
||||||
|
auto timer1 = Timer::CreateManualResetTimer();
|
||||||
|
|
||||||
|
// None signaled
|
||||||
|
all_result = WaitAll({timer0.get(), timer1.get()}, false, 1ms);
|
||||||
|
REQUIRE(all_result == WaitResult::kTimeout);
|
||||||
|
any_result = WaitAny({timer0.get(), timer1.get()}, false, 1ms);
|
||||||
|
REQUIRE(any_result.first == WaitResult::kTimeout);
|
||||||
|
REQUIRE(any_result.second == 0);
|
||||||
|
|
||||||
|
// Some signaled
|
||||||
|
REQUIRE(timer1->SetOnce(1ms));
|
||||||
|
all_result = WaitAll({timer0.get(), timer1.get()}, false, 10ms);
|
||||||
|
REQUIRE(all_result == WaitResult::kTimeout);
|
||||||
|
any_result = WaitAny({timer0.get(), timer1.get()}, false, 10ms);
|
||||||
|
REQUIRE(any_result.first == WaitResult::kSuccess);
|
||||||
|
REQUIRE(any_result.second == 1);
|
||||||
|
|
||||||
|
// All signaled
|
||||||
|
REQUIRE(timer0->SetOnce(1ms));
|
||||||
|
all_result = WaitAll({timer0.get(), timer1.get()}, false, 10ms);
|
||||||
|
REQUIRE(all_result == WaitResult::kSuccess);
|
||||||
|
REQUIRE(timer0->SetOnce(1ms));
|
||||||
|
Sleep(1ms);
|
||||||
|
any_result = WaitAny({timer0.get(), timer1.get()}, false, 10ms);
|
||||||
|
REQUIRE(any_result.first == WaitResult::kSuccess);
|
||||||
|
REQUIRE(any_result.second == 0);
|
||||||
|
|
||||||
|
// Check that timer0 reset
|
||||||
|
any_result = WaitAny({timer0.get(), timer1.get()}, false, 10ms);
|
||||||
|
REQUIRE(any_result.first == WaitResult::kSuccess);
|
||||||
|
REQUIRE(any_result.second == 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("Create and Trigger Timer Callbacks", "Timer") {
|
||||||
|
// TODO(bwrsandman): Check which thread performs callback and timing of
|
||||||
|
// callback
|
||||||
|
REQUIRE(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("Set and Test Current Thread ID", "Thread") {
|
||||||
|
// System ID
|
||||||
|
auto system_id = current_thread_system_id();
|
||||||
|
REQUIRE(system_id > 0);
|
||||||
|
|
||||||
|
// Thread ID
|
||||||
|
auto thread_id = current_thread_id();
|
||||||
|
REQUIRE(thread_id == system_id);
|
||||||
|
|
||||||
|
// Set a new thread id
|
||||||
|
const uint32_t new_thread_id = 0xDEADBEEF;
|
||||||
|
set_current_thread_id(new_thread_id);
|
||||||
|
REQUIRE(current_thread_id() == new_thread_id);
|
||||||
|
|
||||||
|
// Set back original thread id of system
|
||||||
|
set_current_thread_id(std::numeric_limits<uint32_t>::max());
|
||||||
|
REQUIRE(current_thread_id() == system_id);
|
||||||
|
|
||||||
|
// TODO(bwrsandman): Test on Thread object
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("Set and Test Current Thread Name", "Thread") {
|
||||||
|
auto current_thread = Thread::GetCurrentThread();
|
||||||
|
REQUIRE(current_thread);
|
||||||
|
auto old_thread_name = current_thread->name();
|
||||||
|
|
||||||
|
std::string new_thread_name = "Threading Test";
|
||||||
|
REQUIRE_NOTHROW(set_name(new_thread_name));
|
||||||
|
|
||||||
|
// Restore the old catch.hpp thread name
|
||||||
|
REQUIRE_NOTHROW(set_name(old_thread_name));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("Create and Run Thread", "Thread") {
|
||||||
|
std::unique_ptr<Thread> thread;
|
||||||
|
WaitResult result;
|
||||||
|
Thread::CreationParameters params = {};
|
||||||
|
auto func = [] { Sleep(20ns); };
|
||||||
|
|
||||||
|
// Create most basic case of thread
|
||||||
|
thread = Thread::Create(params, func);
|
||||||
|
REQUIRE(thread->native_handle() != nullptr);
|
||||||
|
REQUIRE_NOTHROW(thread->affinity_mask());
|
||||||
|
REQUIRE(thread->name().empty());
|
||||||
|
result = Wait(thread.get(), false, 5ms);
|
||||||
|
REQUIRE(result == WaitResult::kSuccess);
|
||||||
|
|
||||||
|
// Add thread name
|
||||||
|
std::string new_name = "Test thread name";
|
||||||
|
thread = Thread::Create(params, func);
|
||||||
|
auto name = thread->name();
|
||||||
|
INFO(name.c_str());
|
||||||
|
REQUIRE(name.empty());
|
||||||
|
thread->set_name(new_name);
|
||||||
|
REQUIRE(thread->name() == new_name);
|
||||||
|
result = Wait(thread.get(), false, 5ms);
|
||||||
|
REQUIRE(result == WaitResult::kSuccess);
|
||||||
|
|
||||||
|
// Use Terminate to end an infinitely looping thread
|
||||||
|
thread = Thread::Create(params, [] {
|
||||||
|
while (true) {
|
||||||
|
Sleep(1ms);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
result = Wait(thread.get(), false, 5ms);
|
||||||
|
REQUIRE(result == WaitResult::kTimeout);
|
||||||
|
thread->Terminate(-1);
|
||||||
|
result = Wait(thread.get(), false, 5ms);
|
||||||
|
REQUIRE(result == WaitResult::kSuccess);
|
||||||
|
|
||||||
|
// Call Exit from inside an infinitely looping thread
|
||||||
|
thread = Thread::Create(params, [] {
|
||||||
|
while (true) {
|
||||||
|
Thread::Exit(-1);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
result = Wait(thread.get(), false, 5ms);
|
||||||
|
REQUIRE(result == WaitResult::kSuccess);
|
||||||
|
|
||||||
|
// Call timeout wait on self
|
||||||
|
result = Wait(Thread::GetCurrentThread(), false, 5ms);
|
||||||
|
REQUIRE(result == WaitResult::kTimeout);
|
||||||
|
|
||||||
|
params.stack_size = 16 * 1024;
|
||||||
|
thread = Thread::Create(params, [] {
|
||||||
|
while (true) {
|
||||||
|
Thread::Exit(-1);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
REQUIRE(thread != nullptr);
|
||||||
|
result = Wait(thread.get(), false, 5ms);
|
||||||
|
REQUIRE(result == WaitResult::kSuccess);
|
||||||
|
|
||||||
|
// TODO(bwrsandman): Test with different priorities
|
||||||
|
// TODO(bwrsandman): Test setting and getting thread affinity
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("Test Suspending Thread", "Thread") {
|
||||||
|
std::unique_ptr<Thread> thread;
|
||||||
|
WaitResult result;
|
||||||
|
Thread::CreationParameters params = {};
|
||||||
|
auto func = [] { Sleep(20us); };
|
||||||
|
|
||||||
|
// Create initially suspended
|
||||||
|
params.create_suspended = true;
|
||||||
|
thread = threading::Thread::Create(params, func);
|
||||||
|
result = threading::Wait(thread.get(), false, 5ms);
|
||||||
|
REQUIRE(result == threading::WaitResult::kTimeout);
|
||||||
|
thread->Resume();
|
||||||
|
result = threading::Wait(thread.get(), false, 5ms);
|
||||||
|
REQUIRE(result == threading::WaitResult::kSuccess);
|
||||||
|
params.create_suspended = false;
|
||||||
|
|
||||||
|
// Create and then suspend
|
||||||
|
thread = threading::Thread::Create(params, func);
|
||||||
|
thread->Suspend();
|
||||||
|
result = threading::Wait(thread.get(), false, 5ms);
|
||||||
|
REQUIRE(result == threading::WaitResult::kTimeout);
|
||||||
|
thread->Resume();
|
||||||
|
result = threading::Wait(thread.get(), false, 5ms);
|
||||||
|
REQUIRE(result == threading::WaitResult::kSuccess);
|
||||||
|
|
||||||
|
// Test recursive suspend
|
||||||
|
thread = threading::Thread::Create(params, func);
|
||||||
|
thread->Suspend();
|
||||||
|
thread->Suspend();
|
||||||
|
result = threading::Wait(thread.get(), false, 5ms);
|
||||||
|
REQUIRE(result == threading::WaitResult::kTimeout);
|
||||||
|
thread->Resume();
|
||||||
|
result = threading::Wait(thread.get(), false, 5ms);
|
||||||
|
REQUIRE(result == threading::WaitResult::kTimeout);
|
||||||
|
thread->Resume();
|
||||||
|
result = threading::Wait(thread.get(), false, 5ms);
|
||||||
|
REQUIRE(result == threading::WaitResult::kSuccess);
|
||||||
|
|
||||||
|
// Test suspend count
|
||||||
|
uint32_t suspend_count = 0;
|
||||||
|
thread = threading::Thread::Create(params, func);
|
||||||
|
thread->Suspend(&suspend_count);
|
||||||
|
REQUIRE(suspend_count == 0);
|
||||||
|
thread->Suspend(&suspend_count);
|
||||||
|
REQUIRE(suspend_count == 1);
|
||||||
|
thread->Suspend(&suspend_count);
|
||||||
|
REQUIRE(suspend_count == 2);
|
||||||
|
thread->Resume(&suspend_count);
|
||||||
|
REQUIRE(suspend_count == 3);
|
||||||
|
thread->Resume(&suspend_count);
|
||||||
|
REQUIRE(suspend_count == 2);
|
||||||
|
thread->Resume(&suspend_count);
|
||||||
|
REQUIRE(suspend_count == 1);
|
||||||
|
thread->Suspend(&suspend_count);
|
||||||
|
REQUIRE(suspend_count == 0);
|
||||||
|
thread->Resume(&suspend_count);
|
||||||
|
REQUIRE(suspend_count == 1);
|
||||||
|
result = threading::Wait(thread.get(), false, 5ms);
|
||||||
|
REQUIRE(result == threading::WaitResult::kSuccess);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("Test Thread QueueUserCallback", "Thread") {
|
||||||
|
std::unique_ptr<Thread> thread;
|
||||||
|
WaitResult result;
|
||||||
|
Thread::CreationParameters params = {};
|
||||||
|
std::atomic_int order;
|
||||||
|
int is_modified;
|
||||||
|
int has_finished;
|
||||||
|
auto callback = [&is_modified, &order] {
|
||||||
|
is_modified = std::atomic_fetch_add_explicit(
|
||||||
|
&order, 1, std::memory_order::memory_order_relaxed);
|
||||||
|
};
|
||||||
|
|
||||||
|
// Without alertable
|
||||||
|
order = 0;
|
||||||
|
is_modified = -1;
|
||||||
|
has_finished = -1;
|
||||||
|
thread = Thread::Create(params, [&has_finished, &order] {
|
||||||
|
// Not using Alertable so callback is not registered
|
||||||
|
Sleep(9ms);
|
||||||
|
has_finished = std::atomic_fetch_add_explicit(
|
||||||
|
&order, 1, std::memory_order::memory_order_relaxed);
|
||||||
|
});
|
||||||
|
result = Wait(thread.get(), true, 5ms);
|
||||||
|
REQUIRE(result == WaitResult::kTimeout);
|
||||||
|
REQUIRE(is_modified == -1);
|
||||||
|
thread->QueueUserCallback(callback);
|
||||||
|
result = Wait(thread.get(), true, 10ms);
|
||||||
|
REQUIRE(result == WaitResult::kSuccess);
|
||||||
|
REQUIRE(is_modified == -1);
|
||||||
|
REQUIRE(has_finished == 0);
|
||||||
|
|
||||||
|
// With alertable
|
||||||
|
order = 0;
|
||||||
|
is_modified = -1;
|
||||||
|
has_finished = -1;
|
||||||
|
thread = Thread::Create(params, [&has_finished, &order] {
|
||||||
|
// Using Alertable so callback is registered
|
||||||
|
AlertableSleep(9ms);
|
||||||
|
has_finished = std::atomic_fetch_add_explicit(
|
||||||
|
&order, 1, std::memory_order::memory_order_relaxed);
|
||||||
|
});
|
||||||
|
result = Wait(thread.get(), true, 5ms);
|
||||||
|
REQUIRE(result == WaitResult::kTimeout);
|
||||||
|
REQUIRE(is_modified == -1);
|
||||||
|
thread->QueueUserCallback(callback);
|
||||||
|
result = Wait(thread.get(), true, 10ms);
|
||||||
|
REQUIRE(result == WaitResult::kSuccess);
|
||||||
|
REQUIRE(is_modified == 0);
|
||||||
|
REQUIRE(has_finished == 1);
|
||||||
|
|
||||||
|
// Test Exit command with QueueUserCallback
|
||||||
|
order = 0;
|
||||||
|
is_modified = -1;
|
||||||
|
has_finished = -1;
|
||||||
|
thread = Thread::Create(params, [&is_modified, &has_finished, &order] {
|
||||||
|
is_modified = std::atomic_fetch_add_explicit(
|
||||||
|
&order, 1, std::memory_order::memory_order_relaxed);
|
||||||
|
// Using Alertable so callback is registered
|
||||||
|
AlertableSleep(20ms);
|
||||||
|
has_finished = std::atomic_fetch_add_explicit(
|
||||||
|
&order, 1, std::memory_order::memory_order_relaxed);
|
||||||
|
});
|
||||||
|
result = Wait(thread.get(), true, 10ms);
|
||||||
|
REQUIRE(result == WaitResult::kTimeout);
|
||||||
|
thread->QueueUserCallback([] { Thread::Exit(0); });
|
||||||
|
result = Wait(thread.get(), true, 50ms);
|
||||||
|
REQUIRE(result == WaitResult::kSuccess);
|
||||||
|
REQUIRE(is_modified == 0);
|
||||||
|
REQUIRE(has_finished == -1);
|
||||||
|
|
||||||
|
// TODO(bwrsandman): Test alertable wait returning kUserCallback by using IO
|
||||||
|
// callbacks.
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace test
|
||||||
|
} // namespace base
|
||||||
|
} // namespace xe
|
|
@ -32,21 +32,19 @@ class Fence {
|
||||||
Fence() : signaled_(false) {}
|
Fence() : signaled_(false) {}
|
||||||
void Signal() {
|
void Signal() {
|
||||||
std::unique_lock<std::mutex> lock(mutex_);
|
std::unique_lock<std::mutex> lock(mutex_);
|
||||||
signaled_.store(true);
|
signaled_ = true;
|
||||||
cond_.notify_all();
|
cond_.notify_all();
|
||||||
}
|
}
|
||||||
void Wait() {
|
void Wait() {
|
||||||
std::unique_lock<std::mutex> lock(mutex_);
|
std::unique_lock<std::mutex> lock(mutex_);
|
||||||
while (!signaled_.load()) {
|
cond_.wait(lock, [this] { return signaled_; });
|
||||||
cond_.wait(lock);
|
signaled_ = false;
|
||||||
}
|
|
||||||
signaled_.store(false);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
std::mutex mutex_;
|
std::mutex mutex_;
|
||||||
std::condition_variable cond_;
|
std::condition_variable cond_;
|
||||||
std::atomic<bool> signaled_;
|
bool signaled_;
|
||||||
};
|
};
|
||||||
|
|
||||||
// Returns the total number of logical processors in the host system.
|
// Returns the total number of logical processors in the host system.
|
||||||
|
@ -307,12 +305,12 @@ class Timer : public WaitHandle {
|
||||||
std::chrono::milliseconds period,
|
std::chrono::milliseconds period,
|
||||||
std::function<void()> opt_callback = nullptr) = 0;
|
std::function<void()> opt_callback = nullptr) = 0;
|
||||||
template <typename Rep, typename Period>
|
template <typename Rep, typename Period>
|
||||||
void SetRepeating(std::chrono::nanoseconds due_time,
|
bool SetRepeating(std::chrono::nanoseconds due_time,
|
||||||
std::chrono::duration<Rep, Period> period,
|
std::chrono::duration<Rep, Period> period,
|
||||||
std::function<void()> opt_callback = nullptr) {
|
std::function<void()> opt_callback = nullptr) {
|
||||||
SetRepeating(due_time,
|
return SetRepeating(
|
||||||
std::chrono::duration_cast<std::chrono::milliseconds>(period),
|
due_time, std::chrono::duration_cast<std::chrono::milliseconds>(period),
|
||||||
std::move(opt_callback));
|
std::move(opt_callback));
|
||||||
}
|
}
|
||||||
|
|
||||||
// Stops the timer before it can be set to the signaled state and cancels
|
// Stops the timer before it can be set to the signaled state and cancels
|
||||||
|
@ -358,7 +356,7 @@ class Thread : public WaitHandle {
|
||||||
virtual uint32_t system_id() const = 0;
|
virtual uint32_t system_id() const = 0;
|
||||||
|
|
||||||
// Returns the current name of the thread, if previously specified.
|
// Returns the current name of the thread, if previously specified.
|
||||||
std::string name() const { return name_; }
|
virtual std::string name() const { return name_; }
|
||||||
|
|
||||||
// Sets the name of the thread, used in debugging and logging.
|
// Sets the name of the thread, used in debugging and logging.
|
||||||
virtual void set_name(std::string name) { name_ = std::move(name); }
|
virtual void set_name(std::string name) { name_ = std::move(name); }
|
||||||
|
@ -390,7 +388,7 @@ class Thread : public WaitHandle {
|
||||||
|
|
||||||
// Decrements a thread's suspend count. When the suspend count is decremented
|
// Decrements a thread's suspend count. When the suspend count is decremented
|
||||||
// to zero, the execution of the thread is resumed.
|
// to zero, the execution of the thread is resumed.
|
||||||
virtual bool Resume(uint32_t* out_new_suspend_count = nullptr) = 0;
|
virtual bool Resume(uint32_t* out_previous_suspend_count = nullptr) = 0;
|
||||||
|
|
||||||
// Suspends the specified thread.
|
// Suspends the specified thread.
|
||||||
virtual bool Suspend(uint32_t* out_previous_suspend_count = nullptr) = 0;
|
virtual bool Suspend(uint32_t* out_previous_suspend_count = nullptr) = 0;
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -378,16 +378,16 @@ class Win32Thread : public Win32Handle<Thread> {
|
||||||
QueueUserAPC(DispatchApc, handle_, reinterpret_cast<ULONG_PTR>(apc_data));
|
QueueUserAPC(DispatchApc, handle_, reinterpret_cast<ULONG_PTR>(apc_data));
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Resume(uint32_t* out_new_suspend_count = nullptr) override {
|
bool Resume(uint32_t* out_previous_suspend_count = nullptr) override {
|
||||||
if (out_new_suspend_count) {
|
if (out_previous_suspend_count) {
|
||||||
*out_new_suspend_count = 0;
|
*out_previous_suspend_count = 0;
|
||||||
}
|
}
|
||||||
DWORD result = ResumeThread(handle_);
|
DWORD result = ResumeThread(handle_);
|
||||||
if (result == UINT_MAX) {
|
if (result == UINT_MAX) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if (out_new_suspend_count) {
|
if (out_previous_suspend_count) {
|
||||||
*out_new_suspend_count = result;
|
*out_previous_suspend_count = result;
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
|
@ -73,7 +73,7 @@ bool CommandProcessor::Initialize(
|
||||||
WorkerThreadMain();
|
WorkerThreadMain();
|
||||||
return 0;
|
return 0;
|
||||||
}));
|
}));
|
||||||
worker_thread_->set_name("GraphicsSystem Command Processor");
|
worker_thread_->set_name("GPU Commands");
|
||||||
worker_thread_->Create();
|
worker_thread_->Create();
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
|
|
@ -129,7 +129,7 @@ X_STATUS GraphicsSystem::Setup(cpu::Processor* processor,
|
||||||
}));
|
}));
|
||||||
// As we run vblank interrupts the debugger must be able to suspend us.
|
// As we run vblank interrupts the debugger must be able to suspend us.
|
||||||
vsync_worker_thread_->set_can_debugger_suspend(true);
|
vsync_worker_thread_->set_can_debugger_suspend(true);
|
||||||
vsync_worker_thread_->set_name("GraphicsSystem Vsync");
|
vsync_worker_thread_->set_name("GPU VSync");
|
||||||
vsync_worker_thread_->Create();
|
vsync_worker_thread_->Create();
|
||||||
|
|
||||||
if (cvars::trace_gpu_stream) {
|
if (cvars::trace_gpu_stream) {
|
||||||
|
|
|
@ -242,10 +242,7 @@ object_ref<XThread> KernelState::LaunchModule(object_ref<UserModule> module) {
|
||||||
module->entry_point(), 0, X_CREATE_SUSPENDED, true, true));
|
module->entry_point(), 0, X_CREATE_SUSPENDED, true, true));
|
||||||
|
|
||||||
// We know this is the 'main thread'.
|
// We know this is the 'main thread'.
|
||||||
char thread_name[32];
|
thread->set_name("Main XThread");
|
||||||
std::snprintf(thread_name, xe::countof(thread_name), "Main XThread%08X",
|
|
||||||
thread->handle());
|
|
||||||
thread->set_name(thread_name);
|
|
||||||
|
|
||||||
X_STATUS result = thread->Create();
|
X_STATUS result = thread->Create();
|
||||||
if (XFAILED(result)) {
|
if (XFAILED(result)) {
|
||||||
|
@ -340,7 +337,7 @@ void KernelState::SetExecutableModule(object_ref<UserModule> module) {
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}));
|
}));
|
||||||
dispatch_thread_->set_name("Kernel Dispatch Thread");
|
dispatch_thread_->set_name("Kernel Dispatch");
|
||||||
dispatch_thread_->Create();
|
dispatch_thread_->Create();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue