12bd812c5STeresa Johnson //===- unittests/Threading.cpp - Thread tests -----------------------------===//
22bd812c5STeresa Johnson //
32946cd70SChandler Carruth // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
42946cd70SChandler Carruth // See https://llvm.org/LICENSE.txt for license information.
52946cd70SChandler Carruth // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
62bd812c5STeresa Johnson //
72bd812c5STeresa Johnson //===----------------------------------------------------------------------===//
82bd812c5STeresa Johnson 
92bd812c5STeresa Johnson #include "llvm/Support/Threading.h"
102bd812c5STeresa Johnson #include "llvm/Support/thread.h"
112bd812c5STeresa Johnson #include "gtest/gtest.h"
122bd812c5STeresa Johnson 
13a9c3c176SSam McCall #include <atomic>
14a9c3c176SSam McCall #include <condition_variable>
15a9c3c176SSam McCall 
162bd812c5STeresa Johnson using namespace llvm;
172bd812c5STeresa Johnson 
182bd812c5STeresa Johnson namespace {
192bd812c5STeresa Johnson 
TEST(Threading,PhysicalConcurrency)202bd812c5STeresa Johnson TEST(Threading, PhysicalConcurrency) {
21c0ef9e43STeresa Johnson   auto Num = heavyweight_hardware_concurrency();
222bd812c5STeresa Johnson   // Since Num is unsigned this will also catch us trying to
232bd812c5STeresa Johnson   // return -1.
248404aeb5SAlexandre Ganea   ASSERT_LE(Num.compute_thread_count(),
258404aeb5SAlexandre Ganea             hardware_concurrency().compute_thread_count());
262bd812c5STeresa Johnson }
272bd812c5STeresa Johnson 
28a9c3c176SSam McCall #if LLVM_ENABLE_THREADS
29a9c3c176SSam McCall 
30a9c3c176SSam McCall class Notification {
31a9c3c176SSam McCall public:
notify()32a9c3c176SSam McCall   void notify() {
33a9c3c176SSam McCall     {
34a9c3c176SSam McCall       std::lock_guard<std::mutex> Lock(M);
35a9c3c176SSam McCall       Notified = true;
3618e6a65bSSam McCall       // Broadcast with the lock held, so it's safe to destroy the Notification
3718e6a65bSSam McCall       // after wait() returns.
38a9c3c176SSam McCall       CV.notify_all();
39a9c3c176SSam McCall     }
4018e6a65bSSam McCall   }
41a9c3c176SSam McCall 
wait()42a9c3c176SSam McCall   bool wait() {
43a9c3c176SSam McCall     std::unique_lock<std::mutex> Lock(M);
44a9c3c176SSam McCall     using steady_clock = std::chrono::steady_clock;
45a9c3c176SSam McCall     auto Deadline = steady_clock::now() +
46a9c3c176SSam McCall                     std::chrono::duration_cast<steady_clock::duration>(
47a9c3c176SSam McCall                         std::chrono::duration<double>(5));
48a9c3c176SSam McCall     return CV.wait_until(Lock, Deadline, [this] { return Notified; });
49a9c3c176SSam McCall   }
50a9c3c176SSam McCall 
51a9c3c176SSam McCall private:
52a9c3c176SSam McCall   bool Notified = false;
53a9c3c176SSam McCall   mutable std::condition_variable CV;
54a9c3c176SSam McCall   mutable std::mutex M;
55a9c3c176SSam McCall };
56a9c3c176SSam McCall 
TEST(Threading,RunOnThreadSyncAsync)57a9c3c176SSam McCall TEST(Threading, RunOnThreadSyncAsync) {
58a9c3c176SSam McCall   Notification ThreadStarted, ThreadAdvanced, ThreadFinished;
59a9c3c176SSam McCall 
60a9c3c176SSam McCall   auto ThreadFunc = [&] {
61a9c3c176SSam McCall     ThreadStarted.notify();
62a9c3c176SSam McCall     ASSERT_TRUE(ThreadAdvanced.wait());
63a9c3c176SSam McCall     ThreadFinished.notify();
64a9c3c176SSam McCall   };
65a9c3c176SSam McCall 
6648c68a63STim Northover   llvm::thread Thread(ThreadFunc);
6748c68a63STim Northover   Thread.detach();
68a9c3c176SSam McCall   ASSERT_TRUE(ThreadStarted.wait());
69a9c3c176SSam McCall   ThreadAdvanced.notify();
70a9c3c176SSam McCall   ASSERT_TRUE(ThreadFinished.wait());
71a9c3c176SSam McCall }
72a9c3c176SSam McCall 
TEST(Threading,RunOnThreadSync)73a9c3c176SSam McCall TEST(Threading, RunOnThreadSync) {
74a9c3c176SSam McCall   std::atomic_bool Executed(false);
7548c68a63STim Northover   llvm::thread Thread(
76a9c3c176SSam McCall       [](void *Arg) { *static_cast<std::atomic_bool *>(Arg) = true; },
77a9c3c176SSam McCall       &Executed);
7848c68a63STim Northover   Thread.join();
79a9c3c176SSam McCall   ASSERT_EQ(Executed, true);
80a9c3c176SSam McCall }
8148c68a63STim Northover 
8248c68a63STim Northover #if defined(__APPLE__)
TEST(Threading,AppleStackSize)8348c68a63STim Northover TEST(Threading, AppleStackSize) {
8448c68a63STim Northover   llvm::thread Thread([] {
85*85cb4f99STim Northover     volatile unsigned char Var[8 * 1024 * 1024 - 10240];
8648c68a63STim Northover     Var[0] = 0xff;
8748c68a63STim Northover     ASSERT_EQ(Var[0], 0xff);
8848c68a63STim Northover   });
8948c68a63STim Northover   Thread.join();
9048c68a63STim Northover }
9148c68a63STim Northover #endif
92a9c3c176SSam McCall #endif
93a9c3c176SSam McCall 
942bd812c5STeresa Johnson } // end anon namespace
95