1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
| //===-- NativeThreadDarwin.cpp -------------------------------- -*- C++ -*-===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
#include "NativeThreadDarwin.h"
// C includes
#include <libproc.h>
// LLDB includes
#include "lldb/Utility/Stream.h"
#include "NativeProcessDarwin.h"
using namespace lldb;
using namespace lldb_private;
using namespace lldb_private::process_darwin;
uint64_t NativeThreadDarwin::GetGloballyUniqueThreadIDForMachPortID(
::thread_t mach_port_id) {
thread_identifier_info_data_t tident;
mach_msg_type_number_t tident_count = THREAD_IDENTIFIER_INFO_COUNT;
auto mach_err = ::thread_info(mach_port_id, THREAD_IDENTIFIER_INFO,
(thread_info_t)&tident, &tident_count);
if (mach_err != KERN_SUCCESS) {
// When we fail to get thread info for the supposed port, assume it is
// really a globally unique thread id already, or return the best thing we
// can, which is the thread port.
return mach_port_id;
}
return tident.thread_id;
}
NativeThreadDarwin::NativeThreadDarwin(NativeProcessDarwin *process,
bool is_64_bit,
lldb::tid_t unique_thread_id,
::thread_t mach_thread_port)
: NativeThreadProtocol(process, unique_thread_id),
m_mach_thread_port(mach_thread_port), m_basic_info(),
m_proc_threadinfo() {}
bool NativeThreadDarwin::GetIdentifierInfo() {
// Don't try to get the thread info once and cache it for the life of the
// thread. It changes over time, for instance if the thread name changes,
// then the thread_handle also changes... So you have to refetch it every
// time.
mach_msg_type_number_t count = THREAD_IDENTIFIER_INFO_COUNT;
kern_return_t kret = ::thread_info(m_mach_thread_port, THREAD_IDENTIFIER_INFO,
(thread_info_t)&m_ident_info, &count);
return kret == KERN_SUCCESS;
return false;
}
std::string NativeThreadDarwin::GetName() {
std::string name;
if (GetIdentifierInfo()) {
auto process_sp = GetProcess();
if (!process_sp) {
name = "<unavailable>";
return name;
}
int len = ::proc_pidinfo(process_sp->GetID(), PROC_PIDTHREADINFO,
m_ident_info.thread_handle, &m_proc_threadinfo,
sizeof(m_proc_threadinfo));
if (len && m_proc_threadinfo.pth_name[0])
name = m_proc_threadinfo.pth_name;
}
return name;
}
lldb::StateType NativeThreadDarwin::GetState() {
// TODO implement
return eStateInvalid;
}
bool NativeThreadDarwin::GetStopReason(ThreadStopInfo &stop_info,
std::string &description) {
// TODO implement
return false;
}
NativeRegisterContextSP NativeThreadDarwin::GetRegisterContext() {
// TODO implement
return NativeRegisterContextSP();
}
Status NativeThreadDarwin::SetWatchpoint(lldb::addr_t addr, size_t size,
uint32_t watch_flags, bool hardware) {
Status error;
error.SetErrorString("not yet implemented");
return error;
}
Status NativeThreadDarwin::RemoveWatchpoint(lldb::addr_t addr) {
Status error;
error.SetErrorString("not yet implemented");
return error;
}
void NativeThreadDarwin::Dump(Stream &stream) const {
// This is what we really want once we have the thread class wired up.
#if 0
DNBLogThreaded("[%3u] #%3u tid: 0x%8.8" PRIx64 ", pc: 0x%16.16" PRIx64 ", sp: 0x%16.16" PRIx64 ", user: %d.%6.6d, system: %d.%6.6d, cpu: %2d, policy: %2d, run_state: %2d (%s), flags: %2d, suspend_count: %2d (current %2d), sleep_time: %d",
index,
m_seq_id,
m_unique_id,
GetPC(INVALID_NUB_ADDRESS),
GetSP(INVALID_NUB_ADDRESS),
m_basic_info.user_time.seconds, m_basic_info.user_time.microseconds,
m_basic_info.system_time.seconds, m_basic_info.system_time.microseconds,
m_basic_info.cpu_usage,
m_basic_info.policy,
m_basic_info.run_state,
thread_run_state,
m_basic_info.flags,
m_basic_info.suspend_count, m_suspend_count,
m_basic_info.sleep_time);
#else
// Here's all we have right now.
stream.Printf("tid: 0x%8.8" PRIx64 ", thread port: 0x%4.4x", GetID(),
m_mach_thread_port);
#endif
}
bool NativeThreadDarwin::NotifyException(MachException::Data &exc) {
// TODO implement this.
#if 0
// Allow the arch specific protocol to process (MachException::Data &)exc
// first before possible reassignment of m_stop_exception with exc. See
// also MachThread::GetStopException().
bool handled = m_arch_up->NotifyException(exc);
if (m_stop_exception.IsValid())
{
// We may have more than one exception for a thread, but we need to
// only remember the one that we will say is the reason we stopped. We
// may have been single stepping and also gotten a signal exception, so
// just remember the most pertinent one.
if (m_stop_exception.IsBreakpoint())
m_stop_exception = exc;
}
else
{
m_stop_exception = exc;
}
return handled;
#else
// Pretend we handled it.
return true;
#endif
}
bool NativeThreadDarwin::ShouldStop(bool &step_more) const {
// TODO: implement this
#if 0
// See if this thread is at a breakpoint?
DNBBreakpoint *bp = CurrentBreakpoint();
if (bp)
{
// This thread is sitting at a breakpoint, ask the breakpoint if we
// should be stopping here.
return true;
}
else
{
if (m_arch_up->StepNotComplete())
{
step_more = true;
return false;
}
// The thread state is used to let us know what the thread was trying
// to do. MachThread::ThreadWillResume() will set the thread state to
// various values depending if the thread was the current thread and if
// it was to be single stepped, or resumed.
if (GetState() == eStateRunning)
{
// If our state is running, then we should continue as we are in
// the process of stepping over a breakpoint.
return false;
}
else
{
// Stop if we have any kind of valid exception for this thread.
if (GetStopException().IsValid())
return true;
}
}
return false;
#else
return false;
#endif
}
void NativeThreadDarwin::ThreadDidStop() {
// TODO implement this.
#if 0
// This thread has existed prior to resuming under debug nub control, and
// has just been stopped. Do any cleanup that needs to be done after
// running.
// The thread state and breakpoint will still have the same values as they
// had prior to resuming the thread, so it makes it easy to check if we
// were trying to step a thread, or we tried to resume while being at a
// breakpoint.
// When this method gets called, the process state is still in the state it
// was in while running so we can act accordingly.
m_arch_up->ThreadDidStop();
// We may have suspended this thread so the primary thread could step
// without worrying about race conditions, so lets restore our suspend
// count.
RestoreSuspendCountAfterStop();
// Update the basic information for a thread
MachThread::GetBasicInfo(m_mach_port_number, &m_basic_info);
if (m_basic_info.suspend_count > 0)
SetState(eStateSuspended);
else
SetState(eStateStopped);
#endif
}
bool NativeThreadDarwin::MachPortNumberIsValid(::thread_t thread) {
return thread != (::thread_t)(0);
}
const struct thread_basic_info *NativeThreadDarwin::GetBasicInfo() const {
if (GetBasicInfo(m_mach_thread_port, &m_basic_info))
return &m_basic_info;
return NULL;
}
bool NativeThreadDarwin::GetBasicInfo(::thread_t thread,
struct thread_basic_info *basicInfoPtr) {
if (MachPortNumberIsValid(thread)) {
unsigned int info_count = THREAD_BASIC_INFO_COUNT;
kern_return_t err = ::thread_info(thread, THREAD_BASIC_INFO,
(thread_info_t)basicInfoPtr, &info_count);
if (err == KERN_SUCCESS)
return true;
}
::memset(basicInfoPtr, 0, sizeof(struct thread_basic_info));
return false;
}
bool NativeThreadDarwin::IsUserReady() const {
if (m_basic_info.run_state == 0)
GetBasicInfo();
switch (m_basic_info.run_state) {
default:
case TH_STATE_UNINTERRUPTIBLE:
break;
case TH_STATE_RUNNING:
case TH_STATE_STOPPED:
case TH_STATE_WAITING:
case TH_STATE_HALTED:
return true;
}
return false;
}
NativeProcessDarwinSP NativeThreadDarwin::GetNativeProcessDarwinSP() {
return std::static_pointer_cast<NativeProcessDarwin>(GetProcess());
}
|