| // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 
 | // Use of this source code is governed by a BSD-style license that can be | 
 | // found in the LICENSE file. | 
 |  | 
 | #include "base/logging.h" | 
 | #include "base/threading/simple_thread.h" | 
 | #include "base/threading/thread_local.h" | 
 | #include "base/synchronization/waitable_event.h" | 
 | #include "testing/gtest/include/gtest/gtest.h" | 
 |  | 
 | namespace base { | 
 |  | 
 | namespace { | 
 |  | 
 | class ThreadLocalTesterBase : public base::DelegateSimpleThreadPool::Delegate { | 
 |  public: | 
 |   typedef base::ThreadLocalPointer<char> TLPType; | 
 |  | 
 |   ThreadLocalTesterBase(TLPType* tlp, base::WaitableEvent* done) | 
 |       : tlp_(tlp), | 
 |         done_(done) { | 
 |   } | 
 |   ~ThreadLocalTesterBase() override = default; | 
 |  | 
 |  protected: | 
 |   TLPType* tlp_; | 
 |   base::WaitableEvent* done_; | 
 | }; | 
 |  | 
 | class SetThreadLocal : public ThreadLocalTesterBase { | 
 |  public: | 
 |   SetThreadLocal(TLPType* tlp, base::WaitableEvent* done) | 
 |       : ThreadLocalTesterBase(tlp, done), val_(nullptr) {} | 
 |   ~SetThreadLocal() override = default; | 
 |  | 
 |   void set_value(char* val) { val_ = val; } | 
 |  | 
 |   void Run() override { | 
 |     DCHECK(!done_->IsSignaled()); | 
 |     tlp_->Set(val_); | 
 |     done_->Signal(); | 
 |   } | 
 |  | 
 |  private: | 
 |   char* val_; | 
 | }; | 
 |  | 
 | class GetThreadLocal : public ThreadLocalTesterBase { | 
 |  public: | 
 |   GetThreadLocal(TLPType* tlp, base::WaitableEvent* done) | 
 |       : ThreadLocalTesterBase(tlp, done), ptr_(nullptr) {} | 
 |   ~GetThreadLocal() override = default; | 
 |  | 
 |   void set_ptr(char** ptr) { ptr_ = ptr; } | 
 |  | 
 |   void Run() override { | 
 |     DCHECK(!done_->IsSignaled()); | 
 |     *ptr_ = tlp_->Get(); | 
 |     done_->Signal(); | 
 |   } | 
 |  | 
 |  private: | 
 |   char** ptr_; | 
 | }; | 
 |  | 
 | }  // namespace | 
 |  | 
 | // In this test, we start 2 threads which will access a ThreadLocalPointer.  We | 
 | // make sure the default is NULL, and the pointers are unique to the threads. | 
 | TEST(ThreadLocalTest, Pointer) { | 
 |   base::DelegateSimpleThreadPool tp1("ThreadLocalTest tp1", 1); | 
 |   base::DelegateSimpleThreadPool tp2("ThreadLocalTest tp1", 1); | 
 |   tp1.Start(); | 
 |   tp2.Start(); | 
 |  | 
 |   base::ThreadLocalPointer<char> tlp; | 
 |  | 
 |   static char* const kBogusPointer = reinterpret_cast<char*>(0x1234); | 
 |  | 
 |   char* tls_val; | 
 |   base::WaitableEvent done(WaitableEvent::ResetPolicy::MANUAL, | 
 |                            WaitableEvent::InitialState::NOT_SIGNALED); | 
 |  | 
 |   GetThreadLocal getter(&tlp, &done); | 
 |   getter.set_ptr(&tls_val); | 
 |  | 
 |   // Check that both threads defaulted to NULL. | 
 |   tls_val = kBogusPointer; | 
 |   done.Reset(); | 
 |   tp1.AddWork(&getter); | 
 |   done.Wait(); | 
 |   EXPECT_EQ(static_cast<char*>(nullptr), tls_val); | 
 |  | 
 |   tls_val = kBogusPointer; | 
 |   done.Reset(); | 
 |   tp2.AddWork(&getter); | 
 |   done.Wait(); | 
 |   EXPECT_EQ(static_cast<char*>(nullptr), tls_val); | 
 |  | 
 |   SetThreadLocal setter(&tlp, &done); | 
 |   setter.set_value(kBogusPointer); | 
 |  | 
 |   // Have thread 1 set their pointer value to kBogusPointer. | 
 |   done.Reset(); | 
 |   tp1.AddWork(&setter); | 
 |   done.Wait(); | 
 |  | 
 |   tls_val = nullptr; | 
 |   done.Reset(); | 
 |   tp1.AddWork(&getter); | 
 |   done.Wait(); | 
 |   EXPECT_EQ(kBogusPointer, tls_val); | 
 |  | 
 |   // Make sure thread 2 is still NULL | 
 |   tls_val = kBogusPointer; | 
 |   done.Reset(); | 
 |   tp2.AddWork(&getter); | 
 |   done.Wait(); | 
 |   EXPECT_EQ(static_cast<char*>(nullptr), tls_val); | 
 |  | 
 |   // Set thread 2 to kBogusPointer + 1. | 
 |   setter.set_value(kBogusPointer + 1); | 
 |  | 
 |   done.Reset(); | 
 |   tp2.AddWork(&setter); | 
 |   done.Wait(); | 
 |  | 
 |   tls_val = nullptr; | 
 |   done.Reset(); | 
 |   tp2.AddWork(&getter); | 
 |   done.Wait(); | 
 |   EXPECT_EQ(kBogusPointer + 1, tls_val); | 
 |  | 
 |   // Make sure thread 1 is still kBogusPointer. | 
 |   tls_val = nullptr; | 
 |   done.Reset(); | 
 |   tp1.AddWork(&getter); | 
 |   done.Wait(); | 
 |   EXPECT_EQ(kBogusPointer, tls_val); | 
 |  | 
 |   tp1.JoinAll(); | 
 |   tp2.JoinAll(); | 
 | } | 
 |  | 
 | TEST(ThreadLocalTest, Boolean) { | 
 |   { | 
 |     base::ThreadLocalBoolean tlb; | 
 |     EXPECT_FALSE(tlb.Get()); | 
 |  | 
 |     tlb.Set(false); | 
 |     EXPECT_FALSE(tlb.Get()); | 
 |  | 
 |     tlb.Set(true); | 
 |     EXPECT_TRUE(tlb.Get()); | 
 |   } | 
 |  | 
 |   // Our slot should have been freed, we're all reset. | 
 |   { | 
 |     base::ThreadLocalBoolean tlb; | 
 |     EXPECT_FALSE(tlb.Get()); | 
 |   } | 
 | } | 
 |  | 
 | }  // namespace base |