blob: ec69a51c3d366caf805d9fb945e9abc6007f9e06 [file] [log] [blame]
Jason Evanscd9a1342012-03-21 18:33:03 -07001#define JEMALLOC_TSD_C_
2#include "jemalloc/internal/jemalloc_internal.h"
3
4/******************************************************************************/
5/* Data. */
6
7static unsigned ncleanups;
8static malloc_tsd_cleanup_t cleanups[MALLOC_TSD_CLEANUPS_MAX];
9
Jason Evans5460aa62014-09-22 21:09:23 -070010malloc_tsd_data(, , tsd_t, TSD_INITIALIZER)
11
Jason Evanscd9a1342012-03-21 18:33:03 -070012/******************************************************************************/
13
14void *
15malloc_tsd_malloc(size_t size)
16{
17
Jason Evans4581b972014-11-27 17:22:36 -020018 return (a0malloc(CACHELINE_CEILING(size)));
Jason Evanscd9a1342012-03-21 18:33:03 -070019}
20
21void
22malloc_tsd_dalloc(void *wrapper)
23{
24
Jason Evans10aff3f2015-01-20 15:37:51 -080025 a0dalloc(wrapper);
Jason Evanscd9a1342012-03-21 18:33:03 -070026}
27
28void
29malloc_tsd_no_cleanup(void *arg)
30{
31
32 not_reached();
33}
34
Mike Hommeya19e87f2012-04-21 21:27:46 -070035#if defined(JEMALLOC_MALLOC_THREAD_CLEANUP) || defined(_WIN32)
Mike Hommeyda99e312012-04-30 12:38:29 +020036#ifndef _WIN32
37JEMALLOC_EXPORT
38#endif
Jason Evanscd9a1342012-03-21 18:33:03 -070039void
40_malloc_thread_cleanup(void)
41{
Mike Hommey8b499712012-04-24 23:22:02 +020042 bool pending[MALLOC_TSD_CLEANUPS_MAX], again;
Jason Evanscd9a1342012-03-21 18:33:03 -070043 unsigned i;
44
45 for (i = 0; i < ncleanups; i++)
46 pending[i] = true;
47
48 do {
49 again = false;
50 for (i = 0; i < ncleanups; i++) {
51 if (pending[i]) {
Mike Hommey13067ec2012-04-18 18:29:49 +020052 pending[i] = cleanups[i]();
Jason Evanscd9a1342012-03-21 18:33:03 -070053 if (pending[i])
54 again = true;
55 }
56 }
57 } while (again);
58}
59#endif
60
61void
Mike Hommey13067ec2012-04-18 18:29:49 +020062malloc_tsd_cleanup_register(bool (*f)(void))
Jason Evanscd9a1342012-03-21 18:33:03 -070063{
64
65 assert(ncleanups < MALLOC_TSD_CLEANUPS_MAX);
Mike Hommey13067ec2012-04-18 18:29:49 +020066 cleanups[ncleanups] = f;
Jason Evanscd9a1342012-03-21 18:33:03 -070067 ncleanups++;
68}
69
70void
Jason Evans5460aa62014-09-22 21:09:23 -070071tsd_cleanup(void *arg)
72{
73 tsd_t *tsd = (tsd_t *)arg;
74
Jason Evans5460aa62014-09-22 21:09:23 -070075 switch (tsd->state) {
Jason Evansd36c7eb2015-09-24 16:53:18 -070076 case tsd_state_uninitialized:
77 /* Do nothing. */
78 break;
Jason Evans5460aa62014-09-22 21:09:23 -070079 case tsd_state_nominal:
Jason Evans1423ee92016-04-17 10:30:25 -070080#define O(n, t) \
Jason Evans5460aa62014-09-22 21:09:23 -070081 n##_cleanup(tsd);
82MALLOC_TSD
83#undef O
84 tsd->state = tsd_state_purgatory;
85 tsd_set(tsd);
86 break;
87 case tsd_state_purgatory:
88 /*
89 * The previous time this destructor was called, we set the
90 * state to tsd_state_purgatory so that other destructors
91 * wouldn't cause re-creation of the tsd. This time, do
92 * nothing, and do not request another callback.
93 */
94 break;
95 case tsd_state_reincarnated:
96 /*
97 * Another destructor deallocated memory after this destructor
98 * was called. Reset state to tsd_state_purgatory and request
99 * another callback.
100 */
101 tsd->state = tsd_state_purgatory;
102 tsd_set(tsd);
103 break;
104 default:
105 not_reached();
106 }
107}
108
Jason Evansb2c0d632016-04-13 23:36:15 -0700109tsd_t *
Jason Evans8bb31982014-10-07 23:14:57 -0700110malloc_tsd_boot0(void)
Jason Evanscd9a1342012-03-21 18:33:03 -0700111{
Jason Evansb2c0d632016-04-13 23:36:15 -0700112 tsd_t *tsd;
Jason Evanscd9a1342012-03-21 18:33:03 -0700113
114 ncleanups = 0;
Jason Evans8bb31982014-10-07 23:14:57 -0700115 if (tsd_boot0())
Jason Evansb2c0d632016-04-13 23:36:15 -0700116 return (NULL);
117 tsd = tsd_fetch();
118 *tsd_arenas_tdata_bypassp_get(tsd) = true;
119 return (tsd);
Jason Evanscd9a1342012-03-21 18:33:03 -0700120}
Mike Hommeya19e87f2012-04-21 21:27:46 -0700121
Jason Evans8bb31982014-10-07 23:14:57 -0700122void
123malloc_tsd_boot1(void)
124{
125
126 tsd_boot1();
Jason Evansdb927b62016-02-19 19:37:10 -0800127 *tsd_arenas_tdata_bypassp_get(tsd_fetch()) = false;
Jason Evans8bb31982014-10-07 23:14:57 -0700128}
129
Mike Hommeya19e87f2012-04-21 21:27:46 -0700130#ifdef _WIN32
131static BOOL WINAPI
132_tls_callback(HINSTANCE hinstDLL, DWORD fdwReason, LPVOID lpvReserved)
133{
134
135 switch (fdwReason) {
136#ifdef JEMALLOC_LAZY_LOCK
137 case DLL_THREAD_ATTACH:
138 isthreaded = true;
139 break;
140#endif
141 case DLL_THREAD_DETACH:
142 _malloc_thread_cleanup();
143 break;
144 default:
145 break;
146 }
147 return (true);
148}
149
Mike Hommeyfd97b1d2012-04-30 12:38:31 +0200150#ifdef _MSC_VER
151# ifdef _M_IX86
152# pragma comment(linker, "/INCLUDE:__tls_used")
rustyxefbee862016-02-02 11:27:18 +0100153# pragma comment(linker, "/INCLUDE:_tls_callback")
Mike Hommeyfd97b1d2012-04-30 12:38:31 +0200154# else
155# pragma comment(linker, "/INCLUDE:_tls_used")
rustyxefbee862016-02-02 11:27:18 +0100156# pragma comment(linker, "/INCLUDE:tls_callback")
Mike Hommeyfd97b1d2012-04-30 12:38:31 +0200157# endif
158# pragma section(".CRT$XLY",long,read)
159#endif
Mike Hommeyda99e312012-04-30 12:38:29 +0200160JEMALLOC_SECTION(".CRT$XLY") JEMALLOC_ATTR(used)
rustyxefbee862016-02-02 11:27:18 +0100161BOOL (WINAPI *const tls_callback)(HINSTANCE hinstDLL,
Jason Evansd508ec72015-07-07 20:28:22 -0700162 DWORD fdwReason, LPVOID lpvReserved) = _tls_callback;
Mike Hommeya19e87f2012-04-21 21:27:46 -0700163#endif
Leonard Crestezcb17fc62013-10-22 00:12:16 +0300164
165#if (!defined(JEMALLOC_MALLOC_THREAD_CLEANUP) && !defined(JEMALLOC_TLS) && \
166 !defined(_WIN32))
167void *
168tsd_init_check_recursion(tsd_init_head_t *head, tsd_init_block_t *block)
169{
170 pthread_t self = pthread_self();
171 tsd_init_block_t *iter;
172
173 /* Check whether this thread has already inserted into the list. */
Jason Evansbcd54242016-05-12 21:07:08 -0700174 malloc_mutex_lock(TSDN_NULL, &head->lock);
Leonard Crestezcb17fc62013-10-22 00:12:16 +0300175 ql_foreach(iter, &head->blocks, link) {
176 if (iter->thread == self) {
Jason Evansbcd54242016-05-12 21:07:08 -0700177 malloc_mutex_unlock(TSDN_NULL, &head->lock);
Leonard Crestezcb17fc62013-10-22 00:12:16 +0300178 return (iter->data);
179 }
180 }
181 /* Insert block into list. */
182 ql_elm_new(block, link);
183 block->thread = self;
184 ql_tail_insert(&head->blocks, block, link);
Jason Evansbcd54242016-05-12 21:07:08 -0700185 malloc_mutex_unlock(TSDN_NULL, &head->lock);
Leonard Crestezcb17fc62013-10-22 00:12:16 +0300186 return (NULL);
187}
188
189void
190tsd_init_finish(tsd_init_head_t *head, tsd_init_block_t *block)
191{
192
Jason Evansbcd54242016-05-12 21:07:08 -0700193 malloc_mutex_lock(TSDN_NULL, &head->lock);
Leonard Crestezcb17fc62013-10-22 00:12:16 +0300194 ql_remove(&head->blocks, block, link);
Jason Evansbcd54242016-05-12 21:07:08 -0700195 malloc_mutex_unlock(TSDN_NULL, &head->lock);
Leonard Crestezcb17fc62013-10-22 00:12:16 +0300196}
197#endif