1 /* Get thread information.
2 Copyright (C) 1999-2003,2007,2009 Free Software Foundation, Inc.
3 This file is part of the GNU C Library.
4 Contributed by Ulrich Drepper <drepper@redhat.com>, 1999.
6 The GNU C Library is free software; you can redistribute it and/or
7 modify it under the terms of the GNU Lesser General Public
8 License as published by the Free Software Foundation; either
9 version 2.1 of the License, or (at your option) any later version.
11 The GNU C Library is distributed in the hope that it will be useful,
12 but WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 Lesser General Public License for more details.
16 You should have received a copy of the GNU Lesser General Public
17 License along with the GNU C Library; if not, see
18 <http://www.gnu.org/licenses/>. */
22 #include "thread_dbP.h"
26 td_thr_get_info (const td_thrhandle_t
*th
, td_thrinfo_t
*infop
)
30 psaddr_t tls
, schedpolicy
, schedprio
, cancelhandling
, tid
, report_events
;
32 LOG ("td_thr_get_info");
34 if (th
->th_unique
== 0)
36 /* Special case for the main thread before initialization. */
40 schedpolicy
= SCHED_OTHER
;
43 err
= DB_GET_VALUE (report_events
, th
->th_ta_p
,
44 __nptl_initial_report_events
, 0);
48 /* Copy the whole descriptor in once so we can access the several
49 fields locally. Excess copying in one go is much better than
50 multiple ps_pdread calls. */
51 err
= DB_GET_STRUCT (copy
, th
->th_ta_p
, th
->th_unique
, pthread
);
55 err
= DB_GET_FIELD_ADDRESS (tls
, th
->th_ta_p
, th
->th_unique
,
56 pthread
, specific
, 0);
60 err
= DB_GET_FIELD_LOCAL (schedpolicy
, th
->th_ta_p
, copy
, pthread
,
64 err
= DB_GET_FIELD_LOCAL (schedprio
, th
->th_ta_p
, copy
, pthread
,
65 schedparam_sched_priority
, 0);
68 err
= DB_GET_FIELD_LOCAL (tid
, th
->th_ta_p
, copy
, pthread
, tid
, 0);
71 err
= DB_GET_FIELD_LOCAL (cancelhandling
, th
->th_ta_p
, copy
, pthread
,
75 err
= DB_GET_FIELD_LOCAL (report_events
, th
->th_ta_p
, copy
, pthread
,
81 /* Fill in information. Clear first to provide reproducable
82 results for the fields we do not fill in. */
83 memset (infop
, '\0', sizeof (td_thrinfo_t
));
85 infop
->ti_tid
= (thread_t
) th
->th_unique
;
86 infop
->ti_tls
= (char *) tls
;
87 infop
->ti_pri
= ((uintptr_t) schedpolicy
== SCHED_OTHER
88 ? 0 : (uintptr_t) schedprio
);
89 infop
->ti_type
= TD_THR_USER
;
91 if ((((int) (uintptr_t) cancelhandling
) & EXITING_BITMASK
) == 0)
92 /* XXX For now there is no way to get more information. */
93 infop
->ti_state
= TD_THR_ACTIVE
;
94 else if ((((int) (uintptr_t) cancelhandling
) & TERMINATED_BITMASK
) == 0)
95 infop
->ti_state
= TD_THR_ZOMBIE
;
97 infop
->ti_state
= TD_THR_UNKNOWN
;
99 /* Initialization which are the same in both cases. */
100 infop
->ti_ta_p
= th
->th_ta_p
;
101 infop
->ti_lid
= tid
== 0 ? ps_getpid (th
->th_ta_p
->ph
) : (uintptr_t) tid
;
102 infop
->ti_traceme
= report_events
!= 0;
105 err
= DB_GET_FIELD_LOCAL (infop
->ti_startfunc
, th
->th_ta_p
, copy
, pthread
,
107 if (copy
!= NULL
&& err
== TD_OK
)
110 for (idx
= 0; idx
< TD_EVENTSIZE
; ++idx
)
113 err
= DB_GET_FIELD_LOCAL (word
, th
->th_ta_p
, copy
, pthread
,
114 eventbuf_eventmask_event_bits
, idx
);
117 infop
->ti_events
.event_bits
[idx
] = (uintptr_t) word
;
119 if (err
== TD_NOAPLIC
)
120 memset (&infop
->ti_events
.event_bits
[idx
], 0,
121 (TD_EVENTSIZE
- idx
) * sizeof infop
->ti_events
.event_bits
[0]);