| //===-- Tests for pthread_exit --------------------------------------------===// |
| // |
| // 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 |
| // |
| //===----------------------------------------------------------------------===// |
| |
| #include "src/pthread/pthread_create.h" |
| #include "src/pthread/pthread_exit.h" |
| #include "src/pthread/pthread_join.h" |
| #include "test/IntegrationTest/test.h" |
| |
| #include <pthread.h> |
| |
| bool dtor_called = false; |
| |
| class A { |
| int val; |
| |
| public: |
| A(int i) { val = i; } |
| |
| void set(int i) { val = i; } |
| |
| ~A() { |
| val = 0; |
| dtor_called = true; |
| } |
| }; |
| |
| thread_local A thread_local_a(123); |
| |
| void *func(void *) { |
| // Touch the thread local variable so that it gets initialized and a callback |
| // for its destructor gets registered with __cxa_thread_atexit. |
| thread_local_a.set(321); |
| LIBC_NAMESPACE::pthread_exit(nullptr); |
| return nullptr; |
| } |
| |
| TEST_MAIN() { |
| pthread_t th; |
| void *retval; |
| |
| ASSERT_EQ(LIBC_NAMESPACE::pthread_create(&th, nullptr, func, nullptr), 0); |
| ASSERT_EQ(LIBC_NAMESPACE::pthread_join(th, &retval), 0); |
| |
| ASSERT_TRUE(dtor_called); |
| LIBC_NAMESPACE::pthread_exit(nullptr); |
| return 0; |
| } |
| |
| extern "C" { |
| |
| using Destructor = void(void *); |
| |
| int __cxa_thread_atexit_impl(Destructor *, void *, void *); |
| |
| // We do not link integration tests to C++ runtime pieces like the libcxxabi. |
| // So, we provide our own simple __cxa_thread_atexit implementation. |
| int __cxa_thread_atexit(Destructor *dtor, void *obj, void *) { |
| return __cxa_thread_atexit_impl(dtor, obj, nullptr); |
| } |
| } |