Home | History | Annotate | Download | only in tests
      1 /*
      2  * Copyright (C) 2014 The Android Open Source Project
      3  *
      4  * Licensed under the Apache License, Version 2.0 (the "License");
      5  * you may not use this file except in compliance with the License.
      6  * You may obtain a copy of the License at
      7  *
      8  *      http://www.apache.org/licenses/LICENSE-2.0
      9  *
     10  * Unless required by applicable law or agreed to in writing, software
     11  * distributed under the License is distributed on an "AS IS" BASIS,
     12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     13  * See the License for the specific language governing permissions and
     14  * limitations under the License.
     15  */
     16 
     17 #include <gtest/gtest.h>
     18 
     19 #include <stdint.h>
     20 
     21 #include <string>
     22 
     23 static std::string class_with_dtor_output;
     24 
     25 class ClassWithDtor {
     26  public:
     27   void set_message(const std::string& msg) {
     28     message = msg;
     29   }
     30 
     31   ~ClassWithDtor() {
     32     class_with_dtor_output += message;
     33   }
     34  private:
     35   std::string message;
     36 };
     37 
     38 #if defined(__clang__) && defined(__aarch64__)
     39 // b/25642296, aarch64 clang compiled "thread_local" does not link.
     40 static ClassWithDtor class_with_dtor;
     41 #else
     42 static thread_local ClassWithDtor class_with_dtor;
     43 #endif
     44 
     45 static void* thread_nop(void* arg) {
     46   class_with_dtor.set_message(*static_cast<std::string*>(arg));
     47   return nullptr;
     48 }
     49 
     50 TEST(thread_local, smoke) {
     51   std::string msg("dtor called.");
     52   pthread_t t;
     53   ASSERT_EQ(0, pthread_create(&t, nullptr, thread_nop, &msg));
     54   ASSERT_EQ(0, pthread_join(t, nullptr));
     55 #if defined(__clang__) && defined(__aarch64__)
     56   GTEST_LOG_(INFO) << "Skipping test, b/25642296, "
     57                    << "thread_local does not work with aarch64 clang/llvm.\n";
     58 #else
     59   ASSERT_EQ("dtor called.", class_with_dtor_output);
     60 #endif
     61 }
     62 
     63 class ClassWithDtorForMainThread {
     64  public:
     65   void set_message(const std::string& msg) {
     66     message = msg;
     67   }
     68 
     69   ~ClassWithDtorForMainThread() {
     70     fprintf(stderr, "%s", message.c_str());
     71   }
     72  private:
     73   std::string message;
     74 };
     75 
     76 static void thread_atexit_main() {
     77 #if defined(__clang__) && defined(__aarch64__)
     78   static ClassWithDtorForMainThread class_with_dtor_for_main_thread;
     79   GTEST_LOG_(INFO) << "Skipping test, b/25642296, "
     80                    << "thread_local does not work with aarch64 clang/llvm.\n";
     81 #else
     82   static thread_local ClassWithDtorForMainThread class_with_dtor_for_main_thread;
     83 #endif
     84   class_with_dtor_for_main_thread.set_message("d-tor for main thread called.");
     85   exit(0);
     86 }
     87 
     88 TEST(thread_local, dtor_for_main_thread) {
     89   ASSERT_EXIT(thread_atexit_main(), testing::ExitedWithCode(0), "d-tor for main thread called.");
     90 }
     91 
     92 extern "C" int __cxa_thread_atexit_impl(void (*fn)(void*), void* arg, void* dso_handle);
     93 
     94 static void thread_atexit_fn1(void* arg) {
     95   std::string* call_sequence = static_cast<std::string*>(arg);
     96   *call_sequence += "one, ";
     97 }
     98 
     99 static void thread_atexit_fn2(void* arg) {
    100   std::string* call_sequence = static_cast<std::string*>(arg);
    101   *call_sequence += "two, ";
    102 }
    103 
    104 static void thread_atexit_from_atexit(void* arg) {
    105   std::string* call_sequence = static_cast<std::string*>(arg);
    106   *call_sequence += "oops, ";
    107 }
    108 
    109 static void thread_atexit_fn3(void* arg) {
    110   __cxa_thread_atexit_impl(thread_atexit_from_atexit, arg, nullptr);
    111   std::string* call_sequence = static_cast<std::string*>(arg);
    112   *call_sequence += "three, ";
    113 }
    114 
    115 static void thread_atexit_fn4(void* arg) {
    116   std::string* call_sequence = static_cast<std::string*>(arg);
    117   *call_sequence += "four, ";
    118 }
    119 
    120 static void thread_atexit_fn5(void* arg) {
    121   std::string* call_sequence = static_cast<std::string*>(arg);
    122   *call_sequence += "five.";
    123 }
    124 
    125 static void* thread_main(void* arg) {
    126   __cxa_thread_atexit_impl(thread_atexit_fn5, arg, nullptr);
    127   __cxa_thread_atexit_impl(thread_atexit_fn4, arg, nullptr);
    128   __cxa_thread_atexit_impl(thread_atexit_fn3, arg, nullptr);
    129   __cxa_thread_atexit_impl(thread_atexit_fn2, arg, nullptr);
    130   __cxa_thread_atexit_impl(thread_atexit_fn1, arg, nullptr);
    131   return nullptr;
    132 }
    133 
    134 TEST(__cxa_thread_atexit_impl, smoke) {
    135   std::string atexit_call_sequence;
    136 
    137   pthread_t t;
    138   ASSERT_EQ(0, pthread_create(&t, nullptr, thread_main, &atexit_call_sequence));
    139   ASSERT_EQ(0, pthread_join(t, nullptr));
    140   ASSERT_EQ("one, two, three, oops, four, five.", atexit_call_sequence);
    141 }
    142 
    143 
    144