1447762daSMichael J. Spencer //===- ThreadLocal.cpp - Thread Local Data ----------------------*- C++ -*-===//
2447762daSMichael J. Spencer //
3447762daSMichael J. Spencer //                     The LLVM Compiler Infrastructure
4447762daSMichael J. Spencer //
5447762daSMichael J. Spencer // This file is distributed under the University of Illinois Open Source
6447762daSMichael J. Spencer // License. See LICENSE.TXT for details.
7447762daSMichael J. Spencer //
8447762daSMichael J. Spencer //===----------------------------------------------------------------------===//
9447762daSMichael J. Spencer //
10447762daSMichael J. Spencer // This file implements the llvm::sys::ThreadLocal class.
11447762daSMichael J. Spencer //
12447762daSMichael J. Spencer //===----------------------------------------------------------------------===//
13447762daSMichael J. Spencer 
14447762daSMichael J. Spencer #include "llvm/Config/config.h"
151bcdd6aeSAlp Toker #include "llvm/Support/Compiler.h"
16447762daSMichael J. Spencer #include "llvm/Support/ThreadLocal.h"
17447762daSMichael J. Spencer 
18447762daSMichael J. Spencer //===----------------------------------------------------------------------===//
19447762daSMichael J. Spencer //=== WARNING: Implementation here must contain only TRULY operating system
20447762daSMichael J. Spencer //===          independent code.
21447762daSMichael J. Spencer //===----------------------------------------------------------------------===//
22447762daSMichael J. Spencer 
23efddf201SDylan Noblesmith #if !defined(LLVM_ENABLE_THREADS) || LLVM_ENABLE_THREADS == 0
24447762daSMichael J. Spencer // Define all methods as no-ops if threading is explicitly disabled
25447762daSMichael J. Spencer namespace llvm {
26447762daSMichael J. Spencer using namespace sys;
27fabf8bfdSHans Wennborg ThreadLocalImpl::ThreadLocalImpl() : data() { }
28447762daSMichael J. Spencer ThreadLocalImpl::~ThreadLocalImpl() { }
29444fd426SArgyrios Kyrtzidis void ThreadLocalImpl::setInstance(const void* d) {
30*cf01800bSChandler Carruth   static_assert(sizeof(d) <= sizeof(data), "size too big");
31444fd426SArgyrios Kyrtzidis   void **pd = reinterpret_cast<void**>(&data);
32444fd426SArgyrios Kyrtzidis   *pd = const_cast<void*>(d);
33444fd426SArgyrios Kyrtzidis }
3446785f94SArgyrios Kyrtzidis const void* ThreadLocalImpl::getInstance() {
35444fd426SArgyrios Kyrtzidis   void **pd = reinterpret_cast<void**>(&data);
3646785f94SArgyrios Kyrtzidis   return *pd;
3746785f94SArgyrios Kyrtzidis }
3846785f94SArgyrios Kyrtzidis void ThreadLocalImpl::removeInstance() {
3946785f94SArgyrios Kyrtzidis   setInstance(0);
40444fd426SArgyrios Kyrtzidis }
41447762daSMichael J. Spencer }
42447762daSMichael J. Spencer #else
43447762daSMichael J. Spencer 
44447762daSMichael J. Spencer #if defined(HAVE_PTHREAD_H) && defined(HAVE_PTHREAD_GETSPECIFIC)
45447762daSMichael J. Spencer 
46447762daSMichael J. Spencer #include <cassert>
47447762daSMichael J. Spencer #include <pthread.h>
48447762daSMichael J. Spencer #include <stdlib.h>
49447762daSMichael J. Spencer 
50447762daSMichael J. Spencer namespace llvm {
51447762daSMichael J. Spencer using namespace sys;
52447762daSMichael J. Spencer 
53c6dc4d75SArgyrios Kyrtzidis ThreadLocalImpl::ThreadLocalImpl() : data() {
54*cf01800bSChandler Carruth   static_assert(sizeof(pthread_key_t) <= sizeof(data), "size too big");
558d19c86cSArgyrios Kyrtzidis   pthread_key_t* key = reinterpret_cast<pthread_key_t*>(&data);
56447762daSMichael J. Spencer   int errorcode = pthread_key_create(key, NULL);
57447762daSMichael J. Spencer   assert(errorcode == 0);
58447762daSMichael J. Spencer   (void) errorcode;
59447762daSMichael J. Spencer }
60447762daSMichael J. Spencer 
61447762daSMichael J. Spencer ThreadLocalImpl::~ThreadLocalImpl() {
628d19c86cSArgyrios Kyrtzidis   pthread_key_t* key = reinterpret_cast<pthread_key_t*>(&data);
63447762daSMichael J. Spencer   int errorcode = pthread_key_delete(*key);
64447762daSMichael J. Spencer   assert(errorcode == 0);
65447762daSMichael J. Spencer   (void) errorcode;
66447762daSMichael J. Spencer }
67447762daSMichael J. Spencer 
68447762daSMichael J. Spencer void ThreadLocalImpl::setInstance(const void* d) {
698d19c86cSArgyrios Kyrtzidis   pthread_key_t* key = reinterpret_cast<pthread_key_t*>(&data);
70447762daSMichael J. Spencer   int errorcode = pthread_setspecific(*key, d);
71447762daSMichael J. Spencer   assert(errorcode == 0);
72447762daSMichael J. Spencer   (void) errorcode;
73447762daSMichael J. Spencer }
74447762daSMichael J. Spencer 
75447762daSMichael J. Spencer const void* ThreadLocalImpl::getInstance() {
768d19c86cSArgyrios Kyrtzidis   pthread_key_t* key = reinterpret_cast<pthread_key_t*>(&data);
77447762daSMichael J. Spencer   return pthread_getspecific(*key);
78447762daSMichael J. Spencer }
79447762daSMichael J. Spencer 
80447762daSMichael J. Spencer void ThreadLocalImpl::removeInstance() {
81447762daSMichael J. Spencer   setInstance(0);
82447762daSMichael J. Spencer }
83447762daSMichael J. Spencer 
84447762daSMichael J. Spencer }
85447762daSMichael J. Spencer 
86447762daSMichael J. Spencer #elif defined(LLVM_ON_UNIX)
87447762daSMichael J. Spencer #include "Unix/ThreadLocal.inc"
88447762daSMichael J. Spencer #elif defined( LLVM_ON_WIN32)
89447762daSMichael J. Spencer #include "Windows/ThreadLocal.inc"
90447762daSMichael J. Spencer #else
91037fc931SDaniel Dunbar #warning Neither LLVM_ON_UNIX nor LLVM_ON_WIN32 set in Support/ThreadLocal.cpp
92447762daSMichael J. Spencer #endif
93447762daSMichael J. Spencer #endif
94