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_any;
155a83710eSEric Fiselier 
165a83710eSEric Fiselier // template <class Lock, class Predicate>
175a83710eSEric Fiselier //   void wait(Lock& lock, Predicate pred);
185a83710eSEric Fiselier 
195a83710eSEric Fiselier #include <condition_variable>
205a83710eSEric Fiselier #include <mutex>
215a83710eSEric Fiselier #include <thread>
225a83710eSEric Fiselier #include <functional>
235a83710eSEric Fiselier #include <cassert>
245a83710eSEric Fiselier 
2556462801SLouis Dionne #include "make_test_thread.h"
267fc6a556SMarshall Clow #include "test_macros.h"
277fc6a556SMarshall Clow 
285a83710eSEric Fiselier std::condition_variable_any cv;
295a83710eSEric Fiselier 
305a83710eSEric Fiselier typedef std::timed_mutex L0;
315a83710eSEric Fiselier typedef std::unique_lock<L0> L1;
325a83710eSEric Fiselier 
335a83710eSEric Fiselier L0 m0;
345a83710eSEric Fiselier 
355a83710eSEric Fiselier int test1 = 0;
365a83710eSEric Fiselier int test2 = 0;
375a83710eSEric Fiselier 
385a83710eSEric Fiselier class Pred
395a83710eSEric Fiselier {
405a83710eSEric Fiselier     int& i_;
415a83710eSEric Fiselier public:
Pred(int & i)425a83710eSEric Fiselier     explicit Pred(int& i) : i_(i) {}
435a83710eSEric Fiselier 
operator ()()445a83710eSEric Fiselier     bool operator()() {return i_ != 0;}
455a83710eSEric Fiselier };
465a83710eSEric Fiselier 
f()475a83710eSEric Fiselier void f()
485a83710eSEric Fiselier {
495a83710eSEric Fiselier     L1 lk(m0);
505a83710eSEric Fiselier     assert(test2 == 0);
515a83710eSEric Fiselier     test1 = 1;
525a83710eSEric Fiselier     cv.notify_one();
535a83710eSEric Fiselier     cv.wait(lk, Pred(test2));
545a83710eSEric Fiselier     assert(test2 != 0);
555a83710eSEric Fiselier }
565a83710eSEric Fiselier 
main(int,char **)572df59c50SJF Bastien int main(int, char**)
585a83710eSEric Fiselier {
595a83710eSEric Fiselier     L1 lk(m0);
6056462801SLouis Dionne     std::thread t = support::make_test_thread(f);
615a83710eSEric Fiselier     assert(test1 == 0);
625a83710eSEric Fiselier     while (test1 == 0)
635a83710eSEric Fiselier         cv.wait(lk);
645a83710eSEric Fiselier     assert(test1 != 0);
655a83710eSEric Fiselier     test2 = 1;
665a83710eSEric Fiselier     lk.unlock();
675a83710eSEric Fiselier     cv.notify_one();
685a83710eSEric Fiselier     t.join();
692df59c50SJF Bastien 
702df59c50SJF Bastien   return 0;
715a83710eSEric Fiselier }
72