~drizzle-trunk/drizzle/development

641.2.2 by Monty Taylor
InnoDB Plugin 1.0.3
1
/*****************************************************************************
2
3
Copyright (c) 1995, 2009, Innobase Oy. All Rights Reserved.
4
5
This program is free software; you can redistribute it and/or modify it under
6
the terms of the GNU General Public License as published by the Free Software
7
Foundation; version 2 of the License.
8
9
This program is distributed in the hope that it will be useful, but WITHOUT
10
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
11
FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
12
13
You should have received a copy of the GNU General Public License along with
14
this program; if not, write to the Free Software Foundation, Inc., 59 Temple
15
Place, Suite 330, Boston, MA 02111-1307 USA
16
17
*****************************************************************************/
18
641.2.3 by Monty Taylor
InnoDB Plugin 1.0.4
19
/**************************************************//**
20
@file thr/thr0loc.c
641.1.2 by Monty Taylor
Imported 1.0.1 with clean - with no changes.
21
The thread local storage
22
23
Created 10/5/1995 Heikki Tuuri
24
*******************************************************/
25
26
#include "thr0loc.h"
27
#ifdef UNIV_NONINL
28
#include "thr0loc.ic"
29
#endif
30
31
#include "sync0sync.h"
32
#include "hash0hash.h"
33
#include "mem0mem.h"
34
#include "srv0srv.h"
35
36
/*
37
	IMPLEMENTATION OF THREAD LOCAL STORAGE
38
	======================================
39
40
The threads sometimes need private data which depends on the thread id.
41
This is implemented as a hash table, where the hash value is calculated
42
from the thread id, to prepare for a large number of threads. The hash table
43
is protected by a mutex. If you need modify the program and put new data to
44
the thread local storage, just add it to struct thr_local_struct in the
45
header file. */
46
641.2.3 by Monty Taylor
InnoDB Plugin 1.0.4
47
/** Mutex protecting thr_local_hash */
641.1.2 by Monty Taylor
Imported 1.0.1 with clean - with no changes.
48
static mutex_t		thr_local_mutex;
49
641.2.3 by Monty Taylor
InnoDB Plugin 1.0.4
50
/** The hash table. The module is not yet initialized when it is NULL. */
641.1.2 by Monty Taylor
Imported 1.0.1 with clean - with no changes.
51
static hash_table_t*	thr_local_hash	= NULL;
52
641.2.3 by Monty Taylor
InnoDB Plugin 1.0.4
53
/** Thread local data */
54
typedef struct thr_local_struct thr_local_t;
55
56
/** @brief Thread local data.
57
The private data for each thread should be put to
641.1.2 by Monty Taylor
Imported 1.0.1 with clean - with no changes.
58
the structure below and the accessor functions written
59
for the field. */
60
struct thr_local_struct{
641.2.3 by Monty Taylor
InnoDB Plugin 1.0.4
61
	os_thread_id_t	id;	/*!< id of the thread which owns this struct */
62
	os_thread_t	handle;	/*!< operating system handle to the thread */
63
	ulint		slot_no;/*!< the index of the slot in the thread table
641.1.2 by Monty Taylor
Imported 1.0.1 with clean - with no changes.
64
				for this thread */
641.2.3 by Monty Taylor
InnoDB Plugin 1.0.4
65
	ibool		in_ibuf;/*!< TRUE if the the thread is doing an ibuf
641.1.2 by Monty Taylor
Imported 1.0.1 with clean - with no changes.
66
				operation */
641.2.3 by Monty Taylor
InnoDB Plugin 1.0.4
67
	hash_node_t	hash;	/*!< hash chain node */
68
	ulint		magic_n;/*!< magic number (THR_LOCAL_MAGIC_N) */
641.1.2 by Monty Taylor
Imported 1.0.1 with clean - with no changes.
69
};
70
641.2.3 by Monty Taylor
InnoDB Plugin 1.0.4
71
/** The value of thr_local_struct::magic_n */
641.1.2 by Monty Taylor
Imported 1.0.1 with clean - with no changes.
72
#define THR_LOCAL_MAGIC_N	1231234
73
641.2.3 by Monty Taylor
InnoDB Plugin 1.0.4
74
/*******************************************************************//**
75
Returns the local storage struct for a thread.
76
@return	local storage */
641.1.2 by Monty Taylor
Imported 1.0.1 with clean - with no changes.
77
static
78
thr_local_t*
79
thr_local_get(
80
/*==========*/
641.2.3 by Monty Taylor
InnoDB Plugin 1.0.4
81
	os_thread_id_t	id)	/*!< in: thread id of the thread */
641.1.2 by Monty Taylor
Imported 1.0.1 with clean - with no changes.
82
{
83
	thr_local_t*	local;
84
85
try_again:
86
	ut_ad(thr_local_hash);
87
	ut_ad(mutex_own(&thr_local_mutex));
88
89
	/* Look for the local struct in the hash table */
90
91
	local = NULL;
92
93
	HASH_SEARCH(hash, thr_local_hash, os_thread_pf(id),
641.2.2 by Monty Taylor
InnoDB Plugin 1.0.3
94
		    thr_local_t*, local,, os_thread_eq(local->id, id));
641.1.2 by Monty Taylor
Imported 1.0.1 with clean - with no changes.
95
	if (local == NULL) {
96
		mutex_exit(&thr_local_mutex);
97
98
		thr_local_create();
99
100
		mutex_enter(&thr_local_mutex);
101
102
		goto try_again;
103
	}
104
105
	ut_ad(local->magic_n == THR_LOCAL_MAGIC_N);
106
107
	return(local);
108
}
109
641.2.3 by Monty Taylor
InnoDB Plugin 1.0.4
110
/*******************************************************************//**
111
Gets the slot number in the thread table of a thread.
112
@return	slot number */
641.1.2 by Monty Taylor
Imported 1.0.1 with clean - with no changes.
113
UNIV_INTERN
114
ulint
115
thr_local_get_slot_no(
116
/*==================*/
641.2.3 by Monty Taylor
InnoDB Plugin 1.0.4
117
	os_thread_id_t	id)	/*!< in: thread id of the thread */
