Owen Anderson | 4a28522 | 2009-06-25 21:58:01 +0000 | [diff] [blame] | 1 | //===- ThreadLocal.cpp - Thread Local Data ----------------------*- C++ -*-===// |
| 2 | // |
| 3 | // The LLVM Compiler Infrastructure |
| 4 | // |
| 5 | // This file is distributed under the University of Illinois Open Source |
| 6 | // License. See LICENSE.TXT for details. |
| 7 | // |
| 8 | //===----------------------------------------------------------------------===// |
| 9 | // |
| 10 | // This file implements the llvm::sys::ThreadLocal class. |
| 11 | // |
| 12 | //===----------------------------------------------------------------------===// |
| 13 | |
| 14 | #include "llvm/Config/config.h" |
Michael J. Spencer | 1f6efa3 | 2010-11-29 18:16:10 +0000 | [diff] [blame] | 15 | #include "llvm/Support/ThreadLocal.h" |
Owen Anderson | 4a28522 | 2009-06-25 21:58:01 +0000 | [diff] [blame] | 16 | |
| 17 | //===----------------------------------------------------------------------===// |
| 18 | //=== WARNING: Implementation here must contain only TRULY operating system |
| 19 | //=== independent code. |
| 20 | //===----------------------------------------------------------------------===// |
| 21 | |
Dylan Noblesmith | 08b73a3 | 2011-11-28 00:48:58 +0000 | [diff] [blame] | 22 | #if !defined(LLVM_ENABLE_THREADS) || LLVM_ENABLE_THREADS == 0 |
Owen Anderson | 4a28522 | 2009-06-25 21:58:01 +0000 | [diff] [blame] | 23 | // Define all methods as no-ops if threading is explicitly disabled |
| 24 | namespace llvm { |
| 25 | using namespace sys; |
| 26 | ThreadLocalImpl::ThreadLocalImpl() { } |
| 27 | ThreadLocalImpl::~ThreadLocalImpl() { } |
Argyrios Kyrtzidis | 1a18e9a | 2012-06-13 16:30:06 +0000 | [diff] [blame] | 28 | void ThreadLocalImpl::setInstance(const void* d) { |
| 29 | typedef int SIZE_TOO_BIG[sizeof(d) <= sizeof(data) ? 1 : -1]; |
| 30 | void **pd = reinterpret_cast<void**>(&data); |
| 31 | *pd = const_cast<void*>(d); |
| 32 | } |
Argyrios Kyrtzidis | da72dd2 | 2012-06-26 17:13:58 +0000 | [diff] [blame] | 33 | const void* ThreadLocalImpl::getInstance() { |
Argyrios Kyrtzidis | 1a18e9a | 2012-06-13 16:30:06 +0000 | [diff] [blame] | 34 | void **pd = reinterpret_cast<void**>(&data); |
Argyrios Kyrtzidis | da72dd2 | 2012-06-26 17:13:58 +0000 | [diff] [blame] | 35 | return *pd; |
| 36 | } |
| 37 | void ThreadLocalImpl::removeInstance() { |
| 38 | setInstance(0); |
Argyrios Kyrtzidis | 1a18e9a | 2012-06-13 16:30:06 +0000 | [diff] [blame] | 39 | } |
Owen Anderson | 4a28522 | 2009-06-25 21:58:01 +0000 | [diff] [blame] | 40 | } |
| 41 | #else |
| 42 | |
Owen Anderson | 27fcfe1 | 2009-06-25 23:10:26 +0000 | [diff] [blame] | 43 | #if defined(HAVE_PTHREAD_H) && defined(HAVE_PTHREAD_GETSPECIFIC) |
Owen Anderson | 4a28522 | 2009-06-25 21:58:01 +0000 | [diff] [blame] | 44 | |
| 45 | #include <cassert> |
| 46 | #include <pthread.h> |
| 47 | #include <stdlib.h> |
| 48 | |
| 49 | namespace llvm { |
| 50 | using namespace sys; |
| 51 | |
Argyrios Kyrtzidis | 3e61374 | 2012-06-12 01:06:16 +0000 | [diff] [blame] | 52 | ThreadLocalImpl::ThreadLocalImpl() : data() { |
Argyrios Kyrtzidis | 793537d | 2012-06-12 00:21:31 +0000 | [diff] [blame] | 53 | typedef int SIZE_TOO_BIG[sizeof(pthread_key_t) <= sizeof(data) ? 1 : -1]; |
| 54 | pthread_key_t* key = reinterpret_cast<pthread_key_t*>(&data); |
Owen Anderson | 4a28522 | 2009-06-25 21:58:01 +0000 | [diff] [blame] | 55 | int errorcode = pthread_key_create(key, NULL); |
| 56 | assert(errorcode == 0); |
Daniel Dunbar | 6f8f606 | 2009-06-26 01:34:35 +0000 | [diff] [blame] | 57 | (void) errorcode; |
Owen Anderson | 4a28522 | 2009-06-25 21:58:01 +0000 | [diff] [blame] | 58 | } |
| 59 | |
| 60 | ThreadLocalImpl::~ThreadLocalImpl() { |
Argyrios Kyrtzidis | 793537d | 2012-06-12 00:21:31 +0000 | [diff] [blame] | 61 | pthread_key_t* key = reinterpret_cast<pthread_key_t*>(&data); |
Owen Anderson | 4a28522 | 2009-06-25 21:58:01 +0000 | [diff] [blame] | 62 | int errorcode = pthread_key_delete(*key); |
Owen Anderson | 7e026b7 | 2009-06-25 23:28:28 +0000 | [diff] [blame] | 63 | assert(errorcode == 0); |
Daniel Dunbar | 6f8f606 | 2009-06-26 01:34:35 +0000 | [diff] [blame] | 64 | (void) errorcode; |
Owen Anderson | 4a28522 | 2009-06-25 21:58:01 +0000 | [diff] [blame] | 65 | } |
| 66 | |
Owen Anderson | 438d394 | 2009-06-25 23:31:18 +0000 | [diff] [blame] | 67 | void ThreadLocalImpl::setInstance(const void* d) { |
Argyrios Kyrtzidis | 793537d | 2012-06-12 00:21:31 +0000 | [diff] [blame] | 68 | pthread_key_t* key = reinterpret_cast<pthread_key_t*>(&data); |
Owen Anderson | 4a28522 | 2009-06-25 21:58:01 +0000 | [diff] [blame] | 69 | int errorcode = pthread_setspecific(*key, d); |
| 70 | assert(errorcode == 0); |
Daniel Dunbar | 6f8f606 | 2009-06-26 01:34:35 +0000 | [diff] [blame] | 71 | (void) errorcode; |
Owen Anderson | 4a28522 | 2009-06-25 21:58:01 +0000 | [diff] [blame] | 72 | } |
| 73 | |
Owen Anderson | 438d394 | 2009-06-25 23:31:18 +0000 | [diff] [blame] | 74 | const void* ThreadLocalImpl::getInstance() { |
Argyrios Kyrtzidis | 793537d | 2012-06-12 00:21:31 +0000 | [diff] [blame] | 75 | pthread_key_t* key = reinterpret_cast<pthread_key_t*>(&data); |
Owen Anderson | 4a28522 | 2009-06-25 21:58:01 +0000 | [diff] [blame] | 76 | return pthread_getspecific(*key); |
| 77 | } |
| 78 | |
Owen Anderson | 826c148 | 2010-07-28 22:49:43 +0000 | [diff] [blame] | 79 | void ThreadLocalImpl::removeInstance() { |
| 80 | setInstance(0); |
| 81 | } |
| 82 | |
Owen Anderson | 4a28522 | 2009-06-25 21:58:01 +0000 | [diff] [blame] | 83 | } |
| 84 | |
| 85 | #elif defined(LLVM_ON_UNIX) |
| 86 | #include "Unix/ThreadLocal.inc" |
| 87 | #elif defined( LLVM_ON_WIN32) |
Michael J. Spencer | 1f6efa3 | 2010-11-29 18:16:10 +0000 | [diff] [blame] | 88 | #include "Windows/ThreadLocal.inc" |
Owen Anderson | 4a28522 | 2009-06-25 21:58:01 +0000 | [diff] [blame] | 89 | #else |
Daniel Dunbar | 2c607b6 | 2011-10-11 20:02:52 +0000 | [diff] [blame] | 90 | #warning Neither LLVM_ON_UNIX nor LLVM_ON_WIN32 set in Support/ThreadLocal.cpp |
Owen Anderson | 4a28522 | 2009-06-25 21:58:01 +0000 | [diff] [blame] | 91 | #endif |
| 92 | #endif |