Some time ago, I introduced shortcut features like dylib-has-no-shared_mutex to encode whether the deployment target supported shared_mutex (say). This made the test suite annotations cleaner. However, the problem with building Lit features on top of other Lit features is that it's easier for them to become stale, especially when they are generated programmatically. Furthermore, it makes the bar for defining configurations from scratch higher, since more features have to be defined. Instead, I think it's better to put the XFAILs in the tests directly, which allows cleaning them up with a simple grep.
110 lines
2.6 KiB
C++
110 lines
2.6 KiB
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
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// UNSUPPORTED: libcpp-has-no-threads
|
|
// UNSUPPORTED: c++03, c++11
|
|
|
|
// dylib support for shared_mutex was added in macosx10.12
|
|
// XFAIL: with_system_cxx_lib=macosx10.11
|
|
// XFAIL: with_system_cxx_lib=macosx10.10
|
|
// XFAIL: with_system_cxx_lib=macosx10.9
|
|
|
|
// ALLOW_RETRIES: 2
|
|
|
|
// <shared_mutex>
|
|
|
|
// template <class Mutex> class shared_lock;
|
|
|
|
// explicit shared_lock(mutex_type& m);
|
|
|
|
// template<class _Mutex> shared_lock(shared_lock<_Mutex>)
|
|
// -> shared_lock<_Mutex>; // C++17
|
|
|
|
#include <shared_mutex>
|
|
#include <thread>
|
|
#include <vector>
|
|
#include <cstdlib>
|
|
#include <cassert>
|
|
|
|
#include "test_macros.h"
|
|
|
|
typedef std::chrono::system_clock Clock;
|
|
typedef Clock::time_point time_point;
|
|
typedef Clock::duration duration;
|
|
typedef std::chrono::milliseconds ms;
|
|
typedef std::chrono::nanoseconds ns;
|
|
|
|
ms WaitTime = ms(250);
|
|
|
|
// Thread sanitizer causes more overhead and will sometimes cause this test
|
|
// to fail. To prevent this we give Thread sanitizer more time to complete the
|
|
// test.
|
|
#if !defined(TEST_HAS_SANITIZERS)
|
|
ms Tolerance = ms(50);
|
|
#else
|
|
ms Tolerance = ms(50 * 5);
|
|
#endif
|
|
|
|
std::shared_timed_mutex m;
|
|
|
|
void f()
|
|
{
|
|
time_point t0 = Clock::now();
|
|
time_point t1;
|
|
{
|
|
std::shared_lock<std::shared_timed_mutex> ul(m);
|
|
t1 = Clock::now();
|
|
}
|
|
ns d = t1 - t0 - WaitTime;
|
|
assert(d < Tolerance); // within tolerance
|
|
}
|
|
|
|
void g()
|
|
{
|
|
time_point t0 = Clock::now();
|
|
time_point t1;
|
|
{
|
|
std::shared_lock<std::shared_timed_mutex> ul(m);
|
|
t1 = Clock::now();
|
|
}
|
|
ns d = t1 - t0;
|
|
assert(d < Tolerance); // within tolerance
|
|
}
|
|
|
|
int main(int, char**)
|
|
{
|
|
std::vector<std::thread> v;
|
|
{
|
|
m.lock();
|
|
for (int i = 0; i < 5; ++i)
|
|
v.push_back(std::thread(f));
|
|
std::this_thread::sleep_for(WaitTime);
|
|
m.unlock();
|
|
for (auto& t : v)
|
|
t.join();
|
|
}
|
|
{
|
|
m.lock_shared();
|
|
for (auto& t : v)
|
|
t = std::thread(g);
|
|
std::thread q(f);
|
|
std::this_thread::sleep_for(WaitTime);
|
|
m.unlock_shared();
|
|
for (auto& t : v)
|
|
t.join();
|
|
q.join();
|
|
}
|
|
|
|
#ifdef __cpp_deduction_guides
|
|
std::shared_lock sl(m);
|
|
static_assert((std::is_same<decltype(sl), std::shared_lock<decltype(m)>>::value), "" );
|
|
#endif
|
|
|
|
return 0;
|
|
}
|