641.1.2 by Monty Taylor
Imported 1.0.1 with clean - with no changes.
118
{
119
	ulint		slot_no;
120
	thr_local_t*	local;
121
122
	mutex_enter(&thr_local_mutex);
123
124
	local = thr_local_get(id);
125
126
	slot_no = local->slot_no;
127
128
	mutex_exit(&thr_local_mutex);
129
130
	return(slot_no);
131
}
132
641.2.3 by Monty Taylor
InnoDB Plugin 1.0.4
133
/*******************************************************************//**
641.1.2 by Monty Taylor
Imported 1.0.1 with clean - with no changes.
134
Sets the slot number in the thread table of a thread. */
135
UNIV_INTERN
136
void
137
thr_local_set_slot_no(
138
/*==================*/
641.2.3 by Monty Taylor
InnoDB Plugin 1.0.4
139
	os_thread_id_t	id,	/*!< in: thread id of the thread */
140
	ulint		slot_no)/*!< in: slot number */
641.1.2 by Monty Taylor
Imported 1.0.1 with clean - with no changes.
141
{
142
	thr_local_t*	local;
143
144
	mutex_enter(&thr_local_mutex);
145
146
	local = thr_local_get(id);
147
148
	local->slot_no = slot_no;
149
150
	mutex_exit(&thr_local_mutex);
151
}
152
641.2.3 by Monty Taylor
InnoDB Plugin 1.0.4
153
/*******************************************************************//**
641.1.2 by Monty Taylor
Imported 1.0.1 with clean - with no changes.
154
Returns pointer to the 'in_ibuf' field within the current thread local
641.2.3 by Monty Taylor
InnoDB Plugin 1.0.4
155
storage.
156
@return	pointer to the in_ibuf field */
641.1.2 by Monty Taylor
Imported 1.0.1 with clean - with no changes.
157
UNIV_INTERN
158
ibool*
159
thr_local_get_in_ibuf_field(void)
160
/*=============================*/
161
{
162
	thr_local_t*	local;
163
164
	mutex_enter(&thr_local_mutex);
165
166
	local = thr_local_get(os_thread_get_curr_id());
167
168
	mutex_exit(&thr_local_mutex);
169
170
	return(&(local->in_ibuf));
171
}
172
641.2.3 by Monty Taylor
InnoDB Plugin 1.0.4
173
/*******************************************************************//**
641.1.2 by Monty Taylor
Imported 1.0.1 with clean - with no changes.
174
Creates a local storage struct for the calling new thread. */
175
UNIV_INTERN
176
void
177
thr_local_create(void)
178
/*==================*/
179
{
180
	thr_local_t*	local;
181
182
	if (thr_local_hash == NULL) {
183
		thr_local_init();
184
	}
185
186
	local = mem_alloc(sizeof(thr_local_t));
187
188
	local->id = os_thread_get_curr_id();
189
	local->handle = os_thread_get_curr();
190
	local->magic_n = THR_LOCAL_MAGIC_N;
191
192
	local->in_ibuf = FALSE;
193
194
	mutex_enter(&thr_local_mutex);
195
196
	HASH_INSERT(thr_local_t, hash, thr_local_hash,
197
		    os_thread_pf(os_thread_get_curr_id()),
198
		    local);
199
200
	mutex_exit(&thr_local_mutex);
201
}
202
641.2.3 by Monty Taylor
InnoDB Plugin 1.0.4
203
/*******************************************************************//**
641.1.2 by Monty Taylor
Imported 1.0.1 with clean - with no changes.
204
Frees the local storage struct for the specified thread. */
205
UNIV_INTERN
206
void
207
thr_local_free(
208
/*===========*/
641.2.3 by Monty Taylor
InnoDB Plugin 1.0.4
209
	os_thread_id_t	id)	/*!< in: thread id */
641.1.2 by Monty Taylor
Imported 1.0.1 with clean - with no changes.
210
{
211
	thr_local_t*	local;
212
213
	mutex_enter(&thr_local_mutex);
214
215
	/* Look for the local struct in the hash table */
216
217
	HASH_SEARCH(hash, thr_local_hash, os_thread_pf(id),
641.2.2 by Monty Taylor
InnoDB Plugin 1.0.3
218
		    thr_local_t*, local,, os_thread_eq(local->id, id));
641.1.2 by Monty Taylor
Imported 1.0.1 with clean - with no changes.
219
	if (local == NULL) {
220
		mutex_exit(&thr_local_mutex);
221
222
		return;
223
	}
224
225
	HASH_DELETE(thr_local_t, hash, thr_local_hash,
226
		    os_thread_pf(id), local);
227
228
	mutex_exit(&thr_local_mutex);
229
230
	ut_a(local->magic_n == THR_LOCAL_MAGIC_N);
231
232
	mem_free(local);
233
}
234
641.2.3 by Monty Taylor
InnoDB Plugin 1.0.4
235
/****************************************************************//**
641.1.2 by Monty Taylor
Imported 1.0.1 with clean - with no changes.
236
Initializes the thread local storage module. */
237
UNIV_INTERN
238
void
239
thr_local_init(void)
240
/*================*/
241
{
242
243
	ut_a(thr_local_hash == NULL);
244
245
	thr_local_hash = hash_create(OS_THREAD_MAX_N + 100);
246
247
	mutex_create(&thr_local_mutex, SYNC_THR_LOCAL);
248
}