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 Rep, class Period, class Predicate>
175a83710eSEric Fiselier // bool
185a83710eSEric Fiselier // wait_for(unique_lock<mutex>& lock,
195a83710eSEric Fiselier // const chrono::duration<Rep, Period>& rel_time,
205a83710eSEric Fiselier // Predicate pred);
215a83710eSEric Fiselier
225a83710eSEric Fiselier #include <condition_variable>
235a83710eSEric Fiselier #include <mutex>
245a83710eSEric Fiselier #include <thread>
255a83710eSEric Fiselier #include <chrono>
265a83710eSEric Fiselier #include <cassert>
275a83710eSEric Fiselier
2856462801SLouis Dionne #include "make_test_thread.h"
297fc6a556SMarshall Clow #include "test_macros.h"
307fc6a556SMarshall Clow
315a83710eSEric Fiselier class Pred
325a83710eSEric Fiselier {
335a83710eSEric Fiselier int& i_;
345a83710eSEric Fiselier public:
Pred(int & i)355a83710eSEric Fiselier explicit Pred(int& i) : i_(i) {}
365a83710eSEric Fiselier
operator ()()375a83710eSEric Fiselier bool operator()() {return i_ != 0;}
385a83710eSEric Fiselier };
395a83710eSEric Fiselier
405a83710eSEric Fiselier std::condition_variable cv;
415a83710eSEric Fiselier std::mutex mut;
425a83710eSEric Fiselier
435a83710eSEric Fiselier int test1 = 0;
445a83710eSEric Fiselier int test2 = 0;
455a83710eSEric Fiselier
465a83710eSEric Fiselier int runs = 0;
475a83710eSEric Fiselier
f()485a83710eSEric Fiselier void f()
495a83710eSEric Fiselier {
505a83710eSEric Fiselier typedef std::chrono::system_clock Clock;
515a83710eSEric Fiselier typedef std::chrono::milliseconds milliseconds;
525a83710eSEric Fiselier std::unique_lock<std::mutex> lk(mut);
535a83710eSEric Fiselier assert(test2 == 0);
545a83710eSEric Fiselier test1 = 1;
555a83710eSEric Fiselier cv.notify_one();
565a83710eSEric Fiselier Clock::time_point t0 = Clock::now();
575a83710eSEric Fiselier bool r = cv.wait_for(lk, milliseconds(250), Pred(test2));
584dc0ed83SStephan T. Lavavej ((void)r); // Prevent unused warning
595a83710eSEric Fiselier Clock::time_point t1 = Clock::now();
605a83710eSEric Fiselier if (runs == 0)
615a83710eSEric Fiselier {
625a83710eSEric Fiselier assert(t1 - t0 < milliseconds(250));
635a83710eSEric Fiselier assert(test2 != 0);
645a83710eSEric Fiselier }
655a83710eSEric Fiselier else
665a83710eSEric Fiselier {
675a83710eSEric Fiselier assert(t1 - t0 - milliseconds(250) < milliseconds(50));
685a83710eSEric Fiselier assert(test2 == 0);
695a83710eSEric Fiselier }
705a83710eSEric Fiselier ++runs;
715a83710eSEric Fiselier }
725a83710eSEric Fiselier
main(int,char **)732df59c50SJF Bastien int main(int, char**)
745a83710eSEric Fiselier {
755a83710eSEric Fiselier {
765a83710eSEric Fiselier std::unique_lock<std::mutex>lk(mut);
7756462801SLouis Dionne std::thread t = support::make_test_thread(f);
785a83710eSEric Fiselier assert(test1 == 0);
795a83710eSEric Fiselier while (test1 == 0)
805a83710eSEric Fiselier cv.wait(lk);
815a83710eSEric Fiselier assert(test1 != 0);
825a83710eSEric Fiselier test2 = 1;
835a83710eSEric Fiselier lk.unlock();
845a83710eSEric Fiselier cv.notify_one();
855a83710eSEric Fiselier t.join();
865a83710eSEric Fiselier }
875a83710eSEric Fiselier test1 = 0;
885a83710eSEric Fiselier test2 = 0;
895a83710eSEric Fiselier {
905a83710eSEric Fiselier std::unique_lock<std::mutex>lk(mut);
9156462801SLouis Dionne std::thread t = support::make_test_thread(f);
925a83710eSEric Fiselier assert(test1 == 0);
935a83710eSEric Fiselier while (test1 == 0)
945a83710eSEric Fiselier cv.wait(lk);
955a83710eSEric Fiselier assert(test1 != 0);
965a83710eSEric Fiselier lk.unlock();
975a83710eSEric Fiselier t.join();
985a83710eSEric Fiselier }
992df59c50SJF Bastien
1002df59c50SJF Bastien return 0;
1015a83710eSEric Fiselier }
102