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 10f03ac381SLouis Dionne // ALLOW_RETRIES: 2 11fe1991b5SKamil Rytarowski 125a83710eSEric Fiselier // <condition_variable> 135a83710eSEric Fiselier 145a83710eSEric Fiselier // class condition_variable_any; 155a83710eSEric Fiselier 165a83710eSEric Fiselier // template <class Lock, class Rep, class Period> 175a83710eSEric Fiselier // cv_status 185a83710eSEric Fiselier // wait_for(Lock& lock, const chrono::duration<Rep, Period>& rel_time); 195a83710eSEric Fiselier 205a83710eSEric Fiselier #include <condition_variable> 215a83710eSEric Fiselier #include <mutex> 225a83710eSEric Fiselier #include <thread> 235a83710eSEric Fiselier #include <chrono> 245a83710eSEric Fiselier #include <cassert> 255a83710eSEric Fiselier 2656462801SLouis Dionne #include "make_test_thread.h" 277fc6a556SMarshall Clow #include "test_macros.h" 287fc6a556SMarshall Clow 295a83710eSEric Fiselier std::condition_variable_any cv; 305a83710eSEric Fiselier 315a83710eSEric Fiselier typedef std::timed_mutex L0; 325a83710eSEric Fiselier typedef std::unique_lock<L0> L1; 335a83710eSEric Fiselier 345a83710eSEric Fiselier L0 m0; 355a83710eSEric Fiselier 365a83710eSEric Fiselier int test1 = 0; 375a83710eSEric Fiselier int test2 = 0; 385a83710eSEric Fiselier 39f8013a35SMartin Storsjö bool expect_timeout = false; 405a83710eSEric Fiselier f()415a83710eSEric Fiseliervoid f() 425a83710eSEric Fiselier { 435a83710eSEric Fiselier typedef std::chrono::system_clock Clock; 445a83710eSEric Fiselier typedef std::chrono::milliseconds milliseconds; 455a83710eSEric Fiselier L1 lk(m0); 465a83710eSEric Fiselier assert(test2 == 0); 475a83710eSEric Fiselier test1 = 1; 485a83710eSEric Fiselier cv.notify_one(); 495a83710eSEric Fiselier Clock::time_point t0 = Clock::now(); 5001aa9e1fSMartin Storsjö Clock::time_point wait_end = t0 + milliseconds(250); 5101aa9e1fSMartin Storsjö Clock::duration d; 5201aa9e1fSMartin Storsjö do { 5301aa9e1fSMartin Storsjö d = wait_end - Clock::now(); 5401aa9e1fSMartin Storsjö if (d <= milliseconds(0)) break; 5501aa9e1fSMartin Storsjö } while (test2 == 0 && cv.wait_for(lk, d) == std::cv_status::no_timeout); 565a83710eSEric Fiselier Clock::time_point t1 = Clock::now(); 57f8013a35SMartin Storsjö if (!expect_timeout) 585a83710eSEric Fiselier { 595a83710eSEric Fiselier assert(t1 - t0 < milliseconds(250)); 605a83710eSEric Fiselier assert(test2 != 0); 615a83710eSEric Fiselier } 625a83710eSEric Fiselier else 635a83710eSEric Fiselier { 645a83710eSEric Fiselier assert(t1 - t0 - milliseconds(250) < milliseconds(50)); 655a83710eSEric Fiselier assert(test2 == 0); 665a83710eSEric Fiselier } 675a83710eSEric Fiselier } 685a83710eSEric Fiselier main(int,char **)692df59c50SJF Bastienint main(int, char**) 705a83710eSEric Fiselier { 715a83710eSEric Fiselier { 725a83710eSEric Fiselier L1 lk(m0); 7356462801SLouis Dionne std::thread t = support::make_test_thread(f); 745a83710eSEric Fiselier assert(test1 == 0); 755a83710eSEric Fiselier while (test1 == 0) 765a83710eSEric Fiselier cv.wait(lk); 775a83710eSEric Fiselier assert(test1 != 0); 785a83710eSEric Fiselier test2 = 1; 795a83710eSEric Fiselier lk.unlock(); 805a83710eSEric Fiselier cv.notify_one(); 815a83710eSEric Fiselier t.join(); 825a83710eSEric Fiselier } 835a83710eSEric Fiselier test1 = 0; 845a83710eSEric Fiselier test2 = 0; 85f8013a35SMartin Storsjö expect_timeout = true; 865a83710eSEric Fiselier { 875a83710eSEric Fiselier L1 lk(m0); 8856462801SLouis Dionne std::thread t = support::make_test_thread(f); 895a83710eSEric Fiselier assert(test1 == 0); 905a83710eSEric Fiselier while (test1 == 0) 915a83710eSEric Fiselier cv.wait(lk); 925a83710eSEric Fiselier assert(test1 != 0); 935a83710eSEric Fiselier lk.unlock(); 945a83710eSEric Fiselier t.join(); 955a83710eSEric Fiselier } 962df59c50SJF Bastien 972df59c50SJF Bastien return 0; 985a83710eSEric Fiselier } 99