 8d86b2e686
			
		
	
	8d86b2e686
	
	
	
		
			
			If you're crazy enough to want this sort of thing, then add -D_LIBCPP_HAS_NO_THREADS to your CXXFLAGS and --param=additiona_features=libcpp-has-no-threads to your lit commnad line. http://reviews.llvm.org/D3969 git-svn-id: https://llvm.org/svn/llvm-project/libcxx/trunk@217271 91177308-0d34-0410-b5e6-96231b3b80d8
		
			
				
	
	
		
			153 lines
		
	
	
		
			2.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			153 lines
		
	
	
		
			2.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| //===----------------------------------------------------------------------===//
 | |
| //
 | |
| //                     The LLVM Compiler Infrastructure
 | |
| //
 | |
| // This file is dual licensed under the MIT and the University of Illinois Open
 | |
| // Source Licenses. See LICENSE.TXT for details.
 | |
| //
 | |
| //===----------------------------------------------------------------------===//
 | |
| //
 | |
| // UNSUPPORTED: libcpp-has-no-threads
 | |
| 
 | |
| // <thread>
 | |
| 
 | |
| // class thread
 | |
| 
 | |
| // template <class F, class ...Args> thread(F&& f, Args&&... args);
 | |
| 
 | |
| #include <thread>
 | |
| #include <new>
 | |
| #include <cstdlib>
 | |
| #include <cassert>
 | |
| 
 | |
| unsigned throw_one = 0xFFFF;
 | |
| 
 | |
| void* operator new(std::size_t s) throw(std::bad_alloc)
 | |
| {
 | |
|     if (throw_one == 0)
 | |
|         throw std::bad_alloc();
 | |
|     --throw_one;
 | |
|     return std::malloc(s);
 | |
| }
 | |
| 
 | |
| void  operator delete(void* p) throw()
 | |
| {
 | |
|     std::free(p);
 | |
| }
 | |
| 
 | |
| bool f_run = false;
 | |
| 
 | |
| void f()
 | |
| {
 | |
|     f_run = true;
 | |
| }
 | |
| 
 | |
| class G
 | |
| {
 | |
|     int alive_;
 | |
| public:
 | |
|     static int n_alive;
 | |
|     static bool op_run;
 | |
| 
 | |
|     G() : alive_(1) {++n_alive;}
 | |
|     G(const G& g) : alive_(g.alive_) {++n_alive;}
 | |
|     ~G() {alive_ = 0; --n_alive;}
 | |
| 
 | |
|     void operator()()
 | |
|     {
 | |
|         assert(alive_ == 1);
 | |
|         assert(n_alive >= 1);
 | |
|         op_run = true;
 | |
|     }
 | |
| 
 | |
|     void operator()(int i, double j)
 | |
|     {
 | |
|         assert(alive_ == 1);
 | |
|         assert(n_alive >= 1);
 | |
|         assert(i == 5);
 | |
|         assert(j == 5.5);
 | |
|         op_run = true;
 | |
|     }
 | |
| };
 | |
| 
 | |
| int G::n_alive = 0;
 | |
| bool G::op_run = false;
 | |
| 
 | |
| #ifndef _LIBCPP_HAS_NO_VARIADICS
 | |
| 
 | |
| class MoveOnly
 | |
| {
 | |
|     MoveOnly(const MoveOnly&);
 | |
| public:
 | |
|     MoveOnly() {}
 | |
|     MoveOnly(MoveOnly&&) {}
 | |
| 
 | |
|     void operator()(MoveOnly&&)
 | |
|     {
 | |
|     }
 | |
| };
 | |
| 
 | |
| #endif
 | |
| 
 | |
| int main()
 | |
| {
 | |
|     {
 | |
|         std::thread t(f);
 | |
|         t.join();
 | |
|         assert(f_run == true);
 | |
|     }
 | |
|     f_run = false;
 | |
|     {
 | |
|         try
 | |
|         {
 | |
|             throw_one = 0;
 | |
|             std::thread t(f);
 | |
|             assert(false);
 | |
|         }
 | |
|         catch (...)
 | |
|         {
 | |
|             throw_one = 0xFFFF;
 | |
|             assert(!f_run);
 | |
|         }
 | |
|     }
 | |
|     {
 | |
|         assert(G::n_alive == 0);
 | |
|         assert(!G::op_run);
 | |
|         std::thread t((G()));
 | |
|         t.join();
 | |
|         assert(G::n_alive == 0);
 | |
|         assert(G::op_run);
 | |
|     }
 | |
|     G::op_run = false;
 | |
|     {
 | |
|         try
 | |
|         {
 | |
|             throw_one = 0;
 | |
|             assert(G::n_alive == 0);
 | |
|             assert(!G::op_run);
 | |
|             std::thread t((G()));
 | |
|             assert(false);
 | |
|         }
 | |
|         catch (...)
 | |
|         {
 | |
|             throw_one = 0xFFFF;
 | |
|             assert(G::n_alive == 0);
 | |
|             assert(!G::op_run);
 | |
|         }
 | |
|     }
 | |
| #ifndef _LIBCPP_HAS_NO_VARIADICS
 | |
|     {
 | |
|         assert(G::n_alive == 0);
 | |
|         assert(!G::op_run);
 | |
|         std::thread t(G(), 5, 5.5);
 | |
|         t.join();
 | |
|         assert(G::n_alive == 0);
 | |
|         assert(G::op_run);
 | |
|     }
 | |
|     {
 | |
|         std::thread t = std::thread(MoveOnly(), MoveOnly());
 | |
|         t.join();
 | |
|     }
 | |
| #endif  // _LIBCPP_HAS_NO_VARIADICS
 | |
| }
 |