15a83710eSEric Fiselier //===----------------------------------------------------------------------===//
25a83710eSEric Fiselier //
357b08b09SChandler Carruth // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
457b08b09SChandler Carruth // See https://llvm.org/LICENSE.txt for license information.
557b08b09SChandler Carruth // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
65a83710eSEric Fiselier //
75a83710eSEric Fiselier //===----------------------------------------------------------------------===//
85a83710eSEric Fiselier //
9*a7f9895cSLouis Dionne // UNSUPPORTED: no-threads
10642048eeSLouis Dionne // ALLOW_RETRIES: 2
115a83710eSEric Fiselier 
125a83710eSEric Fiselier // <condition_variable>
135a83710eSEric Fiselier 
145a83710eSEric Fiselier // class condition_variable;
155a83710eSEric Fiselier 
165a83710eSEric Fiselier // template <class Clock, class Duration>
175a83710eSEric Fiselier //   cv_status
185a83710eSEric Fiselier //   wait_until(unique_lock<mutex>& lock,
195a83710eSEric Fiselier //              const chrono::time_point<Clock, Duration>& abs_time);
205a83710eSEric Fiselier 
215a83710eSEric Fiselier #include <condition_variable>
225a83710eSEric Fiselier #include <mutex>
235a83710eSEric Fiselier #include <thread>
245a83710eSEric Fiselier #include <chrono>
255a83710eSEric Fiselier #include <cassert>
265a83710eSEric Fiselier 
2756462801SLouis Dionne #include "make_test_thread.h"
287fc6a556SMarshall Clow #include "test_macros.h"
297fc6a556SMarshall Clow 
3085e26f56SDan Albert struct TestClock
315a83710eSEric Fiselier {
325a83710eSEric Fiselier     typedef std::chrono::milliseconds duration;
335a83710eSEric Fiselier     typedef duration::rep             rep;
345a83710eSEric Fiselier     typedef duration::period          period;
3585e26f56SDan Albert     typedef std::chrono::time_point<TestClock> time_point;
365a83710eSEric Fiselier     static const bool is_steady =  true;
375a83710eSEric Fiselier 
nowTestClock385a83710eSEric Fiselier     static time_point now()
395a83710eSEric Fiselier     {
405a83710eSEric Fiselier         using namespace std::chrono;
415a83710eSEric Fiselier         return time_point(duration_cast<duration>(
425a83710eSEric Fiselier                 steady_clock::now().time_since_epoch()
435a83710eSEric Fiselier                                                  ));
445a83710eSEric Fiselier     }
455a83710eSEric Fiselier };
465a83710eSEric Fiselier 
475a83710eSEric Fiselier std::condition_variable cv;
485a83710eSEric Fiselier std::mutex mut;
495a83710eSEric Fiselier 
505a83710eSEric Fiselier int test1 = 0;
515a83710eSEric Fiselier int test2 = 0;
525a83710eSEric Fiselier 
535a83710eSEric Fiselier int runs = 0;
545a83710eSEric Fiselier 
5585e26f56SDan Albert template <typename Clock>
f()565a83710eSEric Fiselier void f()
575a83710eSEric Fiselier {
585a83710eSEric Fiselier     std::unique_lock<std::mutex> lk(mut);
595a83710eSEric Fiselier     assert(test2 == 0);
605a83710eSEric Fiselier     test1 = 1;
615a83710eSEric Fiselier     cv.notify_one();
6285e26f56SDan Albert     typename Clock::time_point t0 = Clock::now();
6385e26f56SDan Albert     typename Clock::time_point t = t0 + std::chrono::milliseconds(250);
645a83710eSEric Fiselier     while (test2 == 0 && cv.wait_until(lk, t) == std::cv_status::no_timeout)
655a83710eSEric Fiselier         ;
6685e26f56SDan Albert     typename Clock::time_point t1 = Clock::now();
675a83710eSEric Fiselier     if (runs == 0)
685a83710eSEric Fiselier     {
6985e26f56SDan Albert         assert(t1 - t0 < std::chrono::milliseconds(250));
705a83710eSEric Fiselier         assert(test2 != 0);
715a83710eSEric Fiselier     }
725a83710eSEric Fiselier     else
735a83710eSEric Fiselier     {
7485e26f56SDan Albert         assert(t1 - t0 - std::chrono::milliseconds(250) < std::chrono::milliseconds(50));
755a83710eSEric Fiselier         assert(test2 == 0);
765a83710eSEric Fiselier     }
775a83710eSEric Fiselier     ++runs;
785a83710eSEric Fiselier }
795a83710eSEric Fiselier 
8085e26f56SDan Albert template <typename Clock>
run_test()8185e26f56SDan Albert void run_test()
825a83710eSEric Fiselier {
8385e26f56SDan Albert     runs = 0;
8485e26f56SDan Albert     test1 = 0;
8585e26f56SDan Albert     test2 = 0;
865a83710eSEric Fiselier     {
875a83710eSEric Fiselier         std::unique_lock<std::mutex>lk(mut);
8856462801SLouis Dionne         std::thread t = support::make_test_thread(f<Clock>);
895a83710eSEric Fiselier         assert(test1 == 0);
905a83710eSEric Fiselier         while (test1 == 0)
915a83710eSEric Fiselier             cv.wait(lk);
925a83710eSEric Fiselier         assert(test1 != 0);
935a83710eSEric Fiselier         test2 = 1;
945a83710eSEric Fiselier         lk.unlock();
955a83710eSEric Fiselier         cv.notify_one();
965a83710eSEric Fiselier         t.join();
975a83710eSEric Fiselier     }
985a83710eSEric Fiselier     test1 = 0;
995a83710eSEric Fiselier     test2 = 0;
1005a83710eSEric Fiselier     {
1015a83710eSEric Fiselier         std::unique_lock<std::mutex>lk(mut);
10256462801SLouis Dionne         std::thread t = support::make_test_thread(f<Clock>);
1035a83710eSEric Fiselier         assert(test1 == 0);
1045a83710eSEric Fiselier         while (test1 == 0)
1055a83710eSEric Fiselier             cv.wait(lk);
1065a83710eSEric Fiselier         assert(test1 != 0);
1075a83710eSEric Fiselier         lk.unlock();
1085a83710eSEric Fiselier         t.join();
1095a83710eSEric Fiselier     }
11085e26f56SDan Albert }
1112df59c50SJF Bastien 
main(int,char **)11285e26f56SDan Albert int main(int, char**)
11385e26f56SDan Albert {
11485e26f56SDan Albert     run_test<TestClock>();
11585e26f56SDan Albert     run_test<std::chrono::steady_clock>();
11685e26f56SDan Albert     run_test<std::chrono::system_clock>();
1172df59c50SJF Bastien     return 0;
1185a83710eSEric Fiselier }
119