[threading] Simplify and test Fence
Remove atomic boolean in fence. Variable signaled_ is already protected by mutex. Remove wait loop with single predicate wait protected with mutex. Add Fence Signal and Wait tests Test signaling without waiting. Test signaling before waiting. Test signaling twice before waiting. Test synchronizing threads with fence. Few REQUIRES were used to test as there are no return codes. A failing test may hang indefinitely or cause a segfault which would still register as a fail.
This commit is contained in:
parent
b5ea686475
commit
4280a6451d
|
@ -18,9 +18,56 @@ using namespace threading;
|
||||||
using namespace std::chrono_literals;
|
using namespace std::chrono_literals;
|
||||||
|
|
||||||
TEST_CASE("Fence") {
|
TEST_CASE("Fence") {
|
||||||
// TODO(bwrsandman):
|
std::unique_ptr<threading::Fence> pFence;
|
||||||
REQUIRE(true);
|
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(100ms);
|
||||||
|
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(10ms);
|
||||||
|
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") {
|
TEST_CASE("Get number of logical processors") {
|
||||||
auto count = std::thread::hardware_concurrency();
|
auto count = std::thread::hardware_concurrency();
|
||||||
|
|
|
@ -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.
|
||||||
|
|
Loading…
Reference in New Issue