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++98, c++03
11 
12 // <future>
13 
14 // class shared_future<R>
15 
16 // template <class Clock, class Duration>
17 //   future_status
18 //   wait_until(const chrono::time_point<Clock, Duration>& abs_time) const;
19 
20 #include <future>
21 #include <atomic>
22 #include <cassert>
23 
24 #include "test_macros.h"
25 
26 enum class WorkerThreadState { Uninitialized, AllowedToRun, Exiting };
27 typedef std::chrono::milliseconds ms;
28 
29 std::atomic<WorkerThreadState> thread_state(WorkerThreadState::Uninitialized);
30 
31 void set_worker_thread_state(WorkerThreadState state)
32 {
33   thread_state.store(state, std::memory_order_relaxed);
34 }
35 
36 void wait_for_worker_thread_state(WorkerThreadState state)
37 {
38   while (thread_state.load(std::memory_order_relaxed) != state)
39     std::this_thread::yield();
40 }
41 
42 void func1(std::promise<int> p)
43 {
44   wait_for_worker_thread_state(WorkerThreadState::AllowedToRun);
45   p.set_value(3);
46   set_worker_thread_state(WorkerThreadState::Exiting);
47 }
48 
49 int j = 0;
50 
51 void func3(std::promise<int&> p)
52 {
53   wait_for_worker_thread_state(WorkerThreadState::AllowedToRun);
54   j = 5;
55   p.set_value(j);
56   set_worker_thread_state(WorkerThreadState::Exiting);
57 }
58 
59 void func5(std::promise<void> p)
60 {
61   wait_for_worker_thread_state(WorkerThreadState::AllowedToRun);
62   p.set_value();
63   set_worker_thread_state(WorkerThreadState::Exiting);
64 }
65 
66 int main(int, char**)
67 {
68   typedef std::chrono::high_resolution_clock Clock;
69 
70   {
71     typedef int T;
72     std::promise<T> p;
73     std::shared_future<T> f = p.get_future();
74     std::thread(func1, std::move(p)).detach();
75     assert(f.valid());
76     assert(f.wait_until(Clock::now() + ms(10)) == std::future_status::timeout);
77     assert(f.valid());
78 
79     // allow the worker thread to produce the result and wait until the worker is done
80     set_worker_thread_state(WorkerThreadState::AllowedToRun);
81     wait_for_worker_thread_state(WorkerThreadState::Exiting);
82 
83     assert(f.wait_until(Clock::now() + ms(10)) == std::future_status::ready);
84     assert(f.valid());
85     f.wait();
86     assert(f.valid());
87   }
88   {
89     typedef int& T;
90     std::promise<T> p;
91     std::shared_future<T> f = p.get_future();
92     std::thread(func3, std::move(p)).detach();
93     assert(f.valid());
94     assert(f.wait_until(Clock::now() + ms(10)) == std::future_status::timeout);
95     assert(f.valid());
96 
97     // allow the worker thread to produce the result and wait until the worker is done
98     set_worker_thread_state(WorkerThreadState::AllowedToRun);
99     wait_for_worker_thread_state(WorkerThreadState::Exiting);
100 
101     assert(f.wait_until(Clock::now() + ms(10)) == std::future_status::ready);
102     assert(f.valid());
103     f.wait();
104     assert(f.valid());
105   }
106   {
107     typedef void T;
108     std::promise<T> p;
109     std::shared_future<T> f = p.get_future();
110     std::thread(func5, std::move(p)).detach();
111     assert(f.valid());
112     assert(f.wait_until(Clock::now() + ms(10)) == std::future_status::timeout);
113     assert(f.valid());
114 
115     // allow the worker thread to produce the result and wait until the worker is done
116     set_worker_thread_state(WorkerThreadState::AllowedToRun);
117     wait_for_worker_thread_state(WorkerThreadState::Exiting);
118 
119     assert(f.wait_until(Clock::now() + ms(10)) == std::future_status::ready);
120     assert(f.valid());
121     f.wait();
122     assert(f.valid());
123   }
124 
125   return 0;
126 }
127