thrust / dependencies /libcudacxx /libcxx /test /std /thread /thread.threads /thread.thread.class /thread.thread.constr /F.pass.cpp
| //===----------------------------------------------------------------------===// | |
| // | |
| // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. | |
| // See https://llvm.org/LICENSE.txt for license information. | |
| // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception | |
| // | |
| //===----------------------------------------------------------------------===// | |
| // | |
| // UNSUPPORTED: libcpp-has-no-threads | |
| // <thread> | |
| // class thread | |
| // template <class F, class ...Args> thread(F&& f, Args&&... args); | |
| // UNSUPPORTED: sanitizer-new-delete | |
| std::atomic<unsigned> throw_one(0xFFFF); | |
| std::atomic<unsigned> outstanding_new(0); | |
| void* operator new(std::size_t s) TEST_THROW_SPEC(std::bad_alloc) | |
| { | |
| unsigned expected = throw_one; | |
| do { | |
| if (expected == 0) TEST_THROW(std::bad_alloc()); | |
| } while (!throw_one.compare_exchange_weak(expected, expected - 1)); | |
| ++outstanding_new; | |
| void* ret = std::malloc(s); | |
| if (!ret) std::abort(); // placate MSVC's unchecked malloc warning | |
| return ret; | |
| } | |
| void operator delete(void* p) TEST_NOEXCEPT | |
| { | |
| if (!p) return; | |
| --outstanding_new; | |
| 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; | |
| class MoveOnly | |
| { | |
| MoveOnly(const MoveOnly&); | |
| public: | |
| MoveOnly() {} | |
| MoveOnly(MoveOnly&&) {} | |
| void operator()(MoveOnly&&) | |
| { | |
| } | |
| }; | |
| // Test throwing std::bad_alloc | |
| //----------------------------- | |
| // Concerns: | |
| // A Each allocation performed during thread construction should be performed | |
| // in the parent thread so that std::terminate is not called if | |
| // std::bad_alloc is thrown by new. | |
| // B std::thread's constructor should properly handle exceptions and not leak | |
| // memory. | |
| // Plan: | |
| // 1 Create a thread and count the number of allocations, 'numAllocs', it | |
| // performs. | |
| // 2 For each allocation performed run a test where that allocation throws. | |
| // 2.1 check that the exception can be caught in the parent thread. | |
| // 2.2 Check that the functor has not been called. | |
| // 2.3 Check that no memory allocated by the creation of the thread is leaked. | |
| // 3 Finally check that a thread runs successfully if we throw after | |
| // 'numAllocs + 1' allocations. | |
| int numAllocs; | |
| void test_throwing_new_during_thread_creation() { | |
| throw_one = 0xFFF; | |
| { | |
| std::thread t(f); | |
| t.join(); | |
| } | |
| numAllocs = 0xFFF - throw_one; | |
| // i <= numAllocs means the last iteration is expected not to throw. | |
| for (int i=0; i <= numAllocs; ++i) { | |
| throw_one = i; | |
| f_run = false; | |
| unsigned old_outstanding = outstanding_new; | |
| try { | |
| std::thread t(f); | |
| assert(i == numAllocs); // Only final iteration will not throw. | |
| t.join(); | |
| assert(f_run); | |
| } catch (std::bad_alloc const&) { | |
| assert(i < numAllocs); | |
| assert(!f_run); // (2.2) | |
| } | |
| assert(old_outstanding == outstanding_new); // (2.3) | |
| } | |
| f_run = false; | |
| throw_one = 0xFFF; | |
| } | |
| int main(int, char**) | |
| { | |
| test_throwing_new_during_thread_creation(); | |
| { | |
| std::thread t(f); | |
| t.join(); | |
| assert(f_run == true); | |
| } | |
| { | |
| assert(G::n_alive == 0); | |
| assert(!G::op_run); | |
| { | |
| G g; | |
| std::thread t(g); | |
| t.join(); | |
| } | |
| assert(G::n_alive == 0); | |
| assert(G::op_run); | |
| } | |
| G::op_run = false; | |
| // The test below expects `std::thread` to call `new`, which may not be the | |
| // case for all implementations. | |
| LIBCPP_ASSERT(numAllocs > 0); // libc++ should call new. | |
| if (numAllocs > 0) { | |
| try | |
| { | |
| throw_one = 0; | |
| assert(G::n_alive == 0); | |
| assert(!G::op_run); | |
| std::thread t((G())); | |
| assert(false); | |
| } | |
| catch (std::bad_alloc const&) | |
| { | |
| throw_one = 0xFFFF; | |
| assert(G::n_alive == 0); | |
| assert(!G::op_run); | |
| } | |
| } | |
| { | |
| assert(G::n_alive == 0); | |
| assert(!G::op_run); | |
| { | |
| G g; | |
| 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(); | |
| } | |
| return 0; | |
| } | |