Files
clang-p2996/compiler-rt/lib/gwp_asan/tests/mutex_test.cpp
Kostya Kortchinsky 5556616b5b [GWP-ASan] Port tests to Fuchsia
This modifies the tests so that they can be run on Fuchsia:
- add the necessary includes for `set`/`vector` etc
- do the few modifications required to use zxtest instead og gtest

`backtrace.cpp` requires stacktrace support that Fuchsia doesn't have
yet, and `enable_disable.cpp` currently uses `fork()` which Fuchsia
doesn't support yet. I'll revisit this later.

I chose to use `harness.h` to hold my "platform-specific" include and
namespace, and using this header in tests rather than `gtest.h`,
which I am open to change if someone would rather go another direction.

Differential Revision: https://reviews.llvm.org/D91575
2020-11-18 13:36:12 -08:00

90 lines
2.3 KiB
C++

//===-- mutex_test.cpp ------------------------------------------*- C++ -*-===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
#include "gwp_asan/mutex.h"
#include "gwp_asan/tests/harness.h"
#include <atomic>
#include <mutex>
#include <thread>
#include <vector>
using gwp_asan::Mutex;
using gwp_asan::ScopedLock;
TEST(GwpAsanMutexTest, LockUnlockTest) {
Mutex Mu;
ASSERT_TRUE(Mu.tryLock());
ASSERT_FALSE(Mu.tryLock());
Mu.unlock();
Mu.lock();
Mu.unlock();
// Ensure that the mutex actually unlocked.
ASSERT_TRUE(Mu.tryLock());
Mu.unlock();
}
TEST(GwpAsanMutexTest, ScopedLockUnlockTest) {
Mutex Mu;
{ ScopedLock L(Mu); }
// Locking will fail here if the scoped lock failed to unlock.
EXPECT_TRUE(Mu.tryLock());
Mu.unlock();
{
ScopedLock L(Mu);
EXPECT_FALSE(Mu.tryLock()); // Check that the c'tor did lock.
// Manually unlock and check that this succeeds.
Mu.unlock();
EXPECT_TRUE(Mu.tryLock()); // Manually lock.
}
EXPECT_TRUE(Mu.tryLock()); // Assert that the scoped destructor did unlock.
Mu.unlock();
}
static void synchronousIncrementTask(std::atomic<bool> *StartingGun, Mutex *Mu,
unsigned *Counter,
unsigned NumIterations) {
while (!StartingGun) {
// Wait for starting gun.
}
for (unsigned i = 0; i < NumIterations; ++i) {
ScopedLock L(*Mu);
(*Counter)++;
}
}
static void runSynchronisedTest(unsigned NumThreads, unsigned CounterMax) {
std::vector<std::thread> Threads;
ASSERT_TRUE(CounterMax % NumThreads == 0);
std::atomic<bool> StartingGun{false};
Mutex Mu;
unsigned Counter = 0;
for (unsigned i = 0; i < NumThreads; ++i)
Threads.emplace_back(synchronousIncrementTask, &StartingGun, &Mu, &Counter,
CounterMax / NumThreads);
StartingGun = true;
for (auto &T : Threads)
T.join();
EXPECT_EQ(CounterMax, Counter);
}
TEST(GwpAsanMutexTest, SynchronisedCounterTest) {
runSynchronisedTest(4, 1000000);
runSynchronisedTest(100, 1000000);
}