/* Copyright (c) 2013 Microsoft Corporation. All rights reserved. Released under Apache 2.0 license as described in the file LICENSE. Author: Leonardo de Moura */ #include #include #include #include "util/thread.h" #include "util/debug.h" #include "util/shared_mutex.h" #include "util/interrupt.h" using namespace lean; #if !defined(__APPLE__) && defined(LEAN_MULTI_THREAD) void foo() { static LEAN_THREAD_LOCAL std::vector v(1024); if (v.size() != 1024) { std::cerr << "Error\n"; exit(1); } } static void tst1() { unsigned n = 5; for (unsigned i = 0; i < n; i++) { thread t([](){ foo(); }); t.join(); } } static void tst2() { unsigned N = 10; unsigned n = 1; shared_mutex mut; std::vector threads; for (unsigned i = 0; i < N; i++) { threads.emplace_back([&]() { unsigned sum = 0; { shared_lock lock(mut); for (unsigned i = 0; i < 1000000; i++) sum += n; } { exclusive_lock lock(mut); std::cout << sum << "\n"; } }); } for (unsigned i = 0; i < N; i++) threads[i].join(); } static void tst3() { shared_mutex mutex; atomic t2_started(false); atomic t2_done(false); chrono::milliseconds small_delay(10); thread t1([&]() { while (!t2_started) { this_thread::sleep_for(small_delay); } while (!mutex.try_lock()) { this_thread::sleep_for(small_delay); } // test recursive try_lock lean_verify(mutex.try_lock()); mutex.unlock(); // we can only succeed getting the lock if t2 is done lean_assert(t2_done); mutex.unlock(); }); thread t2([&]() { { exclusive_lock lock(mutex); t2_started = true; this_thread::sleep_for(small_delay); } t2_done = true; }); t1.join(); t2.join(); lean_assert(t2_done); } static void tst4() { shared_mutex mutex; atomic t2_started(false); atomic t2_done(false); chrono::milliseconds small_delay(10); thread t1([&]() { while (!t2_started) { this_thread::sleep_for(small_delay); } while (!mutex.try_lock_shared()) { this_thread::sleep_for(small_delay); } // test recursive try_lock_shared lean_verify(mutex.try_lock_shared()); mutex.unlock_shared(); // we can only succeed getting the lock if t2 is done lean_assert(t2_done); mutex.unlock_shared(); }); thread t2([&]() { { exclusive_lock lock(mutex); t2_started = true; this_thread::sleep_for(small_delay); } t2_done = true; }); t1.join(); t2.join(); lean_assert(t2_done); } static void tst5() { shared_mutex mutex; atomic t2_started(false); atomic t1_done(false); chrono::milliseconds small_delay(10); thread t1([&]() { while (!t2_started) { this_thread::sleep_for(small_delay); } lean_verify(mutex.try_lock_shared()); // t2 is also using a shared lock this_thread::sleep_for(small_delay); lean_verify(mutex.try_lock_shared()); this_thread::sleep_for(small_delay); t1_done = true; mutex.unlock_shared(); this_thread::sleep_for(small_delay); mutex.unlock_shared(); }); thread t2([&]() { { shared_lock lock(mutex); t2_started = true; while (!t1_done) { this_thread::sleep_for(small_delay); } } }); t1.join(); t2.join(); } static void tst6() { interruptible_thread t1([]() { try { // Remark: this_thread::sleep_for does not check whether the thread has been interrupted or not. // this_thread::sleep_for(chrono::milliseconds(1000000)); sleep_for(1000000); } catch (interrupted &) { std::cout << "interrupted...\n"; } }); sleep_for(20); t1.request_interrupt(); t1.join(); } #else static void tst1() {} static void tst2() {} static void tst3() {} static void tst4() {} static void tst5() {} static void tst6() {} #endif int main() { save_stack_info(); tst1(); tst2(); tst3(); tst4(); tst5(); tst6(); return has_violations() ? 1 : 0; }