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 Fiseliervoid 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 Albertvoid 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 Albertint 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