1 /* Locate TLS data for a thread.
2 Copyright (C) 2003, 2006, 2007 Free Software Foundation, Inc.
3 This file is part of the GNU C Library.
5 The GNU C Library is free software; you can redistribute it and/or
6 modify it under the terms of the GNU Lesser General Public
7 License as published by the Free Software Foundation; either
8 version 2.1 of the License, or (at your option) any later version.
10 The GNU C Library is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 Lesser General Public License for more details.
15 You should have received a copy of the GNU Lesser General Public
16 License along with the GNU C Library; if not, write to the Free
17 Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
20 #include "thread_dbP.h"
23 td_thr_tlsbase (const td_thrhandle_t
*th
,
24 unsigned long int modid
,
28 psaddr_t dtv
, dtvslot
, dtvptr
;
33 psaddr_t pd
= th
->th_unique
;
36 /* This is the fake handle for the main thread before libpthread
37 initialization. We are using 0 for its th_unique because we can't
38 trust that its thread register has been initialized. But we need
39 a real pointer to have any TLS access work. In case of dlopen'd
40 libpthread, initialization might not be for quite some time. So
41 try looking up the thread register now. Worst case, it's nonzero
42 uninitialized garbage and we get bogus results for TLS access
43 attempted too early. Tough. */
45 td_thrhandle_t main_th
;
46 err
= __td_ta_lookup_th_unique (th
->th_ta_p
, ps_getpid (th
->th_ta_p
->ph
),
49 pd
= main_th
.th_unique
;
54 /* Get the DTV pointer from the thread descriptor. */
55 err
= DB_GET_FIELD (dtv
, th
->th_ta_p
, pd
, pthread
, dtvp
, 0);
59 /* Find the corresponding entry in the DTV. */
60 err
= DB_GET_FIELD_ADDRESS (dtvslot
, th
->th_ta_p
, dtv
, dtv
, dtv
, modid
);
64 /* Extract the TLS block address from that DTV slot. */
65 err
= DB_GET_FIELD (dtvptr
, th
->th_ta_p
, dtvslot
, dtv_t
, pointer_val
, 0);
69 /* It could be that the memory for this module is not allocated for
71 if ((uintptr_t) dtvptr
& 1)