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