dc980ae463411eca0469a1e77daa1752ec5346d1
[llvm-toolchain-13.git] /
1 //===----------------------------------------------------------------------===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 //
9 // UNSUPPORTED: libcpp-has-no-threads
10 // UNSUPPORTED: c++03, c++11
11
12 // dylib support for shared_mutex was added in macosx10.12
13 // XFAIL: use_system_cxx_lib && target={{.+}}-apple-macosx10.{{9|10|11}}
14
15 // ALLOW_RETRIES: 2
16
17 // <shared_mutex>
18
19 // template <class Mutex> class shared_lock;
20
21 // void lock();
22
23 #include <shared_mutex>
24 #include <thread>
25 #include <vector>
26 #include <cstdlib>
27 #include <cassert>
28
29 #include "make_test_thread.h"
30 #include "test_macros.h"
31
32 std::shared_timed_mutex m;
33
34 typedef std::chrono::system_clock Clock;
35 typedef Clock::time_point time_point;
36 typedef Clock::duration duration;
37 typedef std::chrono::milliseconds ms;
38 typedef std::chrono::nanoseconds ns;
39
40 ms WaitTime = ms(250);
41
42 // Thread sanitizer causes more overhead and will sometimes cause this test
43 // to fail. To prevent this we give Thread sanitizer more time to complete the
44 // test.
45 #if !defined(TEST_HAS_SANITIZERS)
46 ms Tolerance = ms(25);
47 #else
48 ms Tolerance = ms(25 * 5);
49 #endif
50
51
52 void f()
53 {
54     std::shared_lock<std::shared_timed_mutex> lk(m, std::defer_lock);
55     time_point t0 = Clock::now();
56     lk.lock();
57     time_point t1 = Clock::now();
58     assert(lk.owns_lock() == true);
59     ns d = t1 - t0 - WaitTime;
60     assert(d < Tolerance);  // within tolerance
61 #ifndef TEST_HAS_NO_EXCEPTIONS
62     try
63     {
64         lk.lock();
65         assert(false);
66     }
67     catch (std::system_error& e)
68     {
69         assert(e.code().value() == EDEADLK);
70     }
71 #endif
72     lk.unlock();
73     lk.release();
74 #ifndef TEST_HAS_NO_EXCEPTIONS
75     try
76     {
77         lk.lock();
78         assert(false);
79     }
80     catch (std::system_error& e)
81     {
82         assert(e.code().value() == EPERM);
83     }
84 #endif
85 }
86
87 int main(int, char**)
88 {
89     m.lock();
90     std::vector<std::thread> v;
91     for (int i = 0; i < 5; ++i)
92         v.push_back(support::make_test_thread(f));
93     std::this_thread::sleep_for(WaitTime);
94     m.unlock();
95     for (auto& t : v)
96         t.join();
97
98   return 0;
99 }