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 | } |
Owen Anderson | 438d394 | 2009-06-25 23:31:18 +0000 | [diff] [blame] | 33 | const void* ThreadLocalImpl::getInstance() { return data; } |
Argyrios Kyrtzidis | 1a18e9a | 2012-06-13 16:30:06 +0000 | [diff] [blame^] | 34 | void ThreadLocalImpl::removeInstance() { |
| 35 | void **pd = reinterpret_cast<void**>(&data); |
| 36 | *pd = 0; |
| 37 | } |
Owen Anderson | 4a28522 | 2009-06-25 21:58:01 +0000 | [diff] [blame] | 38 | } |
| 39 | #else |
| 40 | |
Owen Anderson | 27fcfe1 | 2009-06-25 23:10:26 +0000 | [diff] [blame] | 41 | #if defined(HAVE_PTHREAD_H) && defined(HAVE_PTHREAD_GETSPECIFIC) |
Owen Anderson | 4a28522 | 2009-06-25 21:58:01 +0000 | [diff] [blame] | 42 | |
| 43 | #include <cassert> |
| 44 | #include <pthread.h> |
| 45 | #include <stdlib.h> |
| 46 | |
| 47 | namespace llvm { |
| 48 | using namespace sys; |
| 49 | |
Argyrios Kyrtzidis | 3e61374 | 2012-06-12 01:06:16 +0000 | [diff] [blame] | 50 | ThreadLocalImpl::ThreadLocalImpl() : data() { |
Argyrios Kyrtzidis | 793537d | 2012-06-12 00:21:31 +0000 | [diff] [blame] | 51 | typedef int SIZE_TOO_BIG[sizeof(pthread_key_t) <= sizeof(data) ? 1 : -1]; |
| 52 | pthread_key_t* key = reinterpret_cast<pthread_key_t*>(&data); |
Owen Anderson | 4a28522 | 2009-06-25 21:58:01 +0000 | [diff] [blame] | 53 | int errorcode = pthread_key_create(key, NULL); |
| 54 | assert(errorcode == 0); |
Daniel Dunbar | 6f8f606 | 2009-06-26 01:34:35 +0000 | [diff] [blame] | 55 | (void) errorcode; |
Owen Anderson | 4a28522 | 2009-06-25 21:58:01 +0000 | [diff] [blame] | 56 | } |
| 57 | |
| 58 | ThreadLocalImpl::~ThreadLocalImpl() { |
Argyrios Kyrtzidis | 793537d | 2012-06-12 00:21:31 +0000 | [diff] [blame] | 59 | pthread_key_t* key = reinterpret_cast<pthread_key_t*>(&data); |
Owen Anderson | 4a28522 | 2009-06-25 21:58:01 +0000 | [diff] [blame] | 60 | int errorcode = pthread_key_delete(*key); |
Owen Anderson | 7e026b7 | 2009-06-25 23:28:28 +0000 | [diff] [blame] | 61 | assert(errorcode == 0); |
Daniel Dunbar | 6f8f606 | 2009-06-26 01:34:35 +0000 | [diff] [blame] | 62 | (void) errorcode; |
Owen Anderson | 4a28522 | 2009-06-25 21:58:01 +0000 | [diff] [blame] | 63 | } |
| 64 | |
Owen Anderson | 438d394 | 2009-06-25 23:31:18 +0000 | [diff] [blame] | 65 | void ThreadLocalImpl::setInstance(const void* d) { |
Argyrios Kyrtzidis | 793537d | 2012-06-12 00:21:31 +0000 | [diff] [blame] | 66 | pthread_key_t* key = reinterpret_cast<pthread_key_t*>(&data); |
Owen Anderson | 4a28522 | 2009-06-25 21:58:01 +0000 | [diff] [blame] | 67 | int errorcode = pthread_setspecific(*key, d); |
| 68 | assert(errorcode == 0); |
Daniel Dunbar | 6f8f606 | 2009-06-26 01:34:35 +0000 | [diff] [blame] | 69 | (void) errorcode; |
Owen Anderson | 4a28522 | 2009-06-25 21:58:01 +0000 | [diff] [blame] | 70 | } |
| 71 | |
Owen Anderson | 438d394 | 2009-06-25 23:31:18 +0000 | [diff] [blame] | 72 | const void* ThreadLocalImpl::getInstance() { |
Argyrios Kyrtzidis | 793537d | 2012-06-12 00:21:31 +0000 | [diff] [blame] | 73 | pthread_key_t* key = reinterpret_cast<pthread_key_t*>(&data); |
Owen Anderson | 4a28522 | 2009-06-25 21:58:01 +0000 | [diff] [blame] | 74 | return pthread_getspecific(*key); |
| 75 | } |
| 76 | |
Owen Anderson | 826c148 | 2010-07-28 22:49:43 +0000 | [diff] [blame] | 77 | void ThreadLocalImpl::removeInstance() { |
| 78 | setInstance(0); |
| 79 | } |
| 80 | |
Owen Anderson | 4a28522 | 2009-06-25 21:58:01 +0000 | [diff] [blame] | 81 | } |
| 82 | |
| 83 | #elif defined(LLVM_ON_UNIX) |
| 84 | #include "Unix/ThreadLocal.inc" |
| 85 | #elif defined( LLVM_ON_WIN32) |
Michael J. Spencer | 1f6efa3 | 2010-11-29 18:16:10 +0000 | [diff] [blame] | 86 | #include "Windows/ThreadLocal.inc" |
Owen Anderson | 4a28522 | 2009-06-25 21:58:01 +0000 | [diff] [blame] | 87 | #else |
Daniel Dunbar | 2c607b6 | 2011-10-11 20:02:52 +0000 | [diff] [blame] | 88 | #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] | 89 | #endif |
| 90 | #endif |