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
11389da901SEric Fiselier 
125a83710eSEric Fiselier // <condition_variable>
135a83710eSEric Fiselier 
145a83710eSEric Fiselier // class condition_variable;
155a83710eSEric Fiselier 
165a83710eSEric Fiselier // template <class Rep, class Period>
175a83710eSEric Fiselier //     cv_status
185a83710eSEric Fiselier //     wait_for(unique_lock<mutex>& lock,
195a83710eSEric Fiselier //              const chrono::duration<Rep, Period>& rel_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 
305a83710eSEric Fiselier std::condition_variable cv;
315a83710eSEric Fiselier std::mutex mut;
325a83710eSEric Fiselier 
335a83710eSEric Fiselier int test1 = 0;
345a83710eSEric Fiselier int test2 = 0;
355a83710eSEric Fiselier 
36f8013a35SMartin Storsjö bool expect_timeout = false;
375a83710eSEric Fiselier 
f()385a83710eSEric Fiselier void f()
395a83710eSEric Fiselier {
405a83710eSEric Fiselier     typedef std::chrono::system_clock Clock;
415a83710eSEric Fiselier     typedef std::chrono::milliseconds milliseconds;
425a83710eSEric Fiselier     std::unique_lock<std::mutex> lk(mut);
435a83710eSEric Fiselier     assert(test2 == 0);
445a83710eSEric Fiselier     test1 = 1;
455a83710eSEric Fiselier     cv.notify_one();
465a83710eSEric Fiselier     Clock::time_point t0 = Clock::now();
4701aa9e1fSMartin Storsjö     Clock::time_point wait_end = t0 + milliseconds(250);
4801aa9e1fSMartin Storsjö     Clock::duration d;
4901aa9e1fSMartin Storsjö     do {
5001aa9e1fSMartin Storsjö         d = wait_end - Clock::now();
5101aa9e1fSMartin Storsjö         if (d <= milliseconds(0)) break;
5201aa9e1fSMartin Storsjö     } while (test2 == 0 && cv.wait_for(lk, d) == std::cv_status::no_timeout);
535a83710eSEric Fiselier     Clock::time_point t1 = Clock::now();
54f8013a35SMartin Storsjö     if (!expect_timeout)
555a83710eSEric Fiselier     {
565a83710eSEric Fiselier         assert(t1 - t0 < milliseconds(250));
575a83710eSEric Fiselier         assert(test2 != 0);
585a83710eSEric Fiselier     }
595a83710eSEric Fiselier     else
605a83710eSEric Fiselier     {
615a83710eSEric Fiselier         assert(t1 - t0 - milliseconds(250) < milliseconds(50));
625a83710eSEric Fiselier         assert(test2 == 0);
635a83710eSEric Fiselier     }
645a83710eSEric Fiselier }
655a83710eSEric Fiselier 
main(int,char **)662df59c50SJF Bastien int main(int, char**)
675a83710eSEric Fiselier {
685a83710eSEric Fiselier     {
695a83710eSEric Fiselier         std::unique_lock<std::mutex> lk(mut);
7056462801SLouis Dionne         std::thread t = support::make_test_thread(f);
715a83710eSEric Fiselier         assert(test1 == 0);
725a83710eSEric Fiselier         while (test1 == 0)
735a83710eSEric Fiselier             cv.wait(lk);
745a83710eSEric Fiselier         assert(test1 != 0);
755a83710eSEric Fiselier         test2 = 1;
765a83710eSEric Fiselier         lk.unlock();
775a83710eSEric Fiselier         cv.notify_one();
785a83710eSEric Fiselier         t.join();
795a83710eSEric Fiselier     }
805a83710eSEric Fiselier     test1 = 0;
815a83710eSEric Fiselier     test2 = 0;
82f8013a35SMartin Storsjö     expect_timeout = true;
835a83710eSEric Fiselier     {
845a83710eSEric Fiselier         std::unique_lock<std::mutex> lk(mut);
8556462801SLouis Dionne         std::thread t = support::make_test_thread(f);
865a83710eSEric Fiselier         assert(test1 == 0);
875a83710eSEric Fiselier         while (test1 == 0)
885a83710eSEric Fiselier             cv.wait(lk);
895a83710eSEric Fiselier         assert(test1 != 0);
905a83710eSEric Fiselier         lk.unlock();
915a83710eSEric Fiselier         t.join();
925a83710eSEric Fiselier     }
932df59c50SJF Bastien 
942df59c50SJF Bastien   return 0;
955a83710eSEric Fiselier }
